384 lines
13 KiB
C
Raw Permalink Normal View History

2021-07-24 21:11:47 -07:00
//========== Copyright (c) Valve Corporation, All rights reserved. ==========//
//
// Purpose: Common code for tessellation
//
// $NoKeywords: $
//
//===========================================================================//
#ifndef TESSELLATION_VS_FXC_H_
#define TESSELLATION_VS_FXC_H_
#ifdef SHADER_MODEL_VS_3_0
#define TESSELLATION_MODE_ACC_PATCHES_EXTRA 1
#define TESSELLATION_MODE_ACC_PATCHES_REG 2
struct VS_INPUT
{
float2 UV : POSITION0; // Cartesian UV coordinates
float4 BasisU : TEXCOORD0; // BasisU ( precalculated Bernstein basis functions for U )
float4 BasisV : TEXCOORD1; // BasisV ( precalculated Bernstein basis functions for V )
float4 V0_TanU : POSITION1; // Superprim vertex 0
float4 V0_tc01 : TEXCOORD2;
float4 V0_tc23 : TEXCOORD3;
float4 V1_TanU : POSITION2; // Superprim vertex 1
float4 V1_tc01 : TEXCOORD4;
float4 V1_tc23 : TEXCOORD5;
float4 V2_TanU : POSITION3; // Superprim vertex 2
float4 V2_tc01 : TEXCOORD6;
float4 V2_tc23 : TEXCOORD7;
float4 V3_TanU : POSITION4; // Superprim vertex 3
float4 V3_tc01 : TEXCOORD8;
float4 V3_tc23 : TEXCOORD9;
float PatchID : TEXCOORD10; // ID for this patch
};
void LoadACCPatchPos( float patchIndex, out float3 Bez[16],
float flOneOverSubDHeight, sampler2D sampSubD )
{
float idx = ( patchIndex + 0.5 ) * flOneOverSubDHeight;
[unroll]
for (int i = 0; i < 4; i++)
{
float4 tmp[3];
tmp[0] = tex2Dlod( sampSubD, float4((i * 3 + 0.5) / 30, idx, 0, 0) );
tmp[1] = tex2Dlod( sampSubD, float4((i * 3 + 1.5) / 30, idx, 0, 0) );
tmp[2] = tex2Dlod( sampSubD, float4((i * 3 + 2.5) / 30, idx, 0, 0) );
Bez[4 * i + 0] = tmp[0].xyz;
Bez[4 * i + 1] = float3( tmp[0].w, tmp[1].xy );
Bez[4 * i + 2] = float3( tmp[1].zw, tmp[2].x );
Bez[4 * i + 3] = tmp[2].yzw;
}
}
void LoadACCPatchTan( float patchIndex, out float3 TanU[12], out float3 TanV[12],
float flOneOverSubDHeight, sampler2D sampSubD )
{
float idx = ( patchIndex + 0.5 ) * flOneOverSubDHeight;
// Tangents
[unroll]
for (int i = 0; i < 3; i++)
{
float4 tmp[3];
tmp[0] = tex2Dlod(sampSubD, float4((i * 3 + 0.5 + 12) / 30, idx, 0, 0));
tmp[1] = tex2Dlod(sampSubD, float4((i * 3 + 1.5 + 12) / 30, idx, 0, 0));
tmp[2] = tex2Dlod(sampSubD, float4((i * 3 + 2.5 + 12) / 30, idx, 0, 0));
TanU[4 * i + 0] = tmp[0].xyz;
TanU[4 * i + 1] = float3( tmp[0].w, tmp[1].xy );
TanU[4 * i + 2] = float3( tmp[1].zw, tmp[2].x );
TanU[4 * i + 3] = tmp[2].yzw;
}
// Tangents
[unroll]
for (int i = 0; i < 3; i++)
{
float4 tmp[3];
tmp[0] = tex2Dlod(sampSubD, float4((i * 3 + 0.5 + 21) / 30, idx, 0, 0));
tmp[1] = tex2Dlod(sampSubD, float4((i * 3 + 1.5 + 21) / 30, idx, 0, 0));
tmp[2] = tex2Dlod(sampSubD, float4((i * 3 + 2.5 + 21) / 30, idx, 0, 0));
TanV[4 * i + 0] = tmp[0].xyz;
TanV[4 * i + 1] = float3( tmp[0].w, tmp[1].xy );
TanV[4 * i + 2] = float3( tmp[1].zw, tmp[2].x );
TanV[4 * i + 3] = tmp[2].yzw;
}
}
void EvaluateCubicACCPosPatch( in float4 BasisU, in float4 BasisV, float2 UV, float3 cpP[16], out float3 pos )
{
pos = (BasisU.x * cpP[ 0] + BasisU.y * cpP[ 1] + BasisU.z * cpP[ 2] + BasisU.w * cpP[ 3]) * BasisV.x +
(BasisU.x * cpP[ 4] + BasisU.y * cpP[ 5] + BasisU.z * cpP[ 6] + BasisU.w * cpP[ 7]) * BasisV.y +
(BasisU.x * cpP[ 8] + BasisU.y * cpP[ 9] + BasisU.z * cpP[10] + BasisU.w * cpP[11]) * BasisV.z +
(BasisU.x * cpP[12] + BasisU.y * cpP[13] + BasisU.z * cpP[14] + BasisU.w * cpP[15]) * BasisV.w;
}
//--------------------------------------------------------------------------------------
// Cubic Bernstein basis functions
// http://mathworld.wolfram.com/BernsteinPolynomial.html
//--------------------------------------------------------------------------------------
float4 BernsteinBasis( float t )
{
float invT = 1.0f-t;
return float4( invT*invT*invT, 3.0*t*invT*invT, 3.0*t*t*invT, t*t*t );
}
float3 BersteinBasisQuad( float t )
{
float invT = 1.0f-t;
return float3( invT * invT, 2 * invT * t, t * t );
}
void EvaluateCubicACCTanPatches( in float4 BasisU, in float4 BasisV, float2 UV, float3 cpU[12], float3 cpV[12],
out float3 tanU, out float3 tanV )
{
// quadratic bernstein basis functions
float3 qBasisU = BersteinBasisQuad( UV.x );
float3 qBasisV = BersteinBasisQuad( UV.y );
tanU = (qBasisU.x * cpU[ 0] + qBasisU.y * cpU[ 1] + qBasisU.z * cpU[ 2]) * BasisV.x +
(qBasisU.x * cpU[ 3] + qBasisU.y * cpU[ 4] + qBasisU.z * cpU[ 5]) * BasisV.y +
(qBasisU.x * cpU[ 6] + qBasisU.y * cpU[ 7] + qBasisU.z * cpU[ 8]) * BasisV.z +
(qBasisU.x * cpU[ 9] + qBasisU.y * cpU[10] + qBasisU.z * cpU[11]) * BasisV.w;
tanV = (BasisU.x * cpV[ 0] + BasisU.y * cpV[ 1] + BasisU.z * cpV[ 2] + BasisU.w * cpV[ 3]) * qBasisV.x +
(BasisU.x * cpV[ 4] + BasisU.y * cpV[ 5] + BasisU.z * cpV[ 6] + BasisU.w * cpV[ 7]) * qBasisV.y +
(BasisU.x * cpV[ 8] + BasisU.y * cpV[ 9] + BasisU.z * cpV[10] + BasisU.w * cpV[11]) * qBasisV.z;
}
// We define a patch owner for each edge and vertex of the mesh.
// When sampling a displacement map on the boundaries and corners, owner coords are used
//
// Each patch stores: The superprim verts can store
// all of this data like so:
// -- patch U -->
// | X Y Z W
// p t3|t2 t1|t3 +-----------------------------------+
// a --0-----1-- | tanX | tanY | tanZ | sBWrnk | <- Binormal sign flip bit and wrinkle weight
// t t1|t0 t0|t2 +-----------------------------------+
// c | | | innerU | innerV | edgeVU | edgeVV |
// h t2|t0 t0|t1 +-----------------------------------+
// | --3-----2-- | edgeUU | edgeUV | cornerU| cornerV|
// V t3|t1 t2|t3 +-----------------------------------+
//
float2 ComputeConsistentDisplacementUVs( float2 UV,
float4 V0_tc01, float4 V0_tc23,
float4 V1_tc01, float4 V1_tc23,
float4 V2_tc01, float4 V2_tc23,
float4 V3_tc01, float4 V3_tc23 )
{
// Use the tie-breaking scheme for sampling texture coordinates to avoid cracking
float2 t0[4], t1[4], t2[4], t3[4];
t0[0] = V0_tc01.xy;
t0[1] = V0_tc01.zw;
t0[2] = V0_tc23.xy;
t0[3] = V0_tc23.zw;
t1[0] = V1_tc01.xy;
t1[1] = V1_tc01.zw;
t1[2] = V1_tc23.xy;
t1[3] = V1_tc23.zw;
t2[0] = V2_tc01.xy;
t2[1] = V2_tc01.zw;
t2[2] = V2_tc23.xy;
t2[3] = V2_tc23.zw;
t3[0] = V3_tc01.xy;
t3[1] = V3_tc01.zw;
t3[2] = V3_tc23.xy;
t3[3] = V3_tc23.zw;
float flMaxUV = 0.99;
float flMinUV = 0.01;
int i0 = 2 * (UV.x < flMinUV) + (UV.y < flMinUV);
int i1 = (UV.x > flMaxUV) + 2 * (UV.y < flMinUV);
int i2 = 2 * (UV.x > flMaxUV) + (UV.y > flMaxUV);
int i3 = (UV.x < flMinUV) + 2 * (UV.y > flMaxUV);
float2 bottom = lerp( t0[i0], t1[i1], UV.x );
float2 top = lerp( t3[i3], t2[i2], UV.x );
return lerp( bottom, top, UV.y );
}
void DeCasteljau(float u, float3 p0, float3 p1, float3 p2, float3 p3, out float3 p)
{
float3 q0, q1, q2;
float3 r0, r1;
[isolate]
{
q0 = lerp( p0, p1, u );
q1 = lerp( p1, p2, u );
q2 = lerp( p2, p3, u );
r0 = lerp( q0, q1, u );
r1 = lerp( q1, q2, u );
p = lerp( r0, r1, u );
}
}
void DeCasteljau(float u, float3 p0, float3 p1, float3 p2, float3 p3, out float3 p, out float3 dp)
{
float3 q0, q1, q2;
float3 r0, r1;
[isolate]
{
q0 = lerp( p0, p1, u );
q1 = lerp( p1, p2, u );
q2 = lerp( p2, p3, u );
r0 = lerp( q0, q1, u );
r1 = lerp( q1, q2, u );
p = lerp( r0, r1, u );
}
dp = r0 - r1;
}
void EvaluateBezierRegular( float2 uv, float3 p[16], out float3 pos, out float3 nor )
{
float3 t0, t1, t2, t3;
float3 p0, p1, p2, p3;
[isolate]
{
DeCasteljau( uv.x, p[ 0], p[ 1], p[ 2], p[ 3], p0, t0 );
DeCasteljau( uv.x, p[ 4], p[ 5], p[ 6], p[ 7], p1, t1 );
DeCasteljau( uv.x, p[ 8], p[ 9], p[10], p[11], p2, t2 );
DeCasteljau( uv.x, p[12], p[13], p[14], p[15], p3, t3 );
}
float3 du, dv;
DeCasteljau( uv.y, p0, p1, p2, p3, pos, dv );
DeCasteljau( uv.y, t0, t1, t2, t3, du );
nor = normalize( cross(3 * dv, 3 * du) );
}
void EvaluateBezierPosition( float2 uv, float3 p[16], out float3 pos )
{
float3 t0, t1, t2, t3;
float3 p0, p1, p2, p3;
[isolate]
{
DeCasteljau( uv.x, p[ 0], p[ 1], p[ 2], p[ 3], p0, t0 );
DeCasteljau( uv.x, p[ 4], p[ 5], p[ 6], p[ 7], p1, t1 );
DeCasteljau( uv.x, p[ 8], p[ 9], p[10], p[11], p2, t2 );
DeCasteljau( uv.x, p[12], p[13], p[14], p[15], p3, t3 );
}
DeCasteljau( uv.y, p0, p1, p2, p3, pos );
}
void EvaluateSubdivisionSurface( const VS_INPUT v, float flOneOverSubDHeight, float flDoDisplacement, float flDoWrinkledDisplacements,
sampler2D BezierSampler, sampler2D sampDisplacement,
// Outputs
out float3 vWorldNormal, out float3 vWorldPos,
out float3 vWorldTangentS, out float3 vWorldTangentT, out float flBiTangentSign,
out float flWrinkleWeight,
out float2 vTexUV, out float2 vPatchUV,
bool bTangentFrame = true )
{
float4 vInTan;
float2 vDispUV;
float3 vPatchTangent;
float3 vPatchBiTangent;
float4 vBasisU;
float4 vBasisV;
float flPatchLoadIndex;
// PatchUV is passed in for us
vPatchUV = v.UV;
// compute values for tangent based on patchUV
float4 TanUbottom = lerp( v.V0_TanU, v.V1_TanU, vPatchUV.x );
float4 TanUtop = lerp( v.V3_TanU, v.V2_TanU, vPatchUV.x );
vInTan = lerp( TanUbottom, TanUtop, vPatchUV.y );
// compute values for texcoord based on patchUV
float2 bottom = lerp( v.V0_tc01.xy, v.V1_tc01.xy, vPatchUV.x );
float2 top = lerp( v.V3_tc01.xy, v.V2_tc01.xy, vPatchUV.x );
vTexUV = lerp( bottom, top, vPatchUV.y );
// Compute consistent displacement UVs for crack-free displacement mapping
vDispUV = ComputeConsistentDisplacementUVs( vPatchUV,
v.V0_tc01, v.V0_tc23,
v.V1_tc01, v.V1_tc23,
v.V2_tc01, v.V2_tc23,
v.V3_tc01, v.V3_tc23 );
// Cubic Bernstein basis coefficients are passed in for us
vBasisU = v.BasisU;
vBasisV = v.BasisV;
// Patch load index is passed in for us
flPatchLoadIndex = v.PatchID;
float3 ControlPoints[16];
LoadACCPatchPos( flPatchLoadIndex, ControlPoints, flOneOverSubDHeight, BezierSampler );
#if ( TESSELLATION == TESSELLATION_MODE_ACC_PATCHES_REG )
EvaluateBezierRegular( vPatchUV, ControlPoints, vWorldPos, vWorldNormal );
#else
// We split the loading and evaluation of Position patches and Tangent patches to reduce temp register pressure.
// Load and evaluation position
// EvaluateCubicACCPosPatch( vBasisU, vBasisV, vPatchUV, ControlPoints, vWorldPos );
EvaluateBezierPosition( vPatchUV, ControlPoints, vWorldPos );
// Load and evaluate tangent patches
float3 ControlPointsU[12], ControlPointsV[12];
LoadACCPatchTan( flPatchLoadIndex, ControlPointsU, ControlPointsV, flOneOverSubDHeight, BezierSampler );
EvaluateCubicACCTanPatches( vBasisU, vBasisV, vPatchUV, ControlPointsU, ControlPointsV, vPatchTangent, vPatchBiTangent );
vWorldNormal = normalize( cross( vPatchBiTangent, vPatchTangent ) ); // Compute world normal
#endif
// Up to three scalar displacements for { Neutral, Compress, Stretch }
float3 vDisplacement = tex2Dlod( sampDisplacement, float4( vDispUV, 0, 0 ) );
flBiTangentSign = sign( vInTan.w );
if ( bTangentFrame )
{
vWorldTangentS = normalize( vInTan.xyz - ( vWorldNormal * dot( vInTan.xyz, vWorldNormal ) ) ); // Orthonormalize superprim tangent
vWorldTangentT = cross( vWorldNormal, vWorldTangentS.xyz ) * flBiTangentSign; // Sign encodes Binormal flip
}
else
{
vWorldTangentS = vWorldTangentT = vWorldNormal;
}
flWrinkleWeight = abs( vInTan.w ) - 2.0f; // Convert wrinkle weight to -1 to 1 range for pixel shader to use
float3 vDispCoeff = float3(0,0,0); // { Neutral, Compress, Stretch } Displacement Coefficients
vDispCoeff.y = saturate( -flWrinkleWeight ); // One of these two is zero
vDispCoeff.z = saturate( flWrinkleWeight ); // while the other is in the 0..1 range
vDispCoeff *= flDoWrinkledDisplacements; // Separate control for presence of wrinkled displacements (just multiplying by 0 or 1 here)
vDispCoeff.x = 1.0f - vDispCoeff.y - vDispCoeff.z; // Derive neutral weight since these all sum to one
// Displace along normal, using wrinkle displacement map coefficients
vWorldPos += vWorldNormal * ( flDoDisplacement * dot( vDisplacement, vDispCoeff ) );
}
// Wrapper for no-tangent-frame, no-wrinkle version
void EvaluateSubdivisionSurface( VS_INPUT v, float flOneOverSubDHeight, float flDoDisplacement, float flDoWrinkledDisplacements,
sampler2D BezierSampler, sampler2D DispSampler,
// Outputs
out float3 vWorldNormal, out float3 vWorldPos,
out float2 vUV, out float2 vPatchUV )
{
float3 vDummyA, vDummyB;
float flDummyWrinkle;
float flDummyBinormalFlip;
EvaluateSubdivisionSurface( v, flOneOverSubDHeight, flDoDisplacement, flDoWrinkledDisplacements,
BezierSampler, DispSampler, vWorldNormal, vWorldPos, vDummyA, vDummyB,
flDummyBinormalFlip, flDummyWrinkle, vUV, vPatchUV, false );
}
#endif // SHADER_MODEL_VS_3_0
#endif //#ifndef TESSELLATION_VS_FXC_H_