using System;
using System.Collections.Generic;
using UnityEngine.Experimental.Rendering;
using UnityEngine.Experimental.Rendering.RenderGraphModule;
using UnityEngine.Profiling;
namespace UnityEngine.Rendering.Universal.Internal
{
///
/// Extension of DrawObjectPass that also output Rendering Layers Texture as second render target.
///
internal class DrawObjectsWithRenderingLayersPass : DrawObjectsPass
{
RTHandle[] m_ColorTargetIndentifiers;
RTHandle m_DepthTargetIndentifiers;
public DrawObjectsWithRenderingLayersPass(URPProfileId profilerTag, bool opaque, RenderPassEvent evt, RenderQueueRange renderQueueRange, LayerMask layerMask, StencilState stencilState, int stencilReference) :
base(profilerTag, opaque, evt, renderQueueRange, layerMask, stencilState, stencilReference)
{
m_ColorTargetIndentifiers = new RTHandle[2];
}
public void Setup(RTHandle colorAttachment, RTHandle renderingLayersTexture, RTHandle depthAttachment)
{
if (colorAttachment == null)
throw new ArgumentException("Color attachment can not be null", "colorAttachment");
if (renderingLayersTexture == null)
throw new ArgumentException("Rendering layers attachment can not be null", "renderingLayersTexture");
if (depthAttachment == null)
throw new ArgumentException("Depth attachment can not be null", "depthAttachment");
m_ColorTargetIndentifiers[0] = colorAttachment;
m_ColorTargetIndentifiers[1] = renderingLayersTexture;
m_DepthTargetIndentifiers = depthAttachment;
}
public override void Configure(CommandBuffer cmd, RenderTextureDescriptor cameraTextureDescriptor)
{
ConfigureTarget(m_ColorTargetIndentifiers, m_DepthTargetIndentifiers);
}
protected override void OnExecute(CommandBuffer cmd)
{
CoreUtils.SetKeyword(cmd, ShaderKeywordStrings.WriteRenderingLayers, true);
}
}
///
/// Draw objects into the given color and depth target
///
/// You can use this pass to render objects that have a material and/or shader
/// with the pass names UniversalForward or SRPDefaultUnlit.
///
public class DrawObjectsPass : ScriptableRenderPass
{
FilteringSettings m_FilteringSettings;
RenderStateBlock m_RenderStateBlock;
List m_ShaderTagIdList = new List();
string m_ProfilerTag;
ProfilingSampler m_ProfilingSampler;
bool m_IsOpaque;
///
/// Used to indicate if the active target of the pass is the back buffer
///
public bool m_IsActiveTargetBackBuffer; // TODO: Remove this when we remove non-RG path
///
/// Used to indicate whether transparent objects should receive shadows or not.
///
public bool m_ShouldTransparentsReceiveShadows;
PassData m_PassData;
bool m_UseDepthPriming;
static readonly int s_DrawObjectPassDataPropID = Shader.PropertyToID("_DrawObjectPassData");
///
/// Creates a new DrawObjectsPass instance.
///
///
///
///
/// The RenderPassEvent to use.
///
///
///
///
///
///
///
///
///
public DrawObjectsPass(string profilerTag, ShaderTagId[] shaderTagIds, bool opaque, RenderPassEvent evt, RenderQueueRange renderQueueRange, LayerMask layerMask, StencilState stencilState, int stencilReference)
{
base.profilingSampler = new ProfilingSampler(nameof(DrawObjectsPass));
m_PassData = new PassData();
m_ProfilerTag = profilerTag;
m_ProfilingSampler = new ProfilingSampler(profilerTag);
foreach (ShaderTagId sid in shaderTagIds)
m_ShaderTagIdList.Add(sid);
renderPassEvent = evt;
m_FilteringSettings = new FilteringSettings(renderQueueRange, layerMask);
m_RenderStateBlock = new RenderStateBlock(RenderStateMask.Nothing);
m_IsOpaque = opaque;
m_ShouldTransparentsReceiveShadows = false;
m_IsActiveTargetBackBuffer = false;
if (stencilState.enabled)
{
m_RenderStateBlock.stencilReference = stencilReference;
m_RenderStateBlock.mask = RenderStateMask.Stencil;
m_RenderStateBlock.stencilState = stencilState;
}
}
///
/// Creates a new DrawObjectsPass instance.
///
///
///
///
///
///
///
///
///
///
///
///
public DrawObjectsPass(string profilerTag, bool opaque, RenderPassEvent evt, RenderQueueRange renderQueueRange, LayerMask layerMask, StencilState stencilState, int stencilReference)
: this(profilerTag,
new ShaderTagId[] { new ShaderTagId("SRPDefaultUnlit"), new ShaderTagId("UniversalForward"), new ShaderTagId("UniversalForwardOnly") },
opaque, evt, renderQueueRange, layerMask, stencilState, stencilReference)
{ }
internal DrawObjectsPass(URPProfileId profileId, bool opaque, RenderPassEvent evt, RenderQueueRange renderQueueRange, LayerMask layerMask, StencilState stencilState, int stencilReference)
: this(profileId.GetType().Name, opaque, evt, renderQueueRange, layerMask, stencilState, stencilReference)
{
m_ProfilingSampler = ProfilingSampler.Get(profileId);
}
///
public override void Execute(ScriptableRenderContext context, ref RenderingData renderingData)
{
m_PassData.m_IsOpaque = m_IsOpaque;
m_PassData.m_RenderingData = renderingData;
m_PassData.m_RenderStateBlock = m_RenderStateBlock;
m_PassData.m_FilteringSettings = m_FilteringSettings;
m_PassData.m_ShaderTagIdList = m_ShaderTagIdList;
m_PassData.m_ProfilingSampler = m_ProfilingSampler;
m_PassData.m_IsActiveTargetBackBuffer = m_IsActiveTargetBackBuffer;
m_PassData.pass = this;
CameraSetup(renderingData.commandBuffer, m_PassData, ref renderingData);
ExecutePass(context, m_PassData, ref renderingData, renderingData.cameraData.IsCameraProjectionMatrixFlipped());
}
private static void CameraSetup(CommandBuffer cmd, PassData data, ref RenderingData renderingData)
{
if (renderingData.cameraData.renderer.useDepthPriming && data.m_IsOpaque && (renderingData.cameraData.renderType == CameraRenderType.Base || renderingData.cameraData.clearDepth))
{
data.m_RenderStateBlock.depthState = new DepthState(false, CompareFunction.Equal);
data.m_RenderStateBlock.mask |= RenderStateMask.Depth;
}
else if (data.m_RenderStateBlock.depthState.compareFunction == CompareFunction.Equal)
{
data.m_RenderStateBlock.depthState = new DepthState(true, CompareFunction.LessEqual);
data.m_RenderStateBlock.mask |= RenderStateMask.Depth;
}
}
private static void ExecutePass(ScriptableRenderContext context, PassData data, ref RenderingData renderingData, bool yFlip)
{
var cmd = renderingData.commandBuffer;
using (new ProfilingScope(cmd, data.m_ProfilingSampler))
{
// Global render pass data containing various settings.
// x,y,z are currently unused
// w is used for knowing whether the object is opaque(1) or alpha blended(0)
Vector4 drawObjectPassData = new Vector4(0.0f, 0.0f, 0.0f, (data.m_IsOpaque) ? 1.0f : 0.0f);
cmd.SetGlobalVector(s_DrawObjectPassDataPropID, drawObjectPassData);
if (data.m_RenderingData.cameraData.xrRendering && data.m_IsActiveTargetBackBuffer)
{
cmd.SetViewport(data.m_RenderingData.cameraData.xr.GetViewport());
}
// scaleBias.x = flipSign
// scaleBias.y = scale
// scaleBias.z = bias
// scaleBias.w = unused
float flipSign = yFlip ? -1.0f : 1.0f;
Vector4 scaleBias = (flipSign < 0.0f)
? new Vector4(flipSign, 1.0f, -1.0f, 1.0f)
: new Vector4(flipSign, 0.0f, 1.0f, 1.0f);
cmd.SetGlobalVector(ShaderPropertyId.scaleBiasRt, scaleBias);
// Set a value that can be used by shaders to identify when AlphaToMask functionality may be active
// The material shader alpha clipping logic requires this value in order to function correctly in all cases.
float alphaToMaskAvailable = ((renderingData.cameraData.cameraTargetDescriptor.msaaSamples > 1) && data.m_IsOpaque) ? 1.0f : 0.0f;
cmd.SetGlobalFloat(ShaderPropertyId.alphaToMaskAvailable, alphaToMaskAvailable);
// TODO RENDERGRAPH: do this as a separate pass, so no need of calling OnExecute here...
data.pass.OnExecute(cmd);
context.ExecuteCommandBuffer(cmd);
cmd.Clear();
Camera camera = renderingData.cameraData.camera;
var sortFlags = (data.m_IsOpaque) ? renderingData.cameraData.defaultOpaqueSortFlags : SortingCriteria.CommonTransparent;
if (renderingData.cameraData.renderer.useDepthPriming && data.m_IsOpaque && (renderingData.cameraData.renderType == CameraRenderType.Base || renderingData.cameraData.clearDepth))
sortFlags = SortingCriteria.SortingLayer | SortingCriteria.RenderQueue | SortingCriteria.OptimizeStateChanges | SortingCriteria.CanvasOrder;
var filterSettings = data.m_FilteringSettings;
#if UNITY_EDITOR
// When rendering the preview camera, we want the layer mask to be forced to Everything
if (renderingData.cameraData.isPreviewCamera)
{
filterSettings.layerMask = -1;
}
#endif
DrawingSettings drawSettings = RenderingUtils.CreateDrawingSettings(data.m_ShaderTagIdList, ref renderingData, sortFlags);
var activeDebugHandler = GetActiveDebugHandler(ref renderingData);
if (activeDebugHandler != null)
{
activeDebugHandler.DrawWithDebugRenderState(context, cmd, ref renderingData, ref drawSettings, ref filterSettings, ref data.m_RenderStateBlock,
(ScriptableRenderContext ctx, ref RenderingData data, ref DrawingSettings ds, ref FilteringSettings fs, ref RenderStateBlock rsb) =>
{
ctx.DrawRenderers(data.cullResults, ref ds, ref fs, ref rsb);
});
}
else
{
context.DrawRenderers(renderingData.cullResults, ref drawSettings, ref filterSettings, ref data.m_RenderStateBlock);
// Render objects that did not match any shader pass with error shader
RenderingUtils.RenderObjectsWithError(context, ref renderingData.cullResults, camera, filterSettings, SortingCriteria.None);
}
// Clean up
CoreUtils.SetKeyword(cmd, ShaderKeywordStrings.WriteRenderingLayers, false);
context.ExecuteCommandBuffer(cmd);
cmd.Clear();
}
}
private class PassData
{
internal TextureHandle m_Albedo;
internal TextureHandle m_Depth;
internal RenderingData m_RenderingData;
internal bool m_IsOpaque;
internal RenderStateBlock m_RenderStateBlock;
internal FilteringSettings m_FilteringSettings;
internal List m_ShaderTagIdList;
internal ProfilingSampler m_ProfilingSampler;
internal bool m_ShouldTransparentsReceiveShadows;
internal bool m_IsActiveTargetBackBuffer;
internal DrawObjectsPass pass;
}
internal void Render(RenderGraph renderGraph, TextureHandle colorTarget, TextureHandle depthTarget, TextureHandle mainShadowsTexture, TextureHandle additionalShadowsTexture, ref RenderingData renderingData)
{
Camera camera = renderingData.cameraData.camera;
using (var builder = renderGraph.AddRenderPass("Draw Objects Pass", out var passData,
m_ProfilingSampler))
{
passData.m_Albedo = builder.UseColorBuffer(colorTarget, 0);
passData.m_Depth = builder.UseDepthBuffer(depthTarget, DepthAccess.Write);
if (mainShadowsTexture.IsValid())
builder.ReadTexture(mainShadowsTexture);
if (additionalShadowsTexture.IsValid())
builder.ReadTexture(additionalShadowsTexture);
passData.m_RenderingData = renderingData;
builder.AllowPassCulling(false);
passData.m_IsOpaque = m_IsOpaque;
passData.m_RenderStateBlock = m_RenderStateBlock;
passData.m_FilteringSettings = m_FilteringSettings;
passData.m_ShaderTagIdList = m_ShaderTagIdList;
passData.m_ProfilingSampler = m_ProfilingSampler;
passData.m_ShouldTransparentsReceiveShadows = m_ShouldTransparentsReceiveShadows;
passData.m_IsActiveTargetBackBuffer = m_IsActiveTargetBackBuffer;
passData.pass = this;
builder.SetRenderFunc((PassData data, RenderGraphContext context) =>
{
ref var renderingData = ref data.m_RenderingData;
// TODO RENDERGRAPH figure out where to put XR proj flip logic so that it can be auto handled in render graph
#if ENABLE_VR && ENABLE_XR_MODULE
if (renderingData.cameraData.xr.enabled)
{
// SetRenderTarget might alter the internal device state(winding order).
// Non-stereo buffer is already updated internally when switching render target. We update stereo buffers here to keep the consistency.
bool renderIntoTexture = data.m_Albedo != renderingData.cameraData.xr.renderTarget;
renderingData.cameraData.PushBuiltinShaderConstantsXR(renderingData.commandBuffer, renderIntoTexture);
XRSystemUniversal.MarkShaderProperties(renderingData.commandBuffer, renderingData.cameraData.xrUniversal, renderIntoTexture);
}
#endif
// Currently we only need to call this additional pass when the user
// doesn't want transparent objects to receive shadows
if (!data.m_IsOpaque && !data.m_ShouldTransparentsReceiveShadows)
TransparentSettingsPass.ExecutePass(context.cmd, data.m_ShouldTransparentsReceiveShadows);
bool yFlip = renderingData.cameraData.IsRenderTargetProjectionMatrixFlipped(data.m_Albedo, data.m_Depth);
CameraSetup(context.cmd, data, ref renderingData);
ExecutePass(context.renderContext, data, ref renderingData, yFlip);
});
}
}
///
/// Called before ExecutePass draws the objects.
///
/// The command buffer to use.
protected virtual void OnExecute(CommandBuffer cmd) { }
}
}