using System; using System.Collections.Generic; using System.Diagnostics; using UnityEngine.Experimental.Rendering; namespace UnityEngine.Rendering.Universal { /// /// Contains properties and helper functions that you can use when rendering. /// public static class RenderingUtils { static List m_LegacyShaderPassNames = new List { new ShaderTagId("Always"), new ShaderTagId("ForwardBase"), new ShaderTagId("PrepassBase"), new ShaderTagId("Vertex"), new ShaderTagId("VertexLMRGBM"), new ShaderTagId("VertexLM"), }; static AttachmentDescriptor s_EmptyAttachment = new AttachmentDescriptor(GraphicsFormat.None); internal static AttachmentDescriptor emptyAttachment { get { return s_EmptyAttachment; } } static Mesh s_FullscreenMesh = null; /// /// Returns a mesh that you can use with to render full-screen effects. /// [Obsolete("Use Blitter.BlitCameraTexture instead of CommandBuffer.DrawMesh(fullscreenMesh, ...)")] public static Mesh fullscreenMesh { get { if (s_FullscreenMesh != null) return s_FullscreenMesh; float topV = 1.0f; float bottomV = 0.0f; s_FullscreenMesh = new Mesh { name = "Fullscreen Quad" }; s_FullscreenMesh.SetVertices(new List { new Vector3(-1.0f, -1.0f, 0.0f), new Vector3(-1.0f, 1.0f, 0.0f), new Vector3(1.0f, -1.0f, 0.0f), new Vector3(1.0f, 1.0f, 0.0f) }); s_FullscreenMesh.SetUVs(0, new List { new Vector2(0.0f, bottomV), new Vector2(0.0f, topV), new Vector2(1.0f, bottomV), new Vector2(1.0f, topV) }); s_FullscreenMesh.SetIndices(new[] { 0, 1, 2, 2, 1, 3 }, MeshTopology.Triangles, 0, false); s_FullscreenMesh.UploadMeshData(true); return s_FullscreenMesh; } } internal static bool useStructuredBuffer { // There are some performance issues with StructuredBuffers in some platforms. // We fallback to UBO in those cases. get { // TODO: For now disabling SSBO until figure out Vulkan binding issues. // When enabling this also enable USE_STRUCTURED_BUFFER_FOR_LIGHT_DATA in shader side in Input.hlsl return false; // We don't use SSBO in D3D because we can't figure out without adding shader variants if platforms is D3D10. //GraphicsDeviceType deviceType = SystemInfo.graphicsDeviceType; //return !Application.isMobilePlatform && // (deviceType == GraphicsDeviceType.Metal || deviceType == GraphicsDeviceType.Vulkan || // deviceType == GraphicsDeviceType.PlayStation4 || deviceType == GraphicsDeviceType.PlayStation5 || deviceType == GraphicsDeviceType.XboxOne); } } internal static bool SupportsLightLayers(GraphicsDeviceType type) { // GLES2 does not support bitwise operations. return type != GraphicsDeviceType.OpenGLES2; } static Material s_ErrorMaterial; static Material errorMaterial { get { if (s_ErrorMaterial == null) { // TODO: When importing project, AssetPreviewUpdater::CreatePreviewForAsset will be called multiple times. // This might be in a point that some resources required for the pipeline are not finished importing yet. // Proper fix is to add a fence on asset import. try { s_ErrorMaterial = new Material(Shader.Find("Hidden/Universal Render Pipeline/FallbackError")); } catch { } } return s_ErrorMaterial; } } /// /// Set view and projection matrices. /// This function will set UNITY_MATRIX_V, UNITY_MATRIX_P, UNITY_MATRIX_VP to given view and projection matrices. /// If setInverseMatrices is set to true this function will also set UNITY_MATRIX_I_V and UNITY_MATRIX_I_VP. /// /// CommandBuffer to submit data to GPU. /// View matrix to be set. /// Projection matrix to be set. /// Set this to true if you also need to set inverse camera matrices. public static void SetViewAndProjectionMatrices(CommandBuffer cmd, Matrix4x4 viewMatrix, Matrix4x4 projectionMatrix, bool setInverseMatrices) { Matrix4x4 viewAndProjectionMatrix = projectionMatrix * viewMatrix; cmd.SetGlobalMatrix(ShaderPropertyId.viewMatrix, viewMatrix); cmd.SetGlobalMatrix(ShaderPropertyId.projectionMatrix, projectionMatrix); cmd.SetGlobalMatrix(ShaderPropertyId.viewAndProjectionMatrix, viewAndProjectionMatrix); if (setInverseMatrices) { Matrix4x4 inverseViewMatrix = Matrix4x4.Inverse(viewMatrix); Matrix4x4 inverseProjectionMatrix = Matrix4x4.Inverse(projectionMatrix); Matrix4x4 inverseViewProjection = inverseViewMatrix * inverseProjectionMatrix; cmd.SetGlobalMatrix(ShaderPropertyId.inverseViewMatrix, inverseViewMatrix); cmd.SetGlobalMatrix(ShaderPropertyId.inverseProjectionMatrix, inverseProjectionMatrix); cmd.SetGlobalMatrix(ShaderPropertyId.inverseViewAndProjectionMatrix, inverseViewProjection); } } internal static void SetScaleBiasRt(CommandBuffer cmd, in RenderingData renderingData) { var renderer = renderingData.cameraData.renderer; // SetRenderTarget has logic to flip projection matrix when rendering to render texture. Flip the uv to account for that case. CameraData cameraData = renderingData.cameraData; bool isCameraColorFinalTarget = (cameraData.cameraType == CameraType.Game && renderer.cameraColorTargetHandle.nameID == BuiltinRenderTextureType.CameraTarget && cameraData.camera.targetTexture == null); bool yflip = !isCameraColorFinalTarget; float flipSign = yflip ? -1.0f : 1.0f; Vector4 scaleBiasRt = (flipSign < 0.0f) ? new Vector4(flipSign, 1.0f, -1.0f, 1.0f) : new Vector4(flipSign, 0.0f, 1.0f, 1.0f); cmd.SetGlobalVector(Shader.PropertyToID("_ScaleBiasRt"), scaleBiasRt); } internal static void Blit(CommandBuffer cmd, RTHandle source, Rect viewport, RTHandle destination, RenderBufferLoadAction loadAction, RenderBufferStoreAction storeAction, ClearFlag clearFlag, Color clearColor, Material material, int passIndex = 0) { Vector2 viewportScale = source.useScaling ? new Vector2(source.rtHandleProperties.rtHandleScale.x, source.rtHandleProperties.rtHandleScale.y) : Vector2.one; CoreUtils.SetRenderTarget(cmd, destination, loadAction, storeAction, ClearFlag.None, Color.clear); cmd.SetViewport(viewport); Blitter.BlitTexture(cmd, source, viewportScale, material, passIndex); } internal static void Blit(CommandBuffer cmd, RTHandle source, Rect viewport, RTHandle destinationColor, RenderBufferLoadAction colorLoadAction, RenderBufferStoreAction colorStoreAction, RTHandle destinationDepthStencil, RenderBufferLoadAction depthStencilLoadAction, RenderBufferStoreAction depthStencilStoreAction, ClearFlag clearFlag, Color clearColor, Material material, int passIndex = 0) { Vector2 viewportScale = source.useScaling ? new Vector2(source.rtHandleProperties.rtHandleScale.x, source.rtHandleProperties.rtHandleScale.y) : Vector2.one; CoreUtils.SetRenderTarget(cmd, destinationColor, colorLoadAction, colorStoreAction, destinationDepthStencil, depthStencilLoadAction, depthStencilStoreAction, clearFlag, clearColor); // implicit depth=1.0f stencil=0x0 cmd.SetViewport(viewport); Blitter.BlitTexture(cmd, source, viewportScale, material, passIndex); } internal static void FinalBlit( CommandBuffer cmd, ref CameraData cameraData, RTHandle source, RTHandle destination, RenderBufferLoadAction loadAction, RenderBufferStoreAction storeAction, Material material, int passIndex) { bool isRenderToBackBufferTarget = !cameraData.isSceneViewCamera; #if ENABLE_VR && ENABLE_XR_MODULE if (cameraData.xr.enabled) isRenderToBackBufferTarget = new RenderTargetIdentifier(destination.nameID, 0, CubemapFace.Unknown, -1) == new RenderTargetIdentifier(cameraData.xr.renderTarget, 0, CubemapFace.Unknown, -1); #endif Vector2 viewportScale = source.useScaling ? new Vector2(source.rtHandleProperties.rtHandleScale.x, source.rtHandleProperties.rtHandleScale.y) : Vector2.one; // We y-flip if // 1) we are blitting from render texture to back buffer(UV starts at bottom) and // 2) renderTexture starts UV at top bool yflip = isRenderToBackBufferTarget && cameraData.targetTexture == null && SystemInfo.graphicsUVStartsAtTop; Vector4 scaleBias = yflip ? new Vector4(viewportScale.x, -viewportScale.y, 0, viewportScale.y) : new Vector4(viewportScale.x, viewportScale.y, 0, 0); CoreUtils.SetRenderTarget(cmd, destination, loadAction, storeAction, ClearFlag.None, Color.clear); if (isRenderToBackBufferTarget) cmd.SetViewport(cameraData.pixelRect); // cmd.Blit must be used in Scene View for wireframe mode to make the full screen draw with fill mode // This branch of the if statement must be removed for render graph and the new command list with a novel way of using Blitter with fill mode if (GL.wireframe && cameraData.isSceneViewCamera) { // This set render target is necessary so we change the LOAD state to DontCare. cmd.SetRenderTarget(BuiltinRenderTextureType.CameraTarget, loadAction, storeAction, // color RenderBufferLoadAction.DontCare, RenderBufferStoreAction.DontCare); // depth cmd.Blit(source.nameID, destination.nameID); } else if (source.rt == null) Blitter.BlitTexture(cmd, source.nameID, scaleBias, material, passIndex); // Obsolete usage of RTHandle aliasing a RenderTargetIdentifier else Blitter.BlitTexture(cmd, source, scaleBias, material, passIndex); } // This is used to render materials that contain built-in shader passes not compatible with URP. // It will render those legacy passes with error/pink shader. [Conditional("DEVELOPMENT_BUILD"), Conditional("UNITY_EDITOR")] internal static void RenderObjectsWithError(ScriptableRenderContext context, ref CullingResults cullResults, Camera camera, FilteringSettings filterSettings, SortingCriteria sortFlags) { // TODO: When importing project, AssetPreviewUpdater::CreatePreviewForAsset will be called multiple times. // This might be in a point that some resources required for the pipeline are not finished importing yet. // Proper fix is to add a fence on asset import. if (errorMaterial == null) return; SortingSettings sortingSettings = new SortingSettings(camera) { criteria = sortFlags }; DrawingSettings errorSettings = new DrawingSettings(m_LegacyShaderPassNames[0], sortingSettings) { perObjectData = PerObjectData.None, overrideMaterial = errorMaterial, overrideMaterialPassIndex = 0 }; for (int i = 1; i < m_LegacyShaderPassNames.Count; ++i) errorSettings.SetShaderPassName(i, m_LegacyShaderPassNames[i]); context.DrawRenderers(cullResults, ref errorSettings, ref filterSettings); } // Caches render texture format support. SystemInfo.SupportsRenderTextureFormat and IsFormatSupported allocate memory due to boxing. static Dictionary m_RenderTextureFormatSupport = new Dictionary(); static Dictionary> m_GraphicsFormatSupport = new Dictionary>(); internal static void ClearSystemInfoCache() { m_RenderTextureFormatSupport.Clear(); m_GraphicsFormatSupport.Clear(); } /// /// Checks if a render texture format is supported by the run-time system. /// Similar to , but doesn't allocate memory. /// /// The format to look up. /// Returns true if the graphics card supports the given RenderTextureFormat public static bool SupportsRenderTextureFormat(RenderTextureFormat format) { if (!m_RenderTextureFormatSupport.TryGetValue(format, out var support)) { support = SystemInfo.SupportsRenderTextureFormat(format); m_RenderTextureFormatSupport.Add(format, support); } return support; } /// /// Checks if a texture format is supported by the run-time system. /// Similar to , but doesn't allocate memory. /// /// The format to look up. /// The format usage to look up. /// Returns true if the graphics card supports the given GraphicsFormat public static bool SupportsGraphicsFormat(GraphicsFormat format, FormatUsage usage) { bool support = false; if (!m_GraphicsFormatSupport.TryGetValue(format, out var uses)) { uses = new Dictionary(); support = SystemInfo.IsFormatSupported(format, usage); uses.Add(usage, support); m_GraphicsFormatSupport.Add(format, uses); } else { if (!uses.TryGetValue(usage, out support)) { support = SystemInfo.IsFormatSupported(format, usage); uses.Add(usage, support); } } return support; } /// /// Return the last colorBuffer index actually referring to an existing RenderTarget /// /// /// internal static int GetLastValidColorBufferIndex(RenderTargetIdentifier[] colorBuffers) { int i = colorBuffers.Length - 1; for (; i >= 0; --i) { if (colorBuffers[i] != 0) break; } return i; } /// /// Return the number of items in colorBuffers actually referring to an existing RenderTarget /// /// /// [Obsolete("Use RTHandles for colorBuffers")] internal static uint GetValidColorBufferCount(RenderTargetIdentifier[] colorBuffers) { uint nonNullColorBuffers = 0; if (colorBuffers != null) { foreach (var identifier in colorBuffers) { if (identifier != 0) ++nonNullColorBuffers; } } return nonNullColorBuffers; } /// /// Return the number of items in colorBuffers actually referring to an existing RenderTarget /// /// /// internal static uint GetValidColorBufferCount(RTHandle[] colorBuffers) { uint nonNullColorBuffers = 0; if (colorBuffers != null) { foreach (var identifier in colorBuffers) { if (identifier != null && identifier.nameID != 0) ++nonNullColorBuffers; } } return nonNullColorBuffers; } /// /// Return true if colorBuffers is an actual MRT setup /// /// /// [Obsolete("Use RTHandles for colorBuffers")] internal static bool IsMRT(RenderTargetIdentifier[] colorBuffers) { return GetValidColorBufferCount(colorBuffers) > 1; } /// /// Return true if colorBuffers is an actual MRT setup /// /// /// internal static bool IsMRT(RTHandle[] colorBuffers) { return GetValidColorBufferCount(colorBuffers) > 1; } /// /// Return true if value can be found in source (without recurring to Linq) /// /// /// /// internal static bool Contains(RenderTargetIdentifier[] source, RenderTargetIdentifier value) { foreach (var identifier in source) { if (identifier == value) return true; } return false; } /// /// Return the index where value was found source. Otherwise, return -1. (without recurring to Linq) /// /// /// /// internal static int IndexOf(RenderTargetIdentifier[] source, RenderTargetIdentifier value) { for (int i = 0; i < source.Length; ++i) { if (source[i] == value) return i; } return -1; } /// /// Return the number of RenderTargetIdentifiers in "source" that are valid (not 0) and different from "value" (without recurring to Linq) /// /// /// /// internal static uint CountDistinct(RenderTargetIdentifier[] source, RenderTargetIdentifier value) { uint count = 0; for (int i = 0; i < source.Length; ++i) { if (source[i] != value && source[i] != 0) ++count; } return count; } /// /// Return the index of last valid (i.e different from 0) RenderTargetIdentifiers in "source" (without recurring to Linq) /// /// /// internal static int LastValid(RenderTargetIdentifier[] source) { for (int i = source.Length - 1; i >= 0; --i) { if (source[i] != 0) return i; } return -1; } /// /// Return true if ClearFlag a contains ClearFlag b /// /// /// /// internal static bool Contains(ClearFlag a, ClearFlag b) { return (a & b) == b; } /// /// Return true if "left" and "right" are the same (without recurring to Linq) /// /// /// /// internal static bool SequenceEqual(RenderTargetIdentifier[] left, RenderTargetIdentifier[] right) { if (left.Length != right.Length) return false; for (int i = 0; i < left.Length; ++i) if (left[i] != right[i]) return false; return true; } internal static bool MultisampleDepthResolveSupported() { // Temporarily disabling depth resolve a driver bug on OSX when using some AMD graphics cards. Temporarily disabling depth resolve on that platform // TODO: re-enable once the issue is investigated/fixed if (Application.platform == RuntimePlatform.OSXEditor || Application.platform == RuntimePlatform.OSXPlayer) return false; // Should we also check if the format has stencil and check stencil resolve capability only in that case? return SystemInfo.supportsMultisampleResolveDepth && SystemInfo.supportsMultisampleResolveStencil; } /// /// Return true if handle does not match descriptor /// /// RTHandle to check (can be null) /// Descriptor for the RTHandle to match /// Filtering mode of the RTHandle. /// Addressing mode of the RTHandle. /// Set to true if the depth buffer should be used as a shadow map. /// Anisotropic filtering level. /// Bias applied to mipmaps during filtering. /// Name of the RTHandle. /// Check if the RTHandle has auto scaling enabled if not, check the widths and heights /// internal static bool RTHandleNeedsReAlloc( RTHandle handle, in RenderTextureDescriptor descriptor, FilterMode filterMode, TextureWrapMode wrapMode, bool isShadowMap, int anisoLevel, float mipMapBias, string name, bool scaled) { if (handle == null || handle.rt == null) return true; if (handle.useScaling != scaled) return true; if (!scaled && (handle.rt.width != descriptor.width || handle.rt.height != descriptor.height)) return true; return handle.rt.descriptor.depthBufferBits != descriptor.depthBufferBits || (handle.rt.descriptor.depthBufferBits == (int)DepthBits.None && !isShadowMap && handle.rt.descriptor.graphicsFormat != descriptor.graphicsFormat) || handle.rt.descriptor.dimension != descriptor.dimension || handle.rt.descriptor.enableRandomWrite != descriptor.enableRandomWrite || handle.rt.descriptor.useMipMap != descriptor.useMipMap || handle.rt.descriptor.autoGenerateMips != descriptor.autoGenerateMips || handle.rt.descriptor.msaaSamples != descriptor.msaaSamples || handle.rt.descriptor.bindMS != descriptor.bindMS || handle.rt.descriptor.useDynamicScale != descriptor.useDynamicScale || handle.rt.descriptor.memoryless != descriptor.memoryless || handle.rt.filterMode != filterMode || handle.rt.wrapMode != wrapMode || handle.rt.anisoLevel != anisoLevel || handle.rt.mipMapBias != mipMapBias || handle.name != name; } /// /// Returns the RenderTargetIdentifier of the current camera target. /// /// /// internal static RenderTargetIdentifier GetCameraTargetIdentifier(ref RenderingData renderingData) { // Note: We need to get the cameraData.targetTexture as this will get the targetTexture of the camera stack. // Overlay cameras need to output to the target described in the base camera while doing camera stack. ref CameraData cameraData = ref renderingData.cameraData; RenderTargetIdentifier cameraTarget = (cameraData.targetTexture != null) ? new RenderTargetIdentifier(cameraData.targetTexture) : BuiltinRenderTextureType.CameraTarget; #if ENABLE_VR && ENABLE_XR_MODULE if (cameraData.xr.enabled) { if (cameraData.xr.singlePassEnabled) { cameraTarget = cameraData.xr.renderTarget; } else { int depthSlice = cameraData.xr.GetTextureArraySlice(); cameraTarget = new RenderTargetIdentifier(cameraData.xr.renderTarget, 0, CubemapFace.Unknown, depthSlice); } } #endif return cameraTarget; } /// /// Re-allocate fixed-size RTHandle if it is not allocated or doesn't match the descriptor /// /// RTHandle to check (can be null) /// Descriptor for the RTHandle to match /// Filtering mode of the RTHandle. /// Addressing mode of the RTHandle. /// Set to true if the depth buffer should be used as a shadow map. /// Anisotropic filtering level. /// Bias applied to mipmaps during filtering. /// Name of the RTHandle. /// public static bool ReAllocateIfNeeded( ref RTHandle handle, in RenderTextureDescriptor descriptor, FilterMode filterMode = FilterMode.Point, TextureWrapMode wrapMode = TextureWrapMode.Repeat, bool isShadowMap = false, int anisoLevel = 1, float mipMapBias = 0, string name = "") { if (RTHandleNeedsReAlloc(handle, descriptor, filterMode, wrapMode, isShadowMap, anisoLevel, mipMapBias, name, false)) { handle?.Release(); handle = RTHandles.Alloc(descriptor, filterMode, wrapMode, isShadowMap, anisoLevel, mipMapBias, name); return true; } return false; } /// /// Re-allocate dynamically resized RTHandle if it is not allocated or doesn't match the descriptor /// /// RTHandle to check (can be null) /// Constant scale for the RTHandle size computation. /// Descriptor for the RTHandle to match /// Filtering mode of the RTHandle. /// Addressing mode of the RTHandle. /// Set to true if the depth buffer should be used as a shadow map. /// Anisotropic filtering level. /// Bias applied to mipmaps during filtering. /// Name of the RTHandle. /// If the RTHandle should be re-allocated public static bool ReAllocateIfNeeded( ref RTHandle handle, Vector2 scaleFactor, in RenderTextureDescriptor descriptor, FilterMode filterMode = FilterMode.Point, TextureWrapMode wrapMode = TextureWrapMode.Repeat, bool isShadowMap = false, int anisoLevel = 1, float mipMapBias = 0, string name = "") { var usingConstantScale = handle != null && handle.useScaling && handle.scaleFactor == scaleFactor; if (!usingConstantScale || RTHandleNeedsReAlloc(handle, descriptor, filterMode, wrapMode, isShadowMap, anisoLevel, mipMapBias, name, true)) { handle?.Release(); handle = RTHandles.Alloc(scaleFactor, descriptor, filterMode, wrapMode, isShadowMap, anisoLevel, mipMapBias, name); return true; } return false; } /// /// Re-allocate dynamically resized RTHandle if it is not allocated or doesn't match the descriptor /// /// RTHandle to check (can be null) /// Function used for the RTHandle size computation. /// Descriptor for the RTHandle to match /// Filtering mode of the RTHandle. /// Addressing mode of the RTHandle. /// Set to true if the depth buffer should be used as a shadow map. /// Anisotropic filtering level. /// Bias applied to mipmaps during filtering. /// Name of the RTHandle. /// If an allocation was done public static bool ReAllocateIfNeeded( ref RTHandle handle, ScaleFunc scaleFunc, in RenderTextureDescriptor descriptor, FilterMode filterMode = FilterMode.Point, TextureWrapMode wrapMode = TextureWrapMode.Repeat, bool isShadowMap = false, int anisoLevel = 1, float mipMapBias = 0, string name = "") { var usingScaleFunction = handle != null && handle.useScaling && handle.scaleFactor == Vector2.zero; if (!usingScaleFunction || RTHandleNeedsReAlloc(handle, descriptor, filterMode, wrapMode, isShadowMap, anisoLevel, mipMapBias, name, true)) { handle?.Release(); handle = RTHandles.Alloc(scaleFunc, descriptor, filterMode, wrapMode, isShadowMap, anisoLevel, mipMapBias, name); return true; } return false; } /// /// Creates DrawingSettings based on current the rendering state. /// /// Shader pass tag to render. /// Current rendering state. /// Criteria to sort objects being rendered. /// /// static public DrawingSettings CreateDrawingSettings(ShaderTagId shaderTagId, ref RenderingData renderingData, SortingCriteria sortingCriteria) { Camera camera = renderingData.cameraData.camera; SortingSettings sortingSettings = new SortingSettings(camera) { criteria = sortingCriteria }; DrawingSettings settings = new DrawingSettings(shaderTagId, sortingSettings) { perObjectData = renderingData.perObjectData, mainLightIndex = renderingData.lightData.mainLightIndex, enableDynamicBatching = renderingData.supportsDynamicBatching, // Disable instancing for preview cameras. This is consistent with the built-in forward renderer. Also fixes case 1127324. enableInstancing = camera.cameraType == CameraType.Preview ? false : true, }; return settings; } /// /// Creates DrawingSettings based on current rendering state. /// /// /// List of shader pass tag to render. /// Current rendering state. /// Criteria to sort objects being rendered. /// /// static public DrawingSettings CreateDrawingSettings(List shaderTagIdList, ref RenderingData renderingData, SortingCriteria sortingCriteria) { if (shaderTagIdList == null || shaderTagIdList.Count == 0) { Debug.LogWarning("ShaderTagId list is invalid. DrawingSettings is created with default pipeline ShaderTagId"); return CreateDrawingSettings(new ShaderTagId("UniversalPipeline"), ref renderingData, sortingCriteria); } DrawingSettings settings = CreateDrawingSettings(shaderTagIdList[0], ref renderingData, sortingCriteria); for (int i = 1; i < shaderTagIdList.Count; ++i) settings.SetShaderPassName(i, shaderTagIdList[i]); return settings; } } }