#ifndef UNITY_POSTFX_AMBIENT_OCCLUSION #define UNITY_POSTFX_AMBIENT_OCCLUSION #include "../StdLib.hlsl" #include "../Colors.hlsl" #include "Fog.hlsl" // -------- // Options for further customization // -------- // By default, a 5-tap Gaussian with the linear sampling technique is used // in the bilateral noise filter. It can be replaced with a 7-tap Gaussian // with adaptive sampling by enabling the macro below. Although the // differences are not noticeable in most cases, it may provide preferable // results with some special usage (e.g. NPR without textureing). // #define BLUR_HIGH_QUALITY // By default, a fixed sampling pattern is used in the AO estimator. Although // this gives preferable results in most cases, a completely random sampling // pattern could give aesthetically better results. Disable the macro below // to use such a random pattern instead of the fixed one. #define FIX_SAMPLING_PATTERN // The SampleNormal function normalizes samples from G-buffer because // they're possibly unnormalized. We can eliminate this if it can be said // that there is no wrong shader that outputs unnormalized normals. // #define VALIDATE_NORMALS // The constant below determines the contrast of occlusion. This allows // users to control over/under occlusion. At the moment, this is not exposed // to the editor because it's rarely useful. static const float kContrast = 0.6; // The constant below controls the geometry-awareness of the bilateral // filter. The higher value, the more sensitive it is. static const float kGeometryCoeff = 0.8; // The constants below are used in the AO estimator. Beta is mainly used // for suppressing self-shadowing noise, and Epsilon is used to prevent // calculation underflow. See the paper (Morgan 2011 http://goo.gl/2iz3P) // for further details of these constants. static const float kBeta = 0.002; // -------- // System built-in variables TEXTURE2D_SAMPLER2D(_MainTex, sampler_MainTex); TEXTURE2D_SAMPLER2D(_CameraGBufferTexture2, sampler_CameraGBufferTexture2); TEXTURE2D_SAMPLER2D(_CameraDepthTexture, sampler_CameraDepthTexture); TEXTURE2D_SAMPLER2D(_CameraDepthNormalsTexture, sampler_CameraDepthNormalsTexture); float4 _MainTex_TexelSize; float4 _AOParams; float3 _AOColor; // Sample count #if !defined(SHADER_API_GLES) #define SAMPLE_COUNT _AOParams.w #else // GLES2: In many cases, dynamic looping is not supported. #define SAMPLE_COUNT 3 #endif // Source texture properties TEXTURE2D_SAMPLER2D(_SAOcclusionTexture, sampler_SAOcclusionTexture); float4 _SAOcclusionTexture_TexelSize; // Other parameters #define INTENSITY _AOParams.x #define RADIUS _AOParams.y #define DOWNSAMPLE _AOParams.z // Accessors for packed AO/normal buffer half4 PackAONormal(half ao, half3 n) { return half4(ao, n * 0.5 + 0.5); } half GetPackedAO(half4 p) { return p.r; } half3 GetPackedNormal(half4 p) { return p.gba * 2.0 - 1.0; } // Boundary check for depth sampler // (returns a very large value if it lies out of bounds) float CheckBounds(float2 uv, float d) { float ob = any(uv < 0) + any(uv > 1); #if defined(UNITY_REVERSED_Z) ob += (d <= 0.00001); #else ob += (d >= 0.99999); #endif return ob * 1e8; } // Depth/normal sampling functions float SampleDepth(float2 uv) { float d = Linear01Depth(SAMPLE_DEPTH_TEXTURE_LOD(_CameraDepthTexture, sampler_CameraDepthTexture, UnityStereoTransformScreenSpaceTex(uv), 0)); return d * _ProjectionParams.z + CheckBounds(uv, d); } float3 SampleNormal(float2 uv) { #if defined(SOURCE_GBUFFER) float3 norm = SAMPLE_TEXTURE2D(_CameraGBufferTexture2, sampler_CameraGBufferTexture2, uv).xyz; norm = norm * 2 - any(norm); // gets (0,0,0) when norm == 0 norm = mul((float3x3)unity_WorldToCamera, norm); #if defined(VALIDATE_NORMALS) norm = normalize(norm); #endif return norm; #else float4 cdn = SAMPLE_TEXTURE2D(_CameraDepthNormalsTexture, sampler_CameraDepthNormalsTexture, uv); return DecodeViewNormalStereo(cdn) * float3(1.0, 1.0, -1.0); #endif } float SampleDepthNormal(float2 uv, out float3 normal) { normal = SampleNormal(UnityStereoTransformScreenSpaceTex(uv)); return SampleDepth(uv); } // Normal vector comparer (for geometry-aware weighting) half CompareNormal(half3 d1, half3 d2) { return smoothstep(kGeometryCoeff, 1.0, dot(d1, d2)); } // Trigonometric function utility float2 CosSin(float theta) { float sn, cs; sincos(theta, sn, cs); return float2(cs, sn); } // Pseudo random number generator with 2D coordinates float UVRandom(float u, float v) { float f = dot(float2(12.9898, 78.233), float2(u, v)); return frac(43758.5453 * sin(f)); } // Check if the camera is perspective. // (returns 1.0 when orthographic) float CheckPerspective(float x) { return lerp(x, 1.0, unity_OrthoParams.w); } // Reconstruct view-space position from UV and depth. // p11_22 = (unity_CameraProjection._11, unity_CameraProjection._22) // p13_31 = (unity_CameraProjection._13, unity_CameraProjection._23) float3 ReconstructViewPos(float2 uv, float depth, float2 p11_22, float2 p13_31) { return float3((uv * 2.0 - 1.0 - p13_31) / p11_22 * CheckPerspective(depth), depth); } // Sample point picker float3 PickSamplePoint(float2 uv, float index) { // Uniformaly distributed points on a unit sphere // http://mathworld.wolfram.com/SpherePointPicking.html #if defined(FIX_SAMPLING_PATTERN) float gn = GradientNoise(uv * DOWNSAMPLE); // FIXEME: This was added to avoid a NVIDIA driver issue. // vvvvvvvvvvvv float u = frac(UVRandom(0.0, index + uv.x * 1e-10) + gn) * 2.0 - 1.0; float theta = (UVRandom(1.0, index + uv.x * 1e-10) + gn) * TWO_PI; #else float u = UVRandom(uv.x + _Time.x, uv.y + index) * 2.0 - 1.0; float theta = UVRandom(-uv.x - _Time.x, uv.y + index) * TWO_PI; #endif float3 v = float3(CosSin(theta) * sqrt(1.0 - u * u), u); // Make them distributed between [0, _Radius] float l = sqrt((index + 1.0) / SAMPLE_COUNT) * RADIUS; return v * l; } // // Distance-based AO estimator based on Morgan 2011 // "Alchemy screen-space ambient obscurance algorithm" // http://graphics.cs.williams.edu/papers/AlchemyHPG11/ // float4 FragAO(VaryingsDefault i) : SV_Target { float2 uv = i.texcoord; // Parameters used in coordinate conversion float3x3 proj = (float3x3)unity_CameraProjection; float2 p11_22 = float2(unity_CameraProjection._11, unity_CameraProjection._22); float2 p13_31 = float2(unity_CameraProjection._13, unity_CameraProjection._23); // View space normal and depth float3 norm_o; float depth_o = SampleDepthNormal(uv, norm_o); // Reconstruct the view-space position. float3 vpos_o = ReconstructViewPos(uv, depth_o, p11_22, p13_31); float ao = 0.0; for (int s = 0; s < int(SAMPLE_COUNT); s++) { // Sample point #if defined(SHADER_API_D3D11) // This 'floor(1.0001 * s)' operation is needed to avoid a NVidia shader issue. This issue // is only observed on DX11. float3 v_s1 = PickSamplePoint(uv, floor(1.0001 * s)); #else float3 v_s1 = PickSamplePoint(uv, s); #endif v_s1 = faceforward(v_s1, -norm_o, v_s1); float3 vpos_s1 = vpos_o + v_s1; // Reproject the sample point float3 spos_s1 = mul(proj, vpos_s1); float2 uv_s1_01 = (spos_s1.xy / CheckPerspective(vpos_s1.z) + 1.0) * 0.5; // Depth at the sample point float depth_s1 = SampleDepth(uv_s1_01); // Relative position of the sample point float3 vpos_s2 = ReconstructViewPos(uv_s1_01, depth_s1, p11_22, p13_31); float3 v_s2 = vpos_s2 - vpos_o; // Estimate the obscurance value float a1 = max(dot(v_s2, norm_o) - kBeta * depth_o, 0.0); float a2 = dot(v_s2, v_s2) + EPSILON; ao += a1 / a2; } ao *= RADIUS; // Intensity normalization // Apply other parameters. ao = PositivePow(ao * INTENSITY / SAMPLE_COUNT, kContrast); // Apply fog when enabled (forward-only) #if (APPLY_FORWARD_FOG) float d = Linear01Depth(SAMPLE_DEPTH_TEXTURE(_CameraDepthTexture, sampler_CameraDepthTexture, i.texcoordStereo)); d = ComputeFogDistance(d); ao *= ComputeFog(d); #endif return PackAONormal(ao, norm_o); } // Geometry-aware separable bilateral filter float4 FragBlur(VaryingsDefault i) : SV_Target { #if defined(BLUR_HORIZONTAL) // Horizontal pass: Always use 2 texels interval to match to // the dither pattern. float2 delta = float2(_MainTex_TexelSize.x * 2.0, 0.0); #else // Vertical pass: Apply _Downsample to match to the dither // pattern in the original occlusion buffer. float2 delta = float2(0.0, _MainTex_TexelSize.y / DOWNSAMPLE * 2.0); #endif #if defined(BLUR_HIGH_QUALITY) // High quality 7-tap Gaussian with adaptive sampling half4 p0 = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, i.texcoordStereo); half4 p1a = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord - delta)); half4 p1b = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord + delta)); half4 p2a = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord - delta * 2.0)); half4 p2b = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord + delta * 2.0)); half4 p3a = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord - delta * 3.2307692308)); half4 p3b = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord + delta * 3.2307692308)); #if defined(BLUR_SAMPLE_CENTER_NORMAL) half3 n0 = SampleNormal(i.texcoordStereo); #else half3 n0 = GetPackedNormal(p0); #endif half w0 = 0.37004405286; half w1a = CompareNormal(n0, GetPackedNormal(p1a)) * 0.31718061674; half w1b = CompareNormal(n0, GetPackedNormal(p1b)) * 0.31718061674; half w2a = CompareNormal(n0, GetPackedNormal(p2a)) * 0.19823788546; half w2b = CompareNormal(n0, GetPackedNormal(p2b)) * 0.19823788546; half w3a = CompareNormal(n0, GetPackedNormal(p3a)) * 0.11453744493; half w3b = CompareNormal(n0, GetPackedNormal(p3b)) * 0.11453744493; half s; s = GetPackedAO(p0) * w0; s += GetPackedAO(p1a) * w1a; s += GetPackedAO(p1b) * w1b; s += GetPackedAO(p2a) * w2a; s += GetPackedAO(p2b) * w2b; s += GetPackedAO(p3a) * w3a; s += GetPackedAO(p3b) * w3b; s /= w0 + w1a + w1b + w2a + w2b + w3a + w3b; #else // Fater 5-tap Gaussian with linear sampling half4 p0 = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, i.texcoordStereo); half4 p1a = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord - delta * 1.3846153846)); half4 p1b = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord + delta * 1.3846153846)); half4 p2a = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord - delta * 3.2307692308)); half4 p2b = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, UnityStereoTransformScreenSpaceTex(i.texcoord + delta * 3.2307692308)); #if defined(BLUR_SAMPLE_CENTER_NORMAL) half3 n0 = SampleNormal(i.texcoordStereo); #else half3 n0 = GetPackedNormal(p0); #endif half w0 = 0.2270270270; half w1a = CompareNormal(n0, GetPackedNormal(p1a)) * 0.3162162162; half w1b = CompareNormal(n0, GetPackedNormal(p1b)) * 0.3162162162; half w2a = CompareNormal(n0, GetPackedNormal(p2a)) * 0.0702702703; half w2b = CompareNormal(n0, GetPackedNormal(p2b)) * 0.0702702703; half s; s = GetPackedAO(p0) * w0; s += GetPackedAO(p1a) * w1a; s += GetPackedAO(p1b) * w1b; s += GetPackedAO(p2a) * w2a; s += GetPackedAO(p2b) * w2b; s /= w0 + w1a + w1b + w2a + w2b; #endif return PackAONormal(s, n0); } // Gamma encoding (only needed in gamma lighting mode) half EncodeAO(half x) { #if UNITY_COLORSPACE_GAMMA return 1.0 - max(LinearToSRGB(1.0 - saturate(x)), 0.0); #else return x; #endif } // Geometry-aware bilateral filter (single pass/small kernel) half BlurSmall(TEXTURE2D_ARGS(tex, samp), float2 uv, float2 delta) { half4 p0 = SAMPLE_TEXTURE2D(tex, samp, UnityStereoTransformScreenSpaceTex(uv)); half4 p1 = SAMPLE_TEXTURE2D(tex, samp, UnityStereoTransformScreenSpaceTex(uv + float2(-delta.x, -delta.y))); half4 p2 = SAMPLE_TEXTURE2D(tex, samp, UnityStereoTransformScreenSpaceTex(uv + float2( delta.x, -delta.y))); half4 p3 = SAMPLE_TEXTURE2D(tex, samp, UnityStereoTransformScreenSpaceTex(uv + float2(-delta.x, delta.y))); half4 p4 = SAMPLE_TEXTURE2D(tex, samp, UnityStereoTransformScreenSpaceTex(uv + float2( delta.x, delta.y))); half3 n0 = GetPackedNormal(p0); half w0 = 1.0; half w1 = CompareNormal(n0, GetPackedNormal(p1)); half w2 = CompareNormal(n0, GetPackedNormal(p2)); half w3 = CompareNormal(n0, GetPackedNormal(p3)); half w4 = CompareNormal(n0, GetPackedNormal(p4)); half s; s = GetPackedAO(p0) * w0; s += GetPackedAO(p1) * w1; s += GetPackedAO(p2) * w2; s += GetPackedAO(p3) * w3; s += GetPackedAO(p4) * w4; return s / (w0 + w1 + w2 + w3 + w4); } // Final composition shader float4 FragComposition(VaryingsDefault i) : SV_Target { float2 delta = _SAOcclusionTexture_TexelSize.xy / DOWNSAMPLE; half ao = BlurSmall(TEXTURE2D_PARAM(_SAOcclusionTexture, sampler_SAOcclusionTexture), i.texcoord, delta); ao = EncodeAO(ao); return float4(ao * _AOColor, ao); } #if !SHADER_API_GLES // Excluding the MRT pass under GLES2 struct CompositionOutput { half4 gbuffer0 : SV_Target0; half4 gbuffer3 : SV_Target1; }; CompositionOutput FragCompositionGBuffer(VaryingsDefault i) { // Workaround: _SAOcclusionTexture_Texelsize hasn't been set properly // for some reasons. Use _ScreenParams instead. float2 delta = (_ScreenParams.zw - 1.0) / DOWNSAMPLE; half ao = BlurSmall(TEXTURE2D_PARAM(_SAOcclusionTexture, sampler_SAOcclusionTexture), i.texcoord, delta); CompositionOutput o; o.gbuffer0 = half4(0.0, 0.0, 0.0, ao); o.gbuffer3 = half4((half3)EncodeAO(ao) * _AOColor, 0.0); return o; } #else float4 FragCompositionGBuffer(VaryingsDefault i) : SV_Target { return (0.0).xxxx; } #endif float4 FragDebugOverlay(VaryingsDefault i) : SV_Target { float2 delta = _SAOcclusionTexture_TexelSize.xy / DOWNSAMPLE; half ao = BlurSmall(TEXTURE2D_PARAM(_SAOcclusionTexture, sampler_SAOcclusionTexture), i.texcoord, delta); ao = EncodeAO(ao); return float4(1.0 - ao.xxx, 1.0); } #endif // UNITY_POSTFX_AMBIENT_OCCLUSION