RenderGraphObjectPool.cs 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. using System;
  2. using System.Collections.Generic;
  3. namespace UnityEngine.Experimental.Rendering.RenderGraphModule
  4. {
  5. /// <summary>
  6. /// Helper class provided in the RenderGraphContext to all Render Passes.
  7. /// It allows you to do temporary allocations of various objects during a Render Pass.
  8. /// </summary>
  9. public sealed class RenderGraphObjectPool
  10. {
  11. class SharedObjectPool<T> where T : new()
  12. {
  13. Stack<T> m_Pool = new Stack<T>();
  14. public T Get()
  15. {
  16. var result = m_Pool.Count == 0 ? new T() : m_Pool.Pop();
  17. return result;
  18. }
  19. public void Release(T value)
  20. {
  21. m_Pool.Push(value);
  22. }
  23. static readonly Lazy<SharedObjectPool<T>> s_Instance = new Lazy<SharedObjectPool<T>>();
  24. public static SharedObjectPool<T> sharedPool => s_Instance.Value;
  25. }
  26. Dictionary<(Type, int), Stack<object>> m_ArrayPool = new Dictionary<(Type, int), Stack<object>>();
  27. List<(object, (Type, int))> m_AllocatedArrays = new List<(object, (Type, int))>();
  28. List<MaterialPropertyBlock> m_AllocatedMaterialPropertyBlocks = new List<MaterialPropertyBlock>();
  29. internal RenderGraphObjectPool() { }
  30. /// <summary>
  31. /// Allocate a temporary typed array of a specific size.
  32. /// Unity releases the array at the end of the Render Pass.
  33. /// </summary>
  34. /// <typeparam name="T">Type of the array to be allocated.</typeparam>
  35. /// <param name="size">Number of element in the array.</param>
  36. /// <returns>A new array of type T with size number of elements.</returns>
  37. public T[] GetTempArray<T>(int size)
  38. {
  39. if (!m_ArrayPool.TryGetValue((typeof(T), size), out var stack))
  40. {
  41. stack = new Stack<object>();
  42. m_ArrayPool.Add((typeof(T), size), stack);
  43. }
  44. var result = stack.Count > 0 ? (T[])stack.Pop() : new T[size];
  45. m_AllocatedArrays.Add((result, (typeof(T), size)));
  46. return result;
  47. }
  48. /// <summary>
  49. /// Allocate a temporary MaterialPropertyBlock for the Render Pass.
  50. /// </summary>
  51. /// <returns>A new clean MaterialPropertyBlock.</returns>
  52. public MaterialPropertyBlock GetTempMaterialPropertyBlock()
  53. {
  54. var result = SharedObjectPool<MaterialPropertyBlock>.sharedPool.Get();
  55. result.Clear();
  56. m_AllocatedMaterialPropertyBlocks.Add(result);
  57. return result;
  58. }
  59. internal void ReleaseAllTempAlloc()
  60. {
  61. foreach(var arrayDesc in m_AllocatedArrays)
  62. {
  63. bool result = m_ArrayPool.TryGetValue(arrayDesc.Item2, out var stack);
  64. Debug.Assert(result, "Correct stack type should always be allocated.");
  65. stack.Push(arrayDesc.Item1);
  66. }
  67. m_AllocatedArrays.Clear();
  68. foreach(var mpb in m_AllocatedMaterialPropertyBlocks)
  69. {
  70. SharedObjectPool<MaterialPropertyBlock>.sharedPool.Release(mpb);
  71. }
  72. m_AllocatedMaterialPropertyBlocks.Clear();
  73. }
  74. // Regular pooling API. Only internal use for now
  75. internal T Get<T>() where T : new()
  76. {
  77. var toto = SharedObjectPool<T>.sharedPool;
  78. return toto.Get();
  79. }
  80. internal void Release<T>(T value) where T : new()
  81. {
  82. var toto = SharedObjectPool<T>.sharedPool;
  83. toto.Release(value);
  84. }
  85. }
  86. }