| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463 | //-----------------------------------------------------------------------------// 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 "shaderGen/GLSL/bumpGLSL.h"#include "shaderGen/shaderOp.h"#include "gfx/gfxDevice.h"#include "materials/matInstance.h"#include "materials/processedMaterial.h"#include "materials/materialFeatureTypes.h"#include "shaderGen/shaderGenVars.h"#include "shaderGen/shaderGen.h"void BumpFeatGLSL::processVert(  Vector<ShaderComponent*> &componentList,                                  const MaterialFeatureData &fd ){   MultiLine *meta = new MultiLine;	output = meta;		const bool useTexAnim = fd.features[MFT_TexAnim];   // Output the texture coord.   getOutTexCoord(   "texCoord",                      "vec2",                      useTexAnim,                      meta,                      componentList );   const bool useFoliageTexCoord = fd.features[MFT_Foliage];	if ( fd.features.hasFeature( MFT_DetailNormalMap ) )      addOutDetailTexCoord( componentList, 										meta,										useTexAnim, useFoliageTexCoord);	   // Also output the worldToTanget transform which   // we use to create the world space normal.   getOutWorldToTangent( componentList, meta, fd );}void BumpFeatGLSL::processPix(   Vector<ShaderComponent*> &componentList,                                  const MaterialFeatureData &fd ){   MultiLine *meta = new MultiLine;	output = meta;   // Get the texture coord.   Var *texCoord = getInTexCoord( "texCoord", "vec2", componentList );   // Sample the bumpmap.   Var *bumpMap = getNormalMapTex();	LangElement *texOp = NULL;		//Handle atlased textures   // http://www.infinity-universe.com/Infinity/index.php?option=com_content&task=view&id=65&Itemid=47	if(fd.features[MFT_NormalMapAtlas])   {		// This is a big block of code, so put a comment in the shader code      meta->addStatement( new GenOp( "   // Atlased texture coordinate calculation (see BumpFeat*LSL for details)\r\n") );		      Var *atlasedTex = new Var;      atlasedTex->setName("atlasedBumpCoord");      atlasedTex->setType( "vec2" );      LangElement *atDecl = new DecOp(atlasedTex);		      // Parameters of the texture atlas      Var *atParams  = new Var;      atParams->setType( "float4" );      atParams->setName("bumpAtlasParams");      atParams->uniform = true;      atParams->constSortPos = cspPotentialPrimitive;		      // Parameters of the texture (tile) this object is using in the atlas      Var *tileParams  = new Var;      tileParams->setType( "float4" );      tileParams->setName("bumpAtlasTileParams");      tileParams->uniform = true;      tileParams->constSortPos = cspPotentialPrimitive;				const bool is_sm3 = (GFX->getPixelShaderVersion() > 2.0f);		if(is_sm3)      {         // Figure out the mip level         meta->addStatement( new GenOp( "   float2 _dx_bump = ddx(@ * @.z);\r\n", texCoord, atParams ) );         meta->addStatement( new GenOp( "   float2 _dy_bump = ddy(@ * @.z);\r\n", texCoord, atParams ) );         meta->addStatement( new GenOp( "   float mipLod_bump = 0.5 * log2(max(dot(_dx_bump, _dx_bump), dot(_dy_bump, _dy_bump)));\r\n"));         meta->addStatement( new GenOp( "   mipLod_bump = clamp(mipLod_bump, 0.0, @.w);\r\n", atParams));			         // And the size of the mip level         meta->addStatement(new GenOp("   float mipPixSz_bump = pow(2.0, @.w - mipLod_bump);\r\n", atParams));         meta->addStatement( new GenOp( "   float2 mipSz_bump = mipPixSz_bump / @.xy;\r\n", atParams ) );      }      else      {         meta->addStatement(new GenOp("   float2 mipSz = float2(1.0, 1.0);\r\n"));      }				// Tiling mode      if( true ) // Wrap         meta->addStatement( new GenOp( "   @ = frac(@);\r\n", atDecl, texCoord ) );      else       // Clamp         meta->addStatement(new GenOp("   @ = saturate(@);\r\n", atDecl, texCoord));		      // Finally scale/offset, and correct for filtering      meta->addStatement( new GenOp( "   @ = @ * ((mipSz_bump * @.xy - 1.0) / mipSz_bump) + 0.5 / mipSz_bump + @.xy * @.xy;\r\n", 											  atlasedTex, atlasedTex, atParams, atParams, tileParams));		      // Add a newline      meta->addStatement(new GenOp( "\r\n"));		      if(is_sm3)		{         texOp = new GenOp( "tex2Dlod(@, float4(@, 0.0, mipLod_bump))", bumpMap, texCoord );		}		else		{         texOp = new GenOp( "tex2D(@, @)", bumpMap, texCoord );		}	}	else	{      texOp = new GenOp( "tex2D(@, @)", bumpMap, texCoord );	}		   Var *bumpNorm = new Var( "bumpNormal", "float4" );   meta->addStatement( expandNormalMap( texOp, new DecOp( bumpNorm ), 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( "sampler2D" );      bumpMap->setName( "detailBumpMap" );      bumpMap->uniform = true;      bumpMap->sampler = true;      bumpMap->constNum = Var::getTexUnitNum();		      texCoord = getInTexCoord( "detCoord", "vec2", componentList );      texOp = new GenOp( "tex2D(@, @)", 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 ) );   }	   // We transform it into world space by reversing the    // multiplication by the worldToTanget transform.   Var *wsNormal = new Var( "wsNormal", "vec3" );   Var *worldToTanget = getInWorldToTangent( componentList );   meta->addStatement( new GenOp( "   @ = normalize( tMul( @.xyz, @ ) );\r\n", new DecOp( wsNormal ), bumpNorm, worldToTanget ) );}ShaderFeature::Resources BumpFeatGLSL::getResources( const MaterialFeatureData &fd ){   Resources res;    // If we have no parallax then we bring on the normal tex.   if ( !fd.features[MFT_Parallax] )      res.numTex = 1;   // Only the parallax or diffuse map will add texture   // coords other than us.   if (  !fd.features[MFT_Parallax] &&         !fd.features[MFT_DiffuseMap] &&         !fd.features[MFT_OverlayMap] &&         !fd.features[MFT_DetailMap] )      res.numTexReg++;   // We pass the world to tanget space transform.   res.numTexReg += 3;	// Do we have detail normal mapping?   if ( fd.features[MFT_DetailNormalMap] )   {      res.numTex++;      if ( !fd.features[MFT_DetailMap] )         res.numTexReg++;   }		   return res;}void BumpFeatGLSL::setTexData(   Material::StageData &stageDat,                                 const MaterialFeatureData &fd,                                 RenderPassData &passData,                                 U32 &texIndex ){   // If we had a parallax feature then it takes   // care of hooking up the normal map texture.   if ( fd.features[MFT_Parallax] )      return;   if ( fd.features[MFT_NormalMap] )   {      passData.mTexType[ texIndex ] = Material::Bump;      passData.mSamplerNames[ texIndex ] = "bumpMap";      passData.mTexSlot[ texIndex++ ].texObject = stageDat.getTex( MFT_NormalMap );   }	   if ( fd.features[ MFT_DetailNormalMap ] )   {      passData.mTexType[ texIndex ] = Material::DetailBump;      passData.mSamplerNames[ texIndex ] = "detailBumpMap";      passData.mTexSlot[ texIndex++ ].texObject = stageDat.getTex( MFT_DetailNormalMap );   }}ParallaxFeatGLSL::ParallaxFeatGLSL()   : mIncludeDep(ShaderGen::smCommonShaderPath + String("/gl/torque.glsl" )){   addDependency( &mIncludeDep );}Var* ParallaxFeatGLSL::_getUniformVar( const char *name, const char *type, ConstantSortPosition csp ){   Var *theVar = (Var*)LangElement::find( name );   if ( !theVar )   {      theVar = new Var;      theVar->setType( type );      theVar->setName( name );      theVar->uniform = true;      theVar->constSortPos = csp;   }	   return theVar;}void ParallaxFeatGLSL::processVert( Vector<ShaderComponent*> &componentList, 											  const MaterialFeatureData &fd ){   AssertFatal( GFX->getPixelShaderVersion() >= 2.0,       "ParallaxFeatGLSL::processVert - We don't support SM 1.x!" );	   MultiLine *meta = new MultiLine;	   // Add the texture coords.   getOutTexCoord(   "texCoord",                      "vec2", 						fd.features[MFT_TexAnim], 						meta, 						componentList );	   // Grab the input position.   Var *inPos = (Var*)LangElement::find( "inPosition" );   if ( !inPos )      inPos = (Var*)LangElement::find( "position" );	   // Get the object space eye position and the    // object to tangent space transform.   Var *eyePos = _getUniformVar( "eyePos", "vec3", cspPrimitive );   Var *objToTangentSpace = getOutObjToTangentSpace( componentList, meta, fd );	   // Now send the negative view vector in tangent space to the pixel shader.   ShaderConnector *connectComp = dynamic_cast<ShaderConnector *>( componentList[C_CONNECTOR] );   Var *outNegViewTS = connectComp->getElement( RT_TEXCOORD );   outNegViewTS->setName( "outNegViewTS" );   outNegViewTS->setStructName( "OUT" );   outNegViewTS->setType( "vec3" );   meta->addStatement( new GenOp( "   @ = tMul( @, float3( @.xyz - @ ) );\r\n",       outNegViewTS, objToTangentSpace, inPos, eyePos ) );   // If we have texture anim matrix the tangent   // space view vector may need to be rotated.   Var *texMat = (Var*)LangElement::find( "texMat" );   if ( texMat )   {      meta->addStatement( new GenOp( "   @ = tMul(@, float4(@,0)).xyz;\r\n",         outNegViewTS, texMat, outNegViewTS ) );   }	   output = meta;}void ParallaxFeatGLSL::processPix(  Vector<ShaderComponent*> &componentList, 											 const MaterialFeatureData &fd ){   AssertFatal( GFX->getPixelShaderVersion() >= 2.0,       "ParallaxFeatGLSL::processPix - We don't support SM 1.x!" );	   MultiLine *meta = new MultiLine;	   // Order matters... get this first!   Var *texCoord = getInTexCoord( "texCoord", "vec2", componentList );	   ShaderConnector *connectComp = dynamic_cast<ShaderConnector *>( componentList[C_CONNECTOR] );	   // We need the negative tangent space view vector   // as in parallax mapping we step towards the camera.   Var *negViewTS = (Var*)LangElement::find( "negViewTS" );   if ( !negViewTS )   {      Var *inNegViewTS = (Var*)LangElement::find( "outNegViewTS" );      if ( !inNegViewTS )      {         inNegViewTS = connectComp->getElement( RT_TEXCOORD );         inNegViewTS->setName( "outNegViewTS" );         inNegViewTS->setStructName( "IN" );         inNegViewTS->setType( "vec3" );      }		      negViewTS = new Var( "negViewTS", "vec3" );      meta->addStatement( new GenOp( "   @ = normalize( @ );\r\n", new DecOp( negViewTS ), inNegViewTS ) );   }	   // Get the rest of our inputs.   Var *parallaxInfo = _getUniformVar( "parallaxInfo", "float", cspPotentialPrimitive );   Var *normalMap = getNormalMapTex();	   // Call the library function to do the rest.   if (fd.features.hasFeature(MFT_IsBC3nm, getProcessIndex()))   {      meta->addStatement(new GenOp("   @.xy += parallaxOffsetDxtnm( @, @.xy, @, @ );\r\n",      texCoord, normalMap, texCoord, negViewTS, parallaxInfo));   }   else   {      meta->addStatement(new GenOp("   @.xy += parallaxOffset( @, @.xy, @, @ );\r\n",      texCoord, normalMap, texCoord, negViewTS, parallaxInfo));   }      // TODO: Fix second UV maybe?	   output = meta;}ShaderFeature::Resources ParallaxFeatGLSL::getResources( const MaterialFeatureData &fd ){   AssertFatal( GFX->getPixelShaderVersion() >= 2.0,       "ParallaxFeatGLSL::getResources - We don't support SM 1.x!" );	   Resources res;	   // We add the outViewTS to the outputstructure.   res.numTexReg = 1;	   // If this isn't a deferred then we will be   // creating the normal map here.   if ( !fd.features.hasFeature( MFT_DeferredConditioner ) )      res.numTex = 1;	   return res;}void ParallaxFeatGLSL::setTexData(  Material::StageData &stageDat,											 const MaterialFeatureData &fd,											 RenderPassData &passData,											 U32 &texIndex ){   AssertFatal( GFX->getPixelShaderVersion() >= 2.0,       "ParallaxFeatGLSL::setTexData - We don't support SM 1.x!" );	   GFXTextureObject *tex = stageDat.getTex( MFT_NormalMap );   if ( tex )   {      passData.mSamplerNames[ texIndex ] = "bumpMap";      passData.mTexType[ texIndex ] = Material::Bump;      passData.mTexSlot[ texIndex++ ].texObject = tex;   }}void NormalsOutFeatGLSL::processVert(  Vector<ShaderComponent*> &componentList, 												 const MaterialFeatureData &fd ){   // If we have normal maps then we can count   // on it to generate the world space normal.   if ( fd.features[MFT_NormalMap] )      return;	   MultiLine *meta = new MultiLine;   output = meta;	   ShaderConnector *connectComp = dynamic_cast<ShaderConnector *>( componentList[C_CONNECTOR] );	   Var *outNormal = connectComp->getElement( RT_TEXCOORD );   outNormal->setName( "wsNormal" );   outNormal->setStructName( "OUT" );   outNormal->setType( "vec3" );	   // Find the incoming vertex normal.   Var *inNormal = (Var*)LangElement::find( "normal" );      if ( inNormal )   {      // Transform the normal to world space.      Var *objTrans = getObjTrans( componentList, fd.features[MFT_UseInstancing], meta );      meta->addStatement( new GenOp( "   @ = tMul( @, normalize( vec4(@, 0.0) ) ).xyz;\r\n", outNormal, objTrans, inNormal ) );   }   else   {      // If we don't have a vertex normal... just pass the      // camera facing normal to the pixel shader.      meta->addStatement( new GenOp( "   @ = float3( 0.0, 0.0, 1.0 );\r\n", outNormal ) );   }}void NormalsOutFeatGLSL::processPix(   Vector<ShaderComponent*> &componentList, 												const MaterialFeatureData &fd ){   MultiLine *meta = new MultiLine;   output = meta;	   Var *wsNormal = (Var*)LangElement::find( "wsNormal" );   if ( !wsNormal )   {      ShaderConnector *connectComp = dynamic_cast<ShaderConnector *>( componentList[C_CONNECTOR] );      wsNormal = connectComp->getElement( RT_TEXCOORD );      wsNormal->setName( "wsNormal" );      wsNormal->setStructName( "IN" );      wsNormal->setType( "vec3" );		      // If we loaded the normal its our resposibility      // to normalize it... the interpolators won't.      //      // Note we cast to half here to get partial precision      // optimized code which is an acceptable loss of      // precision for normals and performs much better      // on older Geforce cards.      //      meta->addStatement( new GenOp( "   @ = normalize( half3( @ ) );\r\n", wsNormal, wsNormal ) );         }	   LangElement *normalOut;   Var *outColor = (Var*)LangElement::find(getOutputTargetVarName(ShaderFeature::DefaultTarget));   if ( outColor && !fd.features[MFT_AlphaTest] )      normalOut = new GenOp( "float4( ( -@ + 1 ) * 0.5, @.a )", wsNormal, outColor );   else      normalOut = new GenOp( "float4( ( -@ + 1 ) * 0.5, 1 )", wsNormal );	   meta->addStatement( new GenOp( "   @;\r\n", 											assignColor( normalOut, Material::None ) ) );}
 |