433 lines
18 KiB
HLSL
433 lines
18 KiB
HLSL
|
|
||
|
#ifndef UNIVERSAL_GLOBAL_ILLUMINATION_INCLUDED
|
||
|
#define UNIVERSAL_GLOBAL_ILLUMINATION_INCLUDED
|
||
|
|
||
|
#include "Packages/com.unity.render-pipelines.core/ShaderLibrary/EntityLighting.hlsl"
|
||
|
#include "Packages/com.unity.render-pipelines.core/ShaderLibrary/ImageBasedLighting.hlsl"
|
||
|
#include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/RealtimeLights.hlsl"
|
||
|
|
||
|
// If lightmap is not defined than we evaluate GI (ambient + probes) from SH
|
||
|
// We might do it fully or partially in vertex to save shader ALU
|
||
|
#if !defined(LIGHTMAP_ON)
|
||
|
// TODO: Controls things like these by exposing SHADER_QUALITY levels (low, medium, high)
|
||
|
#if defined(SHADER_API_GLES) || !defined(_NORMALMAP)
|
||
|
// Evaluates SH fully in vertex
|
||
|
#define EVALUATE_SH_VERTEX
|
||
|
#elif !SHADER_HINT_NICE_QUALITY
|
||
|
// Evaluates L2 SH in vertex and L0L1 in pixel
|
||
|
#define EVALUATE_SH_MIXED
|
||
|
#endif
|
||
|
// Otherwise evaluate SH fully per-pixel
|
||
|
#endif
|
||
|
|
||
|
// Renamed -> LIGHTMAP_SHADOW_MIXING
|
||
|
#if !defined(_MIXED_LIGHTING_SUBTRACTIVE) && defined(LIGHTMAP_SHADOW_MIXING) && !defined(SHADOWS_SHADOWMASK)
|
||
|
#define _MIXED_LIGHTING_SUBTRACTIVE
|
||
|
#endif
|
||
|
|
||
|
// Samples SH L0, L1 and L2 terms
|
||
|
half3 SampleSH(half3 normalWS)
|
||
|
{
|
||
|
// LPPV is not supported in Ligthweight Pipeline
|
||
|
real4 SHCoefficients[7];
|
||
|
SHCoefficients[0] = unity_SHAr;
|
||
|
SHCoefficients[1] = unity_SHAg;
|
||
|
SHCoefficients[2] = unity_SHAb;
|
||
|
SHCoefficients[3] = unity_SHBr;
|
||
|
SHCoefficients[4] = unity_SHBg;
|
||
|
SHCoefficients[5] = unity_SHBb;
|
||
|
SHCoefficients[6] = unity_SHC;
|
||
|
|
||
|
return max(half3(0, 0, 0), SampleSH9(SHCoefficients, normalWS));
|
||
|
}
|
||
|
|
||
|
// SH Vertex Evaluation. Depending on target SH sampling might be
|
||
|
// done completely per vertex or mixed with L2 term per vertex and L0, L1
|
||
|
// per pixel. See SampleSHPixel
|
||
|
half3 SampleSHVertex(half3 normalWS)
|
||
|
{
|
||
|
#if defined(EVALUATE_SH_VERTEX)
|
||
|
return SampleSH(normalWS);
|
||
|
#elif defined(EVALUATE_SH_MIXED)
|
||
|
// no max since this is only L2 contribution
|
||
|
return SHEvalLinearL2(normalWS, unity_SHBr, unity_SHBg, unity_SHBb, unity_SHC);
|
||
|
#endif
|
||
|
|
||
|
// Fully per-pixel. Nothing to compute.
|
||
|
return half3(0.0, 0.0, 0.0);
|
||
|
}
|
||
|
|
||
|
// SH Pixel Evaluation. Depending on target SH sampling might be done
|
||
|
// mixed or fully in pixel. See SampleSHVertex
|
||
|
half3 SampleSHPixel(half3 L2Term, half3 normalWS)
|
||
|
{
|
||
|
#if defined(EVALUATE_SH_VERTEX)
|
||
|
return L2Term;
|
||
|
#elif defined(EVALUATE_SH_MIXED)
|
||
|
half3 res = L2Term + SHEvalLinearL0L1(normalWS, unity_SHAr, unity_SHAg, unity_SHAb);
|
||
|
#ifdef UNITY_COLORSPACE_GAMMA
|
||
|
res = LinearToSRGB(res);
|
||
|
#endif
|
||
|
return max(half3(0, 0, 0), res);
|
||
|
#endif
|
||
|
|
||
|
// Default: Evaluate SH fully per-pixel
|
||
|
return SampleSH(normalWS);
|
||
|
}
|
||
|
|
||
|
#if defined(UNITY_DOTS_INSTANCING_ENABLED)
|
||
|
#define LIGHTMAP_NAME unity_Lightmaps
|
||
|
#define LIGHTMAP_INDIRECTION_NAME unity_LightmapsInd
|
||
|
#define LIGHTMAP_SAMPLER_NAME samplerunity_Lightmaps
|
||
|
#define LIGHTMAP_SAMPLE_EXTRA_ARGS staticLightmapUV, unity_LightmapIndex.x
|
||
|
#else
|
||
|
#define LIGHTMAP_NAME unity_Lightmap
|
||
|
#define LIGHTMAP_INDIRECTION_NAME unity_LightmapInd
|
||
|
#define LIGHTMAP_SAMPLER_NAME samplerunity_Lightmap
|
||
|
#define LIGHTMAP_SAMPLE_EXTRA_ARGS staticLightmapUV
|
||
|
#endif
|
||
|
|
||
|
// Sample baked and/or realtime lightmap. Non-Direction and Directional if available.
|
||
|
half3 SampleLightmap(float2 staticLightmapUV, float2 dynamicLightmapUV, half3 normalWS)
|
||
|
{
|
||
|
#ifdef UNITY_LIGHTMAP_FULL_HDR
|
||
|
bool encodedLightmap = false;
|
||
|
#else
|
||
|
bool encodedLightmap = true;
|
||
|
#endif
|
||
|
|
||
|
half4 decodeInstructions = half4(LIGHTMAP_HDR_MULTIPLIER, LIGHTMAP_HDR_EXPONENT, 0.0h, 0.0h);
|
||
|
|
||
|
// The shader library sample lightmap functions transform the lightmap uv coords to apply bias and scale.
|
||
|
// However, universal pipeline already transformed those coords in vertex. We pass half4(1, 1, 0, 0) and
|
||
|
// the compiler will optimize the transform away.
|
||
|
half4 transformCoords = half4(1, 1, 0, 0);
|
||
|
|
||
|
float3 diffuseLighting = 0;
|
||
|
|
||
|
#if defined(LIGHTMAP_ON) && defined(DIRLIGHTMAP_COMBINED)
|
||
|
diffuseLighting = SampleDirectionalLightmap(TEXTURE2D_LIGHTMAP_ARGS(LIGHTMAP_NAME, LIGHTMAP_SAMPLER_NAME),
|
||
|
TEXTURE2D_LIGHTMAP_ARGS(LIGHTMAP_INDIRECTION_NAME, LIGHTMAP_SAMPLER_NAME),
|
||
|
LIGHTMAP_SAMPLE_EXTRA_ARGS, transformCoords, normalWS, encodedLightmap, decodeInstructions);
|
||
|
#elif defined(LIGHTMAP_ON)
|
||
|
diffuseLighting = SampleSingleLightmap(TEXTURE2D_LIGHTMAP_ARGS(LIGHTMAP_NAME, LIGHTMAP_SAMPLER_NAME), LIGHTMAP_SAMPLE_EXTRA_ARGS, transformCoords, encodedLightmap, decodeInstructions);
|
||
|
#endif
|
||
|
|
||
|
#if defined(DYNAMICLIGHTMAP_ON) && defined(DIRLIGHTMAP_COMBINED)
|
||
|
diffuseLighting += SampleDirectionalLightmap(TEXTURE2D_ARGS(unity_DynamicLightmap, samplerunity_DynamicLightmap),
|
||
|
TEXTURE2D_ARGS(unity_DynamicDirectionality, samplerunity_DynamicLightmap),
|
||
|
dynamicLightmapUV, transformCoords, normalWS, false, decodeInstructions);
|
||
|
#elif defined(DYNAMICLIGHTMAP_ON)
|
||
|
diffuseLighting += SampleSingleLightmap(TEXTURE2D_ARGS(unity_DynamicLightmap, samplerunity_DynamicLightmap),
|
||
|
dynamicLightmapUV, transformCoords, false, decodeInstructions);
|
||
|
#endif
|
||
|
|
||
|
return diffuseLighting;
|
||
|
}
|
||
|
|
||
|
// Legacy version of SampleLightmap where Realtime GI is not supported.
|
||
|
half3 SampleLightmap(float2 staticLightmapUV, half3 normalWS)
|
||
|
{
|
||
|
float2 dummyDynamicLightmapUV = float2(0,0);
|
||
|
half3 result = SampleLightmap(staticLightmapUV, dummyDynamicLightmapUV, normalWS);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
// We either sample GI from baked lightmap or from probes.
|
||
|
// If lightmap: sampleData.xy = lightmapUV
|
||
|
// If probe: sampleData.xyz = L2 SH terms
|
||
|
#if defined(LIGHTMAP_ON) && defined(DYNAMICLIGHTMAP_ON)
|
||
|
#define SAMPLE_GI(staticLmName, dynamicLmName, shName, normalWSName) SampleLightmap(staticLmName, dynamicLmName, normalWSName)
|
||
|
#elif defined(DYNAMICLIGHTMAP_ON)
|
||
|
#define SAMPLE_GI(staticLmName, dynamicLmName, shName, normalWSName) SampleLightmap(0, dynamicLmName, normalWSName)
|
||
|
#elif defined(LIGHTMAP_ON)
|
||
|
#define SAMPLE_GI(staticLmName, shName, normalWSName) SampleLightmap(staticLmName, 0, normalWSName)
|
||
|
#else
|
||
|
#define SAMPLE_GI(staticLmName, shName, normalWSName) SampleSHPixel(shName, normalWSName)
|
||
|
#endif
|
||
|
|
||
|
half3 BoxProjectedCubemapDirection(half3 reflectionWS, float3 positionWS, float4 cubemapPositionWS, float4 boxMin, float4 boxMax)
|
||
|
{
|
||
|
// Is this probe using box projection?
|
||
|
if (cubemapPositionWS.w > 0.0f)
|
||
|
{
|
||
|
float3 boxMinMax = (reflectionWS > 0.0f) ? boxMax.xyz : boxMin.xyz;
|
||
|
half3 rbMinMax = half3(boxMinMax - positionWS) / reflectionWS;
|
||
|
|
||
|
half fa = half(min(min(rbMinMax.x, rbMinMax.y), rbMinMax.z));
|
||
|
|
||
|
half3 worldPos = half3(positionWS - cubemapPositionWS.xyz);
|
||
|
|
||
|
half3 result = worldPos + reflectionWS * fa;
|
||
|
return result;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return reflectionWS;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
float CalculateProbeWeight(float3 positionWS, float4 probeBoxMin, float4 probeBoxMax)
|
||
|
{
|
||
|
float blendDistance = probeBoxMax.w;
|
||
|
float3 weightDir = min(positionWS - probeBoxMin.xyz, probeBoxMax.xyz - positionWS) / blendDistance;
|
||
|
return saturate(min(weightDir.x, min(weightDir.y, weightDir.z)));
|
||
|
}
|
||
|
|
||
|
half CalculateProbeVolumeSqrMagnitude(float4 probeBoxMin, float4 probeBoxMax)
|
||
|
{
|
||
|
half3 maxToMin = half3(probeBoxMax.xyz - probeBoxMin.xyz);
|
||
|
return dot(maxToMin, maxToMin);
|
||
|
}
|
||
|
|
||
|
half3 CalculateIrradianceFromReflectionProbes(half3 reflectVector, float3 positionWS, half perceptualRoughness)
|
||
|
{
|
||
|
half probe0Volume = CalculateProbeVolumeSqrMagnitude(unity_SpecCube0_BoxMin, unity_SpecCube0_BoxMax);
|
||
|
half probe1Volume = CalculateProbeVolumeSqrMagnitude(unity_SpecCube1_BoxMin, unity_SpecCube1_BoxMax);
|
||
|
|
||
|
half volumeDiff = probe0Volume - probe1Volume;
|
||
|
float importanceSign = unity_SpecCube1_BoxMin.w;
|
||
|
|
||
|
// A probe is dominant if its importance is higher
|
||
|
// Or have equal importance but smaller volume
|
||
|
bool probe0Dominant = importanceSign > 0.0f || (importanceSign == 0.0f && volumeDiff < -0.0001h);
|
||
|
bool probe1Dominant = importanceSign < 0.0f || (importanceSign == 0.0f && volumeDiff > 0.0001h);
|
||
|
|
||
|
float desiredWeightProbe0 = CalculateProbeWeight(positionWS, unity_SpecCube0_BoxMin, unity_SpecCube0_BoxMax);
|
||
|
float desiredWeightProbe1 = CalculateProbeWeight(positionWS, unity_SpecCube1_BoxMin, unity_SpecCube1_BoxMax);
|
||
|
|
||
|
// Subject the probes weight if the other probe is dominant
|
||
|
float weightProbe0 = probe1Dominant ? min(desiredWeightProbe0, 1.0f - desiredWeightProbe1) : desiredWeightProbe0;
|
||
|
float weightProbe1 = probe0Dominant ? min(desiredWeightProbe1, 1.0f - desiredWeightProbe0) : desiredWeightProbe1;
|
||
|
|
||
|
float totalWeight = weightProbe0 + weightProbe1;
|
||
|
|
||
|
// If either probe 0 or probe 1 is dominant the sum of weights is guaranteed to be 1.
|
||
|
// If neither is dominant this is not guaranteed - only normalize weights if totalweight exceeds 1.
|
||
|
weightProbe0 /= max(totalWeight, 1.0f);
|
||
|
weightProbe1 /= max(totalWeight, 1.0f);
|
||
|
|
||
|
half3 irradiance = half3(0.0h, 0.0h, 0.0h);
|
||
|
half3 originalReflectVector = reflectVector;
|
||
|
half mip = PerceptualRoughnessToMipmapLevel(perceptualRoughness);
|
||
|
|
||
|
// Sample the first reflection probe
|
||
|
if (weightProbe0 > 0.01f)
|
||
|
{
|
||
|
#ifdef _REFLECTION_PROBE_BOX_PROJECTION
|
||
|
reflectVector = BoxProjectedCubemapDirection(originalReflectVector, positionWS, unity_SpecCube0_ProbePosition, unity_SpecCube0_BoxMin, unity_SpecCube0_BoxMax);
|
||
|
#endif // _REFLECTION_PROBE_BOX_PROJECTION
|
||
|
|
||
|
half4 encodedIrradiance = half4(SAMPLE_TEXTURECUBE_LOD(unity_SpecCube0, samplerunity_SpecCube0, reflectVector, mip));
|
||
|
|
||
|
#if defined(UNITY_USE_NATIVE_HDR)
|
||
|
irradiance += weightProbe0 * encodedIrradiance.rbg;
|
||
|
#else
|
||
|
irradiance += weightProbe0 * DecodeHDREnvironment(encodedIrradiance, unity_SpecCube0_HDR);
|
||
|
#endif // UNITY_USE_NATIVE_HDR
|
||
|
}
|
||
|
|
||
|
// Sample the second reflection probe
|
||
|
if (weightProbe1 > 0.01f)
|
||
|
{
|
||
|
#ifdef _REFLECTION_PROBE_BOX_PROJECTION
|
||
|
reflectVector = BoxProjectedCubemapDirection(originalReflectVector, positionWS, unity_SpecCube1_ProbePosition, unity_SpecCube1_BoxMin, unity_SpecCube1_BoxMax);
|
||
|
#endif // _REFLECTION_PROBE_BOX_PROJECTION
|
||
|
half4 encodedIrradiance = half4(SAMPLE_TEXTURECUBE_LOD(unity_SpecCube1, samplerunity_SpecCube1, reflectVector, mip));
|
||
|
|
||
|
#if defined(UNITY_USE_NATIVE_HDR) || defined(UNITY_DOTS_INSTANCING_ENABLED)
|
||
|
irradiance += weightProbe1 * encodedIrradiance.rbg;
|
||
|
#else
|
||
|
irradiance += weightProbe1 * DecodeHDREnvironment(encodedIrradiance, unity_SpecCube1_HDR);
|
||
|
#endif // UNITY_USE_NATIVE_HDR || UNITY_DOTS_INSTANCING_ENABLED
|
||
|
}
|
||
|
|
||
|
// Use any remaining weight to blend to environment reflection cube map
|
||
|
if (totalWeight < 0.99f)
|
||
|
{
|
||
|
half4 encodedIrradiance = half4(SAMPLE_TEXTURECUBE_LOD(_GlossyEnvironmentCubeMap, sampler_GlossyEnvironmentCubeMap, originalReflectVector, mip));
|
||
|
|
||
|
#if defined(UNITY_USE_NATIVE_HDR) || defined(UNITY_DOTS_INSTANCING_ENABLED)
|
||
|
irradiance += (1.0f - totalWeight) * encodedIrradiance.rbg;
|
||
|
#else
|
||
|
irradiance += (1.0f - totalWeight) * DecodeHDREnvironment(encodedIrradiance, _GlossyEnvironmentCubeMap_HDR);
|
||
|
#endif // UNITY_USE_NATIVE_HDR || UNITY_DOTS_INSTANCING_ENABLED
|
||
|
}
|
||
|
|
||
|
return irradiance;
|
||
|
}
|
||
|
|
||
|
half3 GlossyEnvironmentReflection(half3 reflectVector, float3 positionWS, half perceptualRoughness, half occlusion)
|
||
|
{
|
||
|
#if !defined(_ENVIRONMENTREFLECTIONS_OFF)
|
||
|
half3 irradiance;
|
||
|
|
||
|
#ifdef _REFLECTION_PROBE_BLENDING
|
||
|
irradiance = CalculateIrradianceFromReflectionProbes(reflectVector, positionWS, perceptualRoughness);
|
||
|
#else
|
||
|
#ifdef _REFLECTION_PROBE_BOX_PROJECTION
|
||
|
reflectVector = BoxProjectedCubemapDirection(reflectVector, positionWS, unity_SpecCube0_ProbePosition, unity_SpecCube0_BoxMin, unity_SpecCube0_BoxMax);
|
||
|
#endif // _REFLECTION_PROBE_BOX_PROJECTION
|
||
|
half mip = PerceptualRoughnessToMipmapLevel(perceptualRoughness);
|
||
|
half4 encodedIrradiance = half4(SAMPLE_TEXTURECUBE_LOD(unity_SpecCube0, samplerunity_SpecCube0, reflectVector, mip));
|
||
|
|
||
|
#if defined(UNITY_USE_NATIVE_HDR)
|
||
|
irradiance = encodedIrradiance.rgb;
|
||
|
#else
|
||
|
irradiance = DecodeHDREnvironment(encodedIrradiance, unity_SpecCube0_HDR);
|
||
|
#endif // UNITY_USE_NATIVE_HDR
|
||
|
#endif // _REFLECTION_PROBE_BLENDING
|
||
|
return irradiance * occlusion;
|
||
|
#else
|
||
|
return _GlossyEnvironmentColor.rgb * occlusion;
|
||
|
#endif // _ENVIRONMENTREFLECTIONS_OFF
|
||
|
}
|
||
|
|
||
|
half3 GlossyEnvironmentReflection(half3 reflectVector, half perceptualRoughness, half occlusion)
|
||
|
{
|
||
|
#if !defined(_ENVIRONMENTREFLECTIONS_OFF)
|
||
|
half3 irradiance;
|
||
|
half mip = PerceptualRoughnessToMipmapLevel(perceptualRoughness);
|
||
|
half4 encodedIrradiance = half4(SAMPLE_TEXTURECUBE_LOD(unity_SpecCube0, samplerunity_SpecCube0, reflectVector, mip));
|
||
|
|
||
|
#if defined(UNITY_USE_NATIVE_HDR)
|
||
|
irradiance = encodedIrradiance.rgb;
|
||
|
#else
|
||
|
irradiance = DecodeHDREnvironment(encodedIrradiance, unity_SpecCube0_HDR);
|
||
|
#endif // UNITY_USE_NATIVE_HDR
|
||
|
|
||
|
return irradiance * occlusion;
|
||
|
#else
|
||
|
|
||
|
return _GlossyEnvironmentColor.rgb * occlusion;
|
||
|
#endif // _ENVIRONMENTREFLECTIONS_OFF
|
||
|
}
|
||
|
|
||
|
half3 SubtractDirectMainLightFromLightmap(Light mainLight, half3 normalWS, half3 bakedGI)
|
||
|
{
|
||
|
// Let's try to make realtime shadows work on a surface, which already contains
|
||
|
// baked lighting and shadowing from the main sun light.
|
||
|
// Summary:
|
||
|
// 1) Calculate possible value in the shadow by subtracting estimated light contribution from the places occluded by realtime shadow:
|
||
|
// a) preserves other baked lights and light bounces
|
||
|
// b) eliminates shadows on the geometry facing away from the light
|
||
|
// 2) Clamp against user defined ShadowColor.
|
||
|
// 3) Pick original lightmap value, if it is the darkest one.
|
||
|
|
||
|
|
||
|
// 1) Gives good estimate of illumination as if light would've been shadowed during the bake.
|
||
|
// We only subtract the main direction light. This is accounted in the contribution term below.
|
||
|
half shadowStrength = GetMainLightShadowStrength();
|
||
|
half contributionTerm = saturate(dot(mainLight.direction, normalWS));
|
||
|
half3 lambert = mainLight.color * contributionTerm;
|
||
|
half3 estimatedLightContributionMaskedByInverseOfShadow = lambert * (1.0 - mainLight.shadowAttenuation);
|
||
|
half3 subtractedLightmap = bakedGI - estimatedLightContributionMaskedByInverseOfShadow;
|
||
|
|
||
|
// 2) Allows user to define overall ambient of the scene and control situation when realtime shadow becomes too dark.
|
||
|
half3 realtimeShadow = max(subtractedLightmap, _SubtractiveShadowColor.xyz);
|
||
|
realtimeShadow = lerp(bakedGI, realtimeShadow, shadowStrength);
|
||
|
|
||
|
// 3) Pick darkest color
|
||
|
return min(bakedGI, realtimeShadow);
|
||
|
}
|
||
|
|
||
|
half3 GlobalIllumination(BRDFData brdfData, BRDFData brdfDataClearCoat, float clearCoatMask,
|
||
|
half3 bakedGI, half occlusion, float3 positionWS,
|
||
|
half3 normalWS, half3 viewDirectionWS)
|
||
|
{
|
||
|
half3 reflectVector = reflect(-viewDirectionWS, normalWS);
|
||
|
half NoV = saturate(dot(normalWS, viewDirectionWS));
|
||
|
half fresnelTerm = Pow4(1.0 - NoV);
|
||
|
|
||
|
half3 indirectDiffuse = bakedGI;
|
||
|
half3 indirectSpecular = GlossyEnvironmentReflection(reflectVector, positionWS, brdfData.perceptualRoughness, 1.0h);
|
||
|
|
||
|
half3 color = EnvironmentBRDF(brdfData, indirectDiffuse, indirectSpecular, fresnelTerm);
|
||
|
|
||
|
if (IsOnlyAOLightingFeatureEnabled())
|
||
|
{
|
||
|
color = half3(1,1,1); // "Base white" for AO debug lighting mode
|
||
|
}
|
||
|
|
||
|
#if defined(_CLEARCOAT) || defined(_CLEARCOATMAP)
|
||
|
half3 coatIndirectSpecular = GlossyEnvironmentReflection(reflectVector, positionWS, brdfDataClearCoat.perceptualRoughness, 1.0h);
|
||
|
// TODO: "grazing term" causes problems on full roughness
|
||
|
half3 coatColor = EnvironmentBRDFClearCoat(brdfDataClearCoat, clearCoatMask, coatIndirectSpecular, fresnelTerm);
|
||
|
|
||
|
// Blend with base layer using khronos glTF recommended way using NoV
|
||
|
// Smooth surface & "ambiguous" lighting
|
||
|
// NOTE: fresnelTerm (above) is pow4 instead of pow5, but should be ok as blend weight.
|
||
|
half coatFresnel = kDielectricSpec.x + kDielectricSpec.a * fresnelTerm;
|
||
|
return (color * (1.0 - coatFresnel * clearCoatMask) + coatColor) * occlusion;
|
||
|
#else
|
||
|
return color * occlusion;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
// Backwards compatiblity
|
||
|
half3 GlobalIllumination(BRDFData brdfData, half3 bakedGI, half occlusion, float3 positionWS, half3 normalWS, half3 viewDirectionWS)
|
||
|
{
|
||
|
const BRDFData noClearCoat = (BRDFData)0;
|
||
|
return GlobalIllumination(brdfData, noClearCoat, 0.0, bakedGI, occlusion, positionWS, normalWS, viewDirectionWS);
|
||
|
}
|
||
|
|
||
|
half3 GlobalIllumination(BRDFData brdfData, BRDFData brdfDataClearCoat, float clearCoatMask,
|
||
|
half3 bakedGI, half occlusion,
|
||
|
half3 normalWS, half3 viewDirectionWS)
|
||
|
{
|
||
|
half3 reflectVector = reflect(-viewDirectionWS, normalWS);
|
||
|
half NoV = saturate(dot(normalWS, viewDirectionWS));
|
||
|
half fresnelTerm = Pow4(1.0 - NoV);
|
||
|
|
||
|
half3 indirectDiffuse = bakedGI;
|
||
|
half3 indirectSpecular = GlossyEnvironmentReflection(reflectVector, brdfData.perceptualRoughness, half(1.0));
|
||
|
|
||
|
half3 color = EnvironmentBRDF(brdfData, indirectDiffuse, indirectSpecular, fresnelTerm);
|
||
|
|
||
|
#if defined(_CLEARCOAT) || defined(_CLEARCOATMAP)
|
||
|
half3 coatIndirectSpecular = GlossyEnvironmentReflection(reflectVector, brdfDataClearCoat.perceptualRoughness, half(1.0));
|
||
|
// TODO: "grazing term" causes problems on full roughness
|
||
|
half3 coatColor = EnvironmentBRDFClearCoat(brdfDataClearCoat, clearCoatMask, coatIndirectSpecular, fresnelTerm);
|
||
|
|
||
|
// Blend with base layer using khronos glTF recommended way using NoV
|
||
|
// Smooth surface & "ambiguous" lighting
|
||
|
// NOTE: fresnelTerm (above) is pow4 instead of pow5, but should be ok as blend weight.
|
||
|
half coatFresnel = kDielectricSpec.x + kDielectricSpec.a * fresnelTerm;
|
||
|
return (color * (1.0 - coatFresnel * clearCoatMask) + coatColor) * occlusion;
|
||
|
#else
|
||
|
return color * occlusion;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
|
||
|
half3 GlobalIllumination(BRDFData brdfData, half3 bakedGI, half occlusion, half3 normalWS, half3 viewDirectionWS)
|
||
|
{
|
||
|
const BRDFData noClearCoat = (BRDFData)0;
|
||
|
return GlobalIllumination(brdfData, noClearCoat, 0.0, bakedGI, occlusion, normalWS, viewDirectionWS);
|
||
|
}
|
||
|
|
||
|
void MixRealtimeAndBakedGI(inout Light light, half3 normalWS, inout half3 bakedGI)
|
||
|
{
|
||
|
#if defined(LIGHTMAP_ON) && defined(_MIXED_LIGHTING_SUBTRACTIVE)
|
||
|
bakedGI = SubtractDirectMainLightFromLightmap(light, normalWS, bakedGI);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
// Backwards compatibility
|
||
|
void MixRealtimeAndBakedGI(inout Light light, half3 normalWS, inout half3 bakedGI, half4 shadowMask)
|
||
|
{
|
||
|
MixRealtimeAndBakedGI(light, normalWS, bakedGI);
|
||
|
}
|
||
|
|
||
|
void MixRealtimeAndBakedGI(inout Light light, half3 normalWS, inout half3 bakedGI, AmbientOcclusionFactor aoFactor)
|
||
|
{
|
||
|
if (IsLightingFeatureEnabled(DEBUGLIGHTINGFEATUREFLAGS_AMBIENT_OCCLUSION))
|
||
|
{
|
||
|
bakedGI *= aoFactor.indirectAmbientOcclusion;
|
||
|
}
|
||
|
|
||
|
MixRealtimeAndBakedGI(light, normalWS, bakedGI);
|
||
|
}
|
||
|
|
||
|
#endif
|