using System;
using System.Collections.Generic;
using System.Diagnostics;
using UnityEngine.Experimental.Rendering;
using UnityEngine.Experimental.Rendering.RenderGraphModule;
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 TextureDesc descriptor,
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
(DepthBits)handle.rt.descriptor.depthBufferBits != descriptor.depthBufferBits ||
(handle.rt.descriptor.depthBufferBits == (int)DepthBits.None && !descriptor.isShadowMap && handle.rt.descriptor.graphicsFormat != descriptor.colorFormat) ||
handle.rt.descriptor.dimension != descriptor.dimension ||
handle.rt.descriptor.enableRandomWrite != descriptor.enableRandomWrite ||
handle.rt.descriptor.useMipMap != descriptor.useMipMap ||
handle.rt.descriptor.autoGenerateMips != descriptor.autoGenerateMips ||
(MSAASamples)handle.rt.descriptor.msaaSamples != descriptor.msaaSamples ||
handle.rt.descriptor.bindMS != descriptor.bindTextureMS ||
handle.rt.descriptor.useDynamicScale != descriptor.useDynamicScale ||
handle.rt.descriptor.memoryless != descriptor.memoryless ||
handle.rt.filterMode != descriptor.filterMode ||
handle.rt.wrapMode != descriptor.wrapMode ||
handle.rt.anisoLevel != descriptor.anisoLevel ||
handle.rt.mipMapBias != descriptor.mipMapBias ||
handle.name != descriptor.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 = "")
{
TextureDesc requestRTDesc = RTHandleResourcePool.CreateTextureDesc(descriptor, TextureSizeMode.Explicit, anisoLevel, 0, filterMode, wrapMode, name);
if (RTHandleNeedsReAlloc(handle, requestRTDesc, false))
{
if (handle != null && handle.rt != null)
{
TextureDesc currentRTDesc = RTHandleResourcePool.CreateTextureDesc(handle.rt.descriptor, TextureSizeMode.Explicit, handle.rt.anisoLevel, handle.rt.mipMapBias, handle.rt.filterMode, handle.rt.wrapMode, handle.name);
AddStaleResourceToPoolOrRelease(currentRTDesc, handle);
}
if (UniversalRenderPipeline.s_RTHandlePool.TryGetResource(requestRTDesc, out handle))
{
return true;
}
else
{
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;
TextureDesc requestRTDesc = RTHandleResourcePool.CreateTextureDesc(descriptor, TextureSizeMode.Scale, anisoLevel, 0, filterMode, wrapMode);
if (!usingConstantScale || RTHandleNeedsReAlloc(handle, requestRTDesc, true))
{
if (handle != null && handle.rt != null)
{
TextureDesc currentRTDesc = RTHandleResourcePool.CreateTextureDesc(handle.rt.descriptor, TextureSizeMode.Scale, handle.rt.anisoLevel, handle.rt.mipMapBias, handle.rt.filterMode, handle.rt.wrapMode);
AddStaleResourceToPoolOrRelease(currentRTDesc, handle);
}
if (UniversalRenderPipeline.s_RTHandlePool.TryGetResource(requestRTDesc, out handle))
{
return true;
}
else
{
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;
TextureDesc requestRTDesc = RTHandleResourcePool.CreateTextureDesc(descriptor, TextureSizeMode.Functor, anisoLevel, 0, filterMode, wrapMode);
if (!usingScaleFunction || RTHandleNeedsReAlloc(handle, requestRTDesc, true))
{
if (handle != null && handle.rt != null)
{
TextureDesc currentRTDesc = RTHandleResourcePool.CreateTextureDesc(handle.rt.descriptor, TextureSizeMode.Functor, handle.rt.anisoLevel, handle.rt.mipMapBias, handle.rt.filterMode, handle.rt.wrapMode);
AddStaleResourceToPoolOrRelease(currentRTDesc, handle);
}
if (UniversalRenderPipeline.s_RTHandlePool.TryGetResource(requestRTDesc, out handle))
{
return true;
}
else
{
handle = RTHandles.Alloc(scaleFunc, descriptor, filterMode, wrapMode, isShadowMap, anisoLevel, mipMapBias, name);
return true;
}
}
return false;
}
///
/// Resize the rthandle pool's max stale resource capacity. The default value is 32.
/// Increasing the capacity may have a negative impact on the memory usage(dued to staled resources in pool).
/// Increasing the capacity may improve runtime performance (by reducing the runtime RTHandle realloc count in multi view/multi camera setup).
/// Setting capacity will purge the current pool. It is recommended to setup the capacity upfront and not changing it during the runtime.
///
/// Max capacity to set
/// Return true if set successfully. Return false if URP is not initialized and pool does not exist yet.
public static bool SetMaxRTHandlePoolCapacity(int capacity)
{
if (UniversalRenderPipeline.s_RTHandlePool == null)
return false;
UniversalRenderPipeline.s_RTHandlePool.staleResourceCapacity = capacity;
return true;
}
///
/// Add stale rtHandle to pool so that it could be reused in the future.
/// For stale rtHandle failed to add to pool(could happen when pool is reaching its max stale resource capacity), the stale resource will be released.
///
internal static void AddStaleResourceToPoolOrRelease(TextureDesc desc, RTHandle handle)
{
if (!UniversalRenderPipeline.s_RTHandlePool.AddResourceToPool(desc, handle, Time.frameCount))
RTHandles.Release(handle);
}
///
/// 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;
}
}
}