using System;
using Unity.Collections;
using System.Collections.Generic;
#if UNITY_EDITOR
using UnityEditor;
using UnityEditor.Rendering.Universal;
#endif
using UnityEngine.Scripting.APIUpdating;
using Lightmapping = UnityEngine.Experimental.GlobalIllumination.Lightmapping;
using UnityEngine.Experimental.Rendering;
using UnityEngine.Experimental.Rendering.RenderGraphModule;
using UnityEngine.Profiling;
namespace UnityEngine.Rendering.Universal
{
///
/// The main class for the Universal Render Pipeline (URP).
///
public sealed partial class UniversalRenderPipeline : RenderPipeline
{
///
/// The shader tag used in the Universal Render Pipeline (URP)
///
public const string k_ShaderTagName = "UniversalPipeline";
internal static class Profiling
{
private static Dictionary s_HashSamplerCache = new Dictionary();
public static readonly ProfilingSampler unknownSampler = new ProfilingSampler("Unknown");
// Specialization for camera loop to avoid allocations.
public static ProfilingSampler TryGetOrAddCameraSampler(Camera camera)
{
#if UNIVERSAL_PROFILING_NO_ALLOC
return unknownSampler;
#else
ProfilingSampler ps = null;
int cameraId = camera.GetHashCode();
bool exists = s_HashSamplerCache.TryGetValue(cameraId, out ps);
if (!exists)
{
// NOTE: camera.name allocates!
ps = new ProfilingSampler($"{nameof(UniversalRenderPipeline)}.{nameof(RenderSingleCameraInternal)}: {camera.name}");
s_HashSamplerCache.Add(cameraId, ps);
}
return ps;
#endif
}
public static class Pipeline
{
// TODO: Would be better to add Profiling name hooks into RenderPipeline.cs, requires changes outside of Universal.
#if UNITY_2021_1_OR_NEWER
public static readonly ProfilingSampler beginContextRendering = new ProfilingSampler($"{nameof(RenderPipeline)}.{nameof(BeginContextRendering)}");
public static readonly ProfilingSampler endContextRendering = new ProfilingSampler($"{nameof(RenderPipeline)}.{nameof(EndContextRendering)}");
#else
public static readonly ProfilingSampler beginFrameRendering = new ProfilingSampler($"{nameof(RenderPipeline)}.{nameof(BeginFrameRendering)}");
public static readonly ProfilingSampler endFrameRendering = new ProfilingSampler($"{nameof(RenderPipeline)}.{nameof(EndFrameRendering)}");
#endif
public static readonly ProfilingSampler beginCameraRendering = new ProfilingSampler($"{nameof(RenderPipeline)}.{nameof(BeginCameraRendering)}");
public static readonly ProfilingSampler endCameraRendering = new ProfilingSampler($"{nameof(RenderPipeline)}.{nameof(EndCameraRendering)}");
const string k_Name = nameof(UniversalRenderPipeline);
public static readonly ProfilingSampler initializeCameraData = new ProfilingSampler($"{k_Name}.{nameof(InitializeCameraData)}");
public static readonly ProfilingSampler initializeStackedCameraData = new ProfilingSampler($"{k_Name}.{nameof(InitializeStackedCameraData)}");
public static readonly ProfilingSampler initializeAdditionalCameraData = new ProfilingSampler($"{k_Name}.{nameof(InitializeAdditionalCameraData)}");
public static readonly ProfilingSampler initializeRenderingData = new ProfilingSampler($"{k_Name}.{nameof(InitializeRenderingData)}");
public static readonly ProfilingSampler initializeShadowData = new ProfilingSampler($"{k_Name}.{nameof(InitializeShadowData)}");
public static readonly ProfilingSampler initializeLightData = new ProfilingSampler($"{k_Name}.{nameof(InitializeLightData)}");
public static readonly ProfilingSampler getPerObjectLightFlags = new ProfilingSampler($"{k_Name}.{nameof(GetPerObjectLightFlags)}");
public static readonly ProfilingSampler getMainLightIndex = new ProfilingSampler($"{k_Name}.{nameof(GetMainLightIndex)}");
public static readonly ProfilingSampler setupPerFrameShaderConstants = new ProfilingSampler($"{k_Name}.{nameof(SetupPerFrameShaderConstants)}");
public static readonly ProfilingSampler setupPerCameraShaderConstants = new ProfilingSampler($"{k_Name}.{nameof(SetupPerCameraShaderConstants)}");
public static class Renderer
{
const string k_Name = nameof(ScriptableRenderer);
public static readonly ProfilingSampler setupCullingParameters = new ProfilingSampler($"{k_Name}.{nameof(ScriptableRenderer.SetupCullingParameters)}");
public static readonly ProfilingSampler setup = new ProfilingSampler($"{k_Name}.{nameof(ScriptableRenderer.Setup)}");
};
public static class Context
{
const string k_Name = nameof(ScriptableRenderContext);
public static readonly ProfilingSampler submit = new ProfilingSampler($"{k_Name}.{nameof(ScriptableRenderContext.Submit)}");
};
};
}
///
/// The maximum amount of bias allowed for shadows.
///
public static float maxShadowBias
{
get => 10.0f;
}
///
/// The minimum value allowed for render scale.
///
public static float minRenderScale
{
get => 0.1f;
}
///
/// The maximum value allowed for render scale.
///
public static float maxRenderScale
{
get => 2.0f;
}
///
/// The max number of iterations allowed calculating enclosing sphere.
///
public static int maxNumIterationsEnclosingSphere
{
get => 1000;
}
///
/// The max number of lights that can be shaded per object (in the for loop in the shader).
///
public static int maxPerObjectLights
{
// No support to bitfield mask and int[] in gles2. Can't index fast more than 4 lights.
// Check Lighting.hlsl for more details.
get => (SystemInfo.graphicsDeviceType == GraphicsDeviceType.OpenGLES2) ? 4 : 8;
}
///
/// The max number of additional lights that can can affect each GameObject.
///
public static int maxVisibleAdditionalLights
{
get
{
// Must match: Input.hlsl, MAX_VISIBLE_LIGHTS
bool isMobile = GraphicsSettings.HasShaderDefine(BuiltinShaderDefine.SHADER_API_MOBILE);
if (isMobile && (SystemInfo.graphicsDeviceType == GraphicsDeviceType.OpenGLES2 || (SystemInfo.graphicsDeviceType == GraphicsDeviceType.OpenGLES3 && Graphics.minOpenGLESVersion <= OpenGLESVersion.OpenGLES30)))
return ShaderOptions.k_MaxVisibleLightCountLowEndMobile;
// GLES can be selected as platform on Windows (not a mobile platform) but uniform buffer size so we must use a low light count.
return (isMobile || SystemInfo.graphicsDeviceType == GraphicsDeviceType.OpenGLCore || SystemInfo.graphicsDeviceType == GraphicsDeviceType.OpenGLES2 || SystemInfo.graphicsDeviceType == GraphicsDeviceType.OpenGLES3)
? ShaderOptions.k_MaxVisibleLightCountMobile : ShaderOptions.k_MaxVisibleLightCountDesktop;
}
}
// Match with values in Input.hlsl
internal static int lightsPerTile => ((maxVisibleAdditionalLights + 31) / 32) * 32;
internal static int maxZBinWords => 1024 * 4;
internal static int maxTileWords => (maxVisibleAdditionalLights <= 32 ? 1024 : 4096) * 4;
internal static int maxVisibleReflectionProbes => Math.Min(maxVisibleAdditionalLights, 64);
internal const int k_DefaultRenderingLayerMask = 0x00000001;
private readonly DebugDisplaySettingsUI m_DebugDisplaySettingsUI = new DebugDisplaySettingsUI();
private UniversalRenderPipelineGlobalSettings m_GlobalSettings;
///
/// The default Render Pipeline Global Settings.
///
public override RenderPipelineGlobalSettings defaultSettings => m_GlobalSettings;
// flag to keep track of depth buffer requirements by any of the cameras in the stack
internal static bool cameraStackRequiresDepthForPostprocessing = false;
internal static RenderGraph s_RenderGraph;
internal static RTHandleResourcePool s_RTHandlePool;
private static bool useRenderGraph;
// Reference to the asset associated with the pipeline.
// When a pipeline asset is switched in `GraphicsSettings`, the `UniversalRenderPipelineCore.asset` member
// becomes unreliable for the purpose of pipeline and renderer clean-up in the `Dispose` call from
// `RenderPipelineManager.CleanupRenderPipeline`.
// This field provides the correct reference for the purpose of cleaning up the renderers on this pipeline
// asset.
private readonly UniversalRenderPipelineAsset pipelineAsset;
///
public override string ToString() => pipelineAsset?.ToString();
///
/// Creates a new UniversalRenderPipeline instance.
///
/// The UniversalRenderPipelineAsset asset to initialize the pipeline.
///
public UniversalRenderPipeline(UniversalRenderPipelineAsset asset)
{
pipelineAsset = asset;
#if UNITY_EDITOR
m_GlobalSettings = UniversalRenderPipelineGlobalSettings.Ensure();
#else
m_GlobalSettings = UniversalRenderPipelineGlobalSettings.instance;
#endif
SetSupportedRenderingFeatures(pipelineAsset);
// Initial state of the RTHandle system.
// We initialize to screen width/height to avoid multiple realloc that can lead to inflated memory usage (as releasing of memory is delayed).
RTHandles.Initialize(Screen.width, Screen.height);
GraphicsSettings.useScriptableRenderPipelineBatching = asset.useSRPBatcher;
// In QualitySettings.antiAliasing disabled state uses value 0, where in URP 1
int qualitySettingsMsaaSampleCount = QualitySettings.antiAliasing > 0 ? QualitySettings.antiAliasing : 1;
bool msaaSampleCountNeedsUpdate = qualitySettingsMsaaSampleCount != asset.msaaSampleCount;
// Let engine know we have MSAA on for cases where we support MSAA backbuffer
if (msaaSampleCountNeedsUpdate)
{
QualitySettings.antiAliasing = asset.msaaSampleCount;
}
// Configure initial XR settings
MSAASamples msaaSamples = (MSAASamples)Mathf.Clamp(Mathf.NextPowerOfTwo(QualitySettings.antiAliasing), (int)MSAASamples.None, (int)MSAASamples.MSAA8x);
XRSystem.SetDisplayMSAASamples(msaaSamples);
XRSystem.SetRenderScale(asset.renderScale);
Shader.globalRenderPipeline = k_ShaderTagName;
Lightmapping.SetDelegate(lightsDelegate);
CameraCaptureBridge.enabled = true;
RenderingUtils.ClearSystemInfoCache();
DecalProjector.defaultMaterial = asset.decalMaterial;
s_RenderGraph = new RenderGraph("URPRenderGraph");
useRenderGraph = false;
s_RTHandlePool = new RTHandleResourcePool();
DebugManager.instance.RefreshEditor();
m_DebugDisplaySettingsUI.RegisterDebug(UniversalRenderPipelineDebugDisplaySettings.Instance);
QualitySettings.enableLODCrossFade = asset.enableLODCrossFade;
}
///
protected override void Dispose(bool disposing)
{
m_DebugDisplaySettingsUI.UnregisterDebug();
Blitter.Cleanup();
base.Dispose(disposing);
pipelineAsset.DestroyRenderers();
Shader.globalRenderPipeline = string.Empty;
SupportedRenderingFeatures.active = new SupportedRenderingFeatures();
ShaderData.instance.Dispose();
XRSystem.Dispose();
s_RenderGraph.Cleanup();
s_RenderGraph = null;
s_RTHandlePool.Cleanup();
s_RTHandlePool = null;
#if UNITY_EDITOR
SceneViewDrawMode.ResetDrawMode();
#endif
Lightmapping.ResetDelegate();
CameraCaptureBridge.enabled = false;
DisposeAdditionalCameraData();
}
// If the URP gets destroyed, we must clean up all the added URP specific camera data and
// non-GC resources to avoid leaking them.
private void DisposeAdditionalCameraData()
{
foreach (var c in Camera.allCameras)
{
if (c.TryGetComponent(out var acd))
{
acd.taaPersistentData?.DeallocateTargets();
};
}
}
#if UNITY_2021_1_OR_NEWER
///
protected override void Render(ScriptableRenderContext renderContext, Camera[] cameras)
{
Render(renderContext, new List(cameras));
}
#endif
#if UNITY_2021_1_OR_NEWER
///
protected override void Render(ScriptableRenderContext renderContext, List cameras)
#else
///
protected override void Render(ScriptableRenderContext renderContext, Camera[] cameras)
#endif
{
#if RENDER_GRAPH_ENABLED
useRenderGraph = asset.enableRenderGraph;
#else
useRenderGraph = false;
#endif
SetHDRState(cameras);
// When HDR is active we render UI overlay per camera as we want all UI to be calibrated to white paper inside a single pass
// for performance reasons otherwise we render UI overlay after all camera
SupportedRenderingFeatures.active.rendersUIOverlay = HDROutputForAnyDisplayIsActive();
// TODO: Would be better to add Profiling name hooks into RenderPipelineManager.
// C#8 feature, only in >= 2020.2
using var profScope = new ProfilingScope(null, ProfilingSampler.Get(URPProfileId.UniversalRenderTotal));
#if UNITY_2021_1_OR_NEWER
using (new ProfilingScope(null, Profiling.Pipeline.beginContextRendering))
{
BeginContextRendering(renderContext, cameras);
}
#else
using (new ProfilingScope(null, Profiling.Pipeline.beginFrameRendering))
{
BeginFrameRendering(renderContext, cameras);
}
#endif
GraphicsSettings.lightsUseLinearIntensity = (QualitySettings.activeColorSpace == ColorSpace.Linear);
GraphicsSettings.lightsUseColorTemperature = true;
GraphicsSettings.defaultRenderingLayerMask = k_DefaultRenderingLayerMask;
SetupPerFrameShaderConstants();
XRSystem.SetDisplayMSAASamples((MSAASamples)asset.msaaSampleCount);
#if UNITY_EDITOR
// We do not want to start rendering if URP global settings are not ready (m_globalSettings is null)
// or been deleted/moved (m_globalSettings is not necessarily null)
if (m_GlobalSettings == null || UniversalRenderPipelineGlobalSettings.instance == null)
{
m_GlobalSettings = UniversalRenderPipelineGlobalSettings.Ensure();
if(m_GlobalSettings == null) return;
}
#endif
#if DEVELOPMENT_BUILD || UNITY_EDITOR
if (DebugManager.instance.isAnyDebugUIActive)
UniversalRenderPipelineDebugDisplaySettings.Instance.UpdateFrameTiming();
#endif
// URP uses the camera's allowDynamicResolution flag to decide if useDynamicScale should be enabled for camera render targets.
// However, the RTHandle system has an additional setting that controls if useDynamicScale will be set for render targets allocated via RTHandles.
// In order to avoid issues at runtime, we must make the RTHandle system setting consistent with URP's logic. URP already synchronizes the setting
// during initialization, but unfortunately it's possible for external code to overwrite the setting due to RTHandle state being global.
// The best we can do to avoid errors in this situation is to ensure the state is set to the correct value every time we perform rendering.
RTHandles.SetHardwareDynamicResolutionState(true);
SortCameras(cameras);
#if UNITY_2021_1_OR_NEWER
for (int i = 0; i < cameras.Count; ++i)
#else
for (int i = 0; i < cameras.Length; ++i)
#endif
{
var camera = cameras[i];
if (IsGameCamera(camera))
{
RenderCameraStack(renderContext, camera);
}
else
{
using (new ProfilingScope(null, Profiling.Pipeline.beginCameraRendering))
{
BeginCameraRendering(renderContext, camera);
}
#if VISUAL_EFFECT_GRAPH_0_0_1_OR_NEWER
//It should be called before culling to prepare material. When there isn't any VisualEffect component, this method has no effect.
VFX.VFXManager.PrepareCamera(camera);
#endif
UpdateVolumeFramework(camera, null);
RenderSingleCameraInternal(renderContext, camera);
using (new ProfilingScope(null, Profiling.Pipeline.endCameraRendering))
{
EndCameraRendering(renderContext, camera);
}
}
}
s_RenderGraph.EndFrame();
s_RTHandlePool.PurgeUnusedResources(Time.frameCount);
#if UNITY_2021_1_OR_NEWER
using (new ProfilingScope(null, Profiling.Pipeline.endContextRendering))
{
EndContextRendering(renderContext, cameras);
}
#else
using (new ProfilingScope(null, Profiling.Pipeline.endFrameRendering))
{
EndFrameRendering(renderContext, cameras);
}
#endif
#if ENABLE_SHADER_DEBUG_PRINT
ShaderDebugPrintManager.instance.EndFrame();
#endif
}
///
/// Check whether RenderRequest is supported
///
///
///
///
///
protected override bool IsRenderRequestSupported(Camera camera, RequestData data)
{
if (data is StandardRequest)
return true;
else if (data is SingleCameraRequest)
return true;
return false;
}
///
/// Process a render request
///
///
///
///
///
protected override void ProcessRenderRequests(ScriptableRenderContext context, Camera camera, RequestData renderRequest)
{
StandardRequest standardRequest = renderRequest as StandardRequest;
SingleCameraRequest singleRequest = renderRequest as SingleCameraRequest;
if(standardRequest != null || singleRequest != null)
{
RenderTexture destination = standardRequest != null ? standardRequest.destination : singleRequest.destination;
int mipLevel = standardRequest != null ? standardRequest.mipLevel : singleRequest.mipLevel;
int slice = standardRequest != null ? standardRequest.slice : singleRequest.slice;
int face = standardRequest != null ? (int)standardRequest.face : (int)singleRequest.face;
//store data that will be changed
var orignalTarget = camera.targetTexture;
//set data
RenderTexture temporaryRT = null;
RenderTextureDescriptor RTDesc = destination.descriptor;
//need to set use default constructor of RenderTextureDescriptor which doesn't enable allowVerticalFlip which matters for cubemaps.
if (destination.dimension == TextureDimension.Cube)
RTDesc = new RenderTextureDescriptor();
RTDesc.colorFormat = destination.format;
RTDesc.volumeDepth = 1;
RTDesc.msaaSamples = destination.descriptor.msaaSamples;
RTDesc.dimension = TextureDimension.Tex2D;
RTDesc.width = destination.width / (int)Math.Pow(2, mipLevel);
RTDesc.height = destination.height / (int)Math.Pow(2, mipLevel);
RTDesc.width = Mathf.Max(1, RTDesc.width);
RTDesc.height = Mathf.Max(1, RTDesc.height);
//if mip is 0 and target is Texture2D we can immediately render to the requested destination
if(destination.dimension != TextureDimension.Tex2D || mipLevel != 0)
{
temporaryRT = RenderTexture.GetTemporary(RTDesc);
}
camera.targetTexture = temporaryRT ? temporaryRT : destination;
if (standardRequest != null)
{
Render(context, new Camera[] { camera });
}
else
{
camera.gameObject.TryGetComponent(out var additionalCameraData);
RenderSingleCameraInternal(context, camera, ref additionalCameraData);
}
if(temporaryRT)
{
switch(destination.dimension)
{
case TextureDimension.Tex2D:
case TextureDimension.Tex2DArray:
case TextureDimension.Tex3D:
Graphics.CopyTexture(temporaryRT, 0, 0, destination, slice, mipLevel);
break;
case TextureDimension.Cube:
case TextureDimension.CubeArray:
Graphics.CopyTexture(temporaryRT, 0, 0, destination, face + slice * 6, mipLevel);
break;
}
}
//restore data
camera.targetTexture = orignalTarget;
Graphics.SetRenderTarget(orignalTarget);
RenderTexture.ReleaseTemporary(temporaryRT);
}
else
{
Debug.LogWarning("The given RenderRequest type: " + typeof(RequestData).FullName + ", is either invalid or unsupported by the current pipeline");
}
}
///
/// Standalone camera rendering. Use this to render procedural cameras.
/// This method doesn't call BeginCameraRendering and EndCameraRendering callbacks.
///
/// Render context used to record commands during execution.
/// Camera to render.
///
[Obsolete("RenderSingleCamera is obsolete, please use RenderPipeline.SubmitRenderRequest with UniversalRenderer.SingleCameraRequest as RequestData type", false)]
public static void RenderSingleCamera(ScriptableRenderContext context, Camera camera)
{
RenderSingleCameraInternal(context, camera);
}
internal static void RenderSingleCameraInternal(ScriptableRenderContext context, Camera camera)
{
UniversalAdditionalCameraData additionalCameraData = null;
if (IsGameCamera(camera))
camera.gameObject.TryGetComponent(out additionalCameraData);
RenderSingleCameraInternal(context, camera, ref additionalCameraData);
}
internal static void RenderSingleCameraInternal(ScriptableRenderContext context, Camera camera, ref UniversalAdditionalCameraData additionalCameraData)
{
if (additionalCameraData != null && additionalCameraData.renderType != CameraRenderType.Base)
{
Debug.LogWarning("Only Base cameras can be rendered with standalone RenderSingleCamera. Camera will be skipped.");
return;
}
InitializeCameraData(camera, additionalCameraData, true, out var cameraData);
InitializeAdditionalCameraData(camera, additionalCameraData, true, ref cameraData);
#if ADAPTIVE_PERFORMANCE_2_0_0_OR_NEWER
if (asset.useAdaptivePerformance)
ApplyAdaptivePerformance(ref cameraData);
#endif
RenderSingleCamera(context, ref cameraData);
}
static bool TryGetCullingParameters(CameraData cameraData, out ScriptableCullingParameters cullingParams)
{
#if ENABLE_VR && ENABLE_XR_MODULE
if (cameraData.xr.enabled)
{
cullingParams = cameraData.xr.cullingParams;
// Sync the FOV on the camera to match the projection from the XR device
if (!cameraData.camera.usePhysicalProperties && !XRGraphicsAutomatedTests.enabled)
cameraData.camera.fieldOfView = Mathf.Rad2Deg * Mathf.Atan(1.0f / cullingParams.stereoProjectionMatrix.m11) * 2.0f;
return true;
}
#endif
return cameraData.camera.TryGetCullingParameters(false, out cullingParams);
}
///
/// Renders a single camera. This method will do culling, setup and execution of the renderer.
///
/// Render context used to record commands during execution.
/// Camera rendering data. This might contain data inherited from a base camera.
static void RenderSingleCamera(ScriptableRenderContext context, ref CameraData cameraData)
{
Camera camera = cameraData.camera;
var renderer = cameraData.renderer;
if (renderer == null)
{
Debug.LogWarning(string.Format("Trying to render {0} with an invalid renderer. Camera rendering will be skipped.", camera.name));
return;
}
if (!TryGetCullingParameters(cameraData, out var cullingParameters))
return;
ScriptableRenderer.current = renderer;
bool isSceneViewCamera = cameraData.isSceneViewCamera;
// NOTE: Do NOT mix ProfilingScope with named CommandBuffers i.e. CommandBufferPool.Get("name").
// Currently there's an issue which results in mismatched markers.
// The named CommandBuffer will close its "profiling scope" on execution.
// That will orphan ProfilingScope markers as the named CommandBuffer markers are their parents.
// Resulting in following pattern:
// exec(cmd.start, scope.start, cmd.end) and exec(cmd.start, scope.end, cmd.end)
CommandBuffer cmd = CommandBufferPool.Get();
// TODO: move skybox code from C++ to URP in order to remove the call to context.Submit() inside DrawSkyboxPass
// Until then, we can't use nested profiling scopes with XR multipass
CommandBuffer cmdScope = cameraData.xr.enabled ? null : cmd;
ProfilingSampler sampler = Profiling.TryGetOrAddCameraSampler(camera);
using (new ProfilingScope(cmdScope, sampler)) // Enqueues a "BeginSample" command into the CommandBuffer cmd
{
renderer.Clear(cameraData.renderType);
using (new ProfilingScope(null, Profiling.Pipeline.Renderer.setupCullingParameters))
{
renderer.OnPreCullRenderPasses(in cameraData);
renderer.SetupCullingParameters(ref cullingParameters, ref cameraData);
}
context.ExecuteCommandBuffer(cmd); // Send all the commands enqueued so far in the CommandBuffer cmd, to the ScriptableRenderContext context
cmd.Clear();
SetupPerCameraShaderConstants(cmd);
// Emit scene/game view UI. The main game camera UI is always rendered, so this needs to be handled only for different camera types
if (camera.cameraType == CameraType.Reflection || camera.cameraType == CameraType.Preview)
ScriptableRenderContext.EmitGeometryForCamera(camera);
#if UNITY_EDITOR
else if (isSceneViewCamera)
ScriptableRenderContext.EmitWorldGeometryForSceneView(camera);
#endif
// Update camera motion tracking (prev matrices) from cameraData.
// Called and updated only once, as the same camera can be rendered multiple times.
// NOTE: Tracks only the current (this) camera, not shadow views or any other offscreen views.
// NOTE: Shared between both Execute and Render (RG) paths.
if(camera.TryGetComponent(out var additionalCameraData))
additionalCameraData.motionVectorsPersistentData.Update(ref cameraData);
// Update TAA persistent data based on cameraData. Most importantly resize the history render targets.
// NOTE: Persistent data is kept over multiple frames. Its life-time differs from typical resources.
// NOTE: Shared between both Execute and Render (RG) paths.
if (cameraData.taaPersistentData != null)
UpdateTemporalAATargets(ref cameraData);
RTHandles.SetReferenceSize(cameraData.cameraTargetDescriptor.width, cameraData.cameraTargetDescriptor.height);
// Do NOT use cameraData after 'InitializeRenderingData'. CameraData state may diverge otherwise.
// RenderingData takes a copy of the CameraData.
var cullResults = context.Cull(ref cullingParameters);
InitializeRenderingData(asset, ref cameraData, ref cullResults, cmd, out var renderingData);
#if ADAPTIVE_PERFORMANCE_2_0_0_OR_NEWER
if (asset.useAdaptivePerformance)
ApplyAdaptivePerformance(ref renderingData);
#endif
renderer.AddRenderPasses(ref renderingData);
if (useRenderGraph)
{
RecordAndExecuteRenderGraph(s_RenderGraph, context, ref renderingData);
renderer.FinishRenderGraphRendering(context, ref renderingData);
}
else
{
using (new ProfilingScope(null, Profiling.Pipeline.Renderer.setup))
renderer.Setup(context, ref renderingData);
// Timing scope inside
renderer.Execute(context, ref renderingData);
}
} // When ProfilingSample goes out of scope, an "EndSample" command is enqueued into CommandBuffer cmd
context.ExecuteCommandBuffer(cmd); // Sends to ScriptableRenderContext all the commands enqueued since cmd.Clear, i.e the "EndSample" command
CommandBufferPool.Release(cmd);
using (new ProfilingScope(null, Profiling.Pipeline.Context.submit))
{
if (renderer.useRenderPassEnabled && !context.SubmitForRenderPassValidation())
{
renderer.useRenderPassEnabled = false;
CoreUtils.SetKeyword(cmd, ShaderKeywordStrings.RenderPassEnabled, false);
Debug.LogWarning("Rendering command not supported inside a native RenderPass found. Falling back to non-RenderPass rendering path");
}
context.Submit(); // Actually execute the commands that we previously sent to the ScriptableRenderContext context
}
ScriptableRenderer.current = null;
}
///
/// Renders a camera stack. This method calls RenderSingleCamera for each valid camera in the stack.
/// The last camera resolves the final target to screen.
///
/// Render context used to record commands during execution.
/// Camera to render.
static void RenderCameraStack(ScriptableRenderContext context, Camera baseCamera)
{
using var profScope = new ProfilingScope(null, ProfilingSampler.Get(URPProfileId.RenderCameraStack));
baseCamera.TryGetComponent(out var baseCameraAdditionalData);
// Overlay cameras will be rendered stacked while rendering base cameras
if (baseCameraAdditionalData != null && baseCameraAdditionalData.renderType == CameraRenderType.Overlay)
return;
// Renderer contains a stack if it has additional data and the renderer supports stacking
// The renderer is checked if it supports Base camera. Since Base is the only relevant type at this moment.
var renderer = baseCameraAdditionalData?.scriptableRenderer;
bool supportsCameraStacking = renderer != null && renderer.SupportsCameraStackingType(CameraRenderType.Base);
List cameraStack = (supportsCameraStacking) ? baseCameraAdditionalData?.cameraStack : null;
bool anyPostProcessingEnabled = baseCameraAdditionalData != null && baseCameraAdditionalData.renderPostProcessing;
bool mainHdrDisplayOutputActive = HDROutputForMainDisplayIsActive();
int rendererCount = asset.m_RendererDataList.Length;
// We need to know the last active camera in the stack to be able to resolve
// rendering to screen when rendering it. The last camera in the stack is not
// necessarily the last active one as it users might disable it.
int lastActiveOverlayCameraIndex = -1;
if (cameraStack != null)
{
var baseCameraRendererType = baseCameraAdditionalData?.scriptableRenderer.GetType();
bool shouldUpdateCameraStack = false;
cameraStackRequiresDepthForPostprocessing = false;
for (int i = 0; i < cameraStack.Count; ++i)
{
Camera currCamera = cameraStack[i];
if (currCamera == null)
{
shouldUpdateCameraStack = true;
continue;
}
if (currCamera.isActiveAndEnabled)
{
currCamera.TryGetComponent(out var data);
// Checking if the base and the overlay camera is of the same renderer type.
var currCameraRendererType = data?.scriptableRenderer.GetType();
if (currCameraRendererType != baseCameraRendererType)
{
Debug.LogWarning("Only cameras with compatible renderer types can be stacked. " +
$"The camera: {currCamera.name} are using the renderer {currCameraRendererType.Name}, " +
$"but the base camera: {baseCamera.name} are using {baseCameraRendererType.Name}. Will skip rendering");
continue;
}
var overlayRenderer = data.scriptableRenderer;
// Checking if they are the same renderer type but just not supporting Overlay
if ((overlayRenderer.SupportedCameraStackingTypes() & 1 << (int)CameraRenderType.Overlay) == 0)
{
Debug.LogWarning($"The camera: {currCamera.name} is using a renderer of type {renderer.GetType().Name} which does not support Overlay cameras in it's current state.");
continue;
}
if (data == null || data.renderType != CameraRenderType.Overlay)
{
Debug.LogWarning($"Stack can only contain Overlay cameras. The camera: {currCamera.name} " +
$"has a type {data.renderType} that is not supported. Will skip rendering.");
continue;
}
cameraStackRequiresDepthForPostprocessing |= CheckPostProcessForDepth();
anyPostProcessingEnabled |= data.renderPostProcessing;
lastActiveOverlayCameraIndex = i;
}
}
if (shouldUpdateCameraStack)
{
baseCameraAdditionalData.UpdateCameraStack();
}
}
// Post-processing not supported in GLES2.
anyPostProcessingEnabled &= SystemInfo.graphicsDeviceType != GraphicsDeviceType.OpenGLES2;
bool isStackedRendering = lastActiveOverlayCameraIndex != -1;
// Prepare XR rendering
var xrActive = false;
var xrRendering = baseCameraAdditionalData?.allowXRRendering ?? true;
var xrLayout = XRSystem.NewLayout();
xrLayout.AddCamera(baseCamera, xrRendering);
// With XR multi-pass enabled, each camera can be rendered multiple times with different parameters
foreach ((Camera _, XRPass xrPass) in xrLayout.GetActivePasses())
{
if (xrPass.enabled)
{
xrActive = true;
UpdateCameraStereoMatrices(baseCamera, xrPass);
}
using (new ProfilingScope(null, Profiling.Pipeline.beginCameraRendering))
{
BeginCameraRendering(context, baseCamera);
}
// Update volumeframework before initializing additional camera data
UpdateVolumeFramework(baseCamera, baseCameraAdditionalData);
InitializeCameraData(baseCamera, baseCameraAdditionalData, !isStackedRendering, out var baseCameraData);
RenderTextureDescriptor originalTargetDesc = baseCameraData.cameraTargetDescriptor;
#if ENABLE_VR && ENABLE_XR_MODULE
if (xrPass.enabled)
{
baseCameraData.xr = xrPass;
// Helper function for updating cameraData with xrPass Data
// Need to update XRSystem using baseCameraData to handle the case where camera position is modified in BeginCameraRendering
UpdateCameraData(ref baseCameraData, baseCameraData.xr);
// Handle the case where camera position is modified in BeginCameraRendering
xrLayout.ReconfigurePass(baseCameraData.xr, baseCamera);
XRSystemUniversal.BeginLateLatching(baseCamera, baseCameraData.xrUniversal);
}
#endif
// InitializeAdditionalCameraData needs to be initialized after the cameraTargetDescriptor is set because it needs to know the
// msaa level of cameraTargetDescriptor and XR modifications.
InitializeAdditionalCameraData(baseCamera, baseCameraAdditionalData, !isStackedRendering, ref baseCameraData);
#if VISUAL_EFFECT_GRAPH_0_0_1_OR_NEWER
//It should be called before culling to prepare material. When there isn't any VisualEffect component, this method has no effect.
VFX.VFXManager.PrepareCamera(baseCamera);
#endif
#if ADAPTIVE_PERFORMANCE_2_0_0_OR_NEWER
if (asset.useAdaptivePerformance)
ApplyAdaptivePerformance(ref baseCameraData);
#endif
// update the base camera flag so that the scene depth is stored if needed by overlay cameras later in the frame
baseCameraData.postProcessingRequiresDepthTexture |= cameraStackRequiresDepthForPostprocessing;
// Check whether the camera stack final output is HDR
// This is equivalent of UniversalCameraData.isHDROutputActive but without necessiting the base camera to be the last camera in the stack.
bool hdrDisplayOutputActive = mainHdrDisplayOutputActive;
#if ENABLE_VR && ENABLE_XR_MODULE
// If we are rendering to xr then we need to look at the XR Display rather than the main non-xr display.
if (xrPass.enabled)
hdrDisplayOutputActive = xrPass.isHDRDisplayOutputActive;
#endif
bool finalOutputHDR = asset.supportsHDR && hdrDisplayOutputActive // Check whether any HDR display is active and the render pipeline asset allows HDR rendering
&& baseCamera.targetTexture == null && (baseCamera.cameraType == CameraType.Game || baseCamera.cameraType == CameraType.VR) // Check whether the stack outputs to a screen
&& baseCameraData.allowHDROutput; // Check whether the base camera allows HDR output
// Update stack-related parameters
baseCameraData.stackAnyPostProcessingEnabled = anyPostProcessingEnabled;
baseCameraData.stackLastCameraOutputToHDR = finalOutputHDR;
RenderSingleCamera(context, ref baseCameraData);
using (new ProfilingScope(null, Profiling.Pipeline.endCameraRendering))
{
EndCameraRendering(context, baseCamera);
}
// Late latching is not supported after this point
if (baseCameraData.xr.enabled)
XRSystemUniversal.EndLateLatching(baseCamera, baseCameraData.xrUniversal);
if (isStackedRendering)
{
for (int i = 0; i < cameraStack.Count; ++i)
{
var currCamera = cameraStack[i];
if (!currCamera.isActiveAndEnabled)
continue;
currCamera.TryGetComponent(out var currAdditionalCameraData);
// Camera is overlay and enabled
if (currAdditionalCameraData != null)
{
// Copy base settings from base camera data and initialize initialize remaining specific settings for this camera type.
CameraData overlayCameraData = baseCameraData;
overlayCameraData.camera = currCamera;
overlayCameraData.baseCamera = baseCamera;
UpdateCameraStereoMatrices(currAdditionalCameraData.camera, xrPass);
using (new ProfilingScope(null, Profiling.Pipeline.beginCameraRendering))
{
BeginCameraRendering(context, currCamera);
}
#if VISUAL_EFFECT_GRAPH_0_0_1_OR_NEWER
//It should be called before culling to prepare material. When there isn't any VisualEffect component, this method has no effect.
VFX.VFXManager.PrepareCamera(currCamera);
#endif
UpdateVolumeFramework(currCamera, currAdditionalCameraData);
bool lastCamera = i == lastActiveOverlayCameraIndex;
InitializeAdditionalCameraData(currCamera, currAdditionalCameraData, lastCamera, ref overlayCameraData);
overlayCameraData.stackAnyPostProcessingEnabled = anyPostProcessingEnabled;
overlayCameraData.stackLastCameraOutputToHDR = finalOutputHDR;
xrLayout.ReconfigurePass(overlayCameraData.xr, currCamera);
RenderSingleCamera(context, ref overlayCameraData);
using (new ProfilingScope(null, Profiling.Pipeline.endCameraRendering))
{
EndCameraRendering(context, currCamera);
}
}
}
}
if (baseCameraData.xr.enabled)
baseCameraData.cameraTargetDescriptor = originalTargetDesc;
}
if (xrActive)
{
CommandBuffer cmd = CommandBufferPool.Get();
XRSystem.RenderMirrorView(cmd, baseCamera);
context.ExecuteCommandBuffer(cmd);
context.Submit();
CommandBufferPool.Release(cmd);
}
XRSystem.EndLayout();
}
// Used for updating URP cameraData data struct with XRPass data.
static void UpdateCameraData(ref CameraData baseCameraData, in XRPass xr)
{
// Update cameraData viewport for XR
Rect cameraRect = baseCameraData.camera.rect;
Rect xrViewport = xr.GetViewport();
baseCameraData.pixelRect = new Rect(cameraRect.x * xrViewport.width + xrViewport.x,
cameraRect.y * xrViewport.height + xrViewport.y,
cameraRect.width * xrViewport.width,
cameraRect.height * xrViewport.height);
Rect camPixelRect = baseCameraData.pixelRect;
baseCameraData.pixelWidth = (int)System.Math.Round(camPixelRect.width + camPixelRect.x) - (int)System.Math.Round(camPixelRect.x);
baseCameraData.pixelHeight = (int)System.Math.Round(camPixelRect.height + camPixelRect.y) - (int)System.Math.Round(camPixelRect.y);
baseCameraData.aspectRatio = (float)baseCameraData.pixelWidth / (float)baseCameraData.pixelHeight;
// Update cameraData cameraTargetDescriptor for XR. This descriptor is mainly used for configuring intermediate screen space textures
var originalTargetDesc = baseCameraData.cameraTargetDescriptor;
baseCameraData.cameraTargetDescriptor = xr.renderTargetDesc;
if (baseCameraData.isHdrEnabled)
{
baseCameraData.cameraTargetDescriptor.graphicsFormat = originalTargetDesc.graphicsFormat;
}
baseCameraData.cameraTargetDescriptor.msaaSamples = originalTargetDesc.msaaSamples;
if (baseCameraData.isDefaultViewport)
{
// When viewport is default, intermediate textures created with this descriptor will have dynamic resolution enabled.
baseCameraData.cameraTargetDescriptor.useDynamicScale = true;
}
else
{
// Some effects like Vignette computes aspect ratio from width and height. We have to take viewport into consideration if it is not default viewport.
baseCameraData.cameraTargetDescriptor.width = baseCameraData.pixelWidth;
baseCameraData.cameraTargetDescriptor.height = baseCameraData.pixelHeight;
baseCameraData.cameraTargetDescriptor.useDynamicScale = false;
}
}
static void UpdateVolumeFramework(Camera camera, UniversalAdditionalCameraData additionalCameraData)
{
using var profScope = new ProfilingScope(null, ProfilingSampler.Get(URPProfileId.UpdateVolumeFramework));
// We update the volume framework for:
// * All cameras in the editor when not in playmode
// * scene cameras
// * cameras with update mode set to EveryFrame
// * cameras with update mode set to UsePipelineSettings and the URP Asset set to EveryFrame
bool shouldUpdate = camera.cameraType == CameraType.SceneView;
shouldUpdate |= additionalCameraData != null && additionalCameraData.requiresVolumeFrameworkUpdate;
#if UNITY_EDITOR
shouldUpdate |= Application.isPlaying == false;
#endif
// When we have volume updates per-frame disabled...
if (!shouldUpdate && additionalCameraData)
{
// Create a local volume stack and cache the state if it's null
if (additionalCameraData.volumeStack == null)
{
camera.UpdateVolumeStack(additionalCameraData);
}
VolumeManager.instance.stack = additionalCameraData.volumeStack;
return;
}
// When we want to update the volumes every frame...
// We destroy the volumeStack in the additional camera data, if present, to make sure
// it gets recreated and initialized if the update mode gets later changed to ViaScripting...
if (additionalCameraData && additionalCameraData.volumeStack != null)
{
camera.DestroyVolumeStack(additionalCameraData);
}
// Get the mask + trigger and update the stack
camera.GetVolumeLayerMaskAndTrigger(additionalCameraData, out LayerMask layerMask, out Transform trigger);
VolumeManager.instance.ResetMainStack();
VolumeManager.instance.Update(trigger, layerMask);
}
static bool CheckPostProcessForDepth(ref CameraData cameraData)
{
if (!cameraData.postProcessEnabled)
return false;
if ((cameraData.antialiasing == AntialiasingMode.SubpixelMorphologicalAntiAliasing || cameraData.IsTemporalAAEnabled())
&& cameraData.renderType == CameraRenderType.Base)
return true;
return CheckPostProcessForDepth();
}
static bool CheckPostProcessForDepth()
{
var stack = VolumeManager.instance.stack;
if (stack.GetComponent().IsActive())
return true;
if (stack.GetComponent().IsActive())
return true;
return false;
}
static void SetSupportedRenderingFeatures(UniversalRenderPipelineAsset pipelineAsset)
{
#if UNITY_EDITOR
SupportedRenderingFeatures.active = new SupportedRenderingFeatures()
{
reflectionProbeModes = SupportedRenderingFeatures.ReflectionProbeModes.None,
defaultMixedLightingModes = SupportedRenderingFeatures.LightmapMixedBakeModes.Subtractive,
mixedLightingModes = SupportedRenderingFeatures.LightmapMixedBakeModes.Subtractive | SupportedRenderingFeatures.LightmapMixedBakeModes.IndirectOnly | SupportedRenderingFeatures.LightmapMixedBakeModes.Shadowmask,
lightmapBakeTypes = LightmapBakeType.Baked | LightmapBakeType.Mixed | LightmapBakeType.Realtime,
lightmapsModes = LightmapsMode.CombinedDirectional | LightmapsMode.NonDirectional,
lightProbeProxyVolumes = false,
motionVectors = true,
receiveShadows = false,
reflectionProbes = false,
reflectionProbesBlendDistance = true,
particleSystemInstancing = true,
overridesEnableLODCrossFade = true
};
SceneViewDrawMode.SetupDrawMode();
#endif
SupportedRenderingFeatures.active.supportsHDR = pipelineAsset.supportsHDR;
}
static void InitializeCameraData(Camera camera, UniversalAdditionalCameraData additionalCameraData, bool resolveFinalTarget, out CameraData cameraData)
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.initializeCameraData);
cameraData = new CameraData();
InitializeStackedCameraData(camera, additionalCameraData, ref cameraData);
cameraData.camera = camera;
///////////////////////////////////////////////////////////////////
// Descriptor settings /
///////////////////////////////////////////////////////////////////
var renderer = additionalCameraData?.scriptableRenderer;
bool rendererSupportsMSAA = renderer != null && renderer.supportedRenderingFeatures.msaa;
int msaaSamples = 1;
if (camera.allowMSAA && asset.msaaSampleCount > 1 && rendererSupportsMSAA)
msaaSamples = (camera.targetTexture != null) ? camera.targetTexture.antiAliasing : asset.msaaSampleCount;
// Use XR's MSAA if camera is XR camera. XR MSAA needs special handle here because it is not per Camera.
// Multiple cameras could render into the same XR display and they should share the same MSAA level.
if (cameraData.xrRendering && rendererSupportsMSAA && camera.targetTexture == null)
msaaSamples = (int)XRSystem.GetDisplayMSAASamples();
bool needsAlphaChannel = Graphics.preserveFramebufferAlpha;
cameraData.hdrColorBufferPrecision = asset ? asset.hdrColorBufferPrecision : HDRColorBufferPrecision._32Bits;
cameraData.cameraTargetDescriptor = CreateRenderTextureDescriptor(camera, ref cameraData,
cameraData.isHdrEnabled, cameraData.hdrColorBufferPrecision, msaaSamples, needsAlphaChannel, cameraData.requiresOpaqueTexture);
}
///
/// Initialize camera data settings common for all cameras in the stack. Overlay cameras will inherit
/// settings from base camera.
///
/// Base camera to inherit settings from.
/// Component that contains additional base camera data.
/// Camera data to initialize setttings.
static void InitializeStackedCameraData(Camera baseCamera, UniversalAdditionalCameraData baseAdditionalCameraData, ref CameraData cameraData)
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.initializeStackedCameraData);
var settings = asset;
cameraData.targetTexture = baseCamera.targetTexture;
cameraData.cameraType = baseCamera.cameraType;
bool isSceneViewCamera = cameraData.isSceneViewCamera;
///////////////////////////////////////////////////////////////////
// Environment and Post-processing settings /
///////////////////////////////////////////////////////////////////
if (isSceneViewCamera)
{
cameraData.volumeLayerMask = 1; // "Default"
cameraData.volumeTrigger = null;
cameraData.isStopNaNEnabled = false;
cameraData.isDitheringEnabled = false;
cameraData.antialiasing = AntialiasingMode.None;
cameraData.antialiasingQuality = AntialiasingQuality.High;
cameraData.xrRendering = false;
cameraData.allowHDROutput = false;
}
else if (baseAdditionalCameraData != null)
{
cameraData.volumeLayerMask = baseAdditionalCameraData.volumeLayerMask;
cameraData.volumeTrigger = baseAdditionalCameraData.volumeTrigger == null ? baseCamera.transform : baseAdditionalCameraData.volumeTrigger;
cameraData.isStopNaNEnabled = baseAdditionalCameraData.stopNaN && SystemInfo.graphicsShaderLevel >= 35;
cameraData.isDitheringEnabled = baseAdditionalCameraData.dithering;
cameraData.antialiasing = baseAdditionalCameraData.antialiasing;
cameraData.antialiasingQuality = baseAdditionalCameraData.antialiasingQuality;
cameraData.xrRendering = baseAdditionalCameraData.allowXRRendering && XRSystem.displayActive;
cameraData.allowHDROutput = baseAdditionalCameraData.allowHDROutput;
}
else
{
cameraData.volumeLayerMask = 1; // "Default"
cameraData.volumeTrigger = null;
cameraData.isStopNaNEnabled = false;
cameraData.isDitheringEnabled = false;
cameraData.antialiasing = AntialiasingMode.None;
cameraData.antialiasingQuality = AntialiasingQuality.High;
cameraData.xrRendering = XRSystem.displayActive;
cameraData.allowHDROutput = true;
}
///////////////////////////////////////////////////////////////////
// Settings that control output of the camera /
///////////////////////////////////////////////////////////////////
cameraData.isHdrEnabled = baseCamera.allowHDR && settings.supportsHDR;
cameraData.allowHDROutput &= settings.supportsHDR;
Rect cameraRect = baseCamera.rect;
cameraData.pixelRect = baseCamera.pixelRect;
cameraData.pixelWidth = baseCamera.pixelWidth;
cameraData.pixelHeight = baseCamera.pixelHeight;
cameraData.aspectRatio = (float)cameraData.pixelWidth / (float)cameraData.pixelHeight;
cameraData.isDefaultViewport = (!(Math.Abs(cameraRect.x) > 0.0f || Math.Abs(cameraRect.y) > 0.0f ||
Math.Abs(cameraRect.width) < 1.0f || Math.Abs(cameraRect.height) < 1.0f));
bool isScenePreviewOrReflectionCamera = cameraData.cameraType == CameraType.SceneView || cameraData.cameraType == CameraType.Preview || cameraData.cameraType == CameraType.Reflection;
// Discard variations lesser than kRenderScaleThreshold.
// Scale is only enabled for gameview.
const float kRenderScaleThreshold = 0.05f;
bool disableRenderScale = ((Mathf.Abs(1.0f - settings.renderScale) < kRenderScaleThreshold) || isScenePreviewOrReflectionCamera);
cameraData.renderScale = disableRenderScale ? 1.0f : settings.renderScale;
// Convert the upscaling filter selection from the pipeline asset into an image upscaling filter
cameraData.upscalingFilter = ResolveUpscalingFilterSelection(new Vector2(cameraData.pixelWidth, cameraData.pixelHeight), cameraData.renderScale, settings.upscalingFilter);
if (cameraData.renderScale > 1.0f)
{
cameraData.imageScalingMode = ImageScalingMode.Downscaling;
}
else if ((cameraData.renderScale < 1.0f) || (!isScenePreviewOrReflectionCamera && (cameraData.upscalingFilter == ImageUpscalingFilter.FSR)))
{
// When FSR is enabled, we still consider 100% render scale an upscaling operation. (This behavior is only intended for game view cameras)
// This allows us to run the FSR shader passes all the time since they improve visual quality even at 100% scale.
cameraData.imageScalingMode = ImageScalingMode.Upscaling;
}
else
{
cameraData.imageScalingMode = ImageScalingMode.None;
}
cameraData.fsrOverrideSharpness = settings.fsrOverrideSharpness;
cameraData.fsrSharpness = settings.fsrSharpness;
cameraData.xr = XRSystem.emptyPass;
XRSystem.SetRenderScale(cameraData.renderScale);
var commonOpaqueFlags = SortingCriteria.CommonOpaque;
var noFrontToBackOpaqueFlags = SortingCriteria.SortingLayer | SortingCriteria.RenderQueue | SortingCriteria.OptimizeStateChanges | SortingCriteria.CanvasOrder;
bool hasHSRGPU = SystemInfo.hasHiddenSurfaceRemovalOnGPU;
bool canSkipFrontToBackSorting = (baseCamera.opaqueSortMode == OpaqueSortMode.Default && hasHSRGPU) || baseCamera.opaqueSortMode == OpaqueSortMode.NoDistanceSort;
cameraData.defaultOpaqueSortFlags = canSkipFrontToBackSorting ? noFrontToBackOpaqueFlags : commonOpaqueFlags;
cameraData.captureActions = CameraCaptureBridge.GetCaptureActions(baseCamera);
}
///
/// Initialize settings that can be different for each camera in the stack.
///
/// Camera to initialize settings from.
/// Additional camera data component to initialize settings from.
/// True if this is the last camera in the stack and rendering should resolve to camera target.
/// Settings to be initilized.
static void InitializeAdditionalCameraData(Camera camera, UniversalAdditionalCameraData additionalCameraData, bool resolveFinalTarget, ref CameraData cameraData)
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.initializeAdditionalCameraData);
var settings = asset;
bool anyShadowsEnabled = settings.supportsMainLightShadows || settings.supportsAdditionalLightShadows;
cameraData.maxShadowDistance = Mathf.Min(settings.shadowDistance, camera.farClipPlane);
cameraData.maxShadowDistance = (anyShadowsEnabled && cameraData.maxShadowDistance >= camera.nearClipPlane) ? cameraData.maxShadowDistance : 0.0f;
bool isSceneViewCamera = cameraData.isSceneViewCamera;
if (isSceneViewCamera)
{
cameraData.renderType = CameraRenderType.Base;
cameraData.clearDepth = true;
cameraData.postProcessEnabled = CoreUtils.ArePostProcessesEnabled(camera);
cameraData.requiresDepthTexture = settings.supportsCameraDepthTexture;
cameraData.requiresOpaqueTexture = settings.supportsCameraOpaqueTexture;
cameraData.renderer = asset.scriptableRenderer;
cameraData.useScreenCoordOverride = false;
cameraData.screenSizeOverride = cameraData.pixelRect.size;
cameraData.screenCoordScaleBias = Vector2.one;
}
else if (additionalCameraData != null)
{
cameraData.renderType = additionalCameraData.renderType;
cameraData.clearDepth = (additionalCameraData.renderType != CameraRenderType.Base) ? additionalCameraData.clearDepth : true;
cameraData.postProcessEnabled = additionalCameraData.renderPostProcessing;
cameraData.maxShadowDistance = (additionalCameraData.renderShadows) ? cameraData.maxShadowDistance : 0.0f;
cameraData.requiresDepthTexture = additionalCameraData.requiresDepthTexture;
cameraData.requiresOpaqueTexture = additionalCameraData.requiresColorTexture;
cameraData.renderer = additionalCameraData.scriptableRenderer;
cameraData.useScreenCoordOverride = additionalCameraData.useScreenCoordOverride;
cameraData.screenSizeOverride = additionalCameraData.screenSizeOverride;
cameraData.screenCoordScaleBias = additionalCameraData.screenCoordScaleBias;
}
else
{
cameraData.renderType = CameraRenderType.Base;
cameraData.clearDepth = true;
cameraData.postProcessEnabled = false;
cameraData.requiresDepthTexture = settings.supportsCameraDepthTexture;
cameraData.requiresOpaqueTexture = settings.supportsCameraOpaqueTexture;
cameraData.renderer = asset.scriptableRenderer;
cameraData.useScreenCoordOverride = false;
cameraData.screenSizeOverride = cameraData.pixelRect.size;
cameraData.screenCoordScaleBias = Vector2.one;
}
// Disables post if GLes2
cameraData.postProcessEnabled &= SystemInfo.graphicsDeviceType != GraphicsDeviceType.OpenGLES2;
cameraData.requiresDepthTexture |= isSceneViewCamera;
cameraData.postProcessingRequiresDepthTexture = CheckPostProcessForDepth(ref cameraData);
cameraData.resolveFinalTarget = resolveFinalTarget;
// Disable depth and color copy. We should add it in the renderer instead to avoid performance pitfalls
// of camera stacking breaking render pass execution implicitly.
bool isOverlayCamera = (cameraData.renderType == CameraRenderType.Overlay);
if (isOverlayCamera)
{
cameraData.requiresOpaqueTexture = false;
}
// NOTE: TAA depends on XR modifications of cameraTargetDescriptor.
if (additionalCameraData != null)
UpdateTemporalAAData(ref cameraData, additionalCameraData);
Matrix4x4 projectionMatrix = camera.projectionMatrix;
// Overlay cameras inherit viewport from base.
// If the viewport is different between them we might need to patch the projection to adjust aspect ratio
// matrix to prevent squishing when rendering objects in overlay cameras.
if (isOverlayCamera && !camera.orthographic && cameraData.pixelRect != camera.pixelRect)
{
// m00 = (cotangent / aspect), therefore m00 * aspect gives us cotangent.
float cotangent = camera.projectionMatrix.m00 * camera.aspect;
// Get new m00 by dividing by base camera aspectRatio.
float newCotangent = cotangent / cameraData.aspectRatio;
projectionMatrix.m00 = newCotangent;
}
// TAA debug settings
// Affects the jitter set just below. Do not move.
ApplyTaaRenderingDebugOverrides(ref cameraData.taaSettings);
// Depends on the cameraTargetDesc, size and MSAA also XR modifications of those.
Matrix4x4 jitterMat = TemporalAA.CalculateJitterMatrix(ref cameraData);
cameraData.SetViewProjectionAndJitterMatrix(camera.worldToCameraMatrix, projectionMatrix, jitterMat);
cameraData.worldSpaceCameraPos = camera.transform.position;
var backgroundColorSRGB = camera.backgroundColor;
// Get the background color from preferences if preview camera
#if UNITY_EDITOR
if (camera.cameraType == CameraType.Preview && camera.clearFlags != CameraClearFlags.SolidColor)
{
backgroundColorSRGB = CoreRenderPipelinePreferences.previewBackgroundColor;
}
#endif
cameraData.backgroundColor = CoreUtils.ConvertSRGBToActiveColorSpace(backgroundColorSRGB);
cameraData.stackAnyPostProcessingEnabled = cameraData.postProcessEnabled;
cameraData.stackLastCameraOutputToHDR = cameraData.isHDROutputActive;
}
static void InitializeRenderingData(UniversalRenderPipelineAsset settings, ref CameraData cameraData, ref CullingResults cullResults,
CommandBuffer cmd, out RenderingData renderingData)
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.initializeRenderingData);
var isForwardPlus = cameraData.renderer is UniversalRenderer { renderingModeActual: RenderingMode.ForwardPlus };
var visibleLights = cullResults.visibleLights;
int mainLightIndex = GetMainLightIndex(settings, visibleLights);
bool mainLightCastShadows = false;
bool additionalLightsCastShadows = false;
if (cameraData.maxShadowDistance > 0.0f)
{
mainLightCastShadows = (mainLightIndex != -1 && visibleLights[mainLightIndex].light != null &&
visibleLights[mainLightIndex].light.shadows != LightShadows.None);
// If Additional Light Shadows are enabled in the URP Asset
if (settings.supportsAdditionalLightShadows)
{
// If additional lights are shaded per-vertex they cannot cast shadows
if (settings.additionalLightsRenderingMode == LightRenderingMode.PerPixel || isForwardPlus)
{
for (int i = 0; i < visibleLights.Length; ++i)
{
if (i == mainLightIndex)
continue;
ref VisibleLight vl = ref visibleLights.UnsafeElementAtMutable(i);
Light light = vl.light;
// UniversalRP doesn't support additional directional light shadows yet
if ((vl.lightType == LightType.Spot || vl.lightType == LightType.Point) && light != null && light.shadows != LightShadows.None)
{
additionalLightsCastShadows = true;
break;
}
}
}
}
}
renderingData.cullResults = cullResults;
renderingData.cameraData = cameraData;
InitializeLightData(settings, visibleLights, mainLightIndex, out renderingData.lightData);
InitializeShadowData(settings, visibleLights, mainLightCastShadows, additionalLightsCastShadows && !renderingData.lightData.shadeAdditionalLightsPerVertex, isForwardPlus, out renderingData.shadowData);
InitializePostProcessingData(settings, cameraData.stackLastCameraOutputToHDR, out renderingData.postProcessingData);
renderingData.supportsDynamicBatching = settings.supportsDynamicBatching;
renderingData.perObjectData = GetPerObjectLightFlags(renderingData.lightData.additionalLightsCount, isForwardPlus);
renderingData.postProcessingEnabled = cameraData.stackAnyPostProcessingEnabled;
renderingData.commandBuffer = cmd;
CheckAndApplyDebugSettings(ref renderingData);
}
static void InitializeShadowData(UniversalRenderPipelineAsset settings, NativeArray visibleLights, bool mainLightCastShadows, bool additionalLightsCastShadows, bool isForwardPlus, out ShadowData shadowData)
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.initializeShadowData);
m_ShadowBiasData.Clear();
m_ShadowResolutionData.Clear();
for (int i = 0; i < visibleLights.Length; ++i)
{
ref VisibleLight vl = ref visibleLights.UnsafeElementAtMutable(i);
Light light = vl.light;
UniversalAdditionalLightData data = null;
if (light != null)
{
light.gameObject.TryGetComponent(out data);
}
if (data && !data.usePipelineSettings)
m_ShadowBiasData.Add(new Vector4(light.shadowBias, light.shadowNormalBias, 0.0f, 0.0f));
else
m_ShadowBiasData.Add(new Vector4(settings.shadowDepthBias, settings.shadowNormalBias, 0.0f, 0.0f));
if (data && (data.additionalLightsShadowResolutionTier == UniversalAdditionalLightData.AdditionalLightsShadowResolutionTierCustom))
{
m_ShadowResolutionData.Add((int)light.shadowResolution); // native code does not clamp light.shadowResolution between -1 and 3
}
else if (data && (data.additionalLightsShadowResolutionTier != UniversalAdditionalLightData.AdditionalLightsShadowResolutionTierCustom))
{
int resolutionTier = Mathf.Clamp(data.additionalLightsShadowResolutionTier, UniversalAdditionalLightData.AdditionalLightsShadowResolutionTierLow, UniversalAdditionalLightData.AdditionalLightsShadowResolutionTierHigh);
m_ShadowResolutionData.Add(settings.GetAdditionalLightsShadowResolution(resolutionTier));
}
else
{
m_ShadowResolutionData.Add(settings.GetAdditionalLightsShadowResolution(UniversalAdditionalLightData.AdditionalLightsShadowDefaultResolutionTier));
}
}
shadowData.bias = m_ShadowBiasData;
shadowData.resolution = m_ShadowResolutionData;
shadowData.mainLightShadowsEnabled = settings.supportsMainLightShadows && settings.mainLightRenderingMode == LightRenderingMode.PerPixel;
shadowData.supportsMainLightShadows = SystemInfo.supportsShadows && shadowData.mainLightShadowsEnabled && mainLightCastShadows;
// We no longer use screen space shadows in URP.
// This change allows us to have particles & transparent objects receive shadows.
#pragma warning disable 0618
shadowData.requiresScreenSpaceShadowResolve = false;
#pragma warning restore 0618
// On GLES2 we strip the cascade keywords from the lighting shaders, so for consistency we force disable the cascades here too
shadowData.mainLightShadowCascadesCount = SystemInfo.graphicsDeviceType == GraphicsDeviceType.OpenGLES2 ? 1 : settings.shadowCascadeCount;
shadowData.mainLightShadowmapWidth = settings.mainLightShadowmapResolution;
shadowData.mainLightShadowmapHeight = settings.mainLightShadowmapResolution;
switch (shadowData.mainLightShadowCascadesCount)
{
case 1:
shadowData.mainLightShadowCascadesSplit = new Vector3(1.0f, 0.0f, 0.0f);
break;
case 2:
shadowData.mainLightShadowCascadesSplit = new Vector3(settings.cascade2Split, 1.0f, 0.0f);
break;
case 3:
shadowData.mainLightShadowCascadesSplit = new Vector3(settings.cascade3Split.x, settings.cascade3Split.y, 0.0f);
break;
default:
shadowData.mainLightShadowCascadesSplit = settings.cascade4Split;
break;
}
shadowData.mainLightShadowCascadeBorder = settings.cascadeBorder;
shadowData.additionalLightShadowsEnabled = settings.supportsAdditionalLightShadows && (settings.additionalLightsRenderingMode == LightRenderingMode.PerPixel || isForwardPlus);
shadowData.supportsAdditionalLightShadows = SystemInfo.supportsShadows && shadowData.additionalLightShadowsEnabled && additionalLightsCastShadows;
shadowData.additionalLightsShadowmapWidth = shadowData.additionalLightsShadowmapHeight = settings.additionalLightsShadowmapResolution;
shadowData.supportsSoftShadows = settings.supportsSoftShadows && (shadowData.supportsMainLightShadows || shadowData.supportsAdditionalLightShadows);
shadowData.shadowmapDepthBufferBits = 16;
// This will be setup in AdditionalLightsShadowCasterPass.
shadowData.isKeywordAdditionalLightShadowsEnabled = false;
shadowData.isKeywordSoftShadowsEnabled = false;
}
static void InitializePostProcessingData(UniversalRenderPipelineAsset settings, bool isHDROutputActive, out PostProcessingData postProcessingData)
{
postProcessingData.gradingMode = settings.supportsHDR
? settings.colorGradingMode
: ColorGradingMode.LowDynamicRange;
if (isHDROutputActive)
postProcessingData.gradingMode = ColorGradingMode.HighDynamicRange;
postProcessingData.lutSize = settings.colorGradingLutSize;
postProcessingData.useFastSRGBLinearConversion = settings.useFastSRGBLinearConversion;
postProcessingData.supportDataDrivenLensFlare = settings.supportDataDrivenLensFlare;
}
static void InitializeLightData(UniversalRenderPipelineAsset settings, NativeArray visibleLights, int mainLightIndex, out LightData lightData)
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.initializeLightData);
int maxPerObjectAdditionalLights = UniversalRenderPipeline.maxPerObjectLights;
int maxVisibleAdditionalLights = UniversalRenderPipeline.maxVisibleAdditionalLights;
lightData.mainLightIndex = mainLightIndex;
if (settings.additionalLightsRenderingMode != LightRenderingMode.Disabled)
{
lightData.additionalLightsCount =
Math.Min((mainLightIndex != -1) ? visibleLights.Length - 1 : visibleLights.Length,
maxVisibleAdditionalLights);
lightData.maxPerObjectAdditionalLightsCount = Math.Min(settings.maxAdditionalLightsCount, maxPerObjectAdditionalLights);
}
else
{
lightData.additionalLightsCount = 0;
lightData.maxPerObjectAdditionalLightsCount = 0;
}
lightData.supportsAdditionalLights = settings.additionalLightsRenderingMode != LightRenderingMode.Disabled;
lightData.shadeAdditionalLightsPerVertex = settings.additionalLightsRenderingMode == LightRenderingMode.PerVertex;
lightData.visibleLights = visibleLights;
lightData.supportsMixedLighting = settings.supportsMixedLighting;
lightData.reflectionProbeBlending = settings.reflectionProbeBlending;
lightData.reflectionProbeBoxProjection = settings.reflectionProbeBoxProjection;
lightData.supportsLightLayers = RenderingUtils.SupportsLightLayers(SystemInfo.graphicsDeviceType) && settings.useRenderingLayers;
}
private static void ApplyTaaRenderingDebugOverrides(ref TemporalAA.Settings taaSettings)
{
var debugDisplaySettings = UniversalRenderPipelineDebugDisplaySettings.Instance;
DebugDisplaySettingsRendering renderingSettings = debugDisplaySettings.renderingSettings;
switch (renderingSettings.taaDebugMode)
{
case DebugDisplaySettingsRendering.TaaDebugMode.ShowClampedHistory:
taaSettings.m_FrameInfluence = 0;
break;
case DebugDisplaySettingsRendering.TaaDebugMode.ShowRawFrame:
taaSettings.m_FrameInfluence = 1;
break;
case DebugDisplaySettingsRendering.TaaDebugMode.ShowRawFrameNoJitter:
taaSettings.m_FrameInfluence = 1;
taaSettings.jitterScale = 0;
break;
}
}
private static void UpdateTemporalAAData(ref CameraData cameraData, UniversalAdditionalCameraData additionalCameraData)
{
// Initialize shared TAA target desc.
ref var desc = ref cameraData.cameraTargetDescriptor;
cameraData.taaPersistentData = additionalCameraData.taaPersistentData;
cameraData.taaPersistentData.Init(desc.width, desc.height, desc.volumeDepth, desc.graphicsFormat, desc.vrUsage, desc.dimension);
// Update TAA settings
ref var taaSettings = ref additionalCameraData.taaSettings;
cameraData.taaSettings = taaSettings;
// Decrease history clear counter. Typically clear is only 1 frame, but can be many for XR multipass eyes!
taaSettings.resetHistoryFrames -= taaSettings.resetHistoryFrames > 0 ? 1 : 0;
}
private static void UpdateTemporalAATargets(ref CameraData cameraData)
{
if (cameraData.IsTemporalAAEnabled())
{
bool xrMultipassEnabled = false;
#if ENABLE_VR && ENABLE_XR_MODULE
xrMultipassEnabled = cameraData.xr.enabled && !cameraData.xr.singlePassEnabled;
#endif
bool allocation = cameraData.taaPersistentData.AllocateTargets(xrMultipassEnabled);
// Fill new history with current frame
// XR Multipass renders a "frame" per eye
if(allocation)
cameraData.taaSettings.resetHistoryFrames += xrMultipassEnabled ? 2 : 1;
}
else
cameraData.taaPersistentData.DeallocateTargets();
}
static void UpdateCameraStereoMatrices(Camera camera, XRPass xr)
{
#if ENABLE_VR && ENABLE_XR_MODULE
if (xr.enabled)
{
if (xr.singlePassEnabled)
{
for (int i = 0; i < Mathf.Min(2, xr.viewCount); i++)
{
camera.SetStereoProjectionMatrix((Camera.StereoscopicEye)i, xr.GetProjMatrix(i));
camera.SetStereoViewMatrix((Camera.StereoscopicEye)i, xr.GetViewMatrix(i));
}
}
else
{
camera.SetStereoProjectionMatrix((Camera.StereoscopicEye)xr.multipassId, xr.GetProjMatrix(0));
camera.SetStereoViewMatrix((Camera.StereoscopicEye)xr.multipassId, xr.GetViewMatrix(0));
}
}
#endif
}
static PerObjectData GetPerObjectLightFlags(int additionalLightsCount, bool isForwardPlus)
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.getPerObjectLightFlags);
var configuration = PerObjectData.Lightmaps | PerObjectData.LightProbe | PerObjectData.OcclusionProbe | PerObjectData.ShadowMask;
if (!isForwardPlus)
{
configuration |= PerObjectData.ReflectionProbes | PerObjectData.LightData;
}
if (additionalLightsCount > 0 && !isForwardPlus)
{
// In this case we also need per-object indices (unity_LightIndices)
if (!RenderingUtils.useStructuredBuffer)
configuration |= PerObjectData.LightIndices;
}
return configuration;
}
// Main Light is always a directional light
static int GetMainLightIndex(UniversalRenderPipelineAsset settings, NativeArray visibleLights)
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.getMainLightIndex);
int totalVisibleLights = visibleLights.Length;
if (totalVisibleLights == 0 || settings.mainLightRenderingMode != LightRenderingMode.PerPixel)
return -1;
Light sunLight = RenderSettings.sun;
int brightestDirectionalLightIndex = -1;
float brightestLightIntensity = 0.0f;
for (int i = 0; i < totalVisibleLights; ++i)
{
ref VisibleLight currVisibleLight = ref visibleLights.UnsafeElementAtMutable(i);
Light currLight = currVisibleLight.light;
// Particle system lights have the light property as null. We sort lights so all particles lights
// come last. Therefore, if first light is particle light then all lights are particle lights.
// In this case we either have no main light or already found it.
if (currLight == null)
break;
if (currVisibleLight.lightType == LightType.Directional)
{
// Sun source needs be a directional light
if (currLight == sunLight)
return i;
// In case no sun light is present we will return the brightest directional light
if (currLight.intensity > brightestLightIntensity)
{
brightestLightIntensity = currLight.intensity;
brightestDirectionalLightIndex = i;
}
}
}
return brightestDirectionalLightIndex;
}
static void SetupPerFrameShaderConstants()
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.setupPerFrameShaderConstants);
// Required for 2D Unlit Shadergraph master node as it doesn't currently support hidden properties.
Shader.SetGlobalColor(ShaderPropertyId.rendererColor, Color.white);
if (asset.lodCrossFadeDitheringType == LODCrossFadeDitheringType.BayerMatrix)
{
Shader.SetGlobalFloat(ShaderPropertyId.ditheringTextureInvSize, 1.0f / asset.textures.bayerMatrixTex.width);
Shader.SetGlobalTexture(ShaderPropertyId.ditheringTexture, asset.textures.bayerMatrixTex);
}
else if (asset.lodCrossFadeDitheringType == LODCrossFadeDitheringType.BlueNoise)
{
Shader.SetGlobalFloat(ShaderPropertyId.ditheringTextureInvSize, 1.0f / asset.textures.blueNoise64LTex.width);
Shader.SetGlobalTexture(ShaderPropertyId.ditheringTexture, asset.textures.blueNoise64LTex);
}
}
static void SetupPerCameraShaderConstants(CommandBuffer cmd)
{
using var profScope = new ProfilingScope(null, Profiling.Pipeline.setupPerCameraShaderConstants);
// When glossy reflections are OFF in the shader we set a constant color to use as indirect specular
SphericalHarmonicsL2 ambientSH = RenderSettings.ambientProbe;
Color linearGlossyEnvColor = new Color(ambientSH[0, 0], ambientSH[1, 0], ambientSH[2, 0]) * RenderSettings.reflectionIntensity;
Color glossyEnvColor = CoreUtils.ConvertLinearToActiveColorSpace(linearGlossyEnvColor);
cmd.SetGlobalVector(ShaderPropertyId.glossyEnvironmentColor, glossyEnvColor);
// Used as fallback cubemap for reflections
cmd.SetGlobalTexture(ShaderPropertyId.glossyEnvironmentCubeMap, ReflectionProbe.defaultTexture);
cmd.SetGlobalVector(ShaderPropertyId.glossyEnvironmentCubeMapHDR, ReflectionProbe.defaultTextureHDRDecodeValues);
// Ambient
cmd.SetGlobalVector(ShaderPropertyId.ambientSkyColor, CoreUtils.ConvertSRGBToActiveColorSpace(RenderSettings.ambientSkyColor));
cmd.SetGlobalVector(ShaderPropertyId.ambientEquatorColor, CoreUtils.ConvertSRGBToActiveColorSpace(RenderSettings.ambientEquatorColor));
cmd.SetGlobalVector(ShaderPropertyId.ambientGroundColor, CoreUtils.ConvertSRGBToActiveColorSpace(RenderSettings.ambientGroundColor));
// Used when subtractive mode is selected
cmd.SetGlobalVector(ShaderPropertyId.subtractiveShadowColor, CoreUtils.ConvertSRGBToActiveColorSpace(RenderSettings.subtractiveShadowColor));
}
static void CheckAndApplyDebugSettings(ref RenderingData renderingData)
{
var debugDisplaySettings = UniversalRenderPipelineDebugDisplaySettings.Instance;
ref CameraData cameraData = ref renderingData.cameraData;
if (debugDisplaySettings.AreAnySettingsActive && !cameraData.isPreviewCamera)
{
DebugDisplaySettingsRendering renderingSettings = debugDisplaySettings.renderingSettings;
int msaaSamples = cameraData.cameraTargetDescriptor.msaaSamples;
if (!renderingSettings.enableMsaa)
msaaSamples = 1;
if (!renderingSettings.enableHDR)
cameraData.isHdrEnabled = false;
if (!debugDisplaySettings.IsPostProcessingAllowed)
cameraData.postProcessEnabled = false;
cameraData.hdrColorBufferPrecision = asset ? asset.hdrColorBufferPrecision : HDRColorBufferPrecision._32Bits;
cameraData.cameraTargetDescriptor.graphicsFormat = MakeRenderTextureGraphicsFormat(cameraData.isHdrEnabled, cameraData.hdrColorBufferPrecision, true);
cameraData.cameraTargetDescriptor.msaaSamples = msaaSamples;
}
}
///
/// Returns the best supported image upscaling filter based on the provided upscaling filter selection
///
/// Size of the final image
/// Scale being applied to the final image size
/// Upscaling filter selected by the user
/// Either the original filter provided, or the best replacement available
static ImageUpscalingFilter ResolveUpscalingFilterSelection(Vector2 imageSize, float renderScale, UpscalingFilterSelection selection)
{
// By default we just use linear filtering since it's the most compatible choice
ImageUpscalingFilter filter = ImageUpscalingFilter.Linear;
// Fall back to the automatic filter if FSR was selected, but isn't supported on the current platform
// TODO: Investigate how to make FSR work with HDR output.
if ((selection == UpscalingFilterSelection.FSR) && (!FSRUtils.IsSupported()))
{
selection = UpscalingFilterSelection.Auto;
}
switch (selection)
{
case UpscalingFilterSelection.Auto:
{
// The user selected "auto" for their upscaling filter so we should attempt to choose the best filter
// for the current situation. When the current resolution and render scale are compatible with integer
// scaling we use the point sampling filter. Otherwise we just use the default filter (linear).
float pixelScale = (1.0f / renderScale);
bool isIntegerScale = Mathf.Approximately((pixelScale - Mathf.Floor(pixelScale)), 0.0f);
if (isIntegerScale)
{
float widthScale = (imageSize.x / pixelScale);
float heightScale = (imageSize.y / pixelScale);
bool isImageCompatible = (Mathf.Approximately((widthScale - Mathf.Floor(widthScale)), 0.0f) &&
Mathf.Approximately((heightScale - Mathf.Floor(heightScale)), 0.0f));
if (isImageCompatible)
{
filter = ImageUpscalingFilter.Point;
}
}
break;
}
case UpscalingFilterSelection.Linear:
{
// Do nothing since linear is already the default
break;
}
case UpscalingFilterSelection.Point:
{
filter = ImageUpscalingFilter.Point;
break;
}
case UpscalingFilterSelection.FSR:
{
filter = ImageUpscalingFilter.FSR;
break;
}
}
return filter;
}
///
/// Checks if the hardware (main display and platform) and the render pipeline support HDR.
///
/// True if the main display and platform support HDR and HDR output is enabled on the platform.
internal static bool HDROutputForMainDisplayIsActive()
{
bool hdrOutputSupported = SystemInfo.hdrDisplaySupportFlags.HasFlag(HDRDisplaySupportFlags.Supported) && asset.supportsHDR;
bool hdrOutputActive = HDROutputSettings.main.available && HDROutputSettings.main.active;
// TODO: Until we can test it, disable for mobile
return !Application.isMobilePlatform && hdrOutputSupported && hdrOutputActive;
}
// We only want to enable HDR Output for the game view once
// since the game itself might want to control this
internal bool enableHDROnce = true;
///
/// Checks if any of the display devices we can output to are HDR capable and enabled.
///
/// Return true if any of the display devices we can output HDR to have enabled HDR output
internal static bool HDROutputForAnyDisplayIsActive()
{
bool hdrDisplayOutputActive = HDROutputForMainDisplayIsActive();
#if ENABLE_VR && ENABLE_XR_MODULE
// If we are rendering to xr then we need to look at the XR Display rather than the main non-xr display.
if (XRSystem.displayActive)
{
hdrDisplayOutputActive |= XRSystem.isHDRDisplayOutputActive;
}
#endif
return hdrDisplayOutputActive;
}
///
/// Configures the render pipeline to render to HDR output or disables HDR output.
///
#if UNITY_2021_1_OR_NEWER
void SetHDRState(List cameras)
#else
void SetHDRState(Camera[] cameras)
#endif
{
bool hdrOutputActive = HDROutputSettings.main.available && HDROutputSettings.main.active;
// If the pipeline doesn't support HDR rendering, output to SDR.
bool supportsSwitchingHDR = SystemInfo.hdrDisplaySupportFlags.HasFlag(HDRDisplaySupportFlags.RuntimeSwitchable);
bool switchHDRToSDR = supportsSwitchingHDR && !asset.supportsHDR && hdrOutputActive;
if (switchHDRToSDR)
{
HDROutputSettings.main.RequestHDRModeChange(false);
}
#if UNITY_EDITOR
bool requestedHDRModeChange = false;
// Automatically switch to HDR in the editor if it's available
if (supportsSwitchingHDR && asset.supportsHDR && PlayerSettings.useHDRDisplay && HDROutputSettings.main.available)
{
#if UNITY_2021_1_OR_NEWER
int cameraCount = cameras.Count;
#else
int cameraCount = cameras.Length;
#endif
if (cameraCount > 0 && cameras[0].cameraType != CameraType.Game)
{
requestedHDRModeChange = hdrOutputActive;
HDROutputSettings.main.RequestHDRModeChange(false);
}
else if (enableHDROnce)
{
requestedHDRModeChange = !hdrOutputActive;
HDROutputSettings.main.RequestHDRModeChange(true);
enableHDROnce = false;
}
}
if (requestedHDRModeChange || switchHDRToSDR)
{
// Repaint scene views and game views so the HDR mode request is applied
UnityEditorInternal.InternalEditorUtility.RepaintAllViews();
}
#endif
// Make sure HDR auto tonemap is off if the URP is handling it
if (hdrOutputActive)
{
HDROutputSettings.main.automaticHDRTonemapping = false;
}
}
internal static void GetHDROutputLuminanceParameters(HDROutputUtils.HDRDisplayInformation hdrDisplayInformation, ColorGamut hdrDisplayColorGamut, Tonemapping tonemapping, out Vector4 hdrOutputParameters)
{
float minNits = hdrDisplayInformation.minToneMapLuminance;
float maxNits = hdrDisplayInformation.maxToneMapLuminance;
float paperWhite = hdrDisplayInformation.paperWhiteNits;
if (!tonemapping.detectPaperWhite.value)
{
paperWhite = tonemapping.paperWhite.value;
}
if (!tonemapping.detectBrightnessLimits.value)
{
minNits = tonemapping.minNits.value;
maxNits = tonemapping.maxNits.value;
}
hdrOutputParameters = new Vector4(minNits, maxNits, paperWhite, 1f / paperWhite);
}
internal static void GetHDROutputGradingParameters(Tonemapping tonemapping, out Vector4 hdrOutputParameters)
{
int eetfMode = 0;
float hueShift = 0.0f;
switch (tonemapping.mode.value)
{
case TonemappingMode.Neutral:
eetfMode = (int)tonemapping.neutralHDRRangeReductionMode.value;
hueShift = tonemapping.hueShiftAmount.value;
break;
case TonemappingMode.ACES:
eetfMode = (int)tonemapping.acesPreset.value;
break;
}
hdrOutputParameters = new Vector4(eetfMode, hueShift, 0.0f, 0.0f);
}
#if ADAPTIVE_PERFORMANCE_2_0_0_OR_NEWER
static void ApplyAdaptivePerformance(ref CameraData cameraData)
{
var noFrontToBackOpaqueFlags = SortingCriteria.SortingLayer | SortingCriteria.RenderQueue | SortingCriteria.OptimizeStateChanges | SortingCriteria.CanvasOrder;
if (AdaptivePerformance.AdaptivePerformanceRenderSettings.SkipFrontToBackSorting)
cameraData.defaultOpaqueSortFlags = noFrontToBackOpaqueFlags;
var MaxShadowDistanceMultiplier = AdaptivePerformance.AdaptivePerformanceRenderSettings.MaxShadowDistanceMultiplier;
cameraData.maxShadowDistance *= MaxShadowDistanceMultiplier;
var RenderScaleMultiplier = AdaptivePerformance.AdaptivePerformanceRenderSettings.RenderScaleMultiplier;
cameraData.renderScale *= RenderScaleMultiplier;
// TODO
if (!cameraData.xr.enabled)
{
cameraData.cameraTargetDescriptor.width = (int)(cameraData.camera.pixelWidth * cameraData.renderScale);
cameraData.cameraTargetDescriptor.height = (int)(cameraData.camera.pixelHeight * cameraData.renderScale);
}
var antialiasingQualityIndex = (int)cameraData.antialiasingQuality - AdaptivePerformance.AdaptivePerformanceRenderSettings.AntiAliasingQualityBias;
if (antialiasingQualityIndex < 0)
cameraData.antialiasing = AntialiasingMode.None;
cameraData.antialiasingQuality = (AntialiasingQuality)Mathf.Clamp(antialiasingQualityIndex, (int)AntialiasingQuality.Low, (int)AntialiasingQuality.High);
}
static void ApplyAdaptivePerformance(ref RenderingData renderingData)
{
if (AdaptivePerformance.AdaptivePerformanceRenderSettings.SkipDynamicBatching)
renderingData.supportsDynamicBatching = false;
var MainLightShadowmapResolutionMultiplier = AdaptivePerformance.AdaptivePerformanceRenderSettings.MainLightShadowmapResolutionMultiplier;
renderingData.shadowData.mainLightShadowmapWidth = (int)(renderingData.shadowData.mainLightShadowmapWidth * MainLightShadowmapResolutionMultiplier);
renderingData.shadowData.mainLightShadowmapHeight = (int)(renderingData.shadowData.mainLightShadowmapHeight * MainLightShadowmapResolutionMultiplier);
var MainLightShadowCascadesCountBias = AdaptivePerformance.AdaptivePerformanceRenderSettings.MainLightShadowCascadesCountBias;
renderingData.shadowData.mainLightShadowCascadesCount = Mathf.Clamp(renderingData.shadowData.mainLightShadowCascadesCount - MainLightShadowCascadesCountBias, 0, 4);
var shadowQualityIndex = AdaptivePerformance.AdaptivePerformanceRenderSettings.ShadowQualityBias;
for (int i = 0; i < shadowQualityIndex; i++)
{
if (renderingData.shadowData.supportsSoftShadows)
{
renderingData.shadowData.supportsSoftShadows = false;
continue;
}
if (renderingData.shadowData.supportsAdditionalLightShadows)
{
renderingData.shadowData.supportsAdditionalLightShadows = false;
continue;
}
if (renderingData.shadowData.supportsMainLightShadows)
{
renderingData.shadowData.supportsMainLightShadows = false;
continue;
}
break;
}
if (AdaptivePerformance.AdaptivePerformanceRenderSettings.LutBias >= 1 && renderingData.postProcessingData.lutSize == 32)
renderingData.postProcessingData.lutSize = 16;
}
#endif
///
/// Data structure describing the data for a specific render request
///
public class SingleCameraRequest
{
///
/// Target texture
///
public RenderTexture destination = null;
///
/// Target texture mip level
///
public int mipLevel = 0;
///
/// Target texture cubemap face
///
public CubemapFace face = CubemapFace.Unknown;
///
/// Target texture slice
///
public int slice = 0;
}
}
}