using UnityEngine.Experimental.Rendering; using UnityEngine.Experimental.Rendering.Universal; using UnityEngine.Rendering.Universal.Internal; namespace UnityEngine.Rendering.Universal { internal class Renderer2D : ScriptableRenderer { #if UNITY_SWITCH internal const int k_DepthBufferBits = 24; #else internal const int k_DepthBufferBits = 32; #endif const int k_FinalBlitPassQueueOffset = 1; const int k_AfterFinalBlitPassQueueOffset = k_FinalBlitPassQueueOffset + 1; Render2DLightingPass m_Render2DLightingPass; PixelPerfectBackgroundPass m_PixelPerfectBackgroundPass; UpscalePass m_UpscalePass; FinalBlitPass m_FinalBlitPass; DrawScreenSpaceUIPass m_DrawOffscreenUIPass; DrawScreenSpaceUIPass m_DrawOverlayUIPass; Light2DCullResult m_LightCullResult; internal RenderTargetBufferSystem m_ColorBufferSystem; private static readonly ProfilingSampler m_ProfilingSampler = new ProfilingSampler("Create Camera Textures"); bool m_UseDepthStencilBuffer = true; bool m_CreateColorTexture; bool m_CreateDepthTexture; // We probably should declare these names in the base class, // as they must be the same across all ScriptableRenderer types for camera stacking to work. RTHandle m_ColorTextureHandle; RTHandle m_DepthTextureHandle; Material m_BlitMaterial; Material m_BlitHDRMaterial; Material m_SamplingMaterial; Renderer2DData m_Renderer2DData; internal bool createColorTexture => m_CreateColorTexture; internal bool createDepthTexture => m_CreateDepthTexture; PostProcessPasses m_PostProcessPasses; internal ColorGradingLutPass colorGradingLutPass { get => m_PostProcessPasses.colorGradingLutPass; } internal PostProcessPass postProcessPass { get => m_PostProcessPasses.postProcessPass; } internal PostProcessPass finalPostProcessPass { get => m_PostProcessPasses.finalPostProcessPass; } internal RTHandle afterPostProcessColorHandle { get => m_PostProcessPasses.afterPostProcessColor; } internal RTHandle colorGradingLutHandle { get => m_PostProcessPasses.colorGradingLut; } /// public override int SupportedCameraStackingTypes() { return 1 << (int)CameraRenderType.Base | 1 << (int)CameraRenderType.Overlay; } public Renderer2D(Renderer2DData data) : base(data) { m_BlitMaterial = CoreUtils.CreateEngineMaterial(data.coreBlitPS); m_BlitHDRMaterial = CoreUtils.CreateEngineMaterial(data.blitHDROverlay); m_SamplingMaterial = CoreUtils.CreateEngineMaterial(data.samplingShader); m_Render2DLightingPass = new Render2DLightingPass(data, m_BlitMaterial, m_SamplingMaterial); // we should determine why clearing the camera target is set so late in the events... sounds like it could be earlier m_PixelPerfectBackgroundPass = new PixelPerfectBackgroundPass(RenderPassEvent.AfterRenderingTransparents); m_UpscalePass = new UpscalePass(RenderPassEvent.AfterRenderingPostProcessing, m_BlitMaterial); m_FinalBlitPass = new FinalBlitPass(RenderPassEvent.AfterRendering + k_FinalBlitPassQueueOffset, m_BlitMaterial, m_BlitHDRMaterial); m_DrawOffscreenUIPass = new DrawScreenSpaceUIPass(RenderPassEvent.BeforeRenderingPostProcessing, true); m_DrawOverlayUIPass = new DrawScreenSpaceUIPass(RenderPassEvent.AfterRendering + k_AfterFinalBlitPassQueueOffset, false); // after m_FinalBlitPass // RenderTexture format depends on camera and pipeline (HDR, non HDR, etc) // Samples (MSAA) depend on camera and pipeline m_ColorBufferSystem = new RenderTargetBufferSystem("_CameraColorAttachment"); var ppParams = PostProcessParams.Create(); ppParams.blitMaterial = m_BlitMaterial; ppParams.requestHDRFormat = GraphicsFormat.B10G11R11_UFloatPack32; m_PostProcessPasses = new PostProcessPasses(data.postProcessData, ref ppParams); m_UseDepthStencilBuffer = data.useDepthStencilBuffer; m_Renderer2DData = data; supportedRenderingFeatures = new RenderingFeatures(); m_LightCullResult = new Light2DCullResult(); m_Renderer2DData.lightCullResult = m_LightCullResult; // Initialize Blitter if UniversalRenderPipeline hasn't done so bool initBlitter = Blitter.GetBlitMaterial(TextureDimension.Tex2D) == null; var asset = UniversalRenderPipeline.asset; if (asset != null) { foreach (var rendererData in asset.m_RendererDataList) { if (rendererData is UniversalRendererData) { initBlitter = false; break; } } } if (initBlitter) Blitter.Initialize(data.coreBlitPS, data.coreBlitColorAndDepthPS); LensFlareCommonSRP.mergeNeeded = 0; LensFlareCommonSRP.maxLensFlareWithOcclusionTemporalSample = 1; LensFlareCommonSRP.Initialize(); } protected override void Dispose(bool disposing) { m_Renderer2DData.Dispose(); m_PostProcessPasses.Dispose(); m_ColorTextureHandle?.Release(); m_DepthTextureHandle?.Release(); ReleaseRenderTargets(); m_UpscalePass.Dispose(); m_FinalBlitPass?.Dispose(); m_DrawOffscreenUIPass?.Dispose(); m_DrawOverlayUIPass?.Dispose(); CoreUtils.Destroy(m_BlitMaterial); CoreUtils.Destroy(m_BlitHDRMaterial); CoreUtils.Destroy(m_SamplingMaterial); Blitter.Cleanup(); base.Dispose(disposing); } internal override void ReleaseRenderTargets() { m_ColorBufferSystem.Dispose(); m_PostProcessPasses.ReleaseRenderTargets(); } public Renderer2DData GetRenderer2DData() { return m_Renderer2DData; } private struct RenderPassInputSummary { internal bool requiresDepthTexture; internal bool requiresColorTexture; } private RenderPassInputSummary GetRenderPassInputs(ref RenderingData renderingData, ref CameraData cameraData) { RenderPassInputSummary inputSummary = new RenderPassInputSummary(); for (int i = 0; i < activeRenderPassQueue.Count; ++i) { ScriptableRenderPass pass = activeRenderPassQueue[i]; bool needsDepth = (pass.input & ScriptableRenderPassInput.Depth) != ScriptableRenderPassInput.None; bool needsColor = (pass.input & ScriptableRenderPassInput.Color) != ScriptableRenderPassInput.None; inputSummary.requiresDepthTexture |= needsDepth; inputSummary.requiresColorTexture |= needsColor; } inputSummary.requiresColorTexture |= cameraData.postProcessEnabled || cameraData.isHdrEnabled || cameraData.isSceneViewCamera || !cameraData.isDefaultViewport || cameraData.requireSrgbConversion || !cameraData.resolveFinalTarget || m_Renderer2DData.useCameraSortingLayerTexture || !Mathf.Approximately(cameraData.renderScale, 1.0f) || (DebugHandler != null && DebugHandler.WriteToDebugScreenTexture(ref cameraData)); inputSummary.requiresDepthTexture |= (!cameraData.resolveFinalTarget && m_UseDepthStencilBuffer); return inputSummary; } void CreateRenderTextures( ref RenderPassInputSummary renderPassInputs, CommandBuffer cmd, ref CameraData cameraData, bool forceCreateColorTexture, FilterMode colorTextureFilterMode, out RTHandle colorTargetHandle, out RTHandle depthTargetHandle) { ref var cameraTargetDescriptor = ref cameraData.cameraTargetDescriptor; var colorDescriptor = cameraTargetDescriptor; colorDescriptor.depthBufferBits = (int)DepthBits.None; m_ColorBufferSystem.SetCameraSettings(colorDescriptor, colorTextureFilterMode); if (cameraData.renderType == CameraRenderType.Base) { m_CreateColorTexture = renderPassInputs.requiresColorTexture; m_CreateDepthTexture = renderPassInputs.requiresDepthTexture; m_CreateColorTexture |= forceCreateColorTexture; // RTHandles do not support combining color and depth in the same texture so we create them separately m_CreateDepthTexture |= createColorTexture; if (createColorTexture) { if (m_ColorBufferSystem.PeekBackBuffer() == null || m_ColorBufferSystem.PeekBackBuffer().nameID != BuiltinRenderTextureType.CameraTarget) { m_ColorTextureHandle = m_ColorBufferSystem.GetBackBuffer(cmd); cmd.SetGlobalTexture("_CameraColorTexture", m_ColorTextureHandle.nameID); //Set _AfterPostProcessTexture, users might still rely on this although it is now always the cameratarget due to swapbuffer cmd.SetGlobalTexture("_AfterPostProcessTexture", m_ColorTextureHandle.nameID); } m_ColorTextureHandle = m_ColorBufferSystem.PeekBackBuffer(); } if (createDepthTexture) { var depthDescriptor = cameraTargetDescriptor; depthDescriptor.colorFormat = RenderTextureFormat.Depth; depthDescriptor.graphicsFormat = GraphicsFormat.None; depthDescriptor.depthBufferBits = k_DepthBufferBits; if (!cameraData.resolveFinalTarget && m_UseDepthStencilBuffer) depthDescriptor.bindMS = depthDescriptor.msaaSamples > 1 && !SystemInfo.supportsMultisampleAutoResolve && (SystemInfo.supportsMultisampledTextures != 0); RenderingUtils.ReAllocateIfNeeded(ref m_DepthTextureHandle, depthDescriptor, FilterMode.Point, wrapMode: TextureWrapMode.Clamp, name: "_CameraDepthAttachment"); } colorTargetHandle = createColorTexture ? m_ColorTextureHandle : k_CameraTarget; depthTargetHandle = createDepthTexture ? m_DepthTextureHandle : k_CameraTarget; } else // Overlay camera { cameraData.baseCamera.TryGetComponent(out var baseCameraData); var baseRenderer = (Renderer2D)baseCameraData.scriptableRenderer; if (m_ColorBufferSystem != baseRenderer.m_ColorBufferSystem) { m_ColorBufferSystem.Dispose(); m_ColorBufferSystem = baseRenderer.m_ColorBufferSystem; } // These render textures are created by the base camera, but it's the responsibility of the last overlay camera's ScriptableRenderer // to release the textures in its FinishRendering(). m_CreateColorTexture = true; m_CreateDepthTexture = true; m_ColorTextureHandle = baseRenderer.m_ColorTextureHandle; m_DepthTextureHandle = baseRenderer.m_DepthTextureHandle; colorTargetHandle = m_ColorTextureHandle; depthTargetHandle = m_DepthTextureHandle; } } public override void Setup(ScriptableRenderContext context, ref RenderingData renderingData) { ref CameraData cameraData = ref renderingData.cameraData; ref var cameraTargetDescriptor = ref cameraData.cameraTargetDescriptor; bool stackHasPostProcess = renderingData.postProcessingEnabled && m_PostProcessPasses.isCreated; bool hasPostProcess = renderingData.cameraData.postProcessEnabled && m_PostProcessPasses.isCreated; bool lastCameraInStack = cameraData.resolveFinalTarget; var colorTextureFilterMode = FilterMode.Bilinear; PixelPerfectCamera ppc = null; bool ppcUsesOffscreenRT = false; bool ppcUpscaleRT = false; if (DebugHandler != null) { #if UNITY_EDITOR UnityEditorInternal.SpriteMaskUtility.EnableDebugMode(DebugHandler.DebugDisplaySettings.materialSettings.materialDebugMode == DebugMaterialMode.SpriteMask); #endif if (DebugHandler.AreAnySettingsActive) { stackHasPostProcess = stackHasPostProcess && DebugHandler.IsPostProcessingAllowed; hasPostProcess = hasPostProcess && DebugHandler.IsPostProcessingAllowed; } DebugHandler.Setup(context, ref renderingData); if (DebugHandler.IsActiveForCamera(ref cameraData)) { if (DebugHandler.WriteToDebugScreenTexture(ref cameraData)) { RenderTextureDescriptor descriptor = cameraData.cameraTargetDescriptor; DebugHandler.ConfigureColorDescriptorForDebugScreen(ref descriptor, cameraData.pixelWidth, cameraData.pixelHeight); RenderingUtils.ReAllocateIfNeeded(ref DebugHandler.DebugScreenColorHandle, descriptor, name: "_DebugScreenColor"); RenderTextureDescriptor depthDesc = cameraData.cameraTargetDescriptor; DebugHandler.ConfigureDepthDescriptorForDebugScreen(ref depthDesc, k_DepthBufferBits, cameraData.pixelWidth, cameraData.pixelHeight); RenderingUtils.ReAllocateIfNeeded(ref DebugHandler.DebugScreenDepthHandle, depthDesc, name: "_DebugScreenDepth"); } if (DebugHandler.HDRDebugViewIsActive(ref cameraData)) { DebugHandler.hdrDebugViewPass.Setup(ref cameraData, DebugHandler.DebugDisplaySettings.lightingSettings.hdrDebugMode); EnqueuePass(DebugHandler.hdrDebugViewPass); } } } #if UNITY_EDITOR // The scene view camera cannot be uninitialized or skybox when using the 2D renderer. if (cameraData.cameraType == CameraType.SceneView) { renderingData.cameraData.camera.clearFlags = CameraClearFlags.SolidColor; } #endif // Pixel Perfect Camera doesn't support camera stacking. if (cameraData.renderType == CameraRenderType.Base && lastCameraInStack) { cameraData.camera.TryGetComponent(out ppc); if (ppc != null && ppc.enabled) { if (ppc.offscreenRTSize != Vector2Int.zero) { ppcUsesOffscreenRT = true; // Pixel Perfect Camera may request a different RT size than camera VP size. // In that case we need to modify cameraTargetDescriptor here so that all the passes would use the same size. cameraTargetDescriptor.width = ppc.offscreenRTSize.x; cameraTargetDescriptor.height = ppc.offscreenRTSize.y; // If using FullScreenRenderPass with Pixel Perfect, we need to reallocate the size of the RT used var fullScreenRenderPass = activeRenderPassQueue.Find(x => x is FullScreenPassRendererFeature.FullScreenRenderPass) as FullScreenPassRendererFeature.FullScreenRenderPass; fullScreenRenderPass?.ReAllocate(cameraTargetDescriptor); } colorTextureFilterMode = FilterMode.Point; ppcUpscaleRT = ppc.gridSnapping == PixelPerfectCamera.GridSnapping.UpscaleRenderTexture || ppc.requiresUpscalePass; } } RenderPassInputSummary renderPassInputs = GetRenderPassInputs(ref renderingData, ref cameraData); RTHandle colorTargetHandle; RTHandle depthTargetHandle; var cmd = renderingData.commandBuffer; using (new ProfilingScope(cmd, m_ProfilingSampler)) { CreateRenderTextures(ref renderPassInputs, cmd, ref cameraData, ppcUsesOffscreenRT, colorTextureFilterMode, out colorTargetHandle, out depthTargetHandle); } context.ExecuteCommandBuffer(cmd); cmd.Clear(); ConfigureCameraTarget(colorTargetHandle, depthTargetHandle); if (hasPostProcess) { colorGradingLutPass.ConfigureDescriptor(in renderingData.postProcessingData, out var desc, out var filterMode); RenderingUtils.ReAllocateIfNeeded(ref m_PostProcessPasses.m_ColorGradingLut, desc, filterMode, TextureWrapMode.Clamp, name: "_InternalGradingLut"); colorGradingLutPass.Setup(colorGradingLutHandle); EnqueuePass(colorGradingLutPass); } m_Render2DLightingPass.Setup(renderPassInputs.requiresDepthTexture || m_UseDepthStencilBuffer); m_Render2DLightingPass.ConfigureTarget(colorTargetHandle, depthTargetHandle); EnqueuePass(m_Render2DLightingPass); bool shouldRenderUI = cameraData.rendersOverlayUI; bool outputToHDR = cameraData.isHDROutputActive; if (shouldRenderUI && outputToHDR) { m_DrawOffscreenUIPass.Setup(ref cameraData, k_DepthBufferBits); EnqueuePass(m_DrawOffscreenUIPass); } // TODO: Investigate how to make FXAA work with HDR output. bool isFXAAEnabled = cameraData.antialiasing == AntialiasingMode.FastApproximateAntialiasing && !outputToHDR; // When using Upscale Render Texture on a Pixel Perfect Camera, we want all post-processing effects done with a low-res RT, // and only upscale the low-res RT to fullscreen when blitting it to camera target. Also, final post processing pass is not run in this case, // so FXAA is not supported (you don't want to apply FXAA when everything is intentionally pixelated). bool requireFinalPostProcessPass = lastCameraInStack && !ppcUpscaleRT && stackHasPostProcess && isFXAAEnabled; bool hasPassesAfterPostProcessing = activeRenderPassQueue.Find(x => x.renderPassEvent == RenderPassEvent.AfterRenderingPostProcessing) != null; bool needsColorEncoding = DebugHandler == null || !DebugHandler.HDRDebugViewIsActive(ref cameraData); if (hasPostProcess) { var desc = PostProcessPass.GetCompatibleDescriptor(cameraTargetDescriptor, cameraTargetDescriptor.width, cameraTargetDescriptor.height, cameraTargetDescriptor.graphicsFormat, DepthBits.None); RenderingUtils.ReAllocateIfNeeded(ref m_PostProcessPasses.m_AfterPostProcessColor, desc, FilterMode.Point, TextureWrapMode.Clamp, name: "_AfterPostProcessTexture"); postProcessPass.Setup( cameraTargetDescriptor, colorTargetHandle, afterPostProcessColorHandle, depthTargetHandle, colorGradingLutHandle, requireFinalPostProcessPass, afterPostProcessColorHandle.nameID == k_CameraTarget.nameID && needsColorEncoding); EnqueuePass(postProcessPass); } RTHandle finalTargetHandle = colorTargetHandle; if (ppc != null && ppc.enabled && ppc.cropFrame != PixelPerfectCamera.CropFrame.None) { EnqueuePass(m_PixelPerfectBackgroundPass); // Queue PixelPerfect UpscalePass. Only used when using the Stretch Fill option if (ppc.requiresUpscalePass) { int upscaleWidth = ppc.refResolutionX * ppc.pixelRatio; int upscaleHeight = ppc.refResolutionY * ppc.pixelRatio; m_UpscalePass.Setup(colorTargetHandle, upscaleWidth, upscaleHeight, ppc.finalBlitFilterMode, ref renderingData, out finalTargetHandle); EnqueuePass(m_UpscalePass); } } if (requireFinalPostProcessPass) { finalPostProcessPass.SetupFinalPass(finalTargetHandle, hasPassesAfterPostProcessing, needsColorEncoding); EnqueuePass(finalPostProcessPass); } else if (lastCameraInStack && finalTargetHandle != k_CameraTarget) { m_FinalBlitPass.Setup(cameraTargetDescriptor, finalTargetHandle); EnqueuePass(m_FinalBlitPass); } if (shouldRenderUI && !outputToHDR) { EnqueuePass(m_DrawOverlayUIPass); } } public override void SetupCullingParameters(ref ScriptableCullingParameters cullingParameters, ref CameraData cameraData) { cullingParameters.cullingOptions = CullingOptions.None; cullingParameters.isOrthographic = cameraData.camera.orthographic; cullingParameters.shadowDistance = 0.0f; m_LightCullResult.SetupCulling(ref cullingParameters, cameraData.camera); } internal override void SwapColorBuffer(CommandBuffer cmd) { m_ColorBufferSystem.Swap(); //Check if we are using the depth that is attached to color buffer if (m_DepthTextureHandle.nameID != BuiltinRenderTextureType.CameraTarget) ConfigureCameraTarget(m_ColorBufferSystem.GetBackBuffer(cmd), m_DepthTextureHandle); else ConfigureCameraColorTarget(m_ColorBufferSystem.GetBackBuffer(cmd)); m_ColorTextureHandle = m_ColorBufferSystem.GetBackBuffer(cmd); cmd.SetGlobalTexture("_CameraColorTexture", m_ColorTextureHandle.nameID); //Set _AfterPostProcessTexture, users might still rely on this although it is now always the cameratarget due to swapbuffer cmd.SetGlobalTexture("_AfterPostProcessTexture", m_ColorTextureHandle.nameID); } internal override RTHandle GetCameraColorFrontBuffer(CommandBuffer cmd) { return m_ColorBufferSystem.GetFrontBuffer(cmd); } internal override RTHandle GetCameraColorBackBuffer(CommandBuffer cmd) { return m_ColorBufferSystem.GetBackBuffer(cmd); } internal override void EnableSwapBufferMSAA(bool enable) { m_ColorBufferSystem.EnableMSAA(enable); } } }