您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
152 行
6.5 KiB
152 行
6.5 KiB
using UnityEngine.Rendering;
|
|
using UnityEngine.Rendering.PostProcessing;
|
|
|
|
namespace UnityEngine.Experimental.Rendering.HDPipeline
|
|
{
|
|
// This enum allow to identify which render target is used for a specific feature
|
|
public enum GBufferUsage
|
|
{
|
|
None,
|
|
SubsurfaceScattering,
|
|
Normal,
|
|
LightLayers,
|
|
ShadowMask
|
|
}
|
|
|
|
public class GBufferManager : MRTBufferManager
|
|
{
|
|
RenderPipelineMaterial m_DeferredMaterial;
|
|
// This contain the usage for all allocated buffer
|
|
protected GBufferUsage[] m_GBufferUsage;
|
|
// This is the index of the gbuffer use for shadowmask and lightlayers, if any
|
|
protected int m_ShadowMaskIndex = -1;
|
|
protected int m_LightLayers = -1;
|
|
protected HDRenderPipelineAsset m_asset;
|
|
// We need to store current set of RT to bind exactly, as we can have dynamic RT (LightLayers, ShadowMask), we allocated an array for each possible size (to avoid gardbage collect pressure)
|
|
protected RenderTargetIdentifier[][] m_RTIDsArray = new RenderTargetIdentifier[8][];
|
|
|
|
public GBufferManager(HDRenderPipelineAsset asset, RenderPipelineMaterial deferredMaterial)
|
|
: base(deferredMaterial.GetMaterialGBufferCount(asset))
|
|
{
|
|
Debug.Assert(m_BufferCount <= 8);
|
|
|
|
for (int i = 0; i < m_BufferCount; ++i)
|
|
{
|
|
m_RTIDsArray[i] = new RenderTargetIdentifier[i + 1];
|
|
}
|
|
|
|
m_DeferredMaterial = deferredMaterial;
|
|
m_asset = asset;
|
|
}
|
|
|
|
public override void CreateBuffers()
|
|
{
|
|
RenderTextureFormat[] rtFormat;
|
|
bool[] sRGBFlags;
|
|
m_DeferredMaterial.GetMaterialGBufferDescription(m_asset, out rtFormat, out sRGBFlags, out m_GBufferUsage);
|
|
|
|
for (int gbufferIndex = 0; gbufferIndex < m_BufferCount; ++gbufferIndex)
|
|
{
|
|
m_RTs[gbufferIndex] = RTHandles.Alloc(Vector2.one, colorFormat: rtFormat[gbufferIndex], sRGB: sRGBFlags[gbufferIndex], filterMode: FilterMode.Point, name: string.Format("GBuffer{0}", gbufferIndex), enableRandomWrite: gbufferIndex == 1); // normal buffer is used as RWTexture to composite decals in forward
|
|
m_RTIDs[gbufferIndex] = m_RTs[gbufferIndex].nameID;
|
|
m_TextureShaderIDs[gbufferIndex] = HDShaderIDs._GBufferTexture[gbufferIndex];
|
|
|
|
if (m_GBufferUsage[gbufferIndex] == GBufferUsage.ShadowMask)
|
|
m_ShadowMaskIndex = gbufferIndex;
|
|
else if (m_GBufferUsage[gbufferIndex] == GBufferUsage.LightLayers)
|
|
m_LightLayers = gbufferIndex;
|
|
}
|
|
}
|
|
|
|
public override void BindBufferAsTextures(CommandBuffer cmd)
|
|
{
|
|
for (int i = 0; i < m_BufferCount; ++i)
|
|
{
|
|
cmd.SetGlobalTexture(m_TextureShaderIDs[i], m_RTs[i]);
|
|
}
|
|
|
|
// Bind alias for gbuffer usage to simplify shader code (not need to check which gbuffer is the shadowmask or lightlayers)
|
|
if (m_ShadowMaskIndex >= 0)
|
|
cmd.SetGlobalTexture(HDShaderIDs._ShadowMaskTexture, m_RTs[m_ShadowMaskIndex]);
|
|
if (m_LightLayers >= 0)
|
|
cmd.SetGlobalTexture(HDShaderIDs._LightLayersTexture, m_RTs[m_LightLayers]);
|
|
else
|
|
cmd.SetGlobalTexture(HDShaderIDs._LightLayersTexture, RuntimeUtilities.whiteTexture); // This is never use but need to be bind as the read is inside a if
|
|
}
|
|
|
|
// This function will setup the required render target array. This take into account if shadow mask and light layers are enabled or not.
|
|
// Note for the future: Setup works fine as we don't have change per object (like velocity for example). If in the future it is the case
|
|
// the changing per object buffer MUST be the last one so the shader can decide if it write to it or not
|
|
public RenderTargetIdentifier[] GetBuffersRTI(FrameSettings frameSettings)
|
|
{
|
|
// We do two loop to avoid to have to allocate an array every frame
|
|
// Do a first step to know how many RT we require
|
|
int gbufferIndex = 0;
|
|
for (int i = 0; i < m_BufferCount; ++i)
|
|
{
|
|
if (m_GBufferUsage[i] == GBufferUsage.ShadowMask && !frameSettings.enableShadowMask)
|
|
continue; // Skip
|
|
|
|
if (m_GBufferUsage[i] == GBufferUsage.LightLayers && !frameSettings.enableLightLayers)
|
|
continue; // Skip
|
|
|
|
gbufferIndex++;
|
|
}
|
|
|
|
// Now select the correct array and do another loop to fill the array
|
|
RenderTargetIdentifier[] m_RTIDsArrayCurrent = m_RTIDsArray[gbufferIndex - 1];
|
|
|
|
gbufferIndex = 0;
|
|
// nameID can change from one frame to another depending on the msaa flag so so we need to update this array to be sure it's up to date.
|
|
for (int i = 0; i < m_BufferCount; ++i)
|
|
{
|
|
if (m_GBufferUsage[i] == GBufferUsage.ShadowMask && !frameSettings.enableShadowMask)
|
|
continue; // Skip
|
|
|
|
if (m_GBufferUsage[i] == GBufferUsage.LightLayers && !frameSettings.enableLightLayers)
|
|
continue; // Skip
|
|
|
|
m_RTIDsArrayCurrent[gbufferIndex] = m_RTs[i].nameID;
|
|
gbufferIndex++;
|
|
}
|
|
|
|
return m_RTIDsArrayCurrent;
|
|
}
|
|
|
|
public RTHandleSystem.RTHandle GetNormalBuffer(int index)
|
|
{
|
|
int currentIndex = 0;
|
|
for (int gbufferIndex = 0; gbufferIndex < m_BufferCount; ++gbufferIndex)
|
|
{
|
|
if (m_GBufferUsage[gbufferIndex] == GBufferUsage.Normal)
|
|
{
|
|
if (currentIndex == index)
|
|
return m_RTs[gbufferIndex];
|
|
|
|
// This is not the index we are looking for, find the next one
|
|
currentIndex++;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public RTHandleSystem.RTHandle GetSubsurfaceScatteringBuffer(int index)
|
|
{
|
|
int currentIndex = 0;
|
|
for (int gbufferIndex = 0; gbufferIndex < m_BufferCount; ++gbufferIndex)
|
|
{
|
|
if (m_GBufferUsage[gbufferIndex] == GBufferUsage.SubsurfaceScattering)
|
|
{
|
|
if (currentIndex == index)
|
|
return m_RTs[gbufferIndex];
|
|
|
|
// This is not the index we are looking for, find the next one
|
|
currentIndex++;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|
|
}
|