UnityGame/Library/PackageCache/com.unity.render-pipelines.universal/Shaders/Terrain/TerrainDetailLitPasses.hlsl
2024-10-27 10:53:47 +03:00

215 lines
7.0 KiB
HLSL

#include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Core.hlsl"
#include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Lighting.hlsl"
#include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/UnityGBuffer.hlsl"
#include "Packages/com.unity.render-pipelines.core/ShaderLibrary/DebugMipmapStreamingMacros.hlsl"
struct Attributes
{
float4 PositionOS : POSITION;
float2 UV0 : TEXCOORD0;
float2 UV1 : TEXCOORD1;
float3 NormalOS : NORMAL;
half4 Color : COLOR;
UNITY_VERTEX_INPUT_INSTANCE_ID
};
struct Varyings
{
float2 UV01 : TEXCOORD0; // UV0
DECLARE_LIGHTMAP_OR_SH(staticLightmapUV, vertexSH, 1);
half4 Color : TEXCOORD2; // Vertex Color
half4 LightingFog : TEXCOORD3; // Vertex Lighting, Fog Factor
#if defined(REQUIRES_VERTEX_SHADOW_COORD_INTERPOLATOR)
float4 ShadowCoords : TEXCOORD4; // Shadow UVs
#endif
half4 NormalWS : TEXCOORD5;
float3 PositionWS : TEXCOORD6;
#ifdef USE_APV_PROBE_OCCLUSION
float4 probeOcclusion : TEXCOORD7;
#endif
float4 PositionCS : SV_POSITION; // Clip Position
UNITY_VERTEX_INPUT_INSTANCE_ID
UNITY_VERTEX_OUTPUT_STEREO
};
void InitializeInputData(Varyings input, out InputData inputData)
{
inputData = (InputData)0;
inputData.positionCS = input.PositionCS;
inputData.normalWS = half3(0, 1, 0);
inputData.viewDirectionWS = half3(0, 0, 1);
#if defined(REQUIRES_VERTEX_SHADOW_COORD_INTERPOLATOR)
inputData.shadowCoord = input.ShadowCoords;
#elif defined(MAIN_LIGHT_CALCULATE_SHADOWS)
inputData.shadowCoord = TransformWorldToShadowCoord(input.PositionWS);
#else
inputData.shadowCoord = float4(0, 0, 0, 0);
#endif
inputData.fogCoord = input.LightingFog.a;
inputData.vertexLighting = input.LightingFog.rgb;
inputData.normalizedScreenSpaceUV = GetNormalizedScreenSpaceUV(input.PositionCS);
inputData.positionWS = input.PositionWS;
#if !defined(LIGHTMAP_ON) && (defined(PROBE_VOLUMES_L1) || defined(PROBE_VOLUMES_L2))
inputData.bakedGI = SAMPLE_GI(input.vertexSH,
GetAbsolutePositionWS(inputData.positionWS),
input.NormalWS.xyz,
GetWorldSpaceNormalizeViewDir(inputData.positionWS),
inputData.positionCS.xy,
input.probeOcclusion,
inputData.shadowMask);
#else
inputData.bakedGI = SAMPLE_GI(input.staticLightmapUV, input.vertexSH, input.NormalWS.xyz);
inputData.shadowMask = SAMPLE_SHADOWMASK(input.staticLightmapUV);
#endif
#if defined(DEBUG_DISPLAY)
inputData.uv = input.UV01;
#if defined(USE_APV_PROBE_OCCLUSION)
inputData.probeOcclusion = input.probeOcclusion;
#endif
#endif
}
void InitializeSurfaceData(half3 albedo, half alpha, out SurfaceData surfaceData)
{
surfaceData = (SurfaceData)0;
surfaceData.albedo = albedo;
surfaceData.alpha = alpha;
surfaceData.emission = half3(0, 0, 0);
surfaceData.metallic = 0;
surfaceData.occlusion = 0;
surfaceData.smoothness = 1;
surfaceData.specular = half3(0, 0, 0);
surfaceData.clearCoatMask = 0;
surfaceData.clearCoatSmoothness = 1;
surfaceData.normalTS = half3(0, 0, 1);
}
half4 UniversalTerrainLit(InputData inputData, SurfaceData surfaceData)
{
#if defined(DEBUG_DISPLAY)
half4 debugColor;
if (CanDebugOverrideOutputColor(inputData, surfaceData, debugColor))
{
return debugColor;
}
#endif
#if defined(MAIN_LIGHT_CALCULATE_SHADOWS)
half3 lighting = inputData.vertexLighting * MainLightRealtimeShadow(inputData.shadowCoord);
#else
half3 lighting = inputData.vertexLighting;
#endif
half4 color = half4(surfaceData.albedo, surfaceData.alpha);
if (IsLightingFeatureEnabled(DEBUGLIGHTINGFEATUREFLAGS_GLOBAL_ILLUMINATION))
{
lighting += inputData.bakedGI;
}
color.rgb *= lighting;
return color;
}
half4 UniversalTerrainLit(InputData inputData, half3 albedo, half alpha)
{
SurfaceData surfaceData;
InitializeSurfaceData(albedo, alpha, surfaceData);
return UniversalTerrainLit(inputData, surfaceData);
}
Varyings TerrainLitVertex(Attributes input)
{
Varyings output = (Varyings)0;
UNITY_SETUP_INSTANCE_ID(input);
UNITY_INITIALIZE_VERTEX_OUTPUT_STEREO(output);
// Vertex attributes
output.UV01 = TRANSFORM_TEX(input.UV0, _MainTex);
OUTPUT_LIGHTMAP_UV(input.UV1, unity_LightmapST, output.staticLightmapUV);
VertexPositionInputs vertexInput = GetVertexPositionInputs(input.PositionOS.xyz);
output.Color = input.Color;
output.PositionCS = vertexInput.positionCS;
// Shadow Coords
#if defined(REQUIRES_VERTEX_SHADOW_COORD_INTERPOLATOR)
output.ShadowCoords = GetShadowCoord(vertexInput);
#endif
// Vertex Lighting
half3 NormalWS = input.NormalOS;
OUTPUT_SH4(vertexInput.positionWS, NormalWS.xyz, GetWorldSpaceNormalizeViewDir(vertexInput.positionWS), output.vertexSH, output.probeOcclusion);
Light mainLight = GetMainLight();
half3 attenuatedLightColor = mainLight.color * mainLight.distanceAttenuation;
half3 diffuseColor = half3(0, 0, 0);
if (IsLightingFeatureEnabled(DEBUGLIGHTINGFEATUREFLAGS_MAIN_LIGHT))
{
diffuseColor += LightingLambert(attenuatedLightColor, mainLight.direction, NormalWS);
}
#if defined(_ADDITIONAL_LIGHTS) || defined(_ADDITIONAL_LIGHTS_VERTEX)
if (IsLightingFeatureEnabled(DEBUGLIGHTINGFEATUREFLAGS_ADDITIONAL_LIGHTS))
{
int pixelLightCount = GetAdditionalLightsCount();
for (int i = 0; i < pixelLightCount; ++i)
{
Light light = GetAdditionalLight(i, vertexInput.positionWS);
half3 attenuatedLightColor = light.color * light.distanceAttenuation;
diffuseColor += LightingLambert(attenuatedLightColor, light.direction, NormalWS);
}
}
#endif
output.LightingFog.xyz = diffuseColor;
// Fog factor
output.LightingFog.w = ComputeFogFactor(output.PositionCS.z);
output.NormalWS.xyz = NormalWS;
output.PositionWS = vertexInput.positionWS;
return output;
}
half4 TerrainLitForwardFragment(Varyings input) : SV_Target
{
UNITY_SETUP_STEREO_EYE_INDEX_POST_VERTEX(input);
InputData inputData;
InitializeInputData(input, inputData);
SETUP_DEBUG_TEXTURE_DATA_FOR_TERRAIN(inputData);
half4 tex = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, input.UV01);
half4 color = UniversalTerrainLit(inputData, tex.rgb, tex.a);
color.rgb = MixFog(color.rgb, inputData.fogCoord);
return color;
}
FragmentOutput TerrainLitGBufferFragment(Varyings input)
{
UNITY_SETUP_INSTANCE_ID(input);
UNITY_SETUP_STEREO_EYE_INDEX_POST_VERTEX(input);
half4 tex = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, input.UV01);
InputData inputData;
InitializeInputData(input, inputData);
SETUP_DEBUG_TEXTURE_DATA_FOR_TERRAIN(inputData);
SurfaceData surfaceData;
InitializeSurfaceData(tex.rgb, tex.a, surfaceData);
half4 color = UniversalTerrainLit(inputData, tex.rgb, tex.a);
return SurfaceDataToGbuffer(surfaceData, inputData, color.rgb, kLightingInvalid);
}