using System; using UnityEngine.Experimental.Rendering; using UnityEngine.Experimental.Rendering.RenderGraphModule; namespace UnityEngine.Rendering.Universal.Internal { /// /// Renders a shadow map for the main Light. /// public class MainLightShadowCasterPass : ScriptableRenderPass { private static class MainLightShadowConstantBuffer { public static int _WorldToShadow; public static int _ShadowParams; public static int _CascadeShadowSplitSpheres0; public static int _CascadeShadowSplitSpheres1; public static int _CascadeShadowSplitSpheres2; public static int _CascadeShadowSplitSpheres3; public static int _CascadeShadowSplitSphereRadii; public static int _ShadowOffset0; public static int _ShadowOffset1; public static int _ShadowmapSize; } const int k_MaxCascades = 4; const int k_ShadowmapBufferBits = 16; float m_CascadeBorder; float m_MaxShadowDistanceSq; int m_ShadowCasterCascadesCount; int m_MainLightShadowmapID; internal RTHandle m_MainLightShadowmapTexture; internal RTHandle m_EmptyLightShadowmapTexture; Matrix4x4[] m_MainLightShadowMatrices; ShadowSliceData[] m_CascadeSlices; Vector4[] m_CascadeSplitDistances; bool m_CreateEmptyShadowmap; int renderTargetWidth; int renderTargetHeight; ProfilingSampler m_ProfilingSetupSampler = new ProfilingSampler("Setup Main Shadowmap"); /// /// Creates a new MainLightShadowCasterPass instance. /// /// The RenderPassEvent to use. /// public MainLightShadowCasterPass(RenderPassEvent evt) { base.profilingSampler = new ProfilingSampler(nameof(MainLightShadowCasterPass)); renderPassEvent = evt; m_MainLightShadowMatrices = new Matrix4x4[k_MaxCascades + 1]; m_CascadeSlices = new ShadowSliceData[k_MaxCascades]; m_CascadeSplitDistances = new Vector4[k_MaxCascades]; MainLightShadowConstantBuffer._WorldToShadow = Shader.PropertyToID("_MainLightWorldToShadow"); MainLightShadowConstantBuffer._ShadowParams = Shader.PropertyToID("_MainLightShadowParams"); MainLightShadowConstantBuffer._CascadeShadowSplitSpheres0 = Shader.PropertyToID("_CascadeShadowSplitSpheres0"); MainLightShadowConstantBuffer._CascadeShadowSplitSpheres1 = Shader.PropertyToID("_CascadeShadowSplitSpheres1"); MainLightShadowConstantBuffer._CascadeShadowSplitSpheres2 = Shader.PropertyToID("_CascadeShadowSplitSpheres2"); MainLightShadowConstantBuffer._CascadeShadowSplitSpheres3 = Shader.PropertyToID("_CascadeShadowSplitSpheres3"); MainLightShadowConstantBuffer._CascadeShadowSplitSphereRadii = Shader.PropertyToID("_CascadeShadowSplitSphereRadii"); MainLightShadowConstantBuffer._ShadowOffset0 = Shader.PropertyToID("_MainLightShadowOffset0"); MainLightShadowConstantBuffer._ShadowOffset1 = Shader.PropertyToID("_MainLightShadowOffset1"); MainLightShadowConstantBuffer._ShadowmapSize = Shader.PropertyToID("_MainLightShadowmapSize"); m_MainLightShadowmapID = Shader.PropertyToID("_MainLightShadowmapTexture"); m_EmptyLightShadowmapTexture = ShadowUtils.AllocShadowRT(1, 1, k_ShadowmapBufferBits, 1, 0, name: "_EmptyLightShadowmapTexture"); } /// /// Cleans up resources used by the pass. /// public void Dispose() { m_MainLightShadowmapTexture?.Release(); m_EmptyLightShadowmapTexture?.Release(); } /// /// Sets up the pass. /// /// /// True if the pass should be enqueued, otherwise false. /// public bool Setup(ref RenderingData renderingData) { if (!renderingData.shadowData.mainLightShadowsEnabled) return false; using var profScope = new ProfilingScope(null, m_ProfilingSetupSampler); if (!renderingData.shadowData.supportsMainLightShadows) return SetupForEmptyRendering(ref renderingData); Clear(); int shadowLightIndex = renderingData.lightData.mainLightIndex; if (shadowLightIndex == -1) return SetupForEmptyRendering(ref renderingData); VisibleLight shadowLight = renderingData.lightData.visibleLights[shadowLightIndex]; Light light = shadowLight.light; if (light.shadows == LightShadows.None) return SetupForEmptyRendering(ref renderingData); if (shadowLight.lightType != LightType.Directional) { Debug.LogWarning("Only directional lights are supported as main light."); } Bounds bounds; if (!renderingData.cullResults.GetShadowCasterBounds(shadowLightIndex, out bounds)) return SetupForEmptyRendering(ref renderingData); m_ShadowCasterCascadesCount = renderingData.shadowData.mainLightShadowCascadesCount; int shadowResolution = ShadowUtils.GetMaxTileResolutionInAtlas(renderingData.shadowData.mainLightShadowmapWidth, renderingData.shadowData.mainLightShadowmapHeight, m_ShadowCasterCascadesCount); renderTargetWidth = renderingData.shadowData.mainLightShadowmapWidth; renderTargetHeight = (m_ShadowCasterCascadesCount == 2) ? renderingData.shadowData.mainLightShadowmapHeight >> 1 : renderingData.shadowData.mainLightShadowmapHeight; for (int cascadeIndex = 0; cascadeIndex < m_ShadowCasterCascadesCount; ++cascadeIndex) { bool success = ShadowUtils.ExtractDirectionalLightMatrix(ref renderingData.cullResults, ref renderingData.shadowData, shadowLightIndex, cascadeIndex, renderTargetWidth, renderTargetHeight, shadowResolution, light.shadowNearPlane, out m_CascadeSplitDistances[cascadeIndex], out m_CascadeSlices[cascadeIndex]); if (!success) return SetupForEmptyRendering(ref renderingData); } ShadowUtils.ShadowRTReAllocateIfNeeded(ref m_MainLightShadowmapTexture, renderTargetWidth, renderTargetHeight, k_ShadowmapBufferBits, name: "_MainLightShadowmapTexture"); m_MaxShadowDistanceSq = renderingData.cameraData.maxShadowDistance * renderingData.cameraData.maxShadowDistance; m_CascadeBorder = renderingData.shadowData.mainLightShadowCascadeBorder; m_CreateEmptyShadowmap = false; useNativeRenderPass = true; ShadowUtils.ShadowRTReAllocateIfNeeded(ref m_EmptyLightShadowmapTexture, 1, 1, k_ShadowmapBufferBits, name: "_EmptyLightShadowmapTexture"); return true; } bool SetupForEmptyRendering(ref RenderingData renderingData) { if (!renderingData.cameraData.renderer.stripShadowsOffVariants) return false; m_CreateEmptyShadowmap = true; useNativeRenderPass = false; ShadowUtils.ShadowRTReAllocateIfNeeded(ref m_EmptyLightShadowmapTexture, 1, 1, k_ShadowmapBufferBits, name: "_EmptyLightShadowmapTexture"); return true; } /// public override void Configure(CommandBuffer cmd, RenderTextureDescriptor cameraTextureDescriptor) { if (m_CreateEmptyShadowmap) ConfigureTarget(m_EmptyLightShadowmapTexture); else ConfigureTarget(m_MainLightShadowmapTexture); ConfigureClear(ClearFlag.All, Color.black); } /// public override void Execute(ScriptableRenderContext context, ref RenderingData renderingData) { if (m_CreateEmptyShadowmap) { SetEmptyMainLightCascadeShadowmap(ref context, ref renderingData); renderingData.commandBuffer.SetGlobalTexture(m_MainLightShadowmapID, m_EmptyLightShadowmapTexture.nameID); return; } RenderMainLightCascadeShadowmap(ref context, ref renderingData); renderingData.commandBuffer.SetGlobalTexture(m_MainLightShadowmapID, m_MainLightShadowmapTexture.nameID); } void Clear() { for (int i = 0; i < m_MainLightShadowMatrices.Length; ++i) m_MainLightShadowMatrices[i] = Matrix4x4.identity; for (int i = 0; i < m_CascadeSplitDistances.Length; ++i) m_CascadeSplitDistances[i] = new Vector4(0.0f, 0.0f, 0.0f, 0.0f); for (int i = 0; i < m_CascadeSlices.Length; ++i) m_CascadeSlices[i].Clear(); } void SetEmptyMainLightCascadeShadowmap(ref ScriptableRenderContext context, ref RenderingData renderingData) { var cmd = renderingData.commandBuffer; CoreUtils.SetKeyword(cmd, ShaderKeywordStrings.MainLightShadows, true); cmd.SetGlobalVector(MainLightShadowConstantBuffer._ShadowParams, new Vector4(1, 0, 1, 0)); cmd.SetGlobalVector(MainLightShadowConstantBuffer._ShadowmapSize, new Vector4(1f / m_EmptyLightShadowmapTexture.rt.width, 1f / m_EmptyLightShadowmapTexture.rt.height, m_EmptyLightShadowmapTexture.rt.width, m_EmptyLightShadowmapTexture.rt.height)); context.ExecuteCommandBuffer(cmd); cmd.Clear(); } void RenderMainLightCascadeShadowmap(ref ScriptableRenderContext context, ref RenderingData renderingData) { var cullResults = renderingData.cullResults; var lightData = renderingData.lightData; var shadowData = renderingData.shadowData; int shadowLightIndex = lightData.mainLightIndex; if (shadowLightIndex == -1) return; VisibleLight shadowLight = lightData.visibleLights[shadowLightIndex]; var cmd = renderingData.commandBuffer; using (new ProfilingScope(cmd, ProfilingSampler.Get(URPProfileId.MainLightShadow))) { var settings = new ShadowDrawingSettings(cullResults, shadowLightIndex, BatchCullingProjectionType.Orthographic); settings.useRenderingLayerMaskTest = UniversalRenderPipeline.asset.useRenderingLayers; // Need to start by setting the Camera position as that is not set for passes executed before normal rendering cmd.SetGlobalVector(ShaderPropertyId.worldSpaceCameraPos, renderingData.cameraData.worldSpaceCameraPos); for (int cascadeIndex = 0; cascadeIndex < m_ShadowCasterCascadesCount; ++cascadeIndex) { settings.splitData = m_CascadeSlices[cascadeIndex].splitData; Vector4 shadowBias = ShadowUtils.GetShadowBias(ref shadowLight, shadowLightIndex, ref renderingData.shadowData, m_CascadeSlices[cascadeIndex].projectionMatrix, m_CascadeSlices[cascadeIndex].resolution); ShadowUtils.SetupShadowCasterConstantBuffer(cmd, ref shadowLight, shadowBias); CoreUtils.SetKeyword(cmd, ShaderKeywordStrings.CastingPunctualLightShadow, false); ShadowUtils.RenderShadowSlice(cmd, ref context, ref m_CascadeSlices[cascadeIndex], ref settings, m_CascadeSlices[cascadeIndex].projectionMatrix, m_CascadeSlices[cascadeIndex].viewMatrix); } renderingData.shadowData.isKeywordSoftShadowsEnabled = shadowLight.light.shadows == LightShadows.Soft && renderingData.shadowData.supportsSoftShadows; CoreUtils.SetKeyword(cmd, ShaderKeywordStrings.MainLightShadows, renderingData.shadowData.mainLightShadowCascadesCount == 1); CoreUtils.SetKeyword(cmd, ShaderKeywordStrings.MainLightShadowCascades, renderingData.shadowData.mainLightShadowCascadesCount > 1); CoreUtils.SetKeyword(cmd, ShaderKeywordStrings.SoftShadows, renderingData.shadowData.isKeywordSoftShadowsEnabled); SetupMainLightShadowReceiverConstants(cmd, ref shadowLight, ref renderingData.shadowData); } } void SetupMainLightShadowReceiverConstants(CommandBuffer cmd, ref VisibleLight shadowLight, ref ShadowData shadowData) { Light light = shadowLight.light; bool softShadows = shadowLight.light.shadows == LightShadows.Soft && shadowData.supportsSoftShadows; int cascadeCount = m_ShadowCasterCascadesCount; for (int i = 0; i < cascadeCount; ++i) m_MainLightShadowMatrices[i] = m_CascadeSlices[i].shadowTransform; // We setup and additional a no-op WorldToShadow matrix in the last index // because the ComputeCascadeIndex function in Shadows.hlsl can return an index // out of bounds. (position not inside any cascade) and we want to avoid branching Matrix4x4 noOpShadowMatrix = Matrix4x4.zero; noOpShadowMatrix.m22 = (SystemInfo.usesReversedZBuffer) ? 1.0f : 0.0f; for (int i = cascadeCount; i <= k_MaxCascades; ++i) m_MainLightShadowMatrices[i] = noOpShadowMatrix; float invShadowAtlasWidth = 1.0f / renderTargetWidth; float invShadowAtlasHeight = 1.0f / renderTargetHeight; float invHalfShadowAtlasWidth = 0.5f * invShadowAtlasWidth; float invHalfShadowAtlasHeight = 0.5f * invShadowAtlasHeight; float softShadowsProp = ShadowUtils.SoftShadowQualityToShaderProperty(light, softShadows); ShadowUtils.GetScaleAndBiasForLinearDistanceFade(m_MaxShadowDistanceSq, m_CascadeBorder, out float shadowFadeScale, out float shadowFadeBias); cmd.SetGlobalMatrixArray(MainLightShadowConstantBuffer._WorldToShadow, m_MainLightShadowMatrices); cmd.SetGlobalVector(MainLightShadowConstantBuffer._ShadowParams, new Vector4(light.shadowStrength, softShadowsProp, shadowFadeScale, shadowFadeBias)); if (m_ShadowCasterCascadesCount > 1) { cmd.SetGlobalVector(MainLightShadowConstantBuffer._CascadeShadowSplitSpheres0, m_CascadeSplitDistances[0]); cmd.SetGlobalVector(MainLightShadowConstantBuffer._CascadeShadowSplitSpheres1, m_CascadeSplitDistances[1]); cmd.SetGlobalVector(MainLightShadowConstantBuffer._CascadeShadowSplitSpheres2, m_CascadeSplitDistances[2]); cmd.SetGlobalVector(MainLightShadowConstantBuffer._CascadeShadowSplitSpheres3, m_CascadeSplitDistances[3]); cmd.SetGlobalVector(MainLightShadowConstantBuffer._CascadeShadowSplitSphereRadii, new Vector4( m_CascadeSplitDistances[0].w * m_CascadeSplitDistances[0].w, m_CascadeSplitDistances[1].w * m_CascadeSplitDistances[1].w, m_CascadeSplitDistances[2].w * m_CascadeSplitDistances[2].w, m_CascadeSplitDistances[3].w * m_CascadeSplitDistances[3].w)); } // Inside shader soft shadows are controlled through global keyword. // If any additional light has soft shadows it will force soft shadows on main light too. // As it is not trivial finding out which additional light has soft shadows, we will pass main light properties if soft shadows are supported. // This workaround will be removed once we will support soft shadows per light. if (shadowData.supportsSoftShadows) { cmd.SetGlobalVector(MainLightShadowConstantBuffer._ShadowOffset0, new Vector4(-invHalfShadowAtlasWidth, -invHalfShadowAtlasHeight, invHalfShadowAtlasWidth, -invHalfShadowAtlasHeight)); cmd.SetGlobalVector(MainLightShadowConstantBuffer._ShadowOffset1, new Vector4(-invHalfShadowAtlasWidth, invHalfShadowAtlasHeight, invHalfShadowAtlasWidth, invHalfShadowAtlasHeight)); cmd.SetGlobalVector(MainLightShadowConstantBuffer._ShadowmapSize, new Vector4(invShadowAtlasWidth, invShadowAtlasHeight, renderTargetWidth, renderTargetHeight)); } } private class PassData { internal MainLightShadowCasterPass pass; internal RenderGraph graph; internal TextureHandle shadowmapTexture; internal RenderingData renderingData; internal int shadowmapID; internal bool emptyShadowmap; } internal TextureHandle Render(RenderGraph graph, ref RenderingData renderingData) { TextureHandle shadowTexture; using (var builder = graph.AddRenderPass("Main Light Shadowmap", out var passData, base.profilingSampler)) { InitPassData(ref passData, ref renderingData, ref graph); if (!m_CreateEmptyShadowmap) { passData.shadowmapTexture = UniversalRenderer.CreateRenderGraphTexture(graph, m_MainLightShadowmapTexture.rt.descriptor, "Main Shadowmap", true, ShadowUtils.m_ForceShadowPointSampling ? FilterMode.Point : FilterMode.Bilinear); builder.UseDepthBuffer(passData.shadowmapTexture, DepthAccess.Write); } // Need this as shadowmap is only used as Global Texture and not a buffer, so would get culled by RG builder.AllowPassCulling(false); builder.SetRenderFunc((PassData data, RenderGraphContext context) => { if (!data.emptyShadowmap) data.pass.RenderMainLightCascadeShadowmap(ref context.renderContext, ref data.renderingData); }); shadowTexture = passData.shadowmapTexture; } using (var builder = graph.AddRenderPass("Set Main Shadow Globals", out var passData, base.profilingSampler)) { InitPassData(ref passData, ref renderingData, ref graph); passData.shadowmapTexture = shadowTexture; if (shadowTexture.IsValid()) builder.UseDepthBuffer(shadowTexture, DepthAccess.Read); builder.AllowPassCulling(false); builder.SetRenderFunc((PassData data, RenderGraphContext context) => { if (data.emptyShadowmap) { data.pass.SetEmptyMainLightCascadeShadowmap(ref context.renderContext, ref data.renderingData); data.shadowmapTexture = data.graph.defaultResources.defaultShadowTexture; } data.renderingData.commandBuffer.SetGlobalTexture(data.shadowmapID, data.shadowmapTexture); }); return passData.shadowmapTexture; } } void InitPassData(ref PassData passData, ref RenderingData renderingData, ref RenderGraph graph) { passData.pass = this; passData.graph = graph; passData.emptyShadowmap = m_CreateEmptyShadowmap; passData.shadowmapID = m_MainLightShadowmapID; passData.renderingData = renderingData; } }; }