using System; using UnityEngine.Experimental.Rendering; namespace UnityEngine.Rendering.PostProcessing { /// /// Ambient occlusion modes. /// public enum AmbientOcclusionMode { /// /// A standard implementation of ambient obscurance that works on non modern platforms. If /// you target a compute-enabled platform we recommend that you use /// instead. /// ScalableAmbientObscurance, /// /// A modern version of ambient occlusion heavily optimized for consoles and desktop /// platforms. /// MultiScaleVolumetricObscurance } /// /// Quality settings for . /// public enum AmbientOcclusionQuality { /// /// 4 samples + downsampling. /// Lowest, /// /// 6 samples + downsampling. /// Low, /// /// 10 samples + downsampling. /// Medium, /// /// 8 samples. /// High, /// /// 12 samples. /// Ultra } /// /// A volume parameter holding a value. /// [Serializable] public sealed class AmbientOcclusionModeParameter : ParameterOverride { } /// /// A volume parameter holding a value. /// [Serializable] public sealed class AmbientOcclusionQualityParameter : ParameterOverride { } /// /// This class holds settings for the Ambient Occlusion effect. /// [Serializable] [PostProcess(typeof(AmbientOcclusionRenderer), "Unity/Ambient Occlusion")] public sealed class AmbientOcclusion : PostProcessEffectSettings { // Shared parameters /// /// The ambient occlusion method to use. /// [Tooltip("The ambient occlusion method to use. \"Multi Scale Volumetric Obscurance\" is higher quality and faster on desktop & console platforms but requires compute shader support.")] public AmbientOcclusionModeParameter mode = new AmbientOcclusionModeParameter { value = AmbientOcclusionMode.MultiScaleVolumetricObscurance }; /// /// The degree of darkness added by ambient occlusion. /// [Range(0f, 4f), Tooltip("The degree of darkness added by ambient occlusion. Higher values produce darker areas.")] public FloatParameter intensity = new FloatParameter { value = 0f }; /// /// A custom color to use for the ambient occlusion. /// [ColorUsage(false), Tooltip("The custom color to use for the ambient occlusion. The default is black.")] public ColorParameter color = new ColorParameter { value = Color.black }; /// /// Only affects ambient lighting. This mode is only available with the Deferred rendering /// path and HDR rendering. Objects rendered with the Forward rendering path won't get any /// ambient occlusion. /// [Tooltip("Check this box to mark this Volume as to only affect ambient lighting. This mode is only available with the Deferred rendering path and HDR rendering. Objects rendered with the Forward rendering path won't get any ambient occlusion.")] public BoolParameter ambientOnly = new BoolParameter { value = true }; // MSVO-only parameters /// /// The tolerance of the noise filter to changes in the depth pyramid. /// [Range(-8f, 0f)] public FloatParameter noiseFilterTolerance = new FloatParameter { value = 0f }; // Hidden /// /// The tolerance of the bilateral blur filter to depth changes. /// [Range(-8f, -1f)] public FloatParameter blurTolerance = new FloatParameter { value = -4.6f }; // Hidden /// /// The tolerance of the upsampling pass to depth changes. /// [Range(-12f, -1f)] public FloatParameter upsampleTolerance = new FloatParameter { value = -12f }; // Hidden /// /// Modifies the thickness of occluders. This increases dark areas but also introduces dark /// halo around objects. /// [Range(1f, 10f), Tooltip("This modifies the thickness of occluders. It increases the size of dark areas and also introduces a dark halo around objects.")] public FloatParameter thicknessModifier = new FloatParameter { value = 1f }; /// /// Add a bias distance to sampled depth in AO to reduce self-shadowing aliasing artifacts. /// [Range(0f, 0.001f), Tooltip("Add a bias distance to sampled depth in AO to reduce self-shadowing aliasing artifacts. ")] public FloatParameter zBias = new FloatParameter { value = 0.0001f }; // HDRP-only parameters /// /// Modifies he influence of direct lighting on ambient occlusion. This is only used in the /// HD Render Pipeline currently. /// [Range(0f, 1f), Tooltip("Modifies the influence of direct lighting on ambient occlusion.")] public FloatParameter directLightingStrength = new FloatParameter { value = 0f }; // SAO-only parameters /// /// Radius of sample points, which affects extent of darkened areas. /// [Tooltip("The radius of sample points. This affects the size of darkened areas.")] public FloatParameter radius = new FloatParameter { value = 0.25f }; /// /// The number of sample points, which affects quality and performance. Lowest, Low and Medium /// passes are downsampled. High and Ultra are not and should only be used on high-end /// hardware. /// [Tooltip("The number of sample points. This affects both quality and performance. For \"Lowest\", \"Low\", and \"Medium\", passes are downsampled. For \"High\" and \"Ultra\", they are not and therefore you should only \"High\" and \"Ultra\" on high-end hardware.")] public AmbientOcclusionQualityParameter quality = new AmbientOcclusionQualityParameter { value = AmbientOcclusionQuality.Medium }; // SRPs can call this method without a context set (see HDRP). // We need a better way to handle this than checking for a null context, context should // never be null. /// /// Returns true if the effect is currently enabled and supported. /// /// The current post-processing render context /// true if the effect is currently enabled and supported public override bool IsEnabledAndSupported(PostProcessRenderContext context) { bool state = enabled.value && intensity.value > 0f; if (mode.value == AmbientOcclusionMode.ScalableAmbientObscurance) { state &= !RuntimeUtilities.scriptableRenderPipelineActive; if (context != null) { state &= context.resources.shaders.scalableAO && context.resources.shaders.scalableAO.isSupported; } } else if (mode.value == AmbientOcclusionMode.MultiScaleVolumetricObscurance) { if (context != null) { state &= context.resources.shaders.multiScaleAO && context.resources.shaders.multiScaleAO.isSupported && context.resources.computeShaders.multiScaleAODownsample1 && context.resources.computeShaders.multiScaleAODownsample2 && context.resources.computeShaders.multiScaleAORender && context.resources.computeShaders.multiScaleAOUpsample; } state &= SystemInfo.supportsComputeShaders && !RuntimeUtilities.isAndroidOpenGL && !RuntimeUtilities.isWebNonWebGPU #if UNITY_2023_2_OR_NEWER && SystemInfo.IsFormatSupported(GraphicsFormat.R32_SFloat, GraphicsFormatUsage.Render) && SystemInfo.IsFormatSupported(GraphicsFormat.R16_SFloat, GraphicsFormatUsage.Render) && SystemInfo.IsFormatSupported(GraphicsFormat.R8_UNorm, GraphicsFormatUsage.Render); #else && SystemInfo.IsFormatSupported(GraphicsFormat.R32_SFloat, FormatUsage.Render) && SystemInfo.IsFormatSupported(GraphicsFormat.R16_SFloat, FormatUsage.Render) && SystemInfo.IsFormatSupported(GraphicsFormat.R8_UNorm, FormatUsage.Render); #endif } return state; } } internal interface IAmbientOcclusionMethod { DepthTextureMode GetCameraFlags(); void RenderAfterOpaque(PostProcessRenderContext context); void RenderAmbientOnly(PostProcessRenderContext context); void CompositeAmbientOnly(PostProcessRenderContext context); void Release(); } [UnityEngine.Scripting.Preserve] internal sealed class AmbientOcclusionRenderer : PostProcessEffectRenderer { IAmbientOcclusionMethod[] m_Methods; public override void Init() { if (m_Methods == null) { m_Methods = new IAmbientOcclusionMethod[] { new ScalableAO(settings), new MultiScaleVO(settings), }; } } public bool IsAmbientOnly(PostProcessRenderContext context) { var camera = context.camera; return settings.ambientOnly.value && camera.actualRenderingPath == RenderingPath.DeferredShading && camera.allowHDR; } public IAmbientOcclusionMethod Get() { return m_Methods[(int)settings.mode.value]; } public override DepthTextureMode GetCameraFlags() { return Get().GetCameraFlags(); } public override void Release() { foreach (var m in m_Methods) m.Release(); } public ScalableAO GetScalableAO() { return (ScalableAO)m_Methods[(int)AmbientOcclusionMode.ScalableAmbientObscurance]; } public MultiScaleVO GetMultiScaleVO() { return (MultiScaleVO)m_Methods[(int)AmbientOcclusionMode.MultiScaleVolumetricObscurance]; } // Unused public override void Render(PostProcessRenderContext context) { } } }