Graphics/FrameGraph: Add AttachmentProxy
This commit is contained in:
parent
3185e73941
commit
dfa2a0040a
|
|
@ -18,6 +18,7 @@
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
#include <unordered_set>
|
#include <unordered_set>
|
||||||
|
#include <variant>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace Nz
|
namespace Nz
|
||||||
|
|
@ -31,6 +32,7 @@ namespace Nz
|
||||||
~FrameGraph() = default;
|
~FrameGraph() = default;
|
||||||
|
|
||||||
inline std::size_t AddAttachment(FramePassAttachment attachment);
|
inline std::size_t AddAttachment(FramePassAttachment attachment);
|
||||||
|
inline std::size_t AddAttachmentProxy(std::string name, std::size_t attachmentId);
|
||||||
inline void AddBackbufferOutput(std::size_t backbufferOutput);
|
inline void AddBackbufferOutput(std::size_t backbufferOutput);
|
||||||
inline FramePass& AddPass(std::string name);
|
inline FramePass& AddPass(std::string name);
|
||||||
|
|
||||||
|
|
@ -50,6 +52,12 @@ namespace Nz
|
||||||
using PassIdToPhysicalPassIndex = std::unordered_map<std::size_t /*passId*/, std::size_t /*physicalPassId*/>;
|
using PassIdToPhysicalPassIndex = std::unordered_map<std::size_t /*passId*/, std::size_t /*physicalPassId*/>;
|
||||||
using TextureTransition = BakedFrameGraph::TextureTransition;
|
using TextureTransition = BakedFrameGraph::TextureTransition;
|
||||||
|
|
||||||
|
struct AttachmentProxy
|
||||||
|
{
|
||||||
|
std::size_t attachmentId;
|
||||||
|
std::string name;
|
||||||
|
};
|
||||||
|
|
||||||
struct Barrier
|
struct Barrier
|
||||||
{
|
{
|
||||||
std::size_t textureId;
|
std::size_t textureId;
|
||||||
|
|
@ -89,6 +97,7 @@ namespace Nz
|
||||||
std::vector<std::shared_ptr<RenderPass>> renderPasses;
|
std::vector<std::shared_ptr<RenderPass>> renderPasses;
|
||||||
std::vector<PhysicalPassData> physicalPasses;
|
std::vector<PhysicalPassData> physicalPasses;
|
||||||
std::vector<TextureData> textures;
|
std::vector<TextureData> textures;
|
||||||
|
std::vector<std::size_t> texturePool;
|
||||||
AttachmentIdToPassId attachmentLastUse;
|
AttachmentIdToPassId attachmentLastUse;
|
||||||
AttachmentIdToPassMap attachmentReadList;
|
AttachmentIdToPassMap attachmentReadList;
|
||||||
AttachmentIdToPassMap attachmentWriteList;
|
AttachmentIdToPassMap attachmentWriteList;
|
||||||
|
|
@ -105,13 +114,16 @@ namespace Nz
|
||||||
void BuildPhysicalPassDependencies(std::size_t colorAttachmentCount, bool hasDepthStencilAttachment, std::vector<RenderPass::Attachment>& renderPassAttachments, std::vector<RenderPass::SubpassDescription>& subpasses, std::vector<RenderPass::SubpassDependency>& dependencies);
|
void BuildPhysicalPassDependencies(std::size_t colorAttachmentCount, bool hasDepthStencilAttachment, std::vector<RenderPass::Attachment>& renderPassAttachments, std::vector<RenderPass::SubpassDescription>& subpasses, std::vector<RenderPass::SubpassDependency>& dependencies);
|
||||||
void BuildPhysicalPasses();
|
void BuildPhysicalPasses();
|
||||||
void BuildReadWriteList();
|
void BuildReadWriteList();
|
||||||
|
bool HasAttachment(const std::vector<FramePass::Input>& inputs, std::size_t attachmentIndex) const;
|
||||||
void RemoveDuplicatePasses();
|
void RemoveDuplicatePasses();
|
||||||
|
std::size_t ResolveAttachmentIndex(std::size_t attachmentIndex) const;
|
||||||
|
std::size_t RegisterTexture(std::size_t attachmentIndex);
|
||||||
void ReorderPasses();
|
void ReorderPasses();
|
||||||
void TraverseGraph(std::size_t passIndex);
|
void TraverseGraph(std::size_t passIndex);
|
||||||
|
|
||||||
std::vector<std::size_t> m_backbufferOutputs;
|
std::vector<std::size_t> m_backbufferOutputs;
|
||||||
std::vector<FramePass> m_framePasses;
|
std::vector<FramePass> m_framePasses;
|
||||||
std::vector<FramePassAttachment> m_attachments;
|
std::vector<std::variant<FramePassAttachment, AttachmentProxy>> m_attachments;
|
||||||
WorkData m_pending;
|
WorkData m_pending;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -16,6 +16,20 @@ namespace Nz
|
||||||
return id;
|
return id;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline std::size_t FrameGraph::AddAttachmentProxy(std::string name, std::size_t attachmentId)
|
||||||
|
{
|
||||||
|
assert(attachmentId < m_attachments.size());
|
||||||
|
assert(std::holds_alternative<FramePassAttachment>(m_attachments[attachmentId]));
|
||||||
|
|
||||||
|
std::size_t id = m_attachments.size();
|
||||||
|
m_attachments.emplace_back(AttachmentProxy {
|
||||||
|
attachmentId,
|
||||||
|
std::move(name)
|
||||||
|
});
|
||||||
|
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
inline void FrameGraph::AddBackbufferOutput(std::size_t backbufferOutput)
|
inline void FrameGraph::AddBackbufferOutput(std::size_t backbufferOutput)
|
||||||
{
|
{
|
||||||
m_backbufferOutputs.push_back(backbufferOutput);
|
m_backbufferOutputs.push_back(backbufferOutput);
|
||||||
|
|
|
||||||
|
|
@ -45,6 +45,7 @@ namespace Nz
|
||||||
m_pending.physicalPasses.clear();
|
m_pending.physicalPasses.clear();
|
||||||
m_pending.renderPasses.clear();
|
m_pending.renderPasses.clear();
|
||||||
m_pending.textures.clear();
|
m_pending.textures.clear();
|
||||||
|
m_pending.texturePool.clear();
|
||||||
|
|
||||||
BuildReadWriteList();
|
BuildReadWriteList();
|
||||||
|
|
||||||
|
|
@ -179,51 +180,13 @@ namespace Nz
|
||||||
|
|
||||||
void FrameGraph::AssignPhysicalTextures()
|
void FrameGraph::AssignPhysicalTextures()
|
||||||
{
|
{
|
||||||
std::vector<std::size_t> texturePool;
|
|
||||||
|
|
||||||
auto RegisterTexture = [&](std::size_t attachmentIndex)
|
|
||||||
{
|
|
||||||
if (auto it = m_pending.attachmentToTextures.find(attachmentIndex); it == m_pending.attachmentToTextures.end())
|
|
||||||
{
|
|
||||||
const auto& attachmentData = m_attachments[attachmentIndex];
|
|
||||||
|
|
||||||
// Fetch from reuse pool if possible
|
|
||||||
for (auto it = texturePool.begin(); it != texturePool.end(); ++it)
|
|
||||||
{
|
|
||||||
std::size_t textureId = *it;
|
|
||||||
|
|
||||||
TextureData& data = m_pending.textures[textureId];
|
|
||||||
if (data.format != attachmentData.format ||
|
|
||||||
data.width != attachmentData.width ||
|
|
||||||
data.height != attachmentData.height)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
texturePool.erase(it);
|
|
||||||
m_pending.attachmentToTextures.emplace(attachmentIndex, textureId);
|
|
||||||
|
|
||||||
return textureId;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::size_t textureId = m_pending.textures.size();
|
|
||||||
m_pending.attachmentToTextures.emplace(attachmentIndex, textureId);
|
|
||||||
|
|
||||||
TextureData& data = m_pending.textures.emplace_back();
|
|
||||||
data.format = attachmentData.format;
|
|
||||||
data.width = attachmentData.width;
|
|
||||||
data.height = attachmentData.height;
|
|
||||||
|
|
||||||
return textureId;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
return it->second;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Assign last use pass index for every attachment
|
// Assign last use pass index for every attachment
|
||||||
for (std::size_t passIndex : m_pending.passList)
|
for (std::size_t passIndex : m_pending.passList)
|
||||||
{
|
{
|
||||||
const FramePass& framePass = m_framePasses[passIndex];
|
const FramePass& framePass = m_framePasses[passIndex];
|
||||||
framePass.ForEachAttachment([&](std::size_t attachmentId)
|
framePass.ForEachAttachment([&](std::size_t attachmentId)
|
||||||
{
|
{
|
||||||
|
attachmentId = ResolveAttachmentIndex(attachmentId);
|
||||||
m_pending.attachmentLastUse[attachmentId] = passIndex;
|
m_pending.attachmentLastUse[attachmentId] = passIndex;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
@ -274,6 +237,8 @@ namespace Nz
|
||||||
|
|
||||||
framePass.ForEachAttachment([&](std::size_t attachmentId)
|
framePass.ForEachAttachment([&](std::size_t attachmentId)
|
||||||
{
|
{
|
||||||
|
attachmentId = ResolveAttachmentIndex(attachmentId);
|
||||||
|
|
||||||
std::size_t lastUsingPassId = Retrieve(m_pending.attachmentLastUse, attachmentId);
|
std::size_t lastUsingPassId = Retrieve(m_pending.attachmentLastUse, attachmentId);
|
||||||
|
|
||||||
// If this pass is the last one where this attachment is used, push the texture to the reuse pool
|
// If this pass is the last one where this attachment is used, push the texture to the reuse pool
|
||||||
|
|
@ -282,10 +247,10 @@ namespace Nz
|
||||||
std::size_t textureId = Retrieve(m_pending.attachmentToTextures, attachmentId);
|
std::size_t textureId = Retrieve(m_pending.attachmentToTextures, attachmentId);
|
||||||
|
|
||||||
// For input/output depth-stencil buffer, the same texture can be used
|
// For input/output depth-stencil buffer, the same texture can be used
|
||||||
if (texturePool.empty() || texturePool.back() != textureId)
|
if (m_pending.texturePool.empty() || m_pending.texturePool.back() != textureId)
|
||||||
{
|
{
|
||||||
assert(std::find(texturePool.begin(), texturePool.end(), textureId) == texturePool.end());
|
assert(std::find(m_pending.texturePool.begin(), m_pending.texturePool.end(), textureId) == m_pending.texturePool.end());
|
||||||
texturePool.push_back(textureId);
|
m_pending.texturePool.push_back(textureId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
@ -808,9 +773,13 @@ namespace Nz
|
||||||
|
|
||||||
for (const auto& output : subpassOutputs)
|
for (const auto& output : subpassOutputs)
|
||||||
{
|
{
|
||||||
auto inputIt = std::find_if(subpassInputs.begin(), subpassInputs.end(), [&](const auto& input) { return input.attachmentId == output.attachmentId; });
|
bool shouldLoad = false;
|
||||||
|
|
||||||
std::size_t attachmentIndex = RegisterColorOutput(output, inputIt != subpassInputs.end());
|
// load content if read-write
|
||||||
|
if (HasAttachment(subpassInputs, output.attachmentId))
|
||||||
|
shouldLoad = true;
|
||||||
|
|
||||||
|
std::size_t attachmentIndex = RegisterColorOutput(output, shouldLoad);
|
||||||
|
|
||||||
colorAttachments.push_back({
|
colorAttachments.push_back({
|
||||||
attachmentIndex,
|
attachmentIndex,
|
||||||
|
|
@ -946,6 +915,108 @@ namespace Nz
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool FrameGraph::HasAttachment(const std::vector<FramePass::Input>& inputs, std::size_t attachmentIndex) const
|
||||||
|
{
|
||||||
|
attachmentIndex = ResolveAttachmentIndex(attachmentIndex);
|
||||||
|
|
||||||
|
for (const auto& input : inputs)
|
||||||
|
{
|
||||||
|
if (ResolveAttachmentIndex(input.attachmentId) == attachmentIndex)
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t FrameGraph::RegisterTexture(std::size_t attachmentIndex)
|
||||||
|
{
|
||||||
|
if (auto it = m_pending.attachmentToTextures.find(attachmentIndex); it != m_pending.attachmentToTextures.end())
|
||||||
|
return it->second;
|
||||||
|
|
||||||
|
return std::visit([&](auto&& arg) -> std::size_t
|
||||||
|
{
|
||||||
|
using T = std::decay_t<decltype(arg)>;
|
||||||
|
if constexpr (std::is_same_v<T, FramePassAttachment>)
|
||||||
|
{
|
||||||
|
const FramePassAttachment& attachmentData = arg;
|
||||||
|
|
||||||
|
// Fetch from reuse pool if possible
|
||||||
|
for (auto it = m_pending.texturePool.begin(); it != m_pending.texturePool.end(); ++it)
|
||||||
|
{
|
||||||
|
std::size_t textureId = *it;
|
||||||
|
|
||||||
|
TextureData& data = m_pending.textures[textureId];
|
||||||
|
if (data.format != attachmentData.format ||
|
||||||
|
data.width != attachmentData.width ||
|
||||||
|
data.height != attachmentData.height)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
m_pending.texturePool.erase(it);
|
||||||
|
m_pending.attachmentToTextures.emplace(attachmentIndex, textureId);
|
||||||
|
|
||||||
|
return textureId;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t textureId = m_pending.textures.size();
|
||||||
|
m_pending.attachmentToTextures.emplace(attachmentIndex, textureId);
|
||||||
|
|
||||||
|
TextureData& data = m_pending.textures.emplace_back();
|
||||||
|
data.format = attachmentData.format;
|
||||||
|
data.width = attachmentData.width;
|
||||||
|
data.height = attachmentData.height;
|
||||||
|
|
||||||
|
return textureId;
|
||||||
|
}
|
||||||
|
else if constexpr (std::is_same_v<T, AttachmentProxy>)
|
||||||
|
{
|
||||||
|
const AttachmentProxy& proxy = arg;
|
||||||
|
|
||||||
|
std::size_t textureId = RegisterTexture(proxy.attachmentId);
|
||||||
|
m_pending.attachmentToTextures.emplace(attachmentIndex, textureId);
|
||||||
|
|
||||||
|
return textureId;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
static_assert(AlwaysFalse<T>::value, "non-exhaustive visitor");
|
||||||
|
}, m_attachments[attachmentIndex]);
|
||||||
|
}
|
||||||
|
|
||||||
|
void FrameGraph::RemoveDuplicatePasses()
|
||||||
|
{
|
||||||
|
// A way to remove duplicates from a std::vector without sorting it
|
||||||
|
std::vector<bool> seen(m_framePasses.size());
|
||||||
|
|
||||||
|
auto itRead = m_pending.passList.begin();
|
||||||
|
auto itWrite = m_pending.passList.begin();
|
||||||
|
|
||||||
|
while (itRead != m_pending.passList.end())
|
||||||
|
{
|
||||||
|
std::size_t passIndex = *itRead;
|
||||||
|
if (!seen[passIndex])
|
||||||
|
{
|
||||||
|
seen[passIndex] = true;
|
||||||
|
|
||||||
|
if (itRead != itWrite)
|
||||||
|
*itWrite++ = passIndex;
|
||||||
|
else
|
||||||
|
++itWrite;
|
||||||
|
}
|
||||||
|
|
||||||
|
++itRead;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_pending.passList.erase(itWrite, m_pending.passList.end());
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t FrameGraph::ResolveAttachmentIndex(std::size_t attachmentIndex) const
|
||||||
|
{
|
||||||
|
assert(attachmentIndex < m_attachments.size());
|
||||||
|
if (const AttachmentProxy* proxy = std::get_if<AttachmentProxy>(&m_attachments[attachmentIndex]))
|
||||||
|
return proxy->attachmentId;
|
||||||
|
|
||||||
|
return attachmentIndex;
|
||||||
|
}
|
||||||
|
|
||||||
void FrameGraph::ReorderPasses()
|
void FrameGraph::ReorderPasses()
|
||||||
{
|
{
|
||||||
/* TODO */
|
/* TODO */
|
||||||
|
|
@ -984,31 +1055,4 @@ namespace Nz
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FrameGraph::RemoveDuplicatePasses()
|
|
||||||
{
|
|
||||||
// A way to remove duplicates from a std::vector without sorting it
|
|
||||||
std::unordered_set<std::size_t> seen;
|
|
||||||
|
|
||||||
auto itRead = m_pending.passList.begin();
|
|
||||||
auto itWrite = m_pending.passList.begin();
|
|
||||||
|
|
||||||
while (itRead != m_pending.passList.end())
|
|
||||||
{
|
|
||||||
std::size_t passIndex = *itRead;
|
|
||||||
if (seen.find(passIndex) == seen.end())
|
|
||||||
{
|
|
||||||
seen.insert(passIndex);
|
|
||||||
|
|
||||||
if (itRead != itWrite)
|
|
||||||
*itWrite++ = passIndex;
|
|
||||||
else
|
|
||||||
++itWrite;
|
|
||||||
}
|
|
||||||
|
|
||||||
++itRead;
|
|
||||||
}
|
|
||||||
|
|
||||||
m_pending.passList.erase(itWrite, m_pending.passList.end());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue