您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
205 行
9.4 KiB
205 行
9.4 KiB
Shader "Hidden/HDRenderPipeline/SubsurfaceScattering"
|
|
{
|
|
Properties
|
|
{
|
|
[HideInInspector] _DstBlend("", Float) = 1 // Can be set to 1 for blending with specular
|
|
}
|
|
|
|
SubShader
|
|
{
|
|
Pass
|
|
{
|
|
Stencil
|
|
{
|
|
Ref 1 // StencilLightingUsage.SplitLighting
|
|
Comp Equal
|
|
Pass Keep
|
|
}
|
|
|
|
Cull Off
|
|
ZTest Always
|
|
ZWrite Off
|
|
Blend One [_DstBlend]
|
|
|
|
HLSLPROGRAM
|
|
#pragma target 4.5
|
|
#pragma only_renderers d3d11 ps4 vulkan metal // TEMP: until we go further in dev
|
|
// #pragma enable_d3d11_debug_symbols
|
|
|
|
#pragma vertex Vert
|
|
#pragma fragment Frag
|
|
|
|
#pragma multi_compile _ SSS_FILTER_HORIZONTAL_AND_COMBINE
|
|
|
|
// Do not modify these.
|
|
#include "../../../ShaderPass/ShaderPass.cs.hlsl"
|
|
#define SHADERPASS SHADERPASS_SUBSURFACE_SCATTERING
|
|
#define MILLIMETERS_PER_METER 1000
|
|
#define CENTIMETERS_PER_METER 100
|
|
|
|
//-------------------------------------------------------------------------------------
|
|
// Include
|
|
//-------------------------------------------------------------------------------------
|
|
|
|
#include "ShaderLibrary/Common.hlsl"
|
|
#include "../../../ShaderVariables.hlsl"
|
|
#define UNITY_MATERIAL_LIT // Needs to be defined before including Material.hlsl
|
|
#include "../../../Material/Material.hlsl"
|
|
|
|
//-------------------------------------------------------------------------------------
|
|
// Inputs & outputs
|
|
//-------------------------------------------------------------------------------------
|
|
|
|
float4 _WorldScales[SSS_N_PROFILES]; // Size of the world unit in meters (only the X component is used)
|
|
float4 _FilterKernelsBasic[SSS_N_PROFILES][SSS_BASIC_N_SAMPLES]; // RGB = weights, A = radial distance
|
|
float4 _HalfRcpWeightedVariances[SSS_BASIC_N_SAMPLES]; // RGB for chromatic, A for achromatic
|
|
|
|
TEXTURE2D(_IrradianceSource); // Includes transmitted light
|
|
|
|
//-------------------------------------------------------------------------------------
|
|
// Implementation
|
|
//-------------------------------------------------------------------------------------
|
|
|
|
struct Attributes
|
|
{
|
|
uint vertexID : SV_VertexID;
|
|
};
|
|
|
|
struct Varyings
|
|
{
|
|
float4 positionCS : SV_Position;
|
|
};
|
|
|
|
Varyings Vert(Attributes input)
|
|
{
|
|
Varyings output;
|
|
output.positionCS = GetFullScreenTriangleVertexPosition(input.vertexID);
|
|
return output;
|
|
}
|
|
|
|
float4 Frag(Varyings input) : SV_Target
|
|
{
|
|
PositionInputs posInput = GetPositionInput(input.positionCS.xy, _ScreenSize.zw);
|
|
|
|
// Note: When we are in this SubsurfaceScattering shader we know that we are a SSS material. This shader is strongly coupled with the deferred Lit.shader.
|
|
// We can use the material classification facility to help the compiler to know we use SSS material and optimize the code (and don't require to read gbuffer with materialId).
|
|
uint featureFlags = MATERIALFEATUREFLAGS_LIT_SSS;
|
|
|
|
BSDFData bsdfData;
|
|
float3 unused;
|
|
DECODE_FROM_GBUFFER(posInput.positionSS, featureFlags, bsdfData, unused);
|
|
|
|
int profileID = bsdfData.subsurfaceProfile;
|
|
float distScale = bsdfData.subsurfaceRadius;
|
|
float maxDistance = _FilterKernelsBasic[profileID][SSS_BASIC_N_SAMPLES - 1].a;
|
|
|
|
// Take the first (central) sample.
|
|
// TODO: copy its neighborhood into LDS.
|
|
float2 centerPosition = posInput.positionSS;
|
|
float3 centerIrradiance = LOAD_TEXTURE2D(_IrradianceSource, centerPosition).rgb;
|
|
|
|
// Reconstruct the view-space position.
|
|
float2 centerPosSS = posInput.positionNDC;
|
|
float2 cornerPosSS = centerPosSS + 0.5 * _ScreenSize.zw;
|
|
float centerDepth = LOAD_TEXTURE2D(_MainDepthTexture, centerPosition).r;
|
|
float3 centerPosVS = ComputeViewSpacePosition(centerPosSS, centerDepth, UNITY_MATRIX_I_P);
|
|
float3 cornerPosVS = ComputeViewSpacePosition(cornerPosSS, centerDepth, UNITY_MATRIX_I_P);
|
|
|
|
// Rescaling the filter is equivalent to inversely scaling the world.
|
|
float metersPerUnit = _WorldScales[profileID].x / distScale * SSS_BASIC_DISTANCE_SCALE;
|
|
float centimPerUnit = CENTIMETERS_PER_METER * metersPerUnit;
|
|
// Compute the view-space dimensions of the pixel as a quad projected onto geometry.
|
|
float2 unitsPerPixel = 2 * abs(cornerPosVS.xy - centerPosVS.xy);
|
|
float2 pixelsPerCm = rcp(centimPerUnit * unitsPerPixel);
|
|
|
|
// Compute the filtering direction.
|
|
#ifdef SSS_FILTER_HORIZONTAL_AND_COMBINE
|
|
float2 unitDirection = float2(1, 0);
|
|
#else
|
|
float2 unitDirection = float2(0, 1);
|
|
#endif
|
|
|
|
float2 scaledDirection = pixelsPerCm * unitDirection;
|
|
float phi = 0; // Random rotation; unused for now
|
|
float2x2 rotationMatrix = float2x2(cos(phi), -sin(phi), sin(phi), cos(phi));
|
|
float2 rotatedDirection = mul(rotationMatrix, scaledDirection);
|
|
|
|
// Load (1 / (2 * WeightedVariance)) for bilateral weighting.
|
|
#if RBG_BILATERAL_WEIGHTS
|
|
float3 halfRcpVariance = _HalfRcpWeightedVariances[profileID].rgb;
|
|
#else
|
|
float halfRcpVariance = _HalfRcpWeightedVariances[profileID].a;
|
|
#endif
|
|
|
|
float3 albedo = ApplyDiffuseTexturingMode(bsdfData);
|
|
|
|
#ifndef SSS_FILTER_HORIZONTAL_AND_COMBINE
|
|
albedo = float3(1, 1, 1);
|
|
#endif
|
|
|
|
// Take the first (central) sample.
|
|
float2 samplePosition = posInput.positionSS;
|
|
float3 sampleWeight = _FilterKernelsBasic[profileID][0].rgb;
|
|
float3 sampleIrradiance = LOAD_TEXTURE2D(_IrradianceSource, samplePosition).rgb;
|
|
|
|
// We perform point sampling. Therefore, we can avoid the cost
|
|
// of filtering if we stay within the bounds of the current pixel.
|
|
// We use the value of 1 instead of 0.5 as an optimization.
|
|
float maxDistInPixels = maxDistance * max(pixelsPerCm.x, pixelsPerCm.y);
|
|
|
|
[branch]
|
|
if (distScale == 0 || maxDistInPixels < 1)
|
|
{
|
|
#if SSS_DEBUG_LOD
|
|
return float4(0, 0, 1, 1);
|
|
#else
|
|
return float4(albedo * sampleIrradiance, 1);
|
|
#endif
|
|
}
|
|
|
|
#if SSS_DEBUG_LOD
|
|
return float4(0.5, 0.5, 0, 1);
|
|
#endif
|
|
|
|
// Accumulate filtered irradiance and bilateral weights (for renormalization).
|
|
float3 totalIrradiance = sampleWeight * sampleIrradiance;
|
|
float3 totalWeight = sampleWeight;
|
|
|
|
[unroll]
|
|
for (int i = 1; i < SSS_BASIC_N_SAMPLES; i++)
|
|
{
|
|
samplePosition = posInput.positionSS + rotatedDirection * _FilterKernelsBasic[profileID][i].a;
|
|
sampleWeight = _FilterKernelsBasic[profileID][i].rgb;
|
|
sampleIrradiance = LOAD_TEXTURE2D(_IrradianceSource, samplePosition).rgb;
|
|
|
|
if (TestLightingForSSS(sampleIrradiance))
|
|
{
|
|
// Apply bilateral weighting.
|
|
// Ref #1: Skin Rendering by Pseudo–Separable Cross Bilateral Filtering.
|
|
// Ref #2: Separable SSS, Supplementary Materials, Section E.
|
|
float rawDepth = LOAD_TEXTURE2D(_MainDepthTexture, samplePosition).r;
|
|
float sampleDepth = LinearEyeDepth(rawDepth, _ZBufferParams);
|
|
float zDistance = centimPerUnit * sampleDepth - (centimPerUnit * centerPosVS.z);
|
|
sampleWeight *= exp(-zDistance * zDistance * halfRcpVariance);
|
|
|
|
totalIrradiance += sampleWeight * sampleIrradiance;
|
|
totalWeight += sampleWeight;
|
|
}
|
|
else
|
|
{
|
|
// The irradiance is 0. This could happen for 2 reasons.
|
|
// Most likely, the surface fragment does not have an SSS material.
|
|
// Alternatively, our sample comes from a region without any geometry.
|
|
// Our blur is energy-preserving, so 'centerWeight' should be set to 0.
|
|
// We do not terminate the loop since we want to gather the contribution
|
|
// of the remaining samples (e.g. in case of hair covering skin).
|
|
}
|
|
}
|
|
|
|
return float4(albedo * totalIrradiance / totalWeight, 1);
|
|
}
|
|
ENDHLSL
|
|
}
|
|
}
|
|
Fallback Off
|
|
}
|