854 lines
22 KiB
C++
854 lines
22 KiB
C++
/**
|
|
* @file lldrawpoolsimple.cpp
|
|
* @brief LLDrawPoolSimple class implementation
|
|
*
|
|
* $LicenseInfo:firstyear=2002&license=viewerlgpl$
|
|
* Second Life Viewer Source Code
|
|
* Copyright (C) 2010, Linden Research, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation;
|
|
* version 2.1 of the License only.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
* Linden Research, Inc., 945 Battery Street, San Francisco, CA 94111 USA
|
|
* $/LicenseInfo$
|
|
*/
|
|
|
|
#include "llviewerprecompiledheaders.h"
|
|
|
|
#include "lldrawpoolsimple.h"
|
|
|
|
#include "llviewercamera.h"
|
|
#include "lldrawable.h"
|
|
#include "llface.h"
|
|
#include "llsky.h"
|
|
#include "pipeline.h"
|
|
#include "llspatialpartition.h"
|
|
#include "llviewershadermgr.h"
|
|
#include "llrender.h"
|
|
|
|
static LLGLSLShader* simple_shader = NULL;
|
|
static LLGLSLShader* fullbright_shader = NULL;
|
|
|
|
static LLTrace::BlockTimerStatHandle FTM_RENDER_SIMPLE_DEFERRED("Deferred Simple");
|
|
static LLTrace::BlockTimerStatHandle FTM_RENDER_GRASS_DEFERRED("Deferred Grass");
|
|
|
|
void LLDrawPoolGlow::beginPostDeferredPass(S32 pass)
|
|
{
|
|
if (pass == 0)
|
|
{
|
|
gDeferredEmissiveProgram.bind();
|
|
}
|
|
else
|
|
{
|
|
llassert(gDeferredEmissiveProgram.mRiggedVariant);
|
|
gDeferredEmissiveProgram.mRiggedVariant->bind();
|
|
}
|
|
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 2.2f);
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolGlow::renderPostDeferred(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_GLOW);
|
|
LLGLEnable blend(GL_BLEND);
|
|
LLGLDisable test(GL_ALPHA_TEST);
|
|
gGL.flush();
|
|
/// Get rid of z-fighting with non-glow pass.
|
|
LLGLEnable polyOffset(GL_POLYGON_OFFSET_FILL);
|
|
glPolygonOffset(-1.0f, -1.0f);
|
|
gGL.setSceneBlendType(LLRender::BT_ADD);
|
|
|
|
LLGLDepthTest depth(GL_TRUE, GL_FALSE);
|
|
gGL.setColorMask(false, true);
|
|
|
|
if (pass == 0)
|
|
{
|
|
pushBatches(LLRenderPass::PASS_GLOW, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
pushRiggedBatches(LLRenderPass::PASS_GLOW_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
|
|
gGL.setColorMask(true, false);
|
|
gGL.setSceneBlendType(LLRender::BT_ALPHA);
|
|
}
|
|
|
|
void LLDrawPoolGlow::endPostDeferredPass(S32 pass)
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->unbind();
|
|
|
|
LLRenderPass::endRenderPass(pass);
|
|
}
|
|
|
|
S32 LLDrawPoolGlow::getNumPasses()
|
|
{
|
|
if (LLViewerShaderMgr::instance()->getShaderLevel(LLViewerShaderMgr::SHADER_OBJECT) > 0)
|
|
{
|
|
return 2;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolGlow::render(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_GLOW);
|
|
LLGLEnable blend(GL_BLEND);
|
|
LLGLDisable test(GL_ALPHA_TEST);
|
|
gGL.flush();
|
|
/// Get rid of z-fighting with non-glow pass.
|
|
LLGLEnable polyOffset(GL_POLYGON_OFFSET_FILL);
|
|
glPolygonOffset(-1.0f, -1.0f);
|
|
gGL.setSceneBlendType(LLRender::BT_ADD);
|
|
|
|
U32 shader_level = LLViewerShaderMgr::instance()->getShaderLevel(LLViewerShaderMgr::SHADER_OBJECT);
|
|
|
|
//should never get here without basic shaders enabled
|
|
llassert(shader_level > 0);
|
|
|
|
LLGLSLShader* shader = LLPipeline::sUnderWaterRender ? &gObjectEmissiveWaterProgram : &gObjectEmissiveProgram;
|
|
if (pass == 1)
|
|
{
|
|
llassert(shader->mRiggedVariant);
|
|
shader = shader->mRiggedVariant;
|
|
}
|
|
shader->bind();
|
|
if (LLPipeline::sRenderDeferred)
|
|
{
|
|
shader->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 2.2f);
|
|
}
|
|
else
|
|
{
|
|
shader->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 1.f);
|
|
}
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
shader->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
shader->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
|
|
LLGLDepthTest depth(GL_TRUE, GL_FALSE);
|
|
gGL.setColorMask(false, true);
|
|
|
|
if (pass == 0)
|
|
{
|
|
pushBatches(LLRenderPass::PASS_GLOW, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
pushRiggedBatches(LLRenderPass::PASS_GLOW_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
|
|
gGL.setColorMask(true, false);
|
|
gGL.setSceneBlendType(LLRender::BT_ALPHA);
|
|
|
|
if (shader_level > 0 && fullbright_shader)
|
|
{
|
|
shader->unbind();
|
|
}
|
|
}
|
|
|
|
LLDrawPoolSimple::LLDrawPoolSimple() :
|
|
LLRenderPass(POOL_SIMPLE)
|
|
{
|
|
}
|
|
|
|
void LLDrawPoolSimple::prerender()
|
|
{
|
|
mShaderLevel = LLViewerShaderMgr::instance()->getShaderLevel(LLViewerShaderMgr::SHADER_OBJECT);
|
|
}
|
|
|
|
S32 LLDrawPoolSimple::getNumPasses()
|
|
{
|
|
return 2;
|
|
}
|
|
|
|
void LLDrawPoolSimple::beginRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_SIMPLE);
|
|
|
|
if (LLPipeline::sImpostorRender)
|
|
{
|
|
simple_shader = &gObjectSimpleImpostorProgram;
|
|
}
|
|
else if (LLPipeline::sUnderWaterRender)
|
|
{
|
|
simple_shader = &gObjectSimpleWaterProgram;
|
|
}
|
|
else
|
|
{
|
|
simple_shader = &gObjectSimpleProgram;
|
|
}
|
|
|
|
if (pass == 1)
|
|
{
|
|
llassert(simple_shader->mRiggedVariant);
|
|
simple_shader = simple_shader->mRiggedVariant;
|
|
}
|
|
|
|
simple_shader->bind();
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
simple_shader->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
simple_shader->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolSimple::endRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_SIMPLE);
|
|
stop_glerror();
|
|
LLRenderPass::endRenderPass(pass);
|
|
stop_glerror();
|
|
simple_shader->unbind();
|
|
}
|
|
|
|
void LLDrawPoolSimple::render(S32 pass)
|
|
{
|
|
LLGLDisable blend(GL_BLEND);
|
|
|
|
{ //render simple
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_SIMPLE);
|
|
gPipeline.enableLightsDynamic();
|
|
|
|
U32 mask = getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX;
|
|
|
|
if (pass == 0)
|
|
{
|
|
pushBatches(LLRenderPass::PASS_SIMPLE, mask, TRUE, TRUE);
|
|
|
|
if (LLPipeline::sRenderDeferred)
|
|
{ //if deferred rendering is enabled, bump faces aren't registered as simple
|
|
//render bump faces here as simple so bump faces will appear under water
|
|
pushBatches(LLRenderPass::PASS_BUMP, mask, TRUE, TRUE);
|
|
pushBatches(LLRenderPass::PASS_MATERIAL, mask, TRUE, TRUE);
|
|
pushBatches(LLRenderPass::PASS_SPECMAP, mask, TRUE, TRUE);
|
|
pushBatches(LLRenderPass::PASS_NORMMAP, mask, TRUE, TRUE);
|
|
pushBatches(LLRenderPass::PASS_NORMSPEC, mask, TRUE, TRUE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pushRiggedBatches(LLRenderPass::PASS_SIMPLE_RIGGED, mask, TRUE, TRUE);
|
|
|
|
if (LLPipeline::sRenderDeferred)
|
|
{ //if deferred rendering is enabled, bump faces aren't registered as simple
|
|
//render bump faces here as simple so bump faces will appear under water
|
|
pushRiggedBatches(LLRenderPass::PASS_BUMP_RIGGED, mask, TRUE, TRUE);
|
|
pushRiggedBatches(LLRenderPass::PASS_MATERIAL_RIGGED, mask, TRUE, TRUE);
|
|
pushRiggedBatches(LLRenderPass::PASS_SPECMAP_RIGGED, mask, TRUE, TRUE);
|
|
pushRiggedBatches(LLRenderPass::PASS_NORMMAP_RIGGED, mask, TRUE, TRUE);
|
|
pushRiggedBatches(LLRenderPass::PASS_NORMSPEC_RIGGED, mask, TRUE, TRUE);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static LLTrace::BlockTimerStatHandle FTM_RENDER_ALPHA_MASK("Alpha Mask");
|
|
|
|
LLDrawPoolAlphaMask::LLDrawPoolAlphaMask() :
|
|
LLRenderPass(POOL_ALPHA_MASK)
|
|
{
|
|
}
|
|
|
|
void LLDrawPoolAlphaMask::prerender()
|
|
{
|
|
mShaderLevel = LLViewerShaderMgr::instance()->getShaderLevel(LLViewerShaderMgr::SHADER_OBJECT);
|
|
}
|
|
|
|
void LLDrawPoolAlphaMask::beginRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_ALPHA_MASK);
|
|
|
|
if (LLPipeline::sUnderWaterRender)
|
|
{
|
|
simple_shader = &gObjectSimpleWaterAlphaMaskProgram;
|
|
}
|
|
else
|
|
{
|
|
simple_shader = &gObjectSimpleAlphaMaskProgram;
|
|
}
|
|
|
|
if (pass == 1)
|
|
{
|
|
llassert(simple_shader->mRiggedVariant);
|
|
simple_shader = simple_shader->mRiggedVariant;
|
|
}
|
|
|
|
simple_shader->bind();
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
simple_shader->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
simple_shader->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolAlphaMask::endRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_ALPHA_MASK);
|
|
stop_glerror();
|
|
LLRenderPass::endRenderPass(pass);
|
|
stop_glerror();
|
|
if (mShaderLevel > 0)
|
|
{
|
|
simple_shader->unbind();
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolAlphaMask::render(S32 pass)
|
|
{
|
|
LLGLDisable blend(GL_BLEND);
|
|
LL_PROFILE_ZONE_SCOPED;
|
|
|
|
|
|
|
|
simple_shader->bind();
|
|
simple_shader->setMinimumAlpha(0.33f);
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
simple_shader->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
simple_shader->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
|
|
if (pass == 0)
|
|
{
|
|
pushMaskBatches(LLRenderPass::PASS_ALPHA_MASK, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
pushMaskBatches(LLRenderPass::PASS_MATERIAL_ALPHA_MASK, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
pushMaskBatches(LLRenderPass::PASS_SPECMAP_MASK, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
pushMaskBatches(LLRenderPass::PASS_NORMMAP_MASK, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
pushMaskBatches(LLRenderPass::PASS_NORMSPEC_MASK, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
pushRiggedMaskBatches(LLRenderPass::PASS_ALPHA_MASK_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
pushRiggedMaskBatches(LLRenderPass::PASS_MATERIAL_ALPHA_MASK_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
pushRiggedMaskBatches(LLRenderPass::PASS_SPECMAP_MASK_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
pushRiggedMaskBatches(LLRenderPass::PASS_NORMMAP_MASK_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
pushRiggedMaskBatches(LLRenderPass::PASS_NORMSPEC_MASK_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
}
|
|
|
|
LLDrawPoolFullbrightAlphaMask::LLDrawPoolFullbrightAlphaMask() :
|
|
LLRenderPass(POOL_FULLBRIGHT_ALPHA_MASK)
|
|
{
|
|
}
|
|
|
|
void LLDrawPoolFullbrightAlphaMask::prerender()
|
|
{
|
|
mShaderLevel = LLViewerShaderMgr::instance()->getShaderLevel(LLViewerShaderMgr::SHADER_OBJECT);
|
|
}
|
|
|
|
void LLDrawPoolFullbrightAlphaMask::beginRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_ALPHA_MASK);
|
|
|
|
bool rigged = (pass == 1);
|
|
if (LLPipeline::sUnderWaterRender)
|
|
{
|
|
gObjectFullbrightWaterAlphaMaskProgram.bind(rigged);
|
|
}
|
|
else
|
|
{
|
|
gObjectFullbrightAlphaMaskProgram.bind(rigged);
|
|
}
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 1.f);
|
|
}
|
|
else
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
if (LLPipeline::sRenderDeferred)
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 2.2f);
|
|
}
|
|
else
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 1.0f);
|
|
}
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolFullbrightAlphaMask::endRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_ALPHA_MASK);
|
|
stop_glerror();
|
|
LLRenderPass::endRenderPass(pass);
|
|
stop_glerror();
|
|
LLGLSLShader::sCurBoundShaderPtr->unbind();
|
|
}
|
|
|
|
void LLDrawPoolFullbrightAlphaMask::render(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_ALPHA_MASK);
|
|
|
|
if (pass == 0)
|
|
{
|
|
pushMaskBatches(LLRenderPass::PASS_FULLBRIGHT_ALPHA_MASK, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
pushRiggedMaskBatches(LLRenderPass::PASS_FULLBRIGHT_ALPHA_MASK_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
|
|
}
|
|
|
|
//===============================
|
|
//DEFERRED IMPLEMENTATION
|
|
//===============================
|
|
|
|
S32 LLDrawPoolSimple::getNumDeferredPasses()
|
|
{
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
return 2;
|
|
}
|
|
}
|
|
void LLDrawPoolSimple::beginDeferredPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_SIMPLE_DEFERRED);
|
|
|
|
mShader = &gDeferredDiffuseProgram;
|
|
|
|
if (pass == 1)
|
|
{
|
|
llassert(mShader->mRiggedVariant != nullptr);
|
|
mShader = mShader->mRiggedVariant;
|
|
}
|
|
|
|
|
|
mShader->bind();
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
mShader->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
mShader->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolSimple::endDeferredPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_SIMPLE_DEFERRED);
|
|
LLRenderPass::endRenderPass(pass);
|
|
|
|
mShader->unbind();
|
|
}
|
|
|
|
void LLDrawPoolSimple::renderDeferred(S32 pass)
|
|
{
|
|
LL_PROFILE_ZONE_SCOPED;
|
|
LLGLDisable blend(GL_BLEND);
|
|
LLGLDisable alpha_test(GL_ALPHA_TEST);
|
|
|
|
if (pass == 0)
|
|
{ //render simple
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_SIMPLE_DEFERRED);
|
|
pushBatches(LLRenderPass::PASS_SIMPLE, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
//render simple rigged
|
|
pushRiggedBatches(LLRenderPass::PASS_SIMPLE_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
}
|
|
|
|
static LLTrace::BlockTimerStatHandle FTM_RENDER_ALPHA_MASK_DEFERRED("Deferred Alpha Mask");
|
|
|
|
void LLDrawPoolAlphaMask::beginDeferredPass(S32 pass)
|
|
{
|
|
if (pass == 0)
|
|
{
|
|
gDeferredDiffuseAlphaMaskProgram.bind();
|
|
}
|
|
else
|
|
{
|
|
llassert(gDeferredDiffuseAlphaMaskProgram.mRiggedVariant);
|
|
gDeferredDiffuseAlphaMaskProgram.mRiggedVariant->bind();
|
|
}
|
|
|
|
}
|
|
|
|
void LLDrawPoolAlphaMask::endDeferredPass(S32 pass)
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->unbind();
|
|
}
|
|
|
|
void LLDrawPoolAlphaMask::renderDeferred(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_ALPHA_MASK_DEFERRED);
|
|
LLGLSLShader::sCurBoundShaderPtr->setMinimumAlpha(0.33f);
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
|
|
if (pass == 0)
|
|
{
|
|
pushMaskBatches(LLRenderPass::PASS_ALPHA_MASK, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
pushRiggedMaskBatches(LLRenderPass::PASS_ALPHA_MASK_RIGGED, getVertexDataMask() | LLVertexBuffer::MAP_TEXTURE_INDEX, TRUE, TRUE);
|
|
}
|
|
}
|
|
|
|
|
|
// grass drawpool
|
|
LLDrawPoolGrass::LLDrawPoolGrass() :
|
|
LLRenderPass(POOL_GRASS)
|
|
{
|
|
|
|
}
|
|
|
|
void LLDrawPoolGrass::prerender()
|
|
{
|
|
mShaderLevel = LLViewerShaderMgr::instance()->getShaderLevel(LLViewerShaderMgr::SHADER_OBJECT);
|
|
}
|
|
|
|
|
|
void LLDrawPoolGrass::beginRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_GRASS);
|
|
stop_glerror();
|
|
|
|
if (LLPipeline::sUnderWaterRender)
|
|
{
|
|
simple_shader = &gObjectAlphaMaskNonIndexedWaterProgram;
|
|
}
|
|
else
|
|
{
|
|
simple_shader = &gObjectAlphaMaskNonIndexedProgram;
|
|
}
|
|
|
|
if (mShaderLevel > 0)
|
|
{
|
|
simple_shader->bind();
|
|
simple_shader->setMinimumAlpha(0.5f);
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
simple_shader->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
simple_shader->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gGL.flush();
|
|
LLGLSLShader::bindNoShader();
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolGrass::endRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_GRASS);
|
|
LLRenderPass::endRenderPass(pass);
|
|
|
|
if (mShaderLevel > 0)
|
|
{
|
|
simple_shader->unbind();
|
|
}
|
|
else
|
|
{
|
|
gGL.flush();
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolGrass::render(S32 pass)
|
|
{
|
|
LLGLDisable blend(GL_BLEND);
|
|
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_GRASS);
|
|
LLGLEnable test(GL_ALPHA_TEST);
|
|
gGL.setSceneBlendType(LLRender::BT_ALPHA);
|
|
//render grass
|
|
LLRenderPass::pushBatches(LLRenderPass::PASS_GRASS, getVertexDataMask());
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolGrass::beginDeferredPass(S32 pass)
|
|
{
|
|
|
|
}
|
|
|
|
void LLDrawPoolGrass::endDeferredPass(S32 pass)
|
|
{
|
|
|
|
}
|
|
|
|
void LLDrawPoolGrass::renderDeferred(S32 pass)
|
|
{
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_GRASS_DEFERRED);
|
|
gDeferredNonIndexedDiffuseAlphaMaskProgram.bind();
|
|
gDeferredNonIndexedDiffuseAlphaMaskProgram.setMinimumAlpha(0.5f);
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
gDeferredNonIndexedDiffuseAlphaMaskProgram.uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
gDeferredNonIndexedDiffuseAlphaMaskProgram.uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
|
|
//render grass
|
|
LLRenderPass::pushBatches(LLRenderPass::PASS_GRASS, getVertexDataMask());
|
|
}
|
|
}
|
|
|
|
|
|
// Fullbright drawpool
|
|
LLDrawPoolFullbright::LLDrawPoolFullbright() :
|
|
LLRenderPass(POOL_FULLBRIGHT)
|
|
{
|
|
}
|
|
|
|
void LLDrawPoolFullbright::prerender()
|
|
{
|
|
mShaderLevel = LLViewerShaderMgr::instance()->getShaderLevel(LLViewerShaderMgr::SHADER_OBJECT);
|
|
}
|
|
|
|
void LLDrawPoolFullbright::beginPostDeferredPass(S32 pass)
|
|
{
|
|
bool rigged = (pass == 1);
|
|
if (LLPipeline::sUnderWaterRender)
|
|
{
|
|
gDeferredFullbrightWaterProgram.bind(rigged);
|
|
}
|
|
else
|
|
{
|
|
gDeferredFullbrightProgram.bind(rigged);
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolFullbright::renderPostDeferred(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_FULLBRIGHT);
|
|
|
|
gGL.setSceneBlendType(LLRender::BT_ALPHA);
|
|
U32 fullbright_mask = LLVertexBuffer::MAP_VERTEX | LLVertexBuffer::MAP_TEXCOORD0 | LLVertexBuffer::MAP_COLOR | LLVertexBuffer::MAP_TEXTURE_INDEX;
|
|
if (pass == 0)
|
|
{
|
|
pushBatches(LLRenderPass::PASS_FULLBRIGHT, fullbright_mask, TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
pushRiggedBatches(LLRenderPass::PASS_FULLBRIGHT_RIGGED, fullbright_mask, TRUE, TRUE);
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolFullbright::endPostDeferredPass(S32 pass)
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->unbind();
|
|
LLRenderPass::endRenderPass(pass);
|
|
}
|
|
|
|
void LLDrawPoolFullbright::beginRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_FULLBRIGHT);
|
|
|
|
if (LLPipeline::sUnderWaterRender)
|
|
{
|
|
fullbright_shader = &gObjectFullbrightWaterProgram;
|
|
}
|
|
else
|
|
{
|
|
fullbright_shader = &gObjectFullbrightProgram;
|
|
}
|
|
|
|
if (pass == 1)
|
|
{
|
|
llassert(fullbright_shader->mRiggedVariant);
|
|
fullbright_shader = fullbright_shader->mRiggedVariant;
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolFullbright::endRenderPass(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_FULLBRIGHT);
|
|
LLRenderPass::endRenderPass(pass);
|
|
|
|
stop_glerror();
|
|
|
|
if (mShaderLevel > 0)
|
|
{
|
|
fullbright_shader->unbind();
|
|
}
|
|
|
|
stop_glerror();
|
|
}
|
|
|
|
void LLDrawPoolFullbright::render(S32 pass)
|
|
{ //render fullbright
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_FULLBRIGHT);
|
|
gGL.setSceneBlendType(LLRender::BT_ALPHA);
|
|
|
|
stop_glerror();
|
|
|
|
if (mShaderLevel > 0)
|
|
{
|
|
fullbright_shader->bind();
|
|
fullbright_shader->uniform1f(LLViewerShaderMgr::FULLBRIGHT, 1.f);
|
|
fullbright_shader->uniform1f(LLViewerShaderMgr::TEXTURE_GAMMA, 1.f);
|
|
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
fullbright_shader->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
fullbright_shader->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
|
|
U32 fullbright_mask = LLVertexBuffer::MAP_VERTEX | LLVertexBuffer::MAP_TEXCOORD0 | LLVertexBuffer::MAP_COLOR | LLVertexBuffer::MAP_TEXTURE_INDEX;
|
|
|
|
if (pass == 0)
|
|
{
|
|
pushBatches(LLRenderPass::PASS_FULLBRIGHT, fullbright_mask, TRUE, TRUE);
|
|
pushBatches(LLRenderPass::PASS_MATERIAL_ALPHA_EMISSIVE, fullbright_mask, TRUE, TRUE);
|
|
pushBatches(LLRenderPass::PASS_SPECMAP_EMISSIVE, fullbright_mask, TRUE, TRUE);
|
|
pushBatches(LLRenderPass::PASS_NORMMAP_EMISSIVE, fullbright_mask, TRUE, TRUE);
|
|
pushBatches(LLRenderPass::PASS_NORMSPEC_EMISSIVE, fullbright_mask, TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
pushRiggedBatches(LLRenderPass::PASS_FULLBRIGHT_RIGGED, fullbright_mask, TRUE, TRUE);
|
|
pushRiggedBatches(LLRenderPass::PASS_MATERIAL_ALPHA_EMISSIVE_RIGGED, fullbright_mask, TRUE, TRUE);
|
|
pushRiggedBatches(LLRenderPass::PASS_SPECMAP_EMISSIVE_RIGGED, fullbright_mask, TRUE, TRUE);
|
|
pushRiggedBatches(LLRenderPass::PASS_NORMMAP_EMISSIVE_RIGGED, fullbright_mask, TRUE, TRUE);
|
|
pushRiggedBatches(LLRenderPass::PASS_NORMSPEC_EMISSIVE_RIGGED, fullbright_mask, TRUE, TRUE);
|
|
}
|
|
}
|
|
|
|
stop_glerror();
|
|
}
|
|
|
|
S32 LLDrawPoolFullbright::getNumPasses()
|
|
{
|
|
return 2;
|
|
}
|
|
|
|
|
|
void LLDrawPoolFullbrightAlphaMask::beginPostDeferredPass(S32 pass)
|
|
{
|
|
bool rigged = (pass == 1);
|
|
if (LLPipeline::sRenderingHUDs)
|
|
{
|
|
gObjectFullbrightAlphaMaskProgram.bind(rigged);
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 1.0f);
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else if (LLPipeline::sRenderDeferred)
|
|
{
|
|
if (LLPipeline::sUnderWaterRender)
|
|
{
|
|
gDeferredFullbrightAlphaMaskWaterProgram.bind(rigged);
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 2.2f);
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 1);
|
|
}
|
|
else
|
|
{
|
|
gDeferredFullbrightAlphaMaskProgram.bind(rigged);
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 2.2f);
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gObjectFullbrightAlphaMaskProgram.bind(rigged);
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1f(LLShaderMgr::TEXTURE_GAMMA, 1.0f);
|
|
LLGLSLShader::sCurBoundShaderPtr->uniform1i(LLShaderMgr::NO_ATMO, 0);
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolFullbrightAlphaMask::renderPostDeferred(S32 pass)
|
|
{
|
|
LL_RECORD_BLOCK_TIME(FTM_RENDER_FULLBRIGHT);
|
|
LLGLDisable blend(GL_BLEND);
|
|
U32 fullbright_mask = LLVertexBuffer::MAP_VERTEX | LLVertexBuffer::MAP_TEXCOORD0 | LLVertexBuffer::MAP_COLOR | LLVertexBuffer::MAP_TEXTURE_INDEX;
|
|
if (pass == 0)
|
|
{
|
|
pushMaskBatches(LLRenderPass::PASS_FULLBRIGHT_ALPHA_MASK, fullbright_mask, TRUE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
pushRiggedMaskBatches(LLRenderPass::PASS_FULLBRIGHT_ALPHA_MASK_RIGGED, fullbright_mask, TRUE, TRUE);
|
|
}
|
|
}
|
|
|
|
void LLDrawPoolFullbrightAlphaMask::endPostDeferredPass(S32 pass)
|
|
{
|
|
LLGLSLShader::sCurBoundShaderPtr->unbind();
|
|
LLRenderPass::endRenderPass(pass);
|
|
}
|
|
|
|
|