using System; using System.Diagnostics; using UnityEngine.Rendering; namespace UnityEngine.Experimental.Rendering.RenderGraphModule { /// /// Texture resource handle. /// [DebuggerDisplay("Texture ({handle.index})")] public struct TextureHandle { private static TextureHandle s_NullHandle = new TextureHandle(); /// /// Returns a null texture handle /// /// A null texture handle. public static TextureHandle nullHandle { get { return s_NullHandle; } } internal ResourceHandle handle; internal ResourceHandle fallBackResource; internal TextureHandle(int handle, bool shared = false) { this.handle = new ResourceHandle(handle, RenderGraphResourceType.Texture, shared); fallBackResource = s_NullHandle.handle; } /// /// Cast to RenderTargetIdentifier /// /// Input TextureHandle. /// Resource as a RenderTargetIdentifier. public static implicit operator RenderTargetIdentifier(TextureHandle texture) => texture.IsValid() ? RenderGraphResourceRegistry.current.GetTexture(texture) : default(RenderTargetIdentifier); /// /// Cast to Texture /// /// Input TextureHandle. /// Resource as a Texture. public static implicit operator Texture(TextureHandle texture) => texture.IsValid() ? RenderGraphResourceRegistry.current.GetTexture(texture) : null; /// /// Cast to RenderTexture /// /// Input TextureHandle. /// Resource as a RenderTexture. public static implicit operator RenderTexture(TextureHandle texture) => texture.IsValid() ? RenderGraphResourceRegistry.current.GetTexture(texture) : null; /// /// Cast to RTHandle /// /// Input TextureHandle. /// Resource as a RTHandle. public static implicit operator RTHandle(TextureHandle texture) => texture.IsValid() ? RenderGraphResourceRegistry.current.GetTexture(texture) : null; /// /// Return true if the handle is valid. /// /// True if the handle is valid. public bool IsValid() => handle.IsValid(); public void SetFallBackResource(TextureHandle texture) { fallBackResource = texture.handle; } } /// /// The mode that determines the size of a Texture. /// public enum TextureSizeMode { ///Explicit size. Explicit, ///Size automatically scaled by a Vector. Scale, ///Size automatically scaled by a Functor. Functor } #if UNITY_2020_2_OR_NEWER /// /// Subset of the texture desc containing information for fast memory allocation (when platform supports it) /// public struct FastMemoryDesc { ///Whether the texture will be in fast memory. public bool inFastMemory; ///Flag to determine what parts of the render target is spilled if not fully resident in fast memory. public FastMemoryFlags flags; ///How much of the render target is to be switched into fast memory (between 0 and 1). public float residencyFraction; } #endif /// /// Descriptor used to create texture resources /// public struct TextureDesc { ///Texture sizing mode. public TextureSizeMode sizeMode; ///Texture width. public int width; ///Texture height. public int height; ///Number of texture slices.. public int slices; ///Texture scale. public Vector2 scale; ///Texture scale function. public ScaleFunc func; ///Depth buffer bit depth. public DepthBits depthBufferBits; ///Color format. public GraphicsFormat colorFormat; ///Filtering mode. public FilterMode filterMode; ///Addressing mode. public TextureWrapMode wrapMode; ///Texture dimension. public TextureDimension dimension; ///Enable random UAV read/write on the texture. public bool enableRandomWrite; ///Texture needs mip maps. public bool useMipMap; ///Automatically generate mip maps. public bool autoGenerateMips; ///Texture is a shadow map. public bool isShadowMap; ///Anisotropic filtering level. public int anisoLevel; ///Mip map bias. public float mipMapBias; ///Number of MSAA samples. public MSAASamples msaaSamples; ///Bind texture multi sampled. public bool bindTextureMS; ///Texture uses dynamic scaling. public bool useDynamicScale; ///Memory less flag. public RenderTextureMemoryless memoryless; ///Texture name. public string name; #if UNITY_2020_2_OR_NEWER ///Descriptor to determine how the texture will be in fast memory on platform that supports it. public FastMemoryDesc fastMemoryDesc; #endif ///Determines whether the texture will fallback to a black texture if it is read without ever writing to it. public bool fallBackToBlackTexture; // Initial state. Those should not be used in the hash ///Texture needs to be cleared on first use. public bool clearBuffer; ///Clear color. public Color clearColor; void InitDefaultValues(bool dynamicResolution, bool xrReady) { useDynamicScale = dynamicResolution; // XR Ready if (xrReady) { slices = TextureXR.slices; dimension = TextureXR.dimension; } else { slices = 1; dimension = TextureDimension.Tex2D; } } /// /// TextureDesc constructor for a texture using explicit size /// /// Texture width /// Texture height /// Use dynamic resolution /// Set this to true if the Texture is a render texture in an XR setting. public TextureDesc(int width, int height, bool dynamicResolution = false, bool xrReady = false) : this() { // Size related init sizeMode = TextureSizeMode.Explicit; this.width = width; this.height = height; // Important default values not handled by zero construction in this() msaaSamples = MSAASamples.None; InitDefaultValues(dynamicResolution, xrReady); } /// /// TextureDesc constructor for a texture using a fixed scaling /// /// RTHandle scale used for this texture /// Use dynamic resolution /// Set this to true if the Texture is a render texture in an XR setting. public TextureDesc(Vector2 scale, bool dynamicResolution = false, bool xrReady = false) : this() { // Size related init sizeMode = TextureSizeMode.Scale; this.scale = scale; // Important default values not handled by zero construction in this() msaaSamples = MSAASamples.None; dimension = TextureDimension.Tex2D; InitDefaultValues(dynamicResolution, xrReady); } /// /// TextureDesc constructor for a texture using a functor for scaling /// /// Function used to determnine the texture size /// Use dynamic resolution /// Set this to true if the Texture is a render texture in an XR setting. public TextureDesc(ScaleFunc func, bool dynamicResolution = false, bool xrReady = false) : this() { // Size related init sizeMode = TextureSizeMode.Functor; this.func = func; // Important default values not handled by zero construction in this() msaaSamples = MSAASamples.None; dimension = TextureDimension.Tex2D; InitDefaultValues(dynamicResolution, xrReady); } /// /// Copy constructor /// /// public TextureDesc(TextureDesc input) { this = input; } /// /// Hash function /// /// The texture descriptor hash. public override int GetHashCode() { int hashCode = 17; unchecked { switch (sizeMode) { case TextureSizeMode.Explicit: hashCode = hashCode * 23 + width; hashCode = hashCode * 23 + height; break; case TextureSizeMode.Functor: if (func != null) hashCode = hashCode * 23 + func.GetHashCode(); break; case TextureSizeMode.Scale: hashCode = hashCode * 23 + scale.x.GetHashCode(); hashCode = hashCode * 23 + scale.y.GetHashCode(); break; } hashCode = hashCode * 23 + mipMapBias.GetHashCode(); hashCode = hashCode * 23 + slices; hashCode = hashCode * 23 + (int)depthBufferBits; hashCode = hashCode * 23 + (int)colorFormat; hashCode = hashCode * 23 + (int)filterMode; hashCode = hashCode * 23 + (int)wrapMode; hashCode = hashCode * 23 + (int)dimension; hashCode = hashCode * 23 + (int)memoryless; hashCode = hashCode * 23 + anisoLevel; hashCode = hashCode * 23 + (enableRandomWrite ? 1 : 0); hashCode = hashCode * 23 + (useMipMap ? 1 : 0); hashCode = hashCode * 23 + (autoGenerateMips ? 1 : 0); hashCode = hashCode * 23 + (isShadowMap ? 1 : 0); hashCode = hashCode * 23 + (bindTextureMS ? 1 : 0); hashCode = hashCode * 23 + (useDynamicScale ? 1 : 0); hashCode = hashCode * 23 + (int)msaaSamples; #if UNITY_2020_2_OR_NEWER hashCode = hashCode * 23 + (fastMemoryDesc.inFastMemory ? 1 : 0); #endif } return hashCode; } } [DebuggerDisplay("TextureResource ({desc.name})")] class TextureResource : RenderGraphResource { static int m_TextureCreationIndex; public override string GetName() { if (imported && !shared) return graphicsResource != null ? graphicsResource.name : "null resource"; else return desc.name; } // NOTE: // Next two functions should have been implemented in RenderGraphResource but for some reason, // when doing so, it's impossible to break in the Texture version of the virtual function (with VS2017 at least), making this completely un-debuggable. // To work around this, we just copy/pasted the implementation in each final class... public override void CreatePooledGraphicsResource() { Debug.Assert(m_Pool != null, "CreatePooledGraphicsResource should only be called for regular pooled resources"); int hashCode = desc.GetHashCode(); if (graphicsResource != null) throw new InvalidOperationException(string.Format("Trying to create an already created resource ({0}). Resource was probably declared for writing more than once in the same pass.", GetName())); var pool = m_Pool as TexturePool; if (!pool.TryGetResource(hashCode, out graphicsResource)) { CreateGraphicsResource(); } cachedHash = hashCode; pool.RegisterFrameAllocation(cachedHash, graphicsResource); } public override void ReleasePooledGraphicsResource(int frameIndex) { if (graphicsResource == null) throw new InvalidOperationException($"Tried to release a resource ({GetName()}) that was never created. Check that there is at least one pass writing to it first."); // Shared resources don't use the pool var pool = m_Pool as TexturePool; if (pool != null) { pool.ReleaseResource(cachedHash, graphicsResource, frameIndex); pool.UnregisterFrameAllocation(cachedHash, graphicsResource); } Reset(null); } public override void CreateGraphicsResource(string name = "") { // Textures are going to be reused under different aliases along the frame so we can't provide a specific name upon creation. // The name in the desc is going to be used for debugging purpose and render graph visualization. if (name == "") name = $"RenderGraphTexture_{m_TextureCreationIndex++}"; switch (desc.sizeMode) { case TextureSizeMode.Explicit: graphicsResource = RTHandles.Alloc(desc.width, desc.height, desc.slices, desc.depthBufferBits, desc.colorFormat, desc.filterMode, desc.wrapMode, desc.dimension, desc.enableRandomWrite, desc.useMipMap, desc.autoGenerateMips, desc.isShadowMap, desc.anisoLevel, desc.mipMapBias, desc.msaaSamples, desc.bindTextureMS, desc.useDynamicScale, desc.memoryless, name); break; case TextureSizeMode.Scale: graphicsResource = RTHandles.Alloc(desc.scale, desc.slices, desc.depthBufferBits, desc.colorFormat, desc.filterMode, desc.wrapMode, desc.dimension, desc.enableRandomWrite, desc.useMipMap, desc.autoGenerateMips, desc.isShadowMap, desc.anisoLevel, desc.mipMapBias, desc.msaaSamples, desc.bindTextureMS, desc.useDynamicScale, desc.memoryless, name); break; case TextureSizeMode.Functor: graphicsResource = RTHandles.Alloc(desc.func, desc.slices, desc.depthBufferBits, desc.colorFormat, desc.filterMode, desc.wrapMode, desc.dimension, desc.enableRandomWrite, desc.useMipMap, desc.autoGenerateMips, desc.isShadowMap, desc.anisoLevel, desc.mipMapBias, desc.msaaSamples, desc.bindTextureMS, desc.useDynamicScale, desc.memoryless, name); break; } } public override void ReleaseGraphicsResource() { if (graphicsResource != null) graphicsResource.Release(); base.ReleaseGraphicsResource(); } public override void LogCreation(RenderGraphLogger logger) { logger.LogLine($"Created Texture: {desc.name} (Cleared: {desc.clearBuffer})"); } public override void LogRelease(RenderGraphLogger logger) { logger.LogLine($"Released Texture: {desc.name}"); } } class TexturePool : RenderGraphResourcePool { protected override void ReleaseInternalResource(RTHandle res) { res.Release(); } protected override string GetResourceName(RTHandle res) { return res.rt.name; } protected override long GetResourceSize(RTHandle res) { return Profiling.Profiler.GetRuntimeMemorySizeLong(res.rt); } override protected string GetResourceTypeName() { return "Texture"; } override protected int GetSortIndex(RTHandle res) { return res.GetInstanceID(); } // Another C# nicety. // We need to re-implement the whole thing every time because: // - obj.resource.Release is Type specific so it cannot be called on a generic (and there's no shared interface for resources like RTHandle, ComputeBuffers etc) // - We can't use a virtual release function because it will capture 'this' in the lambda for RemoveAll generating GCAlloc in the process. override public void PurgeUnusedResources(int currentFrameIndex) { // Update the frame index for the lambda. Static because we don't want to capture. s_CurrentFrameIndex = currentFrameIndex; m_RemoveList.Clear(); foreach (var kvp in m_ResourcePool) { // WARNING: No foreach here. Sorted list GetEnumerator generates garbage... var list = kvp.Value; var keys = list.Keys; var values = list.Values; for (int i = 0; i < list.Count; ++i) { var value = values[i]; if (ShouldReleaseResource(value.frameIndex, s_CurrentFrameIndex)) { value.resource.Release(); m_RemoveList.Add(keys[i]); } } foreach (var key in m_RemoveList) list.Remove(key); } } } }