| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 |
- #include "Uniforms.hlsl"
- #include "Samplers.hlsl"
- #include "Transform.hlsl"
- #include "ScreenPos.hlsl"
- #include "Lighting.hlsl"
- void VS(float4 iPos : POSITION,
- #ifdef DIRLIGHT
- out float2 oScreenPos : TEXCOORD0,
- #else
- out float4 oScreenPos : TEXCOORD0,
- #endif
- out float3 oFarRay : TEXCOORD1,
- #ifdef ORTHO
- out float3 oNearRay : TEXCOORD2,
- #endif
- out float4 oPos : POSITION)
- {
- float4x3 modelMatrix = iModelMatrix;
- float3 worldPos = GetWorldPos(modelMatrix);
- oPos = GetClipPos(worldPos);
- #ifdef DIRLIGHT
- oScreenPos = GetScreenPosPreDiv(oPos);
- oFarRay = GetFarRay(oPos);
- #ifdef ORTHO
- oNearRay = GetNearRay(oPos);
- #endif
- #else
- oScreenPos = GetScreenPos(oPos);
- oFarRay = GetFarRay(oPos) * oPos.w;
- #ifdef ORTHO
- oNearRay = GetNearRay(oPos) * oPos.w;
- #endif
- #endif
- }
- void PS(
- #ifdef DIRLIGHT
- float2 iScreenPos : TEXCOORD0,
- #else
- float4 iScreenPos : TEXCOORD0,
- #endif
- float3 iFarRay : TEXCOORD1,
- #ifdef ORTHO
- float3 iNearRay : TEXCOORD2,
- #endif
- out float4 oColor : COLOR0)
- {
- // If rendering a directional light quad, optimize out the w divide
- #ifndef FALLBACK
- #ifdef DIRLIGHT
- #ifdef ORTHO
- float depth = Sample(sDepthBuffer, iScreenPos).r;
- float3 worldPos = lerp(iNearRay, iFarRay, depth);
- #else
- #ifdef LINEAR
- float depth = Sample(sDepthBuffer, iScreenPos).r;
- #else
- float depth = ReconstructDepth(Sample(sDepthBuffer, iScreenPos).r);
- #endif
- float3 worldPos = iFarRay * depth;
- #endif
- float4 normalInput = Sample(sNormalBuffer, iScreenPos);
- #else
- #ifdef ORTHO
- float depth = tex2Dproj(sDepthBuffer, iScreenPos).r;
- float3 worldPos = lerp(iNearRay, iFarRay, depth) / iScreenPos.w;
- #else
- #ifdef LINEAR
- float depth = tex2Dproj(sDepthBuffer, iScreenPos).r;
- #else
- float depth = ReconstructDepth(tex2Dproj(sDepthBuffer, iScreenPos).r);
- #endif
- float3 worldPos = iFarRay * depth / iScreenPos.w;
- #endif
- float4 normalInput = tex2Dproj(sNormalBuffer, iScreenPos);
- #endif
- // With specular, normalization greatly improves stability of reflections,
- // considering input is only 8 bits per axis
- #ifdef SPECULAR
- float3 normal = normalize(normalInput.rgb * 2.0 - 1.0);
- #else
- float3 normal = normalInput.rgb * 2.0 - 1.0;
- #endif
- #else
- float3 normal;
- float depth;
- #ifdef DIRLIGHT
- UnpackNormalDepth(Sample(sNormalBuffer, iScreenPos), normal, depth);
- #ifdef ORTHO
- float3 worldPos = lerp(iNearRay, iFarRay, depth);
- #else
- float3 worldPos = iFarRay * depth;
- #endif
- #else
- UnpackNormalDepth(tex2Dproj(sNormalBuffer, iScreenPos), normal, depth);
- #ifdef ORTHO
- float3 worldPos = lerp(iNearRay, iFarRay, depth) / iScreenPos.w;
- #else
- float3 worldPos = iFarRay * depth / iScreenPos.w;
- #endif
- #endif
- #endif
- float3 lightColor;
- float3 lightDir;
- float diff;
- // Accumulate light at half intensity to allow 2x "overburn"
- #ifdef DIRLIGHT
- lightDir = cLightDirPS;
- diff = 0.5 * GetDiffuseDir(normal, lightDir);
- #else
- float3 lightVec = (cLightPosPS.xyz - worldPos) * cLightPosPS.w;
- diff = 0.5 * GetDiffusePointOrSpot(normal, lightVec, lightDir);
- #endif
- #ifdef SHADOW
- #if defined(DIRLIGHT)
- float4x4 shadowMatrix = GetDirShadowMatrix(depth);
- float4 shadowPos = mul(float4(worldPos, 1.0), shadowMatrix);
- diff *= saturate(GetShadow(shadowPos) + GetShadowFade(depth));
- #elif defined(SPOTLIGHT)
- float4 shadowPos = mul(float4(worldPos, 1.0), cShadowProjPS[1]);
- diff *= GetShadow(shadowPos);
- #else
- float3 shadowPos = worldPos - cLightPosPS.xyz;
- diff *= GetCubeShadow(shadowPos);
- #endif
- #endif
- #ifdef SPOTLIGHT
- float4 spotPos = mul(float4(worldPos, 1.0), cShadowProjPS[0]);
- lightColor = spotPos.w > 0.0 ? tex2Dproj(sLightSpotMap, spotPos).rgb * cLightColor.rgb : 0.0;
- #else
- #ifdef CUBEMASK
- lightColor = texCUBE(sLightCubeMap, mul(lightVec, (float3x3)cShadowProjPS[0])).rgb * cLightColor.rgb;
- #else
- lightColor = cLightColor.rgb;
- #endif
- #endif
- #ifdef SPECULAR
- float spec = lightColor.g * GetSpecular(normal, -worldPos, lightDir, normalInput.a * 255.0);
- oColor = diff * float4(lightColor, spec * cLightColor.a);
- #else
- oColor = diff * float4(lightColor, 0.0);
- #endif
- }
|