DynamicResolutionHandler.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279
  1. using System;
  2. namespace UnityEngine.Rendering
  3. {
  4. /// <summary>
  5. /// The format of the delegate used to perofrm dynamic resolution.
  6. /// </summary>
  7. public delegate float PerformDynamicRes();
  8. /// <summary>
  9. /// The type of dynamic resolution scaler. It essentially defines what the output of the scaler is expected to be.
  10. /// </summary>
  11. public enum DynamicResScalePolicyType
  12. {
  13. /// <summary>
  14. /// If is the option, DynamicResolutionHandler expects the scaler to return a screen percentage.
  15. /// The value set will be clamped between the minimum and maximum percentage set in the GlobalDynamicResolutionSettings.
  16. /// </summary>
  17. ReturnsPercentage,
  18. /// <summary>
  19. /// If is the option, DynamicResolutionHandler expects the scaler to return a factor t in the [0..1] such that the final resolution percentage
  20. /// is determined by lerp(minimumPercentage, maximumPercentage, t), where the minimum and maximum percentages are the one set in the GlobalDynamicResolutionSettings.
  21. /// </summary>
  22. ReturnsMinMaxLerpFactor
  23. }
  24. /// <summary>
  25. /// The class responsible to handle dynamic resolution.
  26. /// </summary>
  27. public class DynamicResolutionHandler
  28. {
  29. private bool m_Enabled = false;
  30. private float m_MinScreenFraction = 1.0f;
  31. private float m_MaxScreenFraction = 1.0f;
  32. private float m_CurrentFraction = 1.0f;
  33. private float m_PrevFraction = -1.0f;
  34. private bool m_ForcingRes = false;
  35. private bool m_CurrentCameraRequest = true;
  36. private bool m_ForceSoftwareFallback = false;
  37. private float m_PrevHWScaleWidth = 1.0f;
  38. private float m_PrevHWScaleHeight = 1.0f;
  39. private Vector2Int m_LastScaledSize = new Vector2Int(0, 0);
  40. private DynamicResScalePolicyType m_ScalerType = DynamicResScalePolicyType.ReturnsMinMaxLerpFactor;
  41. // Debug
  42. private Vector2Int cachedOriginalSize;
  43. /// <summary>
  44. /// The filter that is used to upscale the rendering result to the native resolution.
  45. /// </summary>
  46. public DynamicResUpscaleFilter filter { get; set; }
  47. private DynamicResolutionType type;
  48. private PerformDynamicRes m_DynamicResMethod = null;
  49. private static DynamicResolutionHandler s_Instance = new DynamicResolutionHandler();
  50. /// <summary>
  51. /// Get the instance of the global dynamic resolution handler.
  52. /// </summary>
  53. public static DynamicResolutionHandler instance { get { return s_Instance; } }
  54. private DynamicResolutionHandler()
  55. {
  56. m_DynamicResMethod = DefaultDynamicResMethod;
  57. filter = DynamicResUpscaleFilter.Bilinear;
  58. }
  59. // TODO: Eventually we will need to provide a good default implementation for this.
  60. static private float DefaultDynamicResMethod()
  61. {
  62. return 1.0f;
  63. }
  64. private void ProcessSettings(GlobalDynamicResolutionSettings settings)
  65. {
  66. m_Enabled = settings.enabled;
  67. if (!m_Enabled)
  68. {
  69. m_CurrentFraction = 1.0f;
  70. }
  71. else
  72. {
  73. type = settings.dynResType;
  74. float minScreenFrac = Mathf.Clamp(settings.minPercentage / 100.0f, 0.1f, 1.0f);
  75. m_MinScreenFraction = minScreenFrac;
  76. float maxScreenFrac = Mathf.Clamp(settings.maxPercentage / 100.0f, m_MinScreenFraction, 3.0f);
  77. m_MaxScreenFraction = maxScreenFrac;
  78. filter = settings.upsampleFilter;
  79. m_ForcingRes = settings.forceResolution;
  80. if (m_ForcingRes)
  81. {
  82. float fraction = Mathf.Clamp(settings.forcedPercentage / 100.0f, 0.1f, 1.5f);
  83. m_CurrentFraction = fraction;
  84. }
  85. }
  86. }
  87. /// <summary>
  88. /// Set the scaler method used to drive dynamic resolution.
  89. /// </summary>
  90. /// <param name="scaler">The delegate used to determine the resolution percentage used by the dynamic resolution system.</param>
  91. /// <param name="scalerType">The type of scaler that is used, this is used to indicate the return type of the scaler to the dynamic resolution system.</param>
  92. static public void SetDynamicResScaler(PerformDynamicRes scaler, DynamicResScalePolicyType scalerType = DynamicResScalePolicyType.ReturnsMinMaxLerpFactor)
  93. {
  94. s_Instance.m_ScalerType = scalerType;
  95. s_Instance.m_DynamicResMethod = scaler;
  96. }
  97. /// <summary>
  98. /// Set whether the camera that is currently processed by the pipeline has requested dynamic resolution or not.
  99. /// </summary>
  100. /// <param name="cameraRequest">Determines whether the camera has requested dynamic resolution or not.</param>
  101. public void SetCurrentCameraRequest(bool cameraRequest)
  102. {
  103. m_CurrentCameraRequest = cameraRequest;
  104. }
  105. /// <summary>
  106. /// Update the state of the dynamic resolution system.
  107. /// </summary>
  108. /// <param name="settings">The settings that are to be used by the dynamic resolution system.</param>
  109. /// <param name="OnResolutionChange">An action that will be called every time the dynamic resolution system triggers a change in resolution.</param>
  110. public void Update(GlobalDynamicResolutionSettings settings, Action OnResolutionChange = null)
  111. {
  112. ProcessSettings(settings);
  113. if (!m_Enabled) return;
  114. if (!m_ForcingRes)
  115. {
  116. if(m_ScalerType == DynamicResScalePolicyType.ReturnsMinMaxLerpFactor)
  117. {
  118. float currLerp = m_DynamicResMethod();
  119. float lerpFactor = Mathf.Clamp(currLerp, 0.0f, 1.0f);
  120. m_CurrentFraction = Mathf.Lerp(m_MinScreenFraction, m_MaxScreenFraction, lerpFactor);
  121. }
  122. else if(m_ScalerType == DynamicResScalePolicyType.ReturnsPercentage)
  123. {
  124. float percentageRequested = Mathf.Max(m_DynamicResMethod(), 5.0f);
  125. m_CurrentFraction = Mathf.Clamp(percentageRequested / 100.0f, m_MinScreenFraction, m_MaxScreenFraction);
  126. }
  127. }
  128. if (m_CurrentFraction != m_PrevFraction)
  129. {
  130. m_PrevFraction = m_CurrentFraction;
  131. if (!m_ForceSoftwareFallback && type == DynamicResolutionType.Hardware)
  132. {
  133. ScalableBufferManager.ResizeBuffers(m_CurrentFraction, m_CurrentFraction);
  134. }
  135. OnResolutionChange();
  136. }
  137. else
  138. {
  139. // Unity can change the scale factor by itself so we need to trigger the Action if that happens as well.
  140. if (!m_ForceSoftwareFallback && type == DynamicResolutionType.Hardware)
  141. {
  142. if(ScalableBufferManager.widthScaleFactor != m_PrevHWScaleWidth ||
  143. ScalableBufferManager.heightScaleFactor != m_PrevHWScaleHeight)
  144. {
  145. OnResolutionChange();
  146. }
  147. }
  148. }
  149. m_PrevHWScaleWidth = ScalableBufferManager.widthScaleFactor;
  150. m_PrevHWScaleHeight = ScalableBufferManager.heightScaleFactor;
  151. }
  152. /// <summary>
  153. /// Determines whether software dynamic resolution is enabled or not.
  154. /// </summary>
  155. /// <returns>True: Software dynamic resolution is enabled</returns>
  156. public bool SoftwareDynamicResIsEnabled()
  157. {
  158. return m_CurrentCameraRequest && m_Enabled && m_CurrentFraction != 1.0f && (m_ForceSoftwareFallback || type == DynamicResolutionType.Software);
  159. }
  160. /// <summary>
  161. /// Determines whether hardware dynamic resolution is enabled or not.
  162. /// </summary>
  163. /// <returns>True: Hardware dynamic resolution is enabled</returns>
  164. public bool HardwareDynamicResIsEnabled()
  165. {
  166. return !m_ForceSoftwareFallback && m_CurrentCameraRequest && m_Enabled && type == DynamicResolutionType.Hardware;
  167. }
  168. /// <summary>
  169. /// Identifies whether hardware dynamic resolution has been requested and is going to be used.
  170. /// </summary>
  171. /// <returns>True: Hardware dynamic resolution is requested by user and software fallback has not been forced</returns>
  172. public bool RequestsHardwareDynamicResolution()
  173. {
  174. if (m_ForceSoftwareFallback)
  175. return false;
  176. return type == DynamicResolutionType.Hardware;
  177. }
  178. /// <summary>
  179. /// Identifies whether dynamic resolution is enabled and scaling the render targets.
  180. /// </summary>
  181. /// <returns>True: Dynamic resolution is enabled.</returns>
  182. public bool DynamicResolutionEnabled()
  183. {
  184. return m_CurrentCameraRequest && m_Enabled && m_CurrentFraction != 1.0f;
  185. }
  186. /// <summary>
  187. /// Forces software fallback for dynamic resolution. Needs to be called in case Hardware dynamic resolution is requested by the user, but not supported by the platform.
  188. /// </summary>
  189. public void ForceSoftwareFallback()
  190. {
  191. m_ForceSoftwareFallback = true;
  192. }
  193. /// <summary>
  194. /// Applies to the passed size the scale imposed by the dynamic resolution system.
  195. /// </summary>
  196. /// <param name="size">The starting size of the render target that will be scaled by dynamic resolution.</param>
  197. /// <returns>The parameter size scaled by the dynamic resolution system.</returns>
  198. public Vector2Int GetScaledSize(Vector2Int size)
  199. {
  200. cachedOriginalSize = size;
  201. if (!m_Enabled || !m_CurrentCameraRequest)
  202. {
  203. return size;
  204. }
  205. float scaleFractionX = m_CurrentFraction;
  206. float scaleFractionY = m_CurrentFraction;
  207. if (!m_ForceSoftwareFallback && type == DynamicResolutionType.Hardware)
  208. {
  209. scaleFractionX = ScalableBufferManager.widthScaleFactor;
  210. scaleFractionY = ScalableBufferManager.heightScaleFactor;
  211. }
  212. Vector2Int scaledSize = new Vector2Int(Mathf.CeilToInt(size.x * scaleFractionX), Mathf.CeilToInt(size.y * scaleFractionY));
  213. if (m_ForceSoftwareFallback || type != DynamicResolutionType.Hardware)
  214. {
  215. scaledSize.x += (1 & scaledSize.x);
  216. scaledSize.y += (1 & scaledSize.y);
  217. }
  218. m_LastScaledSize = scaledSize;
  219. return scaledSize;
  220. }
  221. /// <summary>
  222. /// Returns the scale that is currently applied by the dynamic resolution system.
  223. /// </summary>
  224. /// <returns>The scale that is currently applied by the dynamic resolution system.</returns>
  225. public float GetCurrentScale()
  226. {
  227. return (m_Enabled && m_CurrentCameraRequest) ? m_CurrentFraction : 1.0f;
  228. }
  229. /// <summary>
  230. /// Returns the latest scaled size that has been produced by GetScaledSize.
  231. /// </summary>
  232. /// <returns>The latest scaled size that has been produced by GetScaledSize.</returns>
  233. public Vector2Int GetLastScaledSize()
  234. {
  235. return m_LastScaledSize;
  236. }
  237. }
  238. }