2021-07-24 21:11:47 -07:00

224 lines
9.5 KiB
Plaintext

//---------------------------------------------------------------------------------//
// Parallax occlusion mapping algorithm implementation. Pixel shader.
//---------------------------------------------------------------------------------//
//..........................................................................................
// Uniform shader parameters declaration
//..........................................................................................
const float4 g_ParallaxParms : register( c0 );
float fShadowSoftening = 0.59f; // fixme: should be a vmt param
float fHeightMapRange = 0.02f; // fixme: should be a vmt param
// unused stuff
float fSpecularExponent = 100.0f; // fixme: should be a vmt param
float fDiffuseBrightness = 1.0f;
float4 cAmbientColor = float4( 1.0f, 1.0f, 1.0f, 1.0f );
float4 cDiffuseColor = float4( 1.0f, 1.0f, 1.0f, 1.0f );
float4 cSpecularColor = float4( 1.0f, 1.0f, 1.0f, 1.0f );
// texture samplers
sampler tBaseMap;
sampler tNormalMap;
//..........................................................................................
// Note: centroid is specified if multisampling is enabled through RenderMonkey's DirectX
// window preferences settings
//..........................................................................................
struct PS_INPUT
{
float2 texCoord : TEXCOORD0;
// float3 vLightTS : TEXCOORD1_centroid; // light vector in tangent space, denormalized
float3 vViewTS : TEXCOORD2_centroid; // view vector in tangent space, denormalized
float2 vParallaxOffsetTS : TEXCOORD3_centroid; // Parallax offset vector in tangent space
float3 vNormalWS : TEXCOORD4_centroid; // Normal vector in world space
float3 vViewWS : TEXCOORD5_centroid; // View vector in world space
float4 vDebug : TEXCOORD6;
};
//..........................................................................................
// Function: ComputeIllumination
//
// Description: Computes phong illumination for the given pixel using its attribute textures
// and a light vector.
//..........................................................................................
float4 ComputeIllumination( float2 texCoord, float3 vLightTS, float3 vViewTS, float fOcclusionShadow )
{
// Sample the normal from the normal map for the given texture sample:
float3 vNormalTS = normalize( tex2D( tNormalMap, texCoord ) * 2 - 1 );
// Sample base map:
float4 cBaseColor = tex2D( tBaseMap, texCoord );
// Compute diffuse color component:
float4 cDiffuse = saturate( dot( vNormalTS, vLightTS )) * cDiffuseColor;
// Compute specular component:
float3 vReflectionTS = normalize( 2 * dot( vViewTS, vNormalTS ) * vNormalTS - vViewTS );
float fRdotL = dot( vReflectionTS, vLightTS );
float4 cSpecular = saturate( pow( fRdotL, fSpecularExponent )) * cSpecularColor;
float4 cFinalColor = (( cAmbientColor + cDiffuse ) * cBaseColor + cSpecular ) * fOcclusionShadow;
return cFinalColor;
}
//...........................................................................................
// Function: ps_main
//
// Description: Computes pixel illumination result due to applying parallax occlusion mapping
// to simulation of view-dependent surface displacement for a given height map
//...........................................................................................
float4 main( PS_INPUT i ) : COLOR0
{
int nMinSamples = g_ParallaxParms.x;
int nMaxSamples = g_ParallaxParms.y;
// Normalize the interpolated vectors:
float3 vViewTS = normalize( i.vViewTS );
float3 vViewWS = normalize( i.vViewWS );
// float3 vLightTS = normalize( i.vLightTS );
float3 vNormalWS = normalize( i.vNormalWS );
float4 cResultColor = float4( 0, 0, 0, 1 );
// Compute all the derivatives:
float2 dx = ddx( i.texCoord );
float2 dy = ddy( i.texCoord );
//===============================================//
// Parallax occlusion mapping offset computation //
//===============================================//
// Utilize dynamic flow control to change the number of samples per ray
// depending on the viewing angle for the surface. Oblique angles require
// smaller step sizes to achieve more accurate precision for computing displacement.
// We express the sampling rate as a linear function of the angle between
// the geometric normal and the view direction ray:
int nNumSteps = (int) lerp( nMaxSamples, nMinSamples, dot( vViewWS, vNormalWS ) );
// Intersect the view ray with the height field profile along the direction of
// the parallax offset ray (computed in the vertex shader. Note that the code is
// designed specifically to take advantage of the dynamic flow control constructs
// in HLSL and is very sensitive to specific syntax. When converting to other examples,
// if still want to use dynamic flow control in the resulting assembly shader,
// care must be applied.
//
// In the below steps we approximate the height field profile as piecewise linear
// curve. We find the pair of endpoints between which the intersection between the
// height field profile and the view ray is found and then compute line segment
// intersection for the view ray and the line segment formed by the two endpoints.
// This intersection is the displacement offset from the original texture coordinate.
// See the above paper for more details about the process and derivation.
//
float fCurrHeight = 0.0;
float fStepSize = 1.0 / (float) nNumSteps;
float fPrevHeight = 1.0;
float fNextHeight = 0.0;
int nStepIndex = 0;
bool bCondition = true;
float2 vTexOffsetPerStep = fStepSize * i.vParallaxOffsetTS;
float2 vTexCurrentOffset = i.texCoord;
float fCurrentBound = 1.0;
float fParallaxAmount = 0.0;
float2 pt1 = 0;
float2 pt2 = 0;
float2 texOffset2 = 0;
while ( nStepIndex < nNumSteps )
{
vTexCurrentOffset -= vTexOffsetPerStep;
// Sample height map which in this case is stored in the alpha channel of the normal map:
fCurrHeight = tex2Dgrad( tNormalMap, vTexCurrentOffset, dx, dy ).a;
fCurrentBound -= fStepSize;
if ( fCurrHeight > fCurrentBound )
{
pt1 = float2( fCurrentBound, fCurrHeight );
pt2 = float2( fCurrentBound + fStepSize, fPrevHeight );
texOffset2 = vTexCurrentOffset - vTexOffsetPerStep;
nStepIndex = nNumSteps + 1;
}
else
{
nStepIndex++;
fPrevHeight = fCurrHeight;
}
} // End of while ( nStepIndex < nNumSteps )
float fDelta2 = pt2.x - pt2.y;
float fDelta1 = pt1.x - pt1.y;
fParallaxAmount = (pt1.x * fDelta2 - pt2.x * fDelta1 ) / ( fDelta2 - fDelta1 );
float2 vParallaxOffset = i.vParallaxOffsetTS * (1 - fParallaxAmount );
// The computed texture offset for the displaced point on the pseudo-extruded surface:
float2 texSample = i.texCoord - vParallaxOffset;
// float2 vLightRayTS = vLightTS.xy * fHeightMapRange;
// Compute the soft blurry shadows taking into account self-occlusion for features of the height
// field:
// float sh0 = tex2Dgrad( tNormalMap, texSample, dx, dy ).a;
// float shA = (tex2Dgrad( tNormalMap, texSample + vLightRayTS * 0.88, dx, dy ).a - sh0 - 0.88 ) * 1 * fShadowSoftening;
// float sh9 = (tex2Dgrad( tNormalMap, texSample + vLightRayTS * 0.77, dx, dy ).a - sh0 - 0.77 ) * 2 * fShadowSoftening;
// float sh8 = (tex2Dgrad( tNormalMap, texSample + vLightRayTS * 0.66, dx, dy ).a - sh0 - 0.66 ) * 4 * fShadowSoftening;
// float sh7 = (tex2Dgrad( tNormalMap, texSample + vLightRayTS * 0.55, dx, dy ).a - sh0 - 0.55 ) * 6 * fShadowSoftening;
// float sh6 = (tex2Dgrad( tNormalMap, texSample + vLightRayTS * 0.44, dx, dy ).a - sh0 - 0.44 ) * 8 * fShadowSoftening;
// float sh5 = (tex2Dgrad( tNormalMap, texSample + vLightRayTS * 0.33, dx, dy ).a - sh0 - 0.33 ) * 10 * fShadowSoftening;
// float sh4 = (tex2Dgrad( tNormalMap, texSample + vLightRayTS * 0.22, dx, dy ).a - sh0 - 0.22 ) * 12 * fShadowSoftening;
// Compute the actual shadow strength:
// float fOcclusionShadow = 1 - max( max( max( max( max( max( shA, sh9 ), sh8 ), sh7 ), sh6 ), sh5 ), sh4 );
// The previous computation overbrightens the image, let's adjust for that:
// fOcclusionShadow = fOcclusionShadow * 0.6 + 0.4;
// Compute resulting color for the pixel:
// cResultColor = ComputeIllumination( texSample, vLightTS, vViewTS, fOcclusionShadow );
// cResultColor.xyz = ( float )nNumSteps / nMaxSamples;
//cResultColor.xyz = float3( vTexOffsetPerStep, 0.0f );
cResultColor.xyz = tex2Dgrad( tNormalMap, texSample, dx, dy ).rgb;
cResultColor.a = 1.0f;
// cResultColor.xyz = float3( vParallaxOffset * 10.0, 0.0f );
// cResultColor.xyz = ( float3 )fParallaxAmount * .1f;
// cResultColor.xyz = float3( i.texCoord - ( int2 )i.texCoord, 0.0f );
// cResultColor.xyz = i.vViewTS;
// cResultColor.xyz = i.vNormalWS.xyz;
// cResultColor.xyz = float3( i.vParallaxOffsetTS * 10.0f, 0.0f );
// cResultColor = i.vDebug;
// cResultColor.xyz = ( float3 )tex2Dgrad( tNormalMap, texSample, dx, dy ).a;
// cResultColor.xyz = float3( vTexOffsetPerStep * 10.0f, 0.0f );
// cResultColor = float4( 1.0f, 0.0f, 0.0f, 1.0f );
return cResultColor;
} // End of float4 ps_main(..)
#if 0
struct PS_INPUT
{
float2 texCoord : TEXCOORD0;
// float3 vLightTS : TEXCOORD1_centroid; // light vector in tangent space, denormalized
float3 vViewTS : TEXCOORD2_centroid; // view vector in tangent space, denormalized
float2 vParallaxOffsetTS : TEXCOORD3_centroid; // Parallax offset vector in tangent space
float3 vNormalWS : TEXCOORD4_centroid; // Normal vector in world space
float3 vViewWS : TEXCOORD5_centroid; // View vector in world space
};
#endif