RTHandleSystem.cs 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791
  1. using System;
  2. using System.Collections.Generic;
  3. using UnityEngine.Assertions;
  4. using UnityEngine.Experimental.Rendering;
  5. namespace UnityEngine.Rendering
  6. {
  7. /// <summary>
  8. /// Scaled function used to compute the size of a RTHandle for the current frame.
  9. /// </summary>
  10. /// <param name="size">Reference size of the RTHandle system for the frame.</param>
  11. /// <returns>The size of the RTHandled computed from the reference size.</returns>
  12. public delegate Vector2Int ScaleFunc(Vector2Int size);
  13. /// <summary>
  14. /// List of properties of the RTHandle System for the current frame.
  15. /// </summary>
  16. public struct RTHandleProperties
  17. {
  18. /// <summary>
  19. /// Size set as reference at the previous frame
  20. /// </summary>
  21. public Vector2Int previousViewportSize;
  22. /// <summary>
  23. /// Size of the render targets at the previous frame
  24. /// </summary>
  25. public Vector2Int previousRenderTargetSize;
  26. /// <summary>
  27. /// Size set as reference at the current frame
  28. /// </summary>
  29. public Vector2Int currentViewportSize;
  30. /// <summary>
  31. /// Size of the render targets at the current frame
  32. /// </summary>
  33. public Vector2Int currentRenderTargetSize;
  34. /// <summary>
  35. /// Scale factor from RTHandleSystem max size to requested reference size (referenceSize/maxSize)
  36. /// (x,y) current frame (z,w) last frame (this is only used for buffered RTHandle Systems)
  37. /// </summary>
  38. public Vector4 rtHandleScale;
  39. }
  40. /// <summary>
  41. /// System managing a set of RTHandle textures
  42. /// </summary>
  43. public partial class RTHandleSystem : IDisposable
  44. {
  45. internal enum ResizeMode
  46. {
  47. Auto,
  48. OnDemand
  49. }
  50. // Parameters for auto-scaled Render Textures
  51. bool m_HardwareDynamicResRequested = false;
  52. bool m_ScaledRTSupportsMSAA = false;
  53. MSAASamples m_ScaledRTCurrentMSAASamples = MSAASamples.None;
  54. HashSet<RTHandle> m_AutoSizedRTs;
  55. RTHandle[] m_AutoSizedRTsArray; // For fast iteration
  56. HashSet<RTHandle> m_ResizeOnDemandRTs;
  57. RTHandleProperties m_RTHandleProperties;
  58. /// <summary>
  59. /// Current properties of the RTHandle System.
  60. /// </summary>
  61. public RTHandleProperties rtHandleProperties { get { return m_RTHandleProperties; } }
  62. int m_MaxWidths = 0;
  63. int m_MaxHeights = 0;
  64. /// <summary>
  65. /// RTHandleSystem constructor.
  66. /// </summary>
  67. public RTHandleSystem()
  68. {
  69. m_AutoSizedRTs = new HashSet<RTHandle>();
  70. m_ResizeOnDemandRTs = new HashSet<RTHandle>();
  71. m_MaxWidths = 1;
  72. m_MaxHeights = 1;
  73. }
  74. /// <summary>
  75. /// Disposable pattern implementation
  76. /// </summary>
  77. public void Dispose()
  78. {
  79. Dispose(true);
  80. }
  81. /// <summary>
  82. /// Initialize the RTHandle system.
  83. /// </summary>
  84. /// <param name="width">Initial reference rendering width.</param>
  85. /// <param name="height">Initial reference rendering height.</param>
  86. /// <param name="scaledRTsupportsMSAA">Set to true if automatically scaled RTHandles should support MSAA</param>
  87. /// <param name="scaledRTMSAASamples">Number of MSAA samples for automatically scaled RTHandles.</param>
  88. public void Initialize(int width, int height, bool scaledRTsupportsMSAA, MSAASamples scaledRTMSAASamples)
  89. {
  90. Debug.Assert(m_AutoSizedRTs.Count == 0, "RTHandle.Initialize should only be called once before allocating any Render Texture. This may be caused by an unreleased RTHandle resource.");
  91. m_MaxWidths = width;
  92. m_MaxHeights = height;
  93. m_ScaledRTSupportsMSAA = scaledRTsupportsMSAA;
  94. m_ScaledRTCurrentMSAASamples = scaledRTMSAASamples;
  95. m_HardwareDynamicResRequested = DynamicResolutionHandler.instance.RequestsHardwareDynamicResolution();
  96. }
  97. /// <summary>
  98. /// Release memory of a RTHandle from the RTHandle System
  99. /// </summary>
  100. /// <param name="rth">RTHandle that should be released.</param>
  101. public void Release(RTHandle rth)
  102. {
  103. if (rth != null)
  104. {
  105. Assert.AreEqual(this, rth.m_Owner);
  106. rth.Release();
  107. }
  108. }
  109. internal void Remove(RTHandle rth)
  110. {
  111. m_AutoSizedRTs.Remove(rth);
  112. }
  113. /// <summary>
  114. /// Sets the reference rendering size for subsequent rendering for the RTHandle System
  115. /// </summary>
  116. /// <param name="width">Reference rendering width for subsequent rendering.</param>
  117. /// <param name="height">Reference rendering height for subsequent rendering.</param>
  118. /// <param name="msaaSamples">Number of MSAA samples for multisampled textures for subsequent rendering.</param>
  119. public void SetReferenceSize(int width, int height, MSAASamples msaaSamples)
  120. {
  121. m_RTHandleProperties.previousViewportSize = m_RTHandleProperties.currentViewportSize;
  122. m_RTHandleProperties.previousRenderTargetSize = m_RTHandleProperties.currentRenderTargetSize;
  123. Vector2 lastFrameMaxSize = new Vector2(GetMaxWidth(), GetMaxHeight());
  124. width = Mathf.Max(width, 1);
  125. height = Mathf.Max(height, 1);
  126. bool sizeChanged = width > GetMaxWidth() || height > GetMaxHeight();
  127. bool msaaSamplesChanged = (msaaSamples != m_ScaledRTCurrentMSAASamples);
  128. if (sizeChanged || msaaSamplesChanged)
  129. {
  130. Resize(width, height, msaaSamples, sizeChanged, msaaSamplesChanged);
  131. }
  132. m_RTHandleProperties.currentViewportSize = new Vector2Int(width, height);
  133. m_RTHandleProperties.currentRenderTargetSize = new Vector2Int(GetMaxWidth(), GetMaxHeight());
  134. // If the currentViewportSize is 0, it mean we are the first frame of rendering (can happen when doing domain reload for example or for reflection probe)
  135. // in this case the scalePrevious below could be invalided. But some effect rely on having a correct value like TAA with the history buffer for the first frame.
  136. // to work around this, when we detect that size is 0, we setup previous size to current size.
  137. if (m_RTHandleProperties.previousViewportSize.x == 0)
  138. {
  139. m_RTHandleProperties.previousViewportSize = m_RTHandleProperties.currentViewportSize;
  140. m_RTHandleProperties.previousRenderTargetSize = m_RTHandleProperties.currentRenderTargetSize;
  141. lastFrameMaxSize = new Vector2(GetMaxWidth(), GetMaxHeight());
  142. }
  143. if (DynamicResolutionHandler.instance.HardwareDynamicResIsEnabled())
  144. {
  145. m_RTHandleProperties.rtHandleScale = new Vector4(1.0f, 1.0f, 1.0f, 1.0f);
  146. }
  147. else
  148. {
  149. Vector2 maxSize = new Vector2(GetMaxWidth(), GetMaxHeight());
  150. Vector2 scaleCurrent = m_RTHandleProperties.currentViewportSize / maxSize;
  151. Vector2 scalePrevious = m_RTHandleProperties.previousViewportSize / lastFrameMaxSize;
  152. m_RTHandleProperties.rtHandleScale = new Vector4(scaleCurrent.x, scaleCurrent.y, scalePrevious.x, scalePrevious.y);
  153. }
  154. }
  155. /// <summary>
  156. /// Enable or disable hardware dynamic resolution for the RTHandle System
  157. /// </summary>
  158. /// <param name="enableHWDynamicRes">State of hardware dynamic resolution.</param>
  159. public void SetHardwareDynamicResolutionState(bool enableHWDynamicRes)
  160. {
  161. if(enableHWDynamicRes != m_HardwareDynamicResRequested && m_AutoSizedRTsArray != null)
  162. {
  163. m_HardwareDynamicResRequested = enableHWDynamicRes;
  164. for (int i = 0, c = m_AutoSizedRTsArray.Length; i < c; ++i)
  165. {
  166. var rth = m_AutoSizedRTsArray[i];
  167. // Grab the render texture
  168. var renderTexture = rth.m_RT;
  169. if(renderTexture)
  170. {
  171. // Free the previous version
  172. renderTexture.Release();
  173. renderTexture.useDynamicScale = m_HardwareDynamicResRequested && rth.m_EnableHWDynamicScale;
  174. // Create the render texture
  175. renderTexture.Create();
  176. }
  177. }
  178. }
  179. }
  180. internal void SwitchResizeMode(RTHandle rth, ResizeMode mode)
  181. {
  182. // Don't do anything is scaling isn't enabled on this RT
  183. // TODO: useScaling should probably be moved to ResizeMode.Fixed or something
  184. if (!rth.useScaling)
  185. return;
  186. switch (mode)
  187. {
  188. case ResizeMode.OnDemand:
  189. m_AutoSizedRTs.Remove(rth);
  190. m_ResizeOnDemandRTs.Add(rth);
  191. break;
  192. case ResizeMode.Auto:
  193. // Resize now so it is consistent with other auto resize RTHs
  194. if (m_ResizeOnDemandRTs.Contains(rth))
  195. DemandResize(rth);
  196. m_ResizeOnDemandRTs.Remove(rth);
  197. m_AutoSizedRTs.Add(rth);
  198. break;
  199. }
  200. }
  201. void DemandResize(RTHandle rth)
  202. {
  203. Assert.IsTrue(m_ResizeOnDemandRTs.Contains(rth), "The RTHandle is not an resize on demand handle in this RTHandleSystem. Please call SwitchToResizeOnDemand(rth, true) before resizing on demand.");
  204. // Grab the render texture
  205. var rt = rth.m_RT;
  206. rth.referenceSize = new Vector2Int(m_MaxWidths, m_MaxHeights);
  207. var scaledSize = rth.GetScaledSize(rth.referenceSize);
  208. scaledSize = Vector2Int.Max(Vector2Int.one, scaledSize);
  209. // Did the size change?
  210. var sizeChanged = rt.width != scaledSize.x || rt.height != scaledSize.y;
  211. // If this is an MSAA texture, did the sample count change?
  212. var msaaSampleChanged = rth.m_EnableMSAA && rt.antiAliasing != (int)m_ScaledRTCurrentMSAASamples;
  213. if (sizeChanged || msaaSampleChanged)
  214. {
  215. // Free this render texture
  216. rt.Release();
  217. // Update the antialiasing count
  218. if (rth.m_EnableMSAA)
  219. rt.antiAliasing = (int)m_ScaledRTCurrentMSAASamples;
  220. // Update the size
  221. rt.width = scaledSize.x;
  222. rt.height = scaledSize.y;
  223. // Generate a new name
  224. rt.name = CoreUtils.GetRenderTargetAutoName(
  225. rt.width,
  226. rt.height,
  227. rt.volumeDepth,
  228. rt.format,
  229. rth.m_Name,
  230. mips: rt.useMipMap,
  231. enableMSAA: rth.m_EnableMSAA,
  232. msaaSamples: m_ScaledRTCurrentMSAASamples
  233. );
  234. // Create the new texture
  235. rt.Create();
  236. }
  237. }
  238. /// <summary>
  239. /// Returns the maximum allocated width of the RTHandle System.
  240. /// </summary>
  241. /// <returns>Maximum allocated width of the RTHandle System.</returns>
  242. public int GetMaxWidth() { return m_MaxWidths; }
  243. /// <summary>
  244. /// Returns the maximum allocated height of the RTHandle System.
  245. /// </summary>
  246. /// <returns>Maximum allocated height of the RTHandle System.</returns>
  247. public int GetMaxHeight() { return m_MaxHeights; }
  248. void Dispose(bool disposing)
  249. {
  250. if (disposing)
  251. {
  252. Array.Resize(ref m_AutoSizedRTsArray, m_AutoSizedRTs.Count);
  253. m_AutoSizedRTs.CopyTo(m_AutoSizedRTsArray);
  254. for (int i = 0, c = m_AutoSizedRTsArray.Length; i < c; ++i)
  255. {
  256. var rt = m_AutoSizedRTsArray[i];
  257. Release(rt);
  258. }
  259. m_AutoSizedRTs.Clear();
  260. Array.Resize(ref m_AutoSizedRTsArray, m_ResizeOnDemandRTs.Count);
  261. m_ResizeOnDemandRTs.CopyTo(m_AutoSizedRTsArray);
  262. for (int i = 0, c = m_AutoSizedRTsArray.Length; i < c; ++i)
  263. {
  264. var rt = m_AutoSizedRTsArray[i];
  265. Release(rt);
  266. }
  267. m_ResizeOnDemandRTs.Clear();
  268. m_AutoSizedRTsArray = null;
  269. }
  270. }
  271. void Resize(int width, int height, MSAASamples msaaSamples, bool sizeChanged, bool msaaSampleChanged)
  272. {
  273. m_MaxWidths = Math.Max(width, m_MaxWidths);
  274. m_MaxHeights = Math.Max(height, m_MaxHeights);
  275. m_ScaledRTCurrentMSAASamples = msaaSamples;
  276. var maxSize = new Vector2Int(m_MaxWidths, m_MaxHeights);
  277. Array.Resize(ref m_AutoSizedRTsArray, m_AutoSizedRTs.Count);
  278. m_AutoSizedRTs.CopyTo(m_AutoSizedRTsArray);
  279. for (int i = 0, c = m_AutoSizedRTsArray.Length; i < c; ++i)
  280. {
  281. // Grab the RT Handle
  282. var rth = m_AutoSizedRTsArray[i];
  283. // If we are only processing MSAA sample count change, make sure this RT is an MSAA one
  284. if (!sizeChanged && msaaSampleChanged && !rth.m_EnableMSAA)
  285. {
  286. continue;
  287. }
  288. // Force its new reference size
  289. rth.referenceSize = maxSize;
  290. // Grab the render texture
  291. var renderTexture = rth.m_RT;
  292. // Free the previous version
  293. renderTexture.Release();
  294. // Get the scaled size
  295. var scaledSize = rth.GetScaledSize(maxSize);
  296. renderTexture.width = Mathf.Max(scaledSize.x, 1);
  297. renderTexture.height = Mathf.Max(scaledSize.y, 1);
  298. // If this is a msaa texture, make sure to update its msaa count
  299. if (rth.m_EnableMSAA)
  300. {
  301. renderTexture.antiAliasing = (int)m_ScaledRTCurrentMSAASamples;
  302. }
  303. // Regenerate the name
  304. renderTexture.name = CoreUtils.GetRenderTargetAutoName(renderTexture.width, renderTexture.height, renderTexture.volumeDepth, renderTexture.format, rth.m_Name, mips: renderTexture.useMipMap, enableMSAA: rth.m_EnableMSAA, msaaSamples: m_ScaledRTCurrentMSAASamples);
  305. // Create the render texture
  306. renderTexture.Create();
  307. }
  308. }
  309. /// <summary>
  310. /// Allocate a new fixed sized RTHandle.
  311. /// </summary>
  312. /// <param name="width">With of the RTHandle.</param>
  313. /// <param name="height">Heigh of the RTHandle.</param>
  314. /// <param name="slices">Number of slices of the RTHandle.</param>
  315. /// <param name="depthBufferBits">Bit depths of a depth buffer.</param>
  316. /// <param name="colorFormat">GraphicsFormat of a color buffer.</param>
  317. /// <param name="filterMode">Filtering mode of the RTHandle.</param>
  318. /// <param name="wrapMode">Addressing mode of the RTHandle.</param>
  319. /// <param name="dimension">Texture dimension of the RTHandle.</param>
  320. /// <param name="enableRandomWrite">Set to true to enable UAV random read writes on the texture.</param>
  321. /// <param name="useMipMap">Set to true if the texture should have mipmaps.</param>
  322. /// <param name="autoGenerateMips">Set to true to automatically generate mipmaps.</param>
  323. /// <param name="isShadowMap">Set to true if the depth buffer should be used as a shadow map.</param>
  324. /// <param name="anisoLevel">Anisotropic filtering level.</param>
  325. /// <param name="mipMapBias">Bias applied to mipmaps during filtering.</param>
  326. /// <param name="msaaSamples">Number of MSAA samples for the RTHandle.</param>
  327. /// <param name="bindTextureMS">Set to true if the texture needs to be bound as a multisampled texture in the shader.</param>
  328. /// <param name="useDynamicScale">Set to true to use hardware dynamic scaling.</param>
  329. /// <param name="memoryless">Use this property to set the render texture memoryless modes.</param>
  330. /// <param name="name">Name of the RTHandle.</param>
  331. /// <returns></returns>
  332. public RTHandle Alloc(
  333. int width,
  334. int height,
  335. int slices = 1,
  336. DepthBits depthBufferBits = DepthBits.None,
  337. GraphicsFormat colorFormat = GraphicsFormat.R8G8B8A8_SRGB,
  338. FilterMode filterMode = FilterMode.Point,
  339. TextureWrapMode wrapMode = TextureWrapMode.Repeat,
  340. TextureDimension dimension = TextureDimension.Tex2D,
  341. bool enableRandomWrite = false,
  342. bool useMipMap = false,
  343. bool autoGenerateMips = true,
  344. bool isShadowMap = false,
  345. int anisoLevel = 1,
  346. float mipMapBias = 0f,
  347. MSAASamples msaaSamples = MSAASamples.None,
  348. bool bindTextureMS = false,
  349. bool useDynamicScale = false,
  350. RenderTextureMemoryless memoryless = RenderTextureMemoryless.None,
  351. string name = ""
  352. )
  353. {
  354. bool enableMSAA = msaaSamples != MSAASamples.None;
  355. if (!enableMSAA && bindTextureMS == true)
  356. {
  357. Debug.LogWarning("RTHandle allocated without MSAA but with bindMS set to true, forcing bindMS to false.");
  358. bindTextureMS = false;
  359. }
  360. // We need to handle this in an explicit way since GraphicsFormat does not expose depth formats. TODO: Get rid of this branch once GraphicsFormat'll expose depth related formats
  361. RenderTexture rt;
  362. if (isShadowMap || depthBufferBits != DepthBits.None)
  363. {
  364. RenderTextureFormat format = isShadowMap ? RenderTextureFormat.Shadowmap : RenderTextureFormat.Depth;
  365. rt = new RenderTexture(width, height, (int)depthBufferBits, format, RenderTextureReadWrite.Linear)
  366. {
  367. hideFlags = HideFlags.HideAndDontSave,
  368. volumeDepth = slices,
  369. filterMode = filterMode,
  370. wrapMode = wrapMode,
  371. dimension = dimension,
  372. enableRandomWrite = enableRandomWrite,
  373. useMipMap = useMipMap,
  374. autoGenerateMips = autoGenerateMips,
  375. anisoLevel = anisoLevel,
  376. mipMapBias = mipMapBias,
  377. antiAliasing = (int)msaaSamples,
  378. bindTextureMS = bindTextureMS,
  379. useDynamicScale = m_HardwareDynamicResRequested && useDynamicScale,
  380. memorylessMode = memoryless,
  381. name = CoreUtils.GetRenderTargetAutoName(width, height, slices, format, name, mips: useMipMap, enableMSAA: enableMSAA, msaaSamples: msaaSamples)
  382. };
  383. }
  384. else
  385. {
  386. rt = new RenderTexture(width, height, (int)depthBufferBits, colorFormat)
  387. {
  388. hideFlags = HideFlags.HideAndDontSave,
  389. volumeDepth = slices,
  390. filterMode = filterMode,
  391. wrapMode = wrapMode,
  392. dimension = dimension,
  393. enableRandomWrite = enableRandomWrite,
  394. useMipMap = useMipMap,
  395. autoGenerateMips = autoGenerateMips,
  396. anisoLevel = anisoLevel,
  397. mipMapBias = mipMapBias,
  398. antiAliasing = (int)msaaSamples,
  399. bindTextureMS = bindTextureMS,
  400. useDynamicScale = m_HardwareDynamicResRequested && useDynamicScale,
  401. memorylessMode = memoryless,
  402. name = CoreUtils.GetRenderTargetAutoName(width, height, slices, GraphicsFormatUtility.GetRenderTextureFormat(colorFormat), name, mips: useMipMap, enableMSAA: enableMSAA, msaaSamples: msaaSamples)
  403. };
  404. }
  405. rt.Create();
  406. var newRT = new RTHandle(this);
  407. newRT.SetRenderTexture(rt);
  408. newRT.useScaling = false;
  409. newRT.m_EnableRandomWrite = enableRandomWrite;
  410. newRT.m_EnableMSAA = enableMSAA;
  411. newRT.m_EnableHWDynamicScale = useDynamicScale;
  412. newRT.m_Name = name;
  413. newRT.referenceSize = new Vector2Int(width, height);
  414. return newRT;
  415. }
  416. // Next two methods are used to allocate RenderTexture that depend on the frame settings (resolution and msaa for now)
  417. // RenderTextures allocated this way are meant to be defined by a scale of camera resolution (full/half/quarter resolution for example).
  418. // The idea is that internally the system will scale up the size of all render texture so that it amortizes with time and not reallocate when a smaller size is required (which is what happens with TemporaryRTs).
  419. // Since MSAA cannot be changed on the fly for a given RenderTexture, a separate instance will be created if the user requires it. This instance will be the one used after the next call of SetReferenceSize if MSAA is required.
  420. /// <summary>
  421. /// Allocate a new automatically sized RTHandle.
  422. /// </summary>
  423. /// <param name="scaleFactor">Constant scale for the RTHandle size computation.</param>
  424. /// <param name="slices">Number of slices of the RTHandle.</param>
  425. /// <param name="depthBufferBits">Bit depths of a depth buffer.</param>
  426. /// <param name="colorFormat">GraphicsFormat of a color buffer.</param>
  427. /// <param name="filterMode">Filtering mode of the RTHandle.</param>
  428. /// <param name="wrapMode">Addressing mode of the RTHandle.</param>
  429. /// <param name="dimension">Texture dimension of the RTHandle.</param>
  430. /// <param name="enableRandomWrite">Set to true to enable UAV random read writes on the texture.</param>
  431. /// <param name="useMipMap">Set to true if the texture should have mipmaps.</param>
  432. /// <param name="autoGenerateMips">Set to true to automatically generate mipmaps.</param>
  433. /// <param name="isShadowMap">Set to true if the depth buffer should be used as a shadow map.</param>
  434. /// <param name="anisoLevel">Anisotropic filtering level.</param>
  435. /// <param name="mipMapBias">Bias applied to mipmaps during filtering.</param>
  436. /// <param name="enableMSAA">Enable MSAA for this RTHandle.</param>
  437. /// <param name="bindTextureMS">Set to true if the texture needs to be bound as a multisampled texture in the shader.</param>
  438. /// <param name="useDynamicScale">Set to true to use hardware dynamic scaling.</param>
  439. /// <param name="memoryless">Use this property to set the render texture memoryless modes.</param>
  440. /// <param name="name">Name of the RTHandle.</param>
  441. /// <returns></returns>
  442. public RTHandle Alloc(
  443. Vector2 scaleFactor,
  444. int slices = 1,
  445. DepthBits depthBufferBits = DepthBits.None,
  446. GraphicsFormat colorFormat = GraphicsFormat.R8G8B8A8_SRGB,
  447. FilterMode filterMode = FilterMode.Point,
  448. TextureWrapMode wrapMode = TextureWrapMode.Repeat,
  449. TextureDimension dimension = TextureDimension.Tex2D,
  450. bool enableRandomWrite = false,
  451. bool useMipMap = false,
  452. bool autoGenerateMips = true,
  453. bool isShadowMap = false,
  454. int anisoLevel = 1,
  455. float mipMapBias = 0f,
  456. bool enableMSAA = false,
  457. bool bindTextureMS = false,
  458. bool useDynamicScale = false,
  459. RenderTextureMemoryless memoryless = RenderTextureMemoryless.None,
  460. string name = ""
  461. )
  462. {
  463. // If an MSAA target is requested, make sure the support was on
  464. if (enableMSAA)
  465. Debug.Assert(m_ScaledRTSupportsMSAA);
  466. int width = Mathf.Max(Mathf.RoundToInt(scaleFactor.x * GetMaxWidth()), 1);
  467. int height = Mathf.Max(Mathf.RoundToInt(scaleFactor.y * GetMaxHeight()), 1);
  468. var rth = AllocAutoSizedRenderTexture(width,
  469. height,
  470. slices,
  471. depthBufferBits,
  472. colorFormat,
  473. filterMode,
  474. wrapMode,
  475. dimension,
  476. enableRandomWrite,
  477. useMipMap,
  478. autoGenerateMips,
  479. isShadowMap,
  480. anisoLevel,
  481. mipMapBias,
  482. enableMSAA,
  483. bindTextureMS,
  484. useDynamicScale,
  485. memoryless,
  486. name
  487. );
  488. rth.referenceSize = new Vector2Int(width, height);
  489. rth.scaleFactor = scaleFactor;
  490. return rth;
  491. }
  492. //
  493. // You can provide your own scaling function for advanced scaling schemes (e.g. scaling to
  494. // the next POT). The function takes a Vec2 as parameter that holds max width & height
  495. // values for the current manager context and returns a Vec2 of the final size in pixels.
  496. //
  497. // var rth = Alloc(
  498. // size => new Vector2Int(size.x / 2, size.y),
  499. // [...]
  500. // );
  501. //
  502. /// <summary>
  503. /// Allocate a new automatically sized RTHandle.
  504. /// </summary>
  505. /// <param name="scaleFunc">Function used for the RTHandle size computation.</param>
  506. /// <param name="slices">Number of slices of the RTHandle.</param>
  507. /// <param name="depthBufferBits">Bit depths of a depth buffer.</param>
  508. /// <param name="colorFormat">GraphicsFormat of a color buffer.</param>
  509. /// <param name="filterMode">Filtering mode of the RTHandle.</param>
  510. /// <param name="wrapMode">Addressing mode of the RTHandle.</param>
  511. /// <param name="dimension">Texture dimension of the RTHandle.</param>
  512. /// <param name="enableRandomWrite">Set to true to enable UAV random read writes on the texture.</param>
  513. /// <param name="useMipMap">Set to true if the texture should have mipmaps.</param>
  514. /// <param name="autoGenerateMips">Set to true to automatically generate mipmaps.</param>
  515. /// <param name="isShadowMap">Set to true if the depth buffer should be used as a shadow map.</param>
  516. /// <param name="anisoLevel">Anisotropic filtering level.</param>
  517. /// <param name="mipMapBias">Bias applied to mipmaps during filtering.</param>
  518. /// <param name="enableMSAA">Enable MSAA for this RTHandle.</param>
  519. /// <param name="bindTextureMS">Set to true if the texture needs to be bound as a multisampled texture in the shader.</param>
  520. /// <param name="useDynamicScale">Set to true to use hardware dynamic scaling.</param>
  521. /// <param name="memoryless">Use this property to set the render texture memoryless modes.</param>
  522. /// <param name="name">Name of the RTHandle.</param>
  523. /// <returns></returns>
  524. public RTHandle Alloc(
  525. ScaleFunc scaleFunc,
  526. int slices = 1,
  527. DepthBits depthBufferBits = DepthBits.None,
  528. GraphicsFormat colorFormat = GraphicsFormat.R8G8B8A8_SRGB,
  529. FilterMode filterMode = FilterMode.Point,
  530. TextureWrapMode wrapMode = TextureWrapMode.Repeat,
  531. TextureDimension dimension = TextureDimension.Tex2D,
  532. bool enableRandomWrite = false,
  533. bool useMipMap = false,
  534. bool autoGenerateMips = true,
  535. bool isShadowMap = false,
  536. int anisoLevel = 1,
  537. float mipMapBias = 0f,
  538. bool enableMSAA = false,
  539. bool bindTextureMS = false,
  540. bool useDynamicScale = false,
  541. RenderTextureMemoryless memoryless = RenderTextureMemoryless.None,
  542. string name = ""
  543. )
  544. {
  545. var scaleFactor = scaleFunc(new Vector2Int(GetMaxWidth(), GetMaxHeight()));
  546. int width = Mathf.Max(scaleFactor.x, 1);
  547. int height = Mathf.Max(scaleFactor.y, 1);
  548. var rth = AllocAutoSizedRenderTexture(width,
  549. height,
  550. slices,
  551. depthBufferBits,
  552. colorFormat,
  553. filterMode,
  554. wrapMode,
  555. dimension,
  556. enableRandomWrite,
  557. useMipMap,
  558. autoGenerateMips,
  559. isShadowMap,
  560. anisoLevel,
  561. mipMapBias,
  562. enableMSAA,
  563. bindTextureMS,
  564. useDynamicScale,
  565. memoryless,
  566. name
  567. );
  568. rth.referenceSize = new Vector2Int(width, height);
  569. rth.scaleFunc = scaleFunc;
  570. return rth;
  571. }
  572. // Internal function
  573. RTHandle AllocAutoSizedRenderTexture(
  574. int width,
  575. int height,
  576. int slices,
  577. DepthBits depthBufferBits,
  578. GraphicsFormat colorFormat,
  579. FilterMode filterMode,
  580. TextureWrapMode wrapMode,
  581. TextureDimension dimension,
  582. bool enableRandomWrite,
  583. bool useMipMap,
  584. bool autoGenerateMips,
  585. bool isShadowMap,
  586. int anisoLevel,
  587. float mipMapBias,
  588. bool enableMSAA,
  589. bool bindTextureMS,
  590. bool useDynamicScale,
  591. RenderTextureMemoryless memoryless,
  592. string name
  593. )
  594. {
  595. // Here user made a mistake in setting up msaa/bindMS, hence the warning
  596. if (!enableMSAA && bindTextureMS == true)
  597. {
  598. Debug.LogWarning("RTHandle allocated without MSAA but with bindMS set to true, forcing bindMS to false.");
  599. bindTextureMS = false;
  600. }
  601. bool allocForMSAA = m_ScaledRTSupportsMSAA ? enableMSAA : false;
  602. // Here we purposefully disable MSAA so we just force the bindMS param to false.
  603. if (!allocForMSAA)
  604. {
  605. bindTextureMS = false;
  606. }
  607. // MSAA Does not support random read/write.
  608. bool UAV = enableRandomWrite;
  609. if (allocForMSAA && (UAV == true))
  610. {
  611. Debug.LogWarning("RTHandle that is MSAA-enabled cannot allocate MSAA RT with 'enableRandomWrite = true'.");
  612. UAV = false;
  613. }
  614. int msaaSamples = allocForMSAA ? (int)m_ScaledRTCurrentMSAASamples : 1;
  615. // We need to handle this in an explicit way since GraphicsFormat does not expose depth formats. TODO: Get rid of this branch once GraphicsFormat'll expose depth related formats
  616. RenderTexture rt;
  617. if (isShadowMap || depthBufferBits != DepthBits.None)
  618. {
  619. RenderTextureFormat format = isShadowMap ? RenderTextureFormat.Shadowmap : RenderTextureFormat.Depth;
  620. GraphicsFormat stencilFormat = isShadowMap ? GraphicsFormat.None : GraphicsFormat.R8_UInt;
  621. rt = new RenderTexture(width, height, (int)depthBufferBits, format, RenderTextureReadWrite.Linear)
  622. {
  623. hideFlags = HideFlags.HideAndDontSave,
  624. volumeDepth = slices,
  625. filterMode = filterMode,
  626. wrapMode = wrapMode,
  627. dimension = dimension,
  628. enableRandomWrite = UAV,
  629. useMipMap = useMipMap,
  630. autoGenerateMips = autoGenerateMips,
  631. anisoLevel = anisoLevel,
  632. mipMapBias = mipMapBias,
  633. antiAliasing = msaaSamples,
  634. bindTextureMS = bindTextureMS,
  635. useDynamicScale = m_HardwareDynamicResRequested && useDynamicScale,
  636. memorylessMode = memoryless,
  637. stencilFormat = stencilFormat,
  638. name = CoreUtils.GetRenderTargetAutoName(width, height, slices, GraphicsFormatUtility.GetRenderTextureFormat(colorFormat), name, mips: useMipMap, enableMSAA: allocForMSAA, msaaSamples: m_ScaledRTCurrentMSAASamples)
  639. };
  640. }
  641. else
  642. {
  643. rt = new RenderTexture(width, height, (int)depthBufferBits, colorFormat)
  644. {
  645. hideFlags = HideFlags.HideAndDontSave,
  646. volumeDepth = slices,
  647. filterMode = filterMode,
  648. wrapMode = wrapMode,
  649. dimension = dimension,
  650. enableRandomWrite = UAV,
  651. useMipMap = useMipMap,
  652. autoGenerateMips = autoGenerateMips,
  653. anisoLevel = anisoLevel,
  654. mipMapBias = mipMapBias,
  655. antiAliasing = msaaSamples,
  656. bindTextureMS = bindTextureMS,
  657. useDynamicScale = m_HardwareDynamicResRequested && useDynamicScale,
  658. memorylessMode = memoryless,
  659. name = CoreUtils.GetRenderTargetAutoName(width, height, slices, GraphicsFormatUtility.GetRenderTextureFormat(colorFormat), name, mips: useMipMap, enableMSAA: allocForMSAA, msaaSamples: m_ScaledRTCurrentMSAASamples)
  660. };
  661. }
  662. rt.Create();
  663. var rth = new RTHandle(this);
  664. rth.SetRenderTexture(rt);
  665. rth.m_EnableMSAA = enableMSAA;
  666. rth.m_EnableRandomWrite = enableRandomWrite;
  667. rth.useScaling = true;
  668. rth.m_EnableHWDynamicScale = useDynamicScale;
  669. rth.m_Name = name;
  670. m_AutoSizedRTs.Add(rth);
  671. return rth;
  672. }
  673. /// <summary>
  674. /// Allocate a RTHandle from a regular Texture.
  675. /// </summary>
  676. /// <param name="texture">Input texture</param>
  677. /// <returns>A new RTHandle referencing the input texture.</returns>
  678. public RTHandle Alloc(Texture texture)
  679. {
  680. var rth = new RTHandle(this);
  681. rth.SetTexture(texture);
  682. rth.m_EnableMSAA = false;
  683. rth.m_EnableRandomWrite = false;
  684. rth.useScaling = false;
  685. rth.m_EnableHWDynamicScale = false;
  686. rth.m_Name = "";
  687. return rth;
  688. }
  689. private static RTHandle Alloc(RTHandle tex)
  690. {
  691. Debug.LogError("Allocation a RTHandle from another one is forbidden.");
  692. return null;
  693. }
  694. internal string DumpRTInfo()
  695. {
  696. string result = "";
  697. Array.Resize(ref m_AutoSizedRTsArray, m_AutoSizedRTs.Count);
  698. m_AutoSizedRTs.CopyTo(m_AutoSizedRTsArray);
  699. for (int i = 0, c = m_AutoSizedRTsArray.Length; i < c; ++i)
  700. {
  701. var rt = m_AutoSizedRTsArray[i].rt;
  702. result = string.Format("{0}\nRT ({1})\t Format: {2} W: {3} H {4}\n", result, i, rt.format, rt.width, rt.height);
  703. }
  704. return result;
  705. }
  706. }
  707. }