forked from Misaki/GhostEngine
Major overhaul of resource barrier and state tracking in RenderGraph: - Introduce ResourceBarrierData for explicit (layout, access, sync) tracking. - Separate aliasing and transition barriers; explicit aliasing support. - Remove BufferHint; infer buffer usage from BufferUsage flags. - Update TextureAccess/BufferAccess to include usage requirements. - Improve enums (BarrierSync, BarrierAccess, BarrierLayout) for D3D12 alignment. - Update D3D12CommandBuffer to use new barrier data and error handling. - Make D3D12DescriptorHeap a class; add ReleaseSampler to IResourceDatabase. - Reset resource pools and aliasing managers each frame. - Batch and flush barriers efficiently per pass. - Update HLSL mesh shader macros to [NumThreads]. - Remove obsolete code and improve documentation. This refactor improves correctness, extensibility, and prepares for advanced features.
317 lines
13 KiB
C#
317 lines
13 KiB
C#
using Ghost.Core;
|
|
using Ghost.Graphics.RHI;
|
|
using System.Diagnostics;
|
|
|
|
namespace Ghost.Graphics.RenderGraphModule;
|
|
|
|
[Flags]
|
|
public enum AccessFlags : byte
|
|
{
|
|
None = 0,
|
|
Read = 1 << 0,
|
|
Write = 1 << 1,
|
|
Discard = 1 << 2,
|
|
|
|
WriteAll = Write | Discard,
|
|
ReadWrite = Read | Write,
|
|
}
|
|
|
|
public interface IRenderGraphBuilder : IDisposable
|
|
{
|
|
/// <summary>
|
|
/// Enables or disables pass culling for the current context.
|
|
/// </summary>
|
|
/// <param name="value">A value indicating whether pass culling is allowed.</param>
|
|
void AllowPassCulling(bool value);
|
|
|
|
/// <summary>
|
|
/// Creates a new texture resource based on the specified desc.
|
|
/// </summary>
|
|
/// <param name="desc">A structure that defines the properties and configuration of the texture to create.</param>
|
|
/// <param name="name">The name of the texture resource.</param>
|
|
/// <returns>An identifier for the newly created texture resource.</returns>
|
|
Identifier<RGTexture> CreateTexture(in RGTextureDesc desc, string name);
|
|
|
|
/// <summary>
|
|
/// Creates a new buffer resource based on the specified desc.
|
|
/// </summary>
|
|
/// <param name="desc">A structure that defines the properties and configuration of the buffer to create.</param>
|
|
/// <param name="name">The name of the buffer resource.</param>
|
|
/// <returns>An identifier for the newly created buffer resource.</returns>
|
|
Identifier<RGBuffer> CreateBuffer(in BufferDesc desc, string name);
|
|
|
|
/// <summary>
|
|
/// Registers the specified texture for use in the current render graph pass with the given access mode.
|
|
/// </summary>
|
|
/// <param name="texture">The identifier of the texture to be used in the render graph pass.</param>
|
|
/// <param name="accessMode">The access mode specifying how the texture will be read or written during the pass.</param>
|
|
/// <returns>An identifier for the texture.</returns>
|
|
Identifier<RGTexture> UseTexture(Identifier<RGTexture> texture, AccessFlags accessMode);
|
|
|
|
/// <summary>
|
|
/// Registers the specified buffer for use in the current render graph pass with the given access mode.
|
|
/// </summary>
|
|
/// <param name="buffer">The identifier of the buffer to be used in the render graph pass.</param>
|
|
/// <param name="accessMode">The access mode specifying how the buffer will be read or written during the pass.</param>
|
|
/// <param name="hint">Optional hint about how the buffer will be used.</param>
|
|
/// <returns>An identifier for the buffer.</returns>
|
|
Identifier<RGBuffer> UseBuffer(Identifier<RGBuffer> buffer, AccessFlags accessMode);
|
|
}
|
|
|
|
public interface IRasterRenderGraphBuilder : IRenderGraphBuilder
|
|
{
|
|
/// <summary>
|
|
/// Binds a texture for random access operations within the current rendering pass.
|
|
/// </summary>
|
|
/// <param name="texture">The identifier of the texture to be used for random access.</param>
|
|
/// <returns>An identifier for the texture.</returns>
|
|
Identifier<RGTexture> UseRandomAccessTexture(Identifier<RGTexture> texture);
|
|
/// <summary>
|
|
/// Specifies that the given buffer will be used for random access operations with the specified access mode within the current context.
|
|
/// </summary>
|
|
/// <param name="buffer">An identifier for the buffer to be used for random access. Must reference a valid buffer resource.</param>
|
|
/// <returns>An identifier for the buffer.</returns>
|
|
Identifier<RGBuffer> UseRandomAccessBuffer(Identifier<RGBuffer> buffer);
|
|
|
|
/// <summary>
|
|
/// Sets the color attachment at the specified index to the given texture.
|
|
/// </summary>
|
|
/// <param name="texture">The identifier of the texture to use as the color attachment.</param>
|
|
/// <param name="index">The zero-based index of the color attachment to set.</param>
|
|
/// <param name="flags">Access flags. Default is Write (assumes partial update). Use WriteAll for fullscreen passes.</param>
|
|
void SetColorAttachment(Identifier<RGTexture> texture, int index, AccessFlags flags = AccessFlags.Write);
|
|
|
|
/// <summary>
|
|
/// Sets the depth attachment for the current render pass using the specified texture.
|
|
/// </summary>
|
|
/// <param name="texture">The identifier of the texture to use as the depth attachment. Cannot be null.</param>
|
|
/// <param name="flags">Access flags. Default is ReadWrite (assumes partial update). Use WriteAll for fullscreen passes.</param>
|
|
void SetDepthAttachment(Identifier<RGTexture> texture, AccessFlags flags = AccessFlags.ReadWrite);
|
|
|
|
/// <summary>
|
|
/// Sets the function used to render a pass with the specified pass data and render context.
|
|
/// </summary>
|
|
/// <typeparam name="TPassData">The type of data associated with the render pass.</typeparam>
|
|
/// <param name="renderFunc">The delegate that defines the rendering logic for the pass.</param>
|
|
void SetRenderFunc<TPassData>(Action<TPassData, IRasterRenderContext> renderFunc)
|
|
where TPassData : class, new();
|
|
}
|
|
|
|
public interface IComputeRenderGraphBuilder : IRenderGraphBuilder
|
|
{
|
|
/// <summary>
|
|
/// Enables or disables asynchronous compute operations.
|
|
/// </summary>
|
|
/// <param name="value">true to enable asynchronous compute; otherwise, false.</param>
|
|
void EnableAsyncCompute(bool value);
|
|
|
|
/// <summary>
|
|
/// Sets the render function to be invoked during the compute rendering process.
|
|
/// </summary>
|
|
/// <typeparam name="TPassData">The type of the data object passed to the render function.</typeparam>
|
|
/// <param name="renderFunc">The delegate that defines the rendering logic to execute.</param>
|
|
void SetRenderFunc<TPassData>(Action<TPassData, IComputeRenderContext> renderFunc)
|
|
where TPassData : class, new();
|
|
}
|
|
|
|
public interface IUnsafeRenderGraphBuilder : IRenderGraphBuilder
|
|
{
|
|
/// <summary>
|
|
/// Binds a texture for random access operations within the current rendering pass.
|
|
/// </summary>
|
|
/// <param name="texture">The identifier of the texture to be used for random access.</param>
|
|
/// <returns>An identifier for the texture.</returns>
|
|
Identifier<RGTexture> UseRandomAccessTexture(Identifier<RGTexture> texture);
|
|
/// <summary>
|
|
/// Specifies that the given buffer will be used for random access operations with the specified access mode within the current context.
|
|
/// </summary>
|
|
/// <param name="buffer">An identifier for the buffer to be used for random access. Must reference a valid buffer resource.</param>
|
|
/// <returns>An identifier for the buffer.</returns>
|
|
Identifier<RGBuffer> UseRandomAccessBuffer(Identifier<RGBuffer> buffer);
|
|
|
|
/// <summary>
|
|
/// Sets the function used to render a pass with the specified pass data and render context.
|
|
/// </summary>
|
|
/// <typeparam name="TPassData">The type of data associated with the render pass.</typeparam>
|
|
/// <param name="renderFunc">The delegate that defines the rendering logic for the pass.</param>
|
|
void SetRenderFunc<TPassData>(Action<TPassData, IUnsafeRenderContext> renderFunc)
|
|
where TPassData : class, new();
|
|
}
|
|
|
|
internal class RenderGraphBuilder : IRasterRenderGraphBuilder, IComputeRenderGraphBuilder, IUnsafeRenderGraphBuilder
|
|
{
|
|
|
|
private RenderGraph _graph = null!;
|
|
private RenderGraphPassBase _pass = null!;
|
|
private RenderGraphResourceRegistry _resources = null!;
|
|
private bool _disposed;
|
|
|
|
internal void Init(RenderGraph graph, RenderGraphPassBase pass, RenderGraphResourceRegistry resources)
|
|
{
|
|
_graph = graph;
|
|
_pass = pass;
|
|
_resources = resources;
|
|
_disposed = false;
|
|
}
|
|
|
|
private void ThrowIfDisposed()
|
|
{
|
|
ObjectDisposedException.ThrowIf(_disposed, this);
|
|
}
|
|
|
|
private Identifier<RGResource> UseResource(Identifier<RGResource> resource, AccessFlags accessFlags, RenderGraphResourceType type)
|
|
{
|
|
if (accessFlags.HasFlag(AccessFlags.Read))
|
|
{
|
|
_pass.resourceReads[(int)type].Add(resource);
|
|
_resources.AddConsumer(resource, _pass.index);
|
|
}
|
|
|
|
if (accessFlags.HasFlag(AccessFlags.Write))
|
|
{
|
|
_pass.resourceWrites[(int)type].Add(resource);
|
|
_resources.SetProducer(resource, _pass.index);
|
|
}
|
|
|
|
return resource;
|
|
}
|
|
|
|
public void AllowPassCulling(bool value)
|
|
{
|
|
_pass.allowCulling = value;
|
|
}
|
|
|
|
public void EnableAsyncCompute(bool value)
|
|
{
|
|
_pass.asyncCompute = value;
|
|
}
|
|
|
|
public Identifier<RGTexture> CreateTexture(in RGTextureDesc desc, string name)
|
|
{
|
|
ThrowIfDisposed();
|
|
|
|
var handle = _resources.CreateTexture(in desc, name);
|
|
_pass.resourceCreates[(int)RenderGraphResourceType.Texture].Add(handle.AsResource());
|
|
_resources.SetProducer(handle.AsResource(), _pass.index);
|
|
return handle;
|
|
}
|
|
|
|
public Identifier<RGBuffer> CreateBuffer(in BufferDesc desc, string name)
|
|
{
|
|
ThrowIfDisposed();
|
|
|
|
var handle = _resources.CreateBuffer(in desc, name);
|
|
_pass.resourceCreates[(int)RenderGraphResourceType.Buffer].Add(handle.AsResource());
|
|
_resources.SetProducer(handle.AsResource(), _pass.index);
|
|
return handle;
|
|
}
|
|
|
|
public Identifier<RGTexture> UseTexture(Identifier<RGTexture> texture, AccessFlags flags)
|
|
{
|
|
ThrowIfDisposed();
|
|
return UseResource(texture.AsResource(), flags, RenderGraphResourceType.Texture).AsTexture();
|
|
}
|
|
|
|
public Identifier<RGBuffer> UseBuffer(Identifier<RGBuffer> buffer, AccessFlags flags)
|
|
{
|
|
ThrowIfDisposed();
|
|
return UseResource(buffer.AsResource(), flags, RenderGraphResourceType.Buffer).AsBuffer();
|
|
}
|
|
|
|
public Identifier<RGTexture> UseRandomAccessTexture(Identifier<RGTexture> texture)
|
|
{
|
|
ThrowIfDisposed();
|
|
|
|
var resource = texture.AsResource();
|
|
UseResource(resource, AccessFlags.ReadWrite, RenderGraphResourceType.Texture);
|
|
_pass.randomAccess.Add(resource);
|
|
return texture;
|
|
}
|
|
|
|
public Identifier<RGBuffer> UseRandomAccessBuffer(Identifier<RGBuffer> buffer)
|
|
{
|
|
ThrowIfDisposed();
|
|
|
|
var resource = buffer.AsResource();
|
|
UseResource(resource, AccessFlags.ReadWrite, RenderGraphResourceType.Buffer);
|
|
_pass.randomAccess.Add(resource);
|
|
return buffer;
|
|
}
|
|
|
|
public void SetColorAttachment(Identifier<RGTexture> texture, int index, AccessFlags flags = AccessFlags.Write)
|
|
{
|
|
ThrowIfDisposed();
|
|
|
|
Debug.Assert(index >= 0 && index < _pass.colorAccess.Length, "Color attachment index out of range.");
|
|
|
|
var id = UseTexture(texture, flags);
|
|
if (_pass.colorAccess[index].id == id || _pass.colorAccess[index].id.IsInvalid)
|
|
{
|
|
_pass.maxColorIndex = Math.Max(_pass.maxColorIndex, index);
|
|
var usage = new ResourceBarrierData(BarrierLayout.RenderTarget, BarrierAccess.RenderTarget, BarrierSync.RenderTarget);
|
|
_pass.colorAccess[index] = new TextureAccess(id, flags, usage);
|
|
}
|
|
else
|
|
{
|
|
throw new InvalidOperationException($"Color attachment at index {index} is already set to a different texture.");
|
|
}
|
|
}
|
|
|
|
public void SetDepthAttachment(Identifier<RGTexture> texture, AccessFlags flags = AccessFlags.ReadWrite)
|
|
{
|
|
ThrowIfDisposed();
|
|
|
|
var id = UseTexture(texture, flags);
|
|
if (_pass.depthAccess.id == id || _pass.depthAccess.id.IsInvalid)
|
|
{
|
|
var layout = flags.HasFlag(AccessFlags.Write) ? BarrierLayout.DepthStencilWrite : BarrierLayout.DepthStencilRead;
|
|
var access = flags.HasFlag(AccessFlags.Write) ? BarrierAccess.DepthStencilWrite : BarrierAccess.DepthStencilRead;
|
|
var sync = BarrierSync.DepthStencil;
|
|
var usage = new ResourceBarrierData(layout, access, sync);
|
|
_pass.depthAccess = new TextureAccess(id, flags, usage);
|
|
}
|
|
else
|
|
{
|
|
throw new InvalidOperationException("Depth attachment is already set to a different texture.");
|
|
}
|
|
}
|
|
|
|
public void SetRenderFunc<TPassData>(Action<TPassData, IRasterRenderContext> renderFunc)
|
|
where TPassData : class, new()
|
|
{
|
|
((RasterRenderGraphPass<TPassData>)_pass).renderFunc = renderFunc;
|
|
}
|
|
|
|
public void SetRenderFunc<TPassData>(Action<TPassData, IComputeRenderContext> renderFunc)
|
|
where TPassData : class, new()
|
|
{
|
|
((ComputeRenderGraphPass<TPassData>)_pass).renderFunc = renderFunc;
|
|
}
|
|
|
|
public void SetRenderFunc<TPassData>(Action<TPassData, IUnsafeRenderContext> renderFunc)
|
|
where TPassData : class, new()
|
|
{
|
|
((UnsafeRenderGraphPass<TPassData>)_pass).renderFunc = renderFunc;
|
|
}
|
|
|
|
public void Dispose()
|
|
|
|
{
|
|
if (_disposed)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (!_pass.HasRenderFunc())
|
|
{
|
|
throw new InvalidOperationException("RenderGraphBuilder must be disposed after setting up the render function.");
|
|
}
|
|
|
|
_graph = null!;
|
|
_pass = null!;
|
|
_resources = null!;
|
|
|
|
_disposed = true;
|
|
}
|
|
}
|