using System; using UnityEngine; using UnityEngine.Rendering.RenderGraphModule; using UnityEngine.Rendering; using UnityEngine.Rendering.Universal; using UnityEngine.Experimental.Rendering; /// /// This renderer feature lets you create single-pass full screen post processing effects without needing to write code. /// [URPHelpURL("renderer-features/renderer-feature-full-screen-pass")] public partial class FullScreenPassRendererFeature : ScriptableRendererFeature { /// /// An injection point for the full screen pass. This is similar to the RenderPassEvent enum but limited to only supported events. /// public enum InjectionPoint { /// /// Inject a full screen pass before transparents are rendered. /// BeforeRenderingTransparents = RenderPassEvent.BeforeRenderingTransparents, /// /// Inject a full screen pass before post processing is rendered. /// BeforeRenderingPostProcessing = RenderPassEvent.BeforeRenderingPostProcessing, /// /// Inject a full screen pass after post processing is rendered. /// AfterRenderingPostProcessing = RenderPassEvent.AfterRenderingPostProcessing } /// /// Specifies at which injection point the pass will be rendered. /// public InjectionPoint injectionPoint = InjectionPoint.AfterRenderingPostProcessing; /// /// Specifies whether the assigned material will need to use the current screen contents as an input texture. /// Disable this to optimize away an extra color copy pass when you know that the assigned material will only need /// to write on top of or hardware blend with the contents of the active color target. /// public bool fetchColorBuffer = true; /// /// A mask of URP textures that the assigned material will need access to. Requesting unused requirements can degrade /// performance unnecessarily as URP might need to run additional rendering passes to generate them. /// public ScriptableRenderPassInput requirements = ScriptableRenderPassInput.None; /// /// The material used to render the full screen pass (typically based on the Fullscreen Shader Graph target). /// public Material passMaterial; /// /// The shader pass index that should be used when rendering the assigned material. /// public int passIndex = 0; /// /// Specifies if the active camera's depth-stencil buffer should be bound when rendering the full screen pass. /// Disabling this will ensure that the material's depth and stencil commands will have no effect (this could also have a slight performance benefit). /// public bool bindDepthStencilAttachment = false; private FullScreenRenderPass m_FullScreenPass; /// public override void Create() { m_FullScreenPass = new FullScreenRenderPass(name); } internal override bool RequireRenderingLayers(bool isDeferred, bool needsGBufferAccurateNormals, out RenderingLayerUtils.Event atEvent, out RenderingLayerUtils.MaskSize maskSize) { atEvent = RenderingLayerUtils.Event.Opaque; maskSize = RenderingLayerUtils.MaskSize.Bits8; return false; } /// public override void AddRenderPasses(ScriptableRenderer renderer, ref RenderingData renderingData) { if (renderingData.cameraData.cameraType == CameraType.Preview || renderingData.cameraData.cameraType == CameraType.Reflection || UniversalRenderer.IsOffscreenDepthTexture(ref renderingData.cameraData)) return; if (passMaterial == null) { Debug.LogWarningFormat("The full screen feature \"{0}\" will not execute - no material is assigned. Please make sure a material is assigned for this feature on the renderer asset.", name); return; } if (passIndex < 0 || passIndex >= passMaterial.passCount) { Debug.LogWarningFormat("The full screen feature \"{0}\" will not execute - the pass index is out of bounds for the material.", name); return; } m_FullScreenPass.renderPassEvent = (RenderPassEvent)injectionPoint; m_FullScreenPass.ConfigureInput(requirements); m_FullScreenPass.SetupMembers(passMaterial, passIndex, fetchColorBuffer, bindDepthStencilAttachment); m_FullScreenPass.requiresIntermediateTexture = fetchColorBuffer; renderer.EnqueuePass(m_FullScreenPass); } /// protected override void Dispose(bool disposing) { m_FullScreenPass.Dispose(); } internal class FullScreenRenderPass : ScriptableRenderPass { private Material m_Material; private int m_PassIndex; private bool m_FetchActiveColor; private bool m_BindDepthStencilAttachment; private RTHandle m_CopiedColor; private static MaterialPropertyBlock s_SharedPropertyBlock = new MaterialPropertyBlock(); public FullScreenRenderPass(string passName) { profilingSampler = new ProfilingSampler(passName); } public void SetupMembers(Material material, int passIndex, bool fetchActiveColor, bool bindDepthStencilAttachment) { m_Material = material; m_PassIndex = passIndex; m_FetchActiveColor = fetchActiveColor; m_BindDepthStencilAttachment = bindDepthStencilAttachment; } [Obsolete(DeprecationMessage.CompatibilityScriptingAPIObsolete, false)] public override void OnCameraSetup(CommandBuffer cmd, ref RenderingData renderingData) { // Disable obsolete warning for internal usage #pragma warning disable CS0618 // FullScreenPass manages its own RenderTarget. // ResetTarget here so that ScriptableRenderer's active attachement can be invalidated when processing this ScriptableRenderPass. ResetTarget(); #pragma warning restore CS0618 if (m_FetchActiveColor) ReAllocate(renderingData.cameraData.cameraTargetDescriptor); } internal void ReAllocate(RenderTextureDescriptor desc) { desc.msaaSamples = 1; desc.depthStencilFormat = GraphicsFormat.None; RenderingUtils.ReAllocateHandleIfNeeded(ref m_CopiedColor, desc, name: "_FullscreenPassColorCopy"); } public void Dispose() { m_CopiedColor?.Release(); } private static void ExecuteCopyColorPass(RasterCommandBuffer cmd, RTHandle sourceTexture) { Blitter.BlitTexture(cmd, sourceTexture, new Vector4(1, 1, 0, 0), 0.0f, false); } private static void ExecuteMainPass(RasterCommandBuffer cmd, RTHandle sourceTexture, Material material, int passIndex) { s_SharedPropertyBlock.Clear(); if (sourceTexture != null) s_SharedPropertyBlock.SetTexture(ShaderPropertyId.blitTexture, sourceTexture); // We need to set the "_BlitScaleBias" uniform for user materials with shaders relying on core Blit.hlsl to work s_SharedPropertyBlock.SetVector(ShaderPropertyId.blitScaleBias, new Vector4(1, 1, 0, 0)); cmd.DrawProcedural(Matrix4x4.identity, material, passIndex, MeshTopology.Triangles, 3, 1, s_SharedPropertyBlock); } [Obsolete(DeprecationMessage.CompatibilityScriptingAPIObsolete, false)] public override void Execute(ScriptableRenderContext context, ref RenderingData renderingData) { ref var cameraData = ref renderingData.cameraData; var cmd = renderingData.commandBuffer; using (new ProfilingScope(cmd, profilingSampler)) { RasterCommandBuffer rasterCmd = CommandBufferHelpers.GetRasterCommandBuffer(cmd); if (m_FetchActiveColor) { CoreUtils.SetRenderTarget(cmd, m_CopiedColor); ExecuteCopyColorPass(rasterCmd, cameraData.renderer.cameraColorTargetHandle); } if (m_BindDepthStencilAttachment) CoreUtils.SetRenderTarget(cmd, cameraData.renderer.cameraColorTargetHandle, cameraData.renderer.cameraDepthTargetHandle); else CoreUtils.SetRenderTarget(cmd, cameraData.renderer.cameraColorTargetHandle); ExecuteMainPass(rasterCmd, m_FetchActiveColor ? m_CopiedColor : null, m_Material, m_PassIndex); } } public override void RecordRenderGraph(RenderGraph renderGraph, ContextContainer frameData) { UniversalResourceData resourcesData = frameData.Get(); UniversalCameraData cameraData = frameData.Get(); TextureHandle source, destination; Debug.Assert(resourcesData.cameraColor.IsValid()); if (m_FetchActiveColor) { var targetDesc = renderGraph.GetTextureDesc(resourcesData.cameraColor); targetDesc.name = "_CameraColorFullScreenPass"; targetDesc.clearBuffer = false; source = resourcesData.activeColorTexture; destination = renderGraph.CreateTexture(targetDesc); using (var builder = renderGraph.AddRasterRenderPass("Copy Color Full Screen", out var passData, profilingSampler)) { passData.inputTexture = source; builder.UseTexture(passData.inputTexture, AccessFlags.Read); builder.SetRenderAttachment(destination, 0, AccessFlags.Write); builder.SetRenderFunc((CopyPassData data, RasterGraphContext rgContext) => { ExecuteCopyColorPass(rgContext.cmd, data.inputTexture); }); } //Swap for next pass; source = destination; } else { source = TextureHandle.nullHandle; } destination = resourcesData.activeColorTexture; using (var builder = renderGraph.AddRasterRenderPass(passName, out var passData, profilingSampler)) { passData.material = m_Material; passData.passIndex = m_PassIndex; passData.inputTexture = source; if(passData.inputTexture.IsValid()) builder.UseTexture(passData.inputTexture, AccessFlags.Read); bool needsColor = (input & ScriptableRenderPassInput.Color) != ScriptableRenderPassInput.None; bool needsDepth = (input & ScriptableRenderPassInput.Depth) != ScriptableRenderPassInput.None; bool needsMotion = (input & ScriptableRenderPassInput.Motion) != ScriptableRenderPassInput.None; bool needsNormal = (input & ScriptableRenderPassInput.Normal) != ScriptableRenderPassInput.None; if (needsColor) { Debug.Assert(resourcesData.cameraOpaqueTexture.IsValid()); builder.UseTexture(resourcesData.cameraOpaqueTexture); } if (needsDepth) { Debug.Assert(resourcesData.cameraDepthTexture.IsValid()); builder.UseTexture(resourcesData.cameraDepthTexture); } if (needsMotion) { Debug.Assert(resourcesData.motionVectorColor.IsValid()); builder.UseTexture(resourcesData.motionVectorColor); Debug.Assert(resourcesData.motionVectorDepth.IsValid()); builder.UseTexture(resourcesData.motionVectorDepth); } if (needsNormal) { Debug.Assert(resourcesData.cameraNormalsTexture.IsValid()); builder.UseTexture(resourcesData.cameraNormalsTexture); } builder.SetRenderAttachment(destination, 0, AccessFlags.Write); if (m_BindDepthStencilAttachment) builder.SetRenderAttachmentDepth(resourcesData.activeDepthTexture, AccessFlags.Write); builder.SetRenderFunc((MainPassData data, RasterGraphContext rgContext) => { ExecuteMainPass(rgContext.cmd, data.inputTexture, data.material, data.passIndex); }); } } private class CopyPassData { internal TextureHandle inputTexture; } private class MainPassData { internal Material material; internal int passIndex; internal TextureHandle inputTexture; } } }