RenderingUtils.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Diagnostics;
  4. using UnityEngine.Scripting.APIUpdating;
  5. #if POST_PROCESSING_STACK_2_0_0_OR_NEWER
  6. using UnityEngine.Rendering.PostProcessing;
  7. #endif
  8. namespace UnityEngine.Rendering.Universal
  9. {
  10. /// <summary>
  11. /// Contains properties and helper functions that you can use when rendering.
  12. /// </summary>
  13. [MovedFrom("UnityEngine.Rendering.LWRP")] public static class RenderingUtils
  14. {
  15. static List<ShaderTagId> m_LegacyShaderPassNames = new List<ShaderTagId>()
  16. {
  17. new ShaderTagId("Always"),
  18. new ShaderTagId("ForwardBase"),
  19. new ShaderTagId("PrepassBase"),
  20. new ShaderTagId("Vertex"),
  21. new ShaderTagId("VertexLMRGBM"),
  22. new ShaderTagId("VertexLM"),
  23. };
  24. static Mesh s_FullscreenMesh = null;
  25. /// <summary>
  26. /// Returns a mesh that you can use with <see cref="CommandBuffer.DrawMesh(Mesh, Matrix4x4, Material)"/> to render full-screen effects.
  27. /// </summary>
  28. public static Mesh fullscreenMesh
  29. {
  30. get
  31. {
  32. if (s_FullscreenMesh != null)
  33. return s_FullscreenMesh;
  34. float topV = 1.0f;
  35. float bottomV = 0.0f;
  36. s_FullscreenMesh = new Mesh { name = "Fullscreen Quad" };
  37. s_FullscreenMesh.SetVertices(new List<Vector3>
  38. {
  39. new Vector3(-1.0f, -1.0f, 0.0f),
  40. new Vector3(-1.0f, 1.0f, 0.0f),
  41. new Vector3(1.0f, -1.0f, 0.0f),
  42. new Vector3(1.0f, 1.0f, 0.0f)
  43. });
  44. s_FullscreenMesh.SetUVs(0, new List<Vector2>
  45. {
  46. new Vector2(0.0f, bottomV),
  47. new Vector2(0.0f, topV),
  48. new Vector2(1.0f, bottomV),
  49. new Vector2(1.0f, topV)
  50. });
  51. s_FullscreenMesh.SetIndices(new[] { 0, 1, 2, 2, 1, 3 }, MeshTopology.Triangles, 0, false);
  52. s_FullscreenMesh.UploadMeshData(true);
  53. return s_FullscreenMesh;
  54. }
  55. }
  56. #if POST_PROCESSING_STACK_2_0_0_OR_NEWER
  57. static readonly int m_PostProcessingTemporaryTargetId = Shader.PropertyToID("_TemporaryColorTexture");
  58. static PostProcessRenderContext m_PostProcessRenderContext;
  59. [Obsolete("The use of the Post-processing Stack V2 is deprecated in the Universal Render Pipeline. Use the builtin post-processing effects instead.")]
  60. public static PostProcessRenderContext postProcessRenderContext
  61. {
  62. get => m_PostProcessRenderContext ?? (m_PostProcessRenderContext = new PostProcessRenderContext());
  63. }
  64. #endif
  65. internal static bool useStructuredBuffer
  66. {
  67. // There are some performance issues with StructuredBuffers in some platforms.
  68. // We fallback to UBO in those cases.
  69. get
  70. {
  71. // TODO: For now disabling SSBO until figure out Vulkan binding issues.
  72. // When enabling this also enable it in shader side in Input.hlsl
  73. return false;
  74. // We don't use SSBO in D3D because we can't figure out without adding shader variants if platforms is D3D10.
  75. //GraphicsDeviceType deviceType = SystemInfo.graphicsDeviceType;
  76. //return !Application.isMobilePlatform &&
  77. // (deviceType == GraphicsDeviceType.Metal || deviceType == GraphicsDeviceType.Vulkan ||
  78. // deviceType == GraphicsDeviceType.PlayStation4 || deviceType == GraphicsDeviceType.XboxOne);
  79. }
  80. }
  81. static Material s_ErrorMaterial;
  82. static Material errorMaterial
  83. {
  84. get
  85. {
  86. if (s_ErrorMaterial == null)
  87. {
  88. // TODO: When importing project, AssetPreviewUpdater::CreatePreviewForAsset will be called multiple times.
  89. // This might be in a point that some resources required for the pipeline are not finished importing yet.
  90. // Proper fix is to add a fence on asset import.
  91. try
  92. {
  93. s_ErrorMaterial = new Material(Shader.Find("Hidden/Universal Render Pipeline/FallbackError"));
  94. }
  95. catch{ }
  96. }
  97. return s_ErrorMaterial;
  98. }
  99. }
  100. #if POST_PROCESSING_STACK_2_0_0_OR_NEWER
  101. #pragma warning disable 0618 // Obsolete
  102. internal static void RenderPostProcessingCompat(CommandBuffer cmd, ref CameraData cameraData, RenderTextureDescriptor sourceDescriptor,
  103. RenderTargetIdentifier source, RenderTargetIdentifier destination, bool opaqueOnly, bool flip)
  104. {
  105. var layer = cameraData.postProcessLayer;
  106. int effectsCount;
  107. if (opaqueOnly)
  108. {
  109. effectsCount = layer.sortedBundles[PostProcessEvent.BeforeTransparent].Count;
  110. }
  111. else
  112. {
  113. effectsCount = layer.sortedBundles[PostProcessEvent.BeforeStack].Count +
  114. layer.sortedBundles[PostProcessEvent.AfterStack].Count;
  115. }
  116. var camera = cameraData.camera;
  117. var postProcessRenderContext = RenderingUtils.postProcessRenderContext;
  118. postProcessRenderContext.Reset();
  119. postProcessRenderContext.camera = camera;
  120. postProcessRenderContext.source = source;
  121. postProcessRenderContext.sourceFormat = sourceDescriptor.colorFormat;
  122. postProcessRenderContext.destination = destination;
  123. postProcessRenderContext.command = cmd;
  124. postProcessRenderContext.flip = flip;
  125. // If there's only one effect in the stack and soure is same as dest we
  126. // create an intermediate blit rendertarget to handle it.
  127. // Otherwise, PostProcessing system will create the intermediate blit targets itself.
  128. if (effectsCount == 1 && source == destination)
  129. {
  130. var rtId = new RenderTargetIdentifier(m_PostProcessingTemporaryTargetId);
  131. var descriptor = sourceDescriptor;
  132. descriptor.msaaSamples = 1;
  133. descriptor.depthBufferBits = 0;
  134. postProcessRenderContext.destination = rtId;
  135. cmd.GetTemporaryRT(m_PostProcessingTemporaryTargetId, descriptor, FilterMode.Point);
  136. if (opaqueOnly)
  137. cameraData.postProcessLayer.RenderOpaqueOnly(postProcessRenderContext);
  138. else
  139. cameraData.postProcessLayer.Render(postProcessRenderContext);
  140. cmd.Blit(rtId, destination);
  141. cmd.ReleaseTemporaryRT(m_PostProcessingTemporaryTargetId);
  142. }
  143. else if (opaqueOnly)
  144. {
  145. cameraData.postProcessLayer.RenderOpaqueOnly(postProcessRenderContext);
  146. }
  147. else
  148. {
  149. cameraData.postProcessLayer.Render(postProcessRenderContext);
  150. }
  151. }
  152. #pragma warning restore 0618
  153. #endif
  154. // This is used to render materials that contain built-in shader passes not compatible with URP.
  155. // It will render those legacy passes with error/pink shader.
  156. [Conditional("DEVELOPMENT_BUILD"), Conditional("UNITY_EDITOR")]
  157. internal static void RenderObjectsWithError(ScriptableRenderContext context, ref CullingResults cullResults, Camera camera, FilteringSettings filterSettings, SortingCriteria sortFlags)
  158. {
  159. // TODO: When importing project, AssetPreviewUpdater::CreatePreviewForAsset will be called multiple times.
  160. // This might be in a point that some resources required for the pipeline are not finished importing yet.
  161. // Proper fix is to add a fence on asset import.
  162. if (errorMaterial == null)
  163. return;
  164. SortingSettings sortingSettings = new SortingSettings(camera) { criteria = sortFlags };
  165. DrawingSettings errorSettings = new DrawingSettings(m_LegacyShaderPassNames[0], sortingSettings)
  166. {
  167. perObjectData = PerObjectData.None,
  168. overrideMaterial = errorMaterial,
  169. overrideMaterialPassIndex = 0
  170. };
  171. for (int i = 1; i < m_LegacyShaderPassNames.Count; ++i)
  172. errorSettings.SetShaderPassName(i, m_LegacyShaderPassNames[i]);
  173. context.DrawRenderers(cullResults, ref errorSettings, ref filterSettings);
  174. }
  175. // Caches render texture format support. SystemInfo.SupportsRenderTextureFormat allocates memory due to boxing.
  176. static Dictionary<RenderTextureFormat, bool> m_RenderTextureFormatSupport = new Dictionary<RenderTextureFormat, bool>();
  177. internal static void ClearSystemInfoCache()
  178. {
  179. m_RenderTextureFormatSupport.Clear();
  180. }
  181. /// <summary>
  182. /// Checks if a render texture format is supported by the run-time system.
  183. /// Similar to <see cref="SystemInfo.SupportsRenderTextureFormat(RenderTextureFormat)"/>, but doesn't allocate memory.
  184. /// </summary>
  185. /// <param name="format">The format to look up.</param>
  186. /// <returns>Returns true if the graphics card supports the given <c>RenderTextureFormat</c></returns>
  187. public static bool SupportsRenderTextureFormat(RenderTextureFormat format)
  188. {
  189. if (!m_RenderTextureFormatSupport.TryGetValue(format, out var support))
  190. {
  191. support = SystemInfo.SupportsRenderTextureFormat(format);
  192. m_RenderTextureFormatSupport.Add(format, support);
  193. }
  194. return support;
  195. }
  196. /// <summary>
  197. /// Return the last colorBuffer index actually referring to an existing RenderTarget
  198. /// </summary>
  199. /// <param name="colorBuffers"></param>
  200. /// <returns></returns>
  201. internal static int GetLastValidColorBufferIndex(RenderTargetIdentifier[] colorBuffers)
  202. {
  203. int i = colorBuffers.Length - 1;
  204. for(; i>=0; --i)
  205. {
  206. if (colorBuffers[i] != 0)
  207. break;
  208. }
  209. return i;
  210. }
  211. /// <summary>
  212. /// Return the number of items in colorBuffers actually referring to an existing RenderTarget
  213. /// </summary>
  214. /// <param name="colorBuffers"></param>
  215. /// <returns></returns>
  216. internal static uint GetValidColorBufferCount(RenderTargetIdentifier[] colorBuffers)
  217. {
  218. uint nonNullColorBuffers = 0;
  219. if (colorBuffers != null)
  220. {
  221. foreach (var identifier in colorBuffers)
  222. {
  223. if (identifier != 0)
  224. ++nonNullColorBuffers;
  225. }
  226. }
  227. return nonNullColorBuffers;
  228. }
  229. /// <summary>
  230. /// Return true if colorBuffers is an actual MRT setup
  231. /// </summary>
  232. /// <param name="colorBuffers"></param>
  233. /// <returns></returns>
  234. internal static bool IsMRT(RenderTargetIdentifier[] colorBuffers)
  235. {
  236. return GetValidColorBufferCount(colorBuffers) > 1;
  237. }
  238. /// <summary>
  239. /// Return true if value can be found in source (without recurring to Linq)
  240. /// </summary>
  241. /// <param name="source"></param>
  242. /// <param name="value"></param>
  243. /// <returns></returns>
  244. internal static bool Contains(RenderTargetIdentifier[] source, RenderTargetIdentifier value)
  245. {
  246. foreach (var identifier in source)
  247. {
  248. if (identifier == value)
  249. return true;
  250. }
  251. return false;
  252. }
  253. /// <summary>
  254. /// Return the index where value was found source. Otherwise, return -1. (without recurring to Linq)
  255. /// </summary>
  256. /// <param name="source"></param>
  257. /// <param name="value"></param>
  258. /// <returns></returns>
  259. internal static int IndexOf(RenderTargetIdentifier[] source, RenderTargetIdentifier value)
  260. {
  261. for (int i = 0; i < source.Length; ++i)
  262. {
  263. if (source[i] == value)
  264. return i;
  265. }
  266. return -1;
  267. }
  268. /// <summary>
  269. /// Return the number of RenderTargetIdentifiers in "source" that are valid (not 0) and different from "value" (without recurring to Linq)
  270. /// </summary>
  271. /// <param name="source"></param>
  272. /// <param name="value"></param>
  273. /// <returns></returns>
  274. internal static uint CountDistinct(RenderTargetIdentifier[] source, RenderTargetIdentifier value)
  275. {
  276. uint count = 0;
  277. for (int i = 0; i < source.Length; ++i)
  278. {
  279. if (source[i] != value && source[i] != 0)
  280. ++count;
  281. }
  282. return count;
  283. }
  284. /// <summary>
  285. /// Return the index of last valid (i.e different from 0) RenderTargetIdentifiers in "source" (without recurring to Linq)
  286. /// </summary>
  287. /// <param name="source"></param>
  288. /// <returns></returns>
  289. internal static int LastValid(RenderTargetIdentifier[] source)
  290. {
  291. for (int i = source.Length-1; i >= 0; --i)
  292. {
  293. if (source[i] != 0)
  294. return i;
  295. }
  296. return -1;
  297. }
  298. /// <summary>
  299. /// Return true if ClearFlag a contains ClearFlag b
  300. /// </summary>
  301. /// <param name="a"></param>
  302. /// <param name="b"></param>
  303. /// <returns></returns>
  304. internal static bool Contains(ClearFlag a, ClearFlag b)
  305. {
  306. return (a & b) == b;
  307. }
  308. /// <summary>
  309. /// Return true if "left" and "right" are the same (without recurring to Linq)
  310. /// </summary>
  311. /// <param name="left"></param>
  312. /// <param name="right"></param>
  313. /// <returns></returns>
  314. internal static bool SequenceEqual(RenderTargetIdentifier[] left, RenderTargetIdentifier[] right)
  315. {
  316. if (left.Length != right.Length)
  317. return false;
  318. for (int i = 0; i < left.Length; ++i)
  319. if (left[i] != right[i])
  320. return false;
  321. return true;
  322. }
  323. }
  324. }