您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
113 行
4.4 KiB
113 行
4.4 KiB
// Library header containing various useful functions for doing moment based shadow maps.
|
|
// Supported flavors are VSM, EVSM and MSM
|
|
|
|
|
|
// conversion helper for VSM flavors
|
|
// Chebychev's inequality (one-tailed version)
|
|
// P( x >= t ) <= pmax(t) := sigma^2 / (sigma^2 + (t - u)^2)
|
|
// for us t is depth, u is E(x) i.d. the blurred depth
|
|
float ShadowMoments_ChebyshevsInequality( float2 moments, float depth, float minVariance, float lightLeakBias )
|
|
{
|
|
// variance sig^2 = E(x^2) - E(x)^2
|
|
float variance = max( moments.y - (moments.x * moments.x), minVariance );
|
|
|
|
// probabilistic upper bound
|
|
float mD = depth - moments.x;
|
|
float p = variance / (variance + mD * mD);
|
|
|
|
p = saturate( (p - lightLeakBias) / (1.0f - lightLeakBias) );
|
|
return max( p, depth <= moments.x );
|
|
}
|
|
|
|
// helper for EVSM
|
|
float2 ShadowMoments_WarpDepth( float depth, float2 exponents )
|
|
{
|
|
// Rescale depth into [-1;1]
|
|
depth = 2.0 * depth - 1.0;
|
|
float pos = exp( exponents.x * depth );
|
|
float neg = -exp(-exponents.y * depth );
|
|
return float2( pos, neg );
|
|
}
|
|
|
|
// helpers for MSM
|
|
// Prepare the moments so there's little quantization error when storing the moments at half
|
|
// precision. This step becomes unnecessary if the moments are stored in 32bit floats.
|
|
float4 ShadowMoments_Encode16MSM( float depth )
|
|
{
|
|
float dsq = depth * depth;
|
|
float4 moments = { depth, dsq, depth * dsq, dsq * dsq };
|
|
float4x4 mat = { - 2.07224649 , 13.7948857237, 0.105877704 , 9.7924062118,
|
|
32.23703778 , -59.4683975703, -1.9077466311, -33.7652110555,
|
|
-68.571074599 , 82.0359750338, 9.3496555107, 47.9456096605,
|
|
39.3703274134, -35.364903257 , -6.6543490743, -23.9728048165 };
|
|
|
|
float4 optimized = mul( moments, mat );
|
|
optimized[0] += 0.035955884801;
|
|
|
|
return optimized;
|
|
}
|
|
|
|
float4 ShadowMoments_Decode16MSM( float4 moments )
|
|
{
|
|
moments[0] -= 0.035955884801;
|
|
float4x4 mat = { 0.2227744146, 0.1549679261, 0.1451988946, 0.163127443,
|
|
0.0771972861, 0.1394629426, 0.2120202157, 0.2591432266,
|
|
0.7926986636, 0.7963415838, 0.7258694464, 0.6539092497,
|
|
0.0319417555, -0.1722823173, -0.2758014811, -0.3376131734 };
|
|
return mul( moments, mat );
|
|
}
|
|
|
|
void ShadowMoments_SolveMSM( float4 moments, float depth, float momentBias, out float3 z, out float4 b )
|
|
{
|
|
// Bias input data to avoid artifacts
|
|
z[0] = depth;
|
|
b = lerp( moments, 0.5.xxxx, momentBias );
|
|
|
|
// Compute a Cholesky factorization of the Hankel matrix B storing only non-trivial entries or related products
|
|
float L32D22 = mad( -b[0], b[1], b[2] );
|
|
float D22 = mad( -b[0], b[0], b[1] );
|
|
float sqDepthVar = mad( -b[1], b[1], b[3] );
|
|
float D33D22 = dot( float2( sqDepthVar, -L32D22 ), float2( D22, L32D22 ) );
|
|
float InvD22 = 1.0 / D22;
|
|
float L32 = L32D22 * InvD22;
|
|
// Obtain a scaled inverse image of bz = ( 1, z[0], z[0]*z[0] )^T
|
|
float3 c = float3( 1.0, z[0], z[0] * z[0] );
|
|
// Forward substitution to solve L * c1 = bz;
|
|
c[1] -= b.x;
|
|
c[2] -= b.y + L32 * c[1];
|
|
// Scaling to solve D * c2 = c1
|
|
c[1] *= InvD22;
|
|
c[2] *= D22 / D33D22;
|
|
// Backward substitution to solve L^T * c3 = c2
|
|
c[1] -= L32 * c[2];
|
|
c[0] -= dot( c.yz, b.xy );
|
|
// Solve the quadratic equation c[0] + c[1] * z + c[2] * z^2 to obtain solutions z[1] and z[2]
|
|
float p = c[1] / c[2];
|
|
float q = c[0] / c[2];
|
|
float D = ((p*p) * 0.25) - q;
|
|
float r = sqrt( D );
|
|
z[1] = -(p * 0.5) - r;
|
|
z[2] = -(p * 0.5) + r;
|
|
}
|
|
|
|
float ShadowMoments_SolveDelta3MSM( float3 z, float2 b, float lightLeakBias )
|
|
{
|
|
// Construct a solution composed of three Dirac-deltas and evaluate its CDF
|
|
float4 switchVal = (z[2] < z[0]) ? float4( z[1], z[0], 1.0, 1.0 )
|
|
: ((z[1] < z[0]) ? float4( z[0], z[1], 0.0, 1.0 ) : 0.0.xxxx);
|
|
|
|
float quotient = (switchVal[0] * z[2] - b[0] * (switchVal[0] + z[2]) + b[1]) / ((z[2] - switchVal[1]) * (z[0] - z[1]));
|
|
float attenuation = saturate( switchVal[2] + switchVal[3] * quotient );
|
|
|
|
return saturate( ((1.0 - attenuation) - lightLeakBias) / (1.0f - lightLeakBias) );
|
|
}
|
|
|
|
float ShadowMoments_SolveDelta4MSM( float3 z, float4 b, float lightLeakBias)
|
|
{
|
|
// Use a solution made of four deltas
|
|
float zFree = ((b[2] - b[1]) * z[0] + b[2] - b[3]) / ((b[1] - b[0]) * z[0] + b[1] - b[2]);
|
|
float w1Factor = (z[0] > zFree) ? 1.0 : 0.0;
|
|
float attenuation = saturate( (b[1] - b[0] + (b[2] - b[0] - (zFree + 1.0) * (b[1] - b[0])) * (zFree - w1Factor - z[0]) / (z[0] * (z[0] - zFree))) / (zFree - w1Factor) + 1.0 - b[0] );
|
|
|
|
return saturate( ((1.0 - attenuation) - lightLeakBias) / (1.0f - lightLeakBias) );
|
|
}
|