Bloom.shader 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. Shader "Hidden/Universal Render Pipeline/Bloom"
  2. {
  3. Properties
  4. {
  5. _MainTex("Source", 2D) = "white" {}
  6. }
  7. HLSLINCLUDE
  8. #pragma multi_compile_local _ _USE_RGBM
  9. #include "Packages/com.unity.render-pipelines.core/ShaderLibrary/Common.hlsl"
  10. #include "Packages/com.unity.render-pipelines.core/ShaderLibrary/Filtering.hlsl"
  11. #include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Core.hlsl"
  12. #include "Packages/com.unity.render-pipelines.universal/Shaders/PostProcessing/Common.hlsl"
  13. TEXTURE2D_X(_MainTex);
  14. TEXTURE2D_X(_MainTexLowMip);
  15. float4 _MainTex_TexelSize;
  16. float4 _MainTexLowMip_TexelSize;
  17. float4 _Params; // x: scatter, y: clamp, z: threshold (linear), w: threshold knee
  18. #define Scatter _Params.x
  19. #define ClampMax _Params.y
  20. #define Threshold _Params.z
  21. #define ThresholdKnee _Params.w
  22. half4 EncodeHDR(half3 color)
  23. {
  24. #if _USE_RGBM
  25. half4 outColor = EncodeRGBM(color);
  26. #else
  27. half4 outColor = half4(color, 1.0);
  28. #endif
  29. #if UNITY_COLORSPACE_GAMMA
  30. return half4(sqrt(outColor.xyz), outColor.w); // linear to γ
  31. #else
  32. return outColor;
  33. #endif
  34. }
  35. half3 DecodeHDR(half4 color)
  36. {
  37. #if UNITY_COLORSPACE_GAMMA
  38. color.xyz *= color.xyz; // γ to linear
  39. #endif
  40. #if _USE_RGBM
  41. return DecodeRGBM(color);
  42. #else
  43. return color.xyz;
  44. #endif
  45. }
  46. half4 FragPrefilter(Varyings input) : SV_Target
  47. {
  48. UNITY_SETUP_STEREO_EYE_INDEX_POST_VERTEX(input);
  49. float2 uv = UnityStereoTransformScreenSpaceTex(input.uv);
  50. half3 color = SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv).xyz;
  51. #if UNITY_COLORSPACE_GAMMA
  52. color = SRGBToLinear(color);
  53. #endif
  54. // User controlled clamp to limit crazy high broken spec
  55. color = min(ClampMax, color);
  56. // Thresholding
  57. half brightness = Max3(color.r, color.g, color.b);
  58. half softness = clamp(brightness - Threshold + ThresholdKnee, 0.0, 2.0 * ThresholdKnee);
  59. softness = (softness * softness) / (4.0 * ThresholdKnee + 1e-4);
  60. half multiplier = max(brightness - Threshold, softness) / max(brightness, 1e-4);
  61. color *= multiplier;
  62. return EncodeHDR(color);
  63. }
  64. half4 FragBlurH(Varyings input) : SV_Target
  65. {
  66. UNITY_SETUP_STEREO_EYE_INDEX_POST_VERTEX(input);
  67. float texelSize = _MainTex_TexelSize.x * 2.0;
  68. float2 uv = UnityStereoTransformScreenSpaceTex(input.uv);
  69. // 9-tap gaussian blur on the downsampled source
  70. half3 c0 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv - float2(texelSize * 4.0, 0.0)));
  71. half3 c1 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv - float2(texelSize * 3.0, 0.0)));
  72. half3 c2 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv - float2(texelSize * 2.0, 0.0)));
  73. half3 c3 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv - float2(texelSize * 1.0, 0.0)));
  74. half3 c4 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv ));
  75. half3 c5 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv + float2(texelSize * 1.0, 0.0)));
  76. half3 c6 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv + float2(texelSize * 2.0, 0.0)));
  77. half3 c7 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv + float2(texelSize * 3.0, 0.0)));
  78. half3 c8 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv + float2(texelSize * 4.0, 0.0)));
  79. half3 color = c0 * 0.01621622 + c1 * 0.05405405 + c2 * 0.12162162 + c3 * 0.19459459
  80. + c4 * 0.22702703
  81. + c5 * 0.19459459 + c6 * 0.12162162 + c7 * 0.05405405 + c8 * 0.01621622;
  82. return EncodeHDR(color);
  83. }
  84. half4 FragBlurV(Varyings input) : SV_Target
  85. {
  86. UNITY_SETUP_STEREO_EYE_INDEX_POST_VERTEX(input);
  87. float texelSize = _MainTex_TexelSize.y;
  88. float2 uv = UnityStereoTransformScreenSpaceTex(input.uv);
  89. // Optimized bilinear 5-tap gaussian on the same-sized source (9-tap equivalent)
  90. half3 c0 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv - float2(0.0, texelSize * 3.23076923)));
  91. half3 c1 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv - float2(0.0, texelSize * 1.38461538)));
  92. half3 c2 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv ));
  93. half3 c3 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv + float2(0.0, texelSize * 1.38461538)));
  94. half3 c4 = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv + float2(0.0, texelSize * 3.23076923)));
  95. half3 color = c0 * 0.07027027 + c1 * 0.31621622
  96. + c2 * 0.22702703
  97. + c3 * 0.31621622 + c4 * 0.07027027;
  98. return EncodeHDR(color);
  99. }
  100. half3 Upsample(float2 uv)
  101. {
  102. half3 highMip = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, uv));
  103. #if _BLOOM_HQ && !defined(SHADER_API_GLES)
  104. half3 lowMip = DecodeHDR(SampleTexture2DBicubic(TEXTURE2D_X_ARGS(_MainTexLowMip, sampler_LinearClamp), uv, _MainTexLowMip_TexelSize.zwxy, (1.0).xx, unity_StereoEyeIndex));
  105. #else
  106. half3 lowMip = DecodeHDR(SAMPLE_TEXTURE2D_X(_MainTexLowMip, sampler_LinearClamp, uv));
  107. #endif
  108. return lerp(highMip, lowMip, Scatter);
  109. }
  110. half4 FragUpsample(Varyings input) : SV_Target
  111. {
  112. UNITY_SETUP_STEREO_EYE_INDEX_POST_VERTEX(input);
  113. half3 color = Upsample(UnityStereoTransformScreenSpaceTex(input.uv));
  114. return EncodeHDR(color);
  115. }
  116. ENDHLSL
  117. SubShader
  118. {
  119. Tags { "RenderType" = "Opaque" "RenderPipeline" = "UniversalPipeline"}
  120. LOD 100
  121. ZTest Always ZWrite Off Cull Off
  122. Pass
  123. {
  124. Name "Bloom Prefilter"
  125. HLSLPROGRAM
  126. #pragma vertex Vert
  127. #pragma fragment FragPrefilter
  128. ENDHLSL
  129. }
  130. Pass
  131. {
  132. Name "Bloom Blur Horizontal"
  133. HLSLPROGRAM
  134. #pragma vertex Vert
  135. #pragma fragment FragBlurH
  136. ENDHLSL
  137. }
  138. Pass
  139. {
  140. Name "Bloom Blur Vertical"
  141. HLSLPROGRAM
  142. #pragma vertex Vert
  143. #pragma fragment FragBlurV
  144. ENDHLSL
  145. }
  146. Pass
  147. {
  148. Name "Bloom Upsample"
  149. HLSLPROGRAM
  150. #pragma vertex Vert
  151. #pragma fragment FragUpsample
  152. #pragma multi_compile_local _ _BLOOM_HQ
  153. ENDHLSL
  154. }
  155. }
  156. }