| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634 | //-----------------------------------------------------------------------------// Copyright (c) 2012 GarageGames, LLC//// Permission is hereby granted, free of charge, to any person obtaining a copy// of this software and associated documentation files (the "Software"), to// deal in the Software without restriction, including without limitation the// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or// sell copies of the Software, and to permit persons to whom the Software is// furnished to do so, subject to the following conditions://// The above copyright notice and this permission notice shall be included in// all copies or substantial portions of the Software.//// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS// IN THE SOFTWARE.//-----------------------------------------------------------------------------#include "platform/platform.h"#include "lighting/advanced/hlsl/advancedLightingFeaturesHLSL.h"#include "lighting/advanced/advancedLightBinManager.h"#include "shaderGen/langElement.h"#include "shaderGen/shaderOp.h"#include "shaderGen/conditionerFeature.h"#include "renderInstance/renderDeferredMgr.h"#include "materials/processedMaterial.h"#include "materials/materialFeatureTypes.h"void DeferredRTLightingFeatHLSL::processPixMacros( Vector<GFXShaderMacro> ¯os,                                                    const MaterialFeatureData &fd  ){   // Skip deferred features, and use forward shading instead   if ( !fd.features[MFT_isDeferred] )   {      Parent::processPixMacros( macros, fd );      return;   }   // Pull in the uncondition method for the light info buffer   NamedTexTarget *texTarget = NamedTexTarget::find( AdvancedLightBinManager::smBufferName );   if ( texTarget && texTarget->getConditioner() )   {      ConditionerMethodDependency *unconditionMethod = texTarget->getConditioner()->getConditionerMethodDependency(ConditionerFeature::UnconditionMethod);      unconditionMethod->createMethodMacro( String::ToLower( AdvancedLightBinManager::smBufferName ) + "Uncondition", macros );      addDependency(unconditionMethod);   }}void DeferredRTLightingFeatHLSL::processVert(   Vector<ShaderComponent*> &componentList,                                                 const MaterialFeatureData &fd ){   // Skip deferred features, and use forward shading instead   if ( !fd.features[MFT_isDeferred] )   {      Parent::processVert( componentList, fd );      return;   }   // Pass screen space position to pixel shader to compute a full screen buffer uv   ShaderConnector *connectComp = dynamic_cast<ShaderConnector *>( componentList[C_CONNECTOR] );   Var *ssPos = connectComp->getElement( RT_TEXCOORD );   ssPos->setName( "screenspacePos" );   ssPos->setStructName( "OUT" );   ssPos->setType( "float4" );   Var *outPosition = (Var*) LangElement::find( "hpos" );   AssertFatal( outPosition, "No hpos, ohnoes." );   output = new GenOp( "   @ = @;\r\n", ssPos, outPosition );}void DeferredRTLightingFeatHLSL::processPix( Vector<ShaderComponent*> &componentList,                                             const MaterialFeatureData &fd ){   // Skip deferred features, and use forward shading instead   if ( !fd.features[MFT_isDeferred] )   {      Parent::processPix(componentList, fd);      return;   }   MultiLine *meta = new MultiLine;   ShaderConnector *connectComp = dynamic_cast<ShaderConnector *>(componentList[C_CONNECTOR]);   Var *ssPos = connectComp->getElement(RT_TEXCOORD);   ssPos->setName("screenspacePos");   ssPos->setStructName("IN");   ssPos->setType("float4");   Var *uvScene = new Var;   uvScene->setType("float2");   uvScene->setName("uvScene");   LangElement *uvSceneDecl = new DecOp(uvScene);   String rtParamName = String::ToString("rtParams%s", "diffuseLightingBuffer");   Var *rtParams = (Var*)LangElement::find(rtParamName);   if (!rtParams)   {      rtParams = new Var;      rtParams->setType("float4");      rtParams->setName(rtParamName);      rtParams->uniform = true;      rtParams->constSortPos = cspPass;   }   meta->addStatement( new GenOp( "   @ = @.xy / @.w;\r\n", uvSceneDecl, ssPos, ssPos ) ); // get the screen coord... its -1 to +1   meta->addStatement( new GenOp( "   @ = ( @ + 1.0 ) / 2.0;\r\n", uvScene, uvScene ) ); // get the screen coord to 0 to 1   meta->addStatement( new GenOp( "   @.y = 1.0 - @.y;\r\n", uvScene, uvScene ) ); // flip the y axis    meta->addStatement( new GenOp( "   @ = ( @ * @.zw ) + @.xy;\r\n", uvScene, uvScene, rtParams, rtParams) ); // scale it down and offset it to the rt size   Var *lightInfoSamp = new Var;   lightInfoSamp->setType( "float4" );   lightInfoSamp->setName( "lightInfoSample" );   // create texture var   Var *lightInfoBuffer = new Var;   lightInfoBuffer->setType( "SamplerState" );   lightInfoBuffer->setName( "lightInfoBuffer" );   lightInfoBuffer->uniform = true;   lightInfoBuffer->sampler = true;   lightInfoBuffer->constNum = Var::getTexUnitNum();     // used as texture unit num here   Var* lightBufferTex =  new Var;   lightBufferTex->setName("lightInfoBufferTex");   lightBufferTex->setType("Texture2D");   lightBufferTex->uniform = true;   lightBufferTex->texture = true;   lightBufferTex->constNum = lightInfoBuffer->constNum;   // Declare the RTLighting variables in this feature, they will either be assigned   // in this feature, or in the tonemap/lightmap feature   Var *d_lightcolor = new Var( "d_lightcolor", "float3" );   meta->addStatement( new GenOp( "   @;\r\n", new DecOp( d_lightcolor ) ) );   Var *d_NL_Att = new Var( "d_NL_Att", "float" );   meta->addStatement( new GenOp( "   @;\r\n", new DecOp( d_NL_Att ) ) );   Var *d_specular = new Var( "d_specular", "float" );   meta->addStatement( new GenOp( "   @;\r\n", new DecOp( d_specular ) ) );      // Perform the uncondition here.   String unconditionLightInfo = String::ToLower( AdvancedLightBinManager::smBufferName ) + "Uncondition";   meta->addStatement(new GenOp(avar("   %s(@.Sample(@, @), @, @, @);\r\n",      unconditionLightInfo.c_str()), lightBufferTex, lightInfoBuffer, uvScene, d_lightcolor, d_NL_Att, d_specular));      // This is kind of weak sauce   if( !fd.features[MFT_VertLit] && !fd.features[MFT_ToneMap] && !fd.features[MFT_LightMap] && !fd.features[MFT_SubSurface] )      meta->addStatement( new GenOp( "   @;\r\n", assignColor( new GenOp( "float4(@, 1.0)", d_lightcolor ), Material::Mul ) ) );   output = meta;}ShaderFeature::Resources DeferredRTLightingFeatHLSL::getResources( const MaterialFeatureData &fd ){   // Skip deferred features, and use forward shading instead   if ( !fd.features[MFT_isDeferred] )      return Parent::getResources( fd );   // HACK: See DeferredRTLightingFeatHLSL::setTexData.   mLastTexIndex = 0;   Resources res;    res.numTex = 1;   res.numTexReg = 1;   return res;}void DeferredRTLightingFeatHLSL::setTexData( Material::StageData &stageDat,                                             const MaterialFeatureData &fd,                                              RenderPassData &passData,                                              U32 &texIndex ){   // Skip deferred features, and use forward shading instead   if ( !fd.features[MFT_isDeferred] )   {      Parent::setTexData( stageDat, fd, passData, texIndex );      return;   }   NamedTexTarget *texTarget = NamedTexTarget::find( AdvancedLightBinManager::smBufferName );   if( texTarget )   {      // HACK: We store this for use in DeferredRTLightingFeatHLSL::processPix()      // which cannot deduce the texture unit itself.      mLastTexIndex = texIndex;      passData.mTexType[ texIndex ] = Material::TexTarget;      passData.mSamplerNames[ texIndex ]= "diffuseLightingBuffer";      passData.mTexSlot[ texIndex++ ].texTarget = texTarget;   }}void DeferredBumpFeatHLSL::processVert(   Vector<ShaderComponent*> &componentList,                                           const MaterialFeatureData &fd ){   if( fd.features[MFT_DeferredConditioner] )   {      // There is an output conditioner active, so we need to supply a transform      // to the pixel shader.       MultiLine *meta = new MultiLine;      // We need the view to tangent space transform in the pixel shader.      getOutViewToTangent( componentList, meta, fd );      const bool useTexAnim = fd.features[MFT_TexAnim];      // Make sure there are texcoords      if( !fd.features[MFT_Parallax] && !fd.features[MFT_DiffuseMap])      {         getOutTexCoord(   "texCoord",                            "float2",                            useTexAnim,                            meta,                            componentList );      }      const bool useFoliageTexCoord = fd.features[MFT_Foliage];      if ( fd.features.hasFeature( MFT_DetailNormalMap ) )            addOutDetailTexCoord( componentList,                                   meta,                                  useTexAnim, useFoliageTexCoord);      output = meta;   }   else if (   fd.materialFeatures[MFT_NormalsOut] ||                !fd.features[MFT_isDeferred] ||                !fd.features[MFT_RTLighting] )   {      Parent::processVert( componentList, fd );      return;   }   else   {      output = NULL;   }}void DeferredBumpFeatHLSL::processPix( Vector<ShaderComponent*> &componentList,                                        const MaterialFeatureData &fd ){   // NULL output in case nothing gets handled   output = NULL;   if( fd.features[MFT_DeferredConditioner] )   {      MultiLine *meta = new MultiLine;      Var *viewToTangent = getInViewToTangent( componentList );      // create texture var      Var *bumpMap = getNormalMapTex();      Var *texCoord = getInTexCoord("texCoord", "float2", componentList);      Var *bumpMapTex = (Var*)LangElement::find("bumpMapTex");      LangElement *texOp = new GenOp("@.Sample(@, @)", bumpMapTex, bumpMap, texCoord);      // create bump normal      Var *bumpNorm = new Var;      bumpNorm->setName( "bumpNormal" );      bumpNorm->setType( "float4" );      LangElement *bumpNormDecl = new DecOp( bumpNorm );      meta->addStatement( expandNormalMap( texOp, bumpNormDecl, bumpNorm, fd ) );      // If we have a detail normal map we add the xy coords of      // it to the base normal map.  This gives us the effect we      // want with few instructions and minial artifacts.      if ( fd.features.hasFeature( MFT_DetailNormalMap ) )      {         bumpMap = new Var;         bumpMap->setType( "SamplerState" );         bumpMap->setName( "detailBumpMap" );         bumpMap->uniform = true;         bumpMap->sampler = true;         bumpMap->constNum = Var::getTexUnitNum();         Var* detailNormalTex = new Var;         detailNormalTex->setName("detailBumpMapTex");         detailNormalTex->setType("Texture2D");         detailNormalTex->uniform = true;         detailNormalTex->texture = true;         detailNormalTex->constNum = bumpMap->constNum;         texCoord = getInTexCoord("detCoord", "float2", componentList);         texOp = new GenOp("@.Sample(@, @)", detailNormalTex, bumpMap, texCoord);         Var *detailBump = new Var;         detailBump->setName( "detailBump" );         detailBump->setType( "float4" );         meta->addStatement( expandNormalMap( texOp, new DecOp( detailBump ), detailBump, fd ) );         Var *detailBumpScale = new Var;         detailBumpScale->setType( "float" );         detailBumpScale->setName( "detailBumpStrength" );         detailBumpScale->uniform = true;         detailBumpScale->constSortPos = cspPass;         meta->addStatement( new GenOp( "   @.xy += @.xy * @;\r\n", bumpNorm, detailBump, detailBumpScale ) );      }      // This var is read from GBufferConditionerHLSL and       // used in the deferred output.      //      // By using the 'half' type here we get a bunch of partial      // precision optimized code on further operations on the normal      // which helps alot on older Geforce cards.      //      Var *gbNormal = new Var;      gbNormal->setName( "gbNormal" );      gbNormal->setType( "half3" );      LangElement *gbNormalDecl = new DecOp( gbNormal );      // Normalize is done later...       // Note: The reverse mul order is intentional. Affine matrix.      meta->addStatement( new GenOp( "   @ = (half3)mul( @.xyz, @ );\r\n", gbNormalDecl, bumpNorm, viewToTangent ) );      output = meta;      return;   }   else if (fd.features[MFT_AccuMap])    {      Var *bumpSample = (Var *)LangElement::find( "bumpSample" );      if (bumpSample == NULL)      {         MultiLine *meta = new MultiLine;         Var *texCoord = getInTexCoord("texCoord", "float2", componentList);         Var *bumpMap = getNormalMapTex();         bumpSample = new Var;         bumpSample->setType("float4");         bumpSample->setName("bumpSample");         LangElement *bumpSampleDecl = new DecOp(bumpSample);         Var *bumpMapTex = (Var *)LangElement::find("bumpMapTex");         output = new GenOp("   @ = @.Sample(@, @);\r\n", bumpSampleDecl, bumpMapTex, bumpMap, texCoord);         if ( fd.features.hasFeature( MFT_DetailNormalMap ) )         {            bumpMap = (Var*)LangElement::find( "detailBumpMap" );            if ( !bumpMap )            {               bumpMap = new Var;               bumpMap->setType( "sampler2D" );               bumpMap->setName( "detailBumpMap" );               bumpMap->uniform = true;               bumpMap->sampler = true;               bumpMap->constNum = Var::getTexUnitNum();            }            bumpMapTex = (Var*)LangElement::find("detailBumpMap");            if (!bumpMapTex)            {               bumpMap->setType("SamplerState");               bumpMapTex = new Var;               bumpMapTex->setName("detailBumpMapTex");               bumpMapTex->setType("Texture2D");               bumpMapTex->uniform = true;               bumpMapTex->texture = true;               bumpMapTex->constNum = bumpMap->constNum;            }            texCoord = getInTexCoord( "detCoord", "float2", componentList );            LangElement *texOp = new GenOp("@.Sample(@, @)", bumpMap, bumpMapTex, texCoord);            Var *detailBump = new Var;            detailBump->setName( "detailBump" );            detailBump->setType( "float4" );            meta->addStatement( expandNormalMap( texOp, new DecOp( detailBump ), detailBump, fd ) );            Var *detailBumpScale = new Var;            detailBumpScale->setType( "float" );            detailBumpScale->setName( "detailBumpStrength" );            detailBumpScale->uniform = true;            detailBumpScale->constSortPos = cspPass;            meta->addStatement( new GenOp( "   @.xy += @.xy * @;\r\n", bumpSample, detailBump, detailBumpScale ) );         }         output = meta;         return;      }   }    else if (   fd.materialFeatures[MFT_NormalsOut] ||                !fd.features[MFT_isDeferred] ||                !fd.features[MFT_RTLighting] )   {      Parent::processPix( componentList, fd );      return;   }   else if (!fd.features[MFT_PBRConfigMap] )   {      Var *bumpSample = (Var *)LangElement::find( "bumpSample" );      if( bumpSample == NULL )      {         Var *texCoord = getInTexCoord( "texCoord", "float2", componentList );         Var *bumpMap = getNormalMapTex();         Var *bumpMapTex = (Var *)LangElement::find("bumpMapTex");         bumpSample = new Var;         bumpSample->setType("float4");         bumpSample->setName("bumpSample");         LangElement *bumpSampleDecl = new DecOp(bumpSample);         output = new GenOp("   @ = @.Sample(@, @);\r\n", bumpSampleDecl, bumpMapTex, bumpMap, texCoord);         return;      }   }   output = NULL;}ShaderFeature::Resources DeferredBumpFeatHLSL::getResources( const MaterialFeatureData &fd ){   if (  fd.materialFeatures[MFT_NormalsOut] ||          !fd.features[MFT_isDeferred] ||          fd.features[MFT_Parallax] ||         !fd.features[MFT_RTLighting] )      return Parent::getResources( fd );   Resources res;    if(!fd.features[MFT_PBRConfigMap])   {      res.numTex = 1;      res.numTexReg = 1;      if (  fd.features[MFT_DeferredConditioner] &&            fd.features.hasFeature( MFT_DetailNormalMap ) )      {         res.numTex += 1;         if ( !fd.features.hasFeature( MFT_DetailMap ) )            res.numTexReg += 1;      }   }   return res;}void DeferredBumpFeatHLSL::setTexData( Material::StageData &stageDat,                                       const MaterialFeatureData &fd,                                        RenderPassData &passData,                                        U32 &texIndex ){   if (  fd.materialFeatures[MFT_NormalsOut] ||          !fd.features[MFT_isDeferred] ||          !fd.features[MFT_RTLighting] )   {      Parent::setTexData( stageDat, fd, passData, texIndex );      return;   }   if (  !fd.features[MFT_DeferredConditioner] && fd.features[MFT_AccuMap] )   {      passData.mTexType[ texIndex ] = Material::Bump;      passData.mSamplerNames[ texIndex ] = "bumpMap";      passData.mTexSlot[ texIndex++ ].texObject = stageDat.getTex( MFT_NormalMap );      if (  fd.features.hasFeature( MFT_DetailNormalMap ) )      {         passData.mTexType[ texIndex ] = Material::DetailBump;         passData.mSamplerNames[texIndex] = "detailBumpMap";         passData.mTexSlot[ texIndex++ ].texObject = stageDat.getTex( MFT_DetailNormalMap );      }   }   else if (  !fd.features[MFT_Parallax] && !fd.features[MFT_PBRConfigMap] &&         ( fd.features[MFT_DeferredConditioner]) )   {      passData.mTexType[ texIndex ] = Material::Bump;      passData.mSamplerNames[ texIndex ] = "bumpMap";      passData.mTexSlot[ texIndex++ ].texObject = stageDat.getTex( MFT_NormalMap );      if (  fd.features[MFT_DeferredConditioner] &&            fd.features.hasFeature( MFT_DetailNormalMap ) )      {         passData.mTexType[ texIndex ] = Material::DetailBump;         passData.mSamplerNames[ texIndex ] = "detailBumpMap";         passData.mTexSlot[ texIndex++ ].texObject = stageDat.getTex( MFT_DetailNormalMap );      }   }}ShaderFeature::Resources DeferredMinnaertHLSL::getResources( const MaterialFeatureData &fd ){   Resources res;   if( fd.features[MFT_isDeferred] && fd.features[MFT_RTLighting] )   {      res.numTex = 1;      res.numTexReg = 1;   }   return res;}void DeferredMinnaertHLSL::setTexData( Material::StageData &stageDat,                                       const MaterialFeatureData &fd,                                        RenderPassData &passData,                                        U32 &texIndex ){   if( fd.features[MFT_isDeferred] && fd.features[MFT_RTLighting] )   {      NamedTexTarget *texTarget = NamedTexTarget::find(RenderDeferredMgr::BufferName);      if ( texTarget )      {         passData.mTexType[texIndex] = Material::TexTarget;         passData.mSamplerNames[texIndex] = "deferredBuffer";         passData.mTexSlot[ texIndex++ ].texTarget = texTarget;      }   }}void DeferredMinnaertHLSL::processPixMacros( Vector<GFXShaderMacro> ¯os,                                              const MaterialFeatureData &fd  ){   if( fd.features[MFT_isDeferred] && fd.features[MFT_RTLighting] )   {      // Pull in the uncondition method for the g buffer      NamedTexTarget *texTarget = NamedTexTarget::find( RenderDeferredMgr::BufferName );      if ( texTarget && texTarget->getConditioner() )      {         ConditionerMethodDependency *unconditionMethod = texTarget->getConditioner()->getConditionerMethodDependency(ConditionerFeature::UnconditionMethod);         unconditionMethod->createMethodMacro( String::ToLower(RenderDeferredMgr::BufferName) + "Uncondition", macros );         addDependency(unconditionMethod);      }   }}void DeferredMinnaertHLSL::processVert(   Vector<ShaderComponent*> &componentList,                                          const MaterialFeatureData &fd ){   // If there is no deferred information, bail on this feature   if( !fd.features[MFT_isDeferred] || !fd.features[MFT_RTLighting] )   {      output = NULL;      return;   }   // Make sure we pass the world space position to the   // pixel shader so we can calculate a view vector.   MultiLine *meta = new MultiLine;   addOutWsPosition( componentList, fd.features[MFT_UseInstancing], meta );   output = meta;}void DeferredMinnaertHLSL::processPix( Vector<ShaderComponent*> &componentList,                                        const MaterialFeatureData &fd ){   // If there is no deferred information, bail on this feature   if( !fd.features[MFT_isDeferred] || !fd.features[MFT_RTLighting] )   {      output = NULL;      return;   }   Var *minnaertConstant = new Var;   minnaertConstant->setType( "float" );   minnaertConstant->setName( "minnaertConstant" );   minnaertConstant->uniform = true;   minnaertConstant->constSortPos = cspPotentialPrimitive;   // create texture var   Var *deferredBuffer = new Var;   deferredBuffer->setType( "SamplerState" );   deferredBuffer->setName( "deferredBuffer" );   deferredBuffer->uniform = true;   deferredBuffer->sampler = true;   deferredBuffer->constNum = Var::getTexUnitNum();     // used as texture unit num here   Var* deferredTex = new Var;   deferredTex->setName("deferredTex");   deferredTex->setType("Texture2D");   deferredTex->uniform = true;   deferredTex->texture = true;   deferredTex->constNum = deferredBuffer->constNum;   // Texture coord   Var *uvScene = (Var*) LangElement::find( "uvScene" );   AssertFatal(uvScene != NULL, "Unable to find UVScene, no RTLighting feature?");   MultiLine *meta = new MultiLine;   // Get the world space view vector.   Var *wsViewVec = getWsView( getInWsPosition( componentList ), meta );   String unconditionDeferredMethod = String::ToLower(RenderDeferredMgr::BufferName) + "Uncondition";   Var *d_NL_Att = (Var*)LangElement::find( "d_NL_Att" );   meta->addStatement(new GenOp(avar("   float4 normalDepth = %s(@, ,@, @);\r\n", unconditionDeferredMethod.c_str()), deferredBuffer, deferredTex, uvScene));   meta->addStatement( new GenOp( "   float vDotN = dot(normalDepth.xyz, @);\r\n", wsViewVec ) );   meta->addStatement( new GenOp( "   float Minnaert = pow( @, @) * pow(vDotN, 1.0 - @);\r\n", d_NL_Att, minnaertConstant, minnaertConstant ) );   meta->addStatement( new GenOp( "   @;\r\n", assignColor( new GenOp( "float4(Minnaert, Minnaert, Minnaert, 1.0)" ), Material::Mul ) ) );   output = meta;}void DeferredSubSurfaceHLSL::processPix(  Vector<ShaderComponent*> &componentList,                                           const MaterialFeatureData &fd ){   Var *subSurfaceParams = new Var;   subSurfaceParams->setType( "float4" );   subSurfaceParams->setName( "subSurfaceParams" );   subSurfaceParams->uniform = true;   subSurfaceParams->constSortPos = cspPotentialPrimitive;   Var *d_lightcolor = (Var*)LangElement::find( "d_lightcolor" );   Var *d_NL_Att = (Var*)LangElement::find( "d_NL_Att" );   MultiLine *meta = new MultiLine;   Var* targ = (Var*)LangElement::find(getOutputTargetVarName(ShaderFeature::DefaultTarget));   if (fd.features[MFT_isDeferred])   {      targ = (Var*)LangElement::find(getOutputTargetVarName(ShaderFeature::RenderTarget3));      meta->addStatement(new GenOp("   @.rgb += @.rgb*@.a;\r\n", targ, subSurfaceParams, subSurfaceParams));      output = meta;      return;   }   output = meta;}
 |