Implement a renderpass cache.

Technically we can generate every possible render-pass up-front based on
the possible combinations of ColorFmt and DepthFmt, but we should only
allocate what the game asks for. Save that pattern for pipelines.
This commit is contained in:
Wunkolo 2023-07-27 10:13:53 -07:00
parent bf8bb5d459
commit 37902cd9d6
2 changed files with 65 additions and 1 deletions

View file

@ -1,3 +1,6 @@
#include <map>
#include <optional>
#include "renderer.hpp"
#include "vk_api.hpp"
@ -49,6 +52,10 @@ class RendererVK final : public Renderer {
std::vector<vk::UniqueImage> topScreenImages = {};
std::vector<vk::UniqueImage> bottomScreenImages = {};
std::map<u32, vk::UniqueRenderPass> renderPassCache;
vk::RenderPass getRenderPass(PICA::ColorFmt colorFormat, std::optional<PICA::DepthFmt> depthFormat);
// Recreate the swapchain, possibly re-using the old one in the case of a resize
vk::Result recreateSwapchain(vk::SurfaceKHR surface, vk::Extent2D swapchainExtent);

View file

@ -8,6 +8,7 @@
#include "helpers.hpp"
#include "renderer_vk/vk_debug.hpp"
#include "renderer_vk/vk_memory.hpp"
#include "renderer_vk/vk_pica.hpp"
// Finds the first queue family that satisfies `queueMask` and excludes `queueExcludeMask` bits
// Returns -1 if not found
@ -24,6 +25,57 @@ static s32 findQueueFamily(
return -1;
}
vk::RenderPass RendererVK::getRenderPass(PICA::ColorFmt colorFormat, std::optional<PICA::DepthFmt> depthFormat) {
u32 renderPassHash = static_cast<u16>(colorFormat);
if (depthFormat.has_value()) {
renderPassHash |= (static_cast<u32>(depthFormat.value()) << 8);
}
// Cache hit
if (renderPassCache.contains(renderPassHash)) {
return renderPassCache.at(renderPassHash).get();
}
// Cache miss
vk::RenderPassCreateInfo renderPassInfo = {};
std::vector<vk::AttachmentDescription> renderPassAttachments = {};
vk::AttachmentDescription colorAttachment = {};
colorAttachment.format = Vulkan::colorFormatToVulkan(colorFormat);
colorAttachment.samples = vk::SampleCountFlagBits::e1;
colorAttachment.loadOp = vk::AttachmentLoadOp::eLoad;
colorAttachment.storeOp = vk::AttachmentStoreOp::eStore;
colorAttachment.stencilLoadOp = vk::AttachmentLoadOp::eLoad;
colorAttachment.stencilStoreOp = vk::AttachmentStoreOp::eStore;
colorAttachment.initialLayout = vk::ImageLayout::eColorAttachmentOptimal;
colorAttachment.finalLayout = vk::ImageLayout::eColorAttachmentOptimal;
renderPassAttachments.emplace_back(colorAttachment);
if (depthFormat.has_value()) {
vk::AttachmentDescription depthAttachment = {};
depthAttachment.format = Vulkan::depthFormatToVulkan(depthFormat.value());
depthAttachment.samples = vk::SampleCountFlagBits::e1;
depthAttachment.loadOp = vk::AttachmentLoadOp::eLoad;
depthAttachment.storeOp = vk::AttachmentStoreOp::eStore;
depthAttachment.stencilLoadOp = vk::AttachmentLoadOp::eLoad;
depthAttachment.stencilStoreOp = vk::AttachmentStoreOp::eStore;
depthAttachment.initialLayout = vk::ImageLayout::eDepthStencilAttachmentOptimal;
depthAttachment.finalLayout = vk::ImageLayout::eDepthStencilAttachmentOptimal;
renderPassAttachments.emplace_back(depthAttachment);
}
renderPassInfo.setAttachments(renderPassAttachments);
if (auto createResult = device->createRenderPassUnique(renderPassInfo); createResult.result == vk::Result::eSuccess) {
return (renderPassCache[renderPassHash] = std::move(createResult.value)).get();
} else {
Helpers::panic("Error creating render pass: %s\n", vk::to_string(createResult.result).c_str());
}
return {};
}
vk::Result RendererVK::recreateSwapchain(vk::SurfaceKHR surface, vk::Extent2D swapchainExtent) {
static constexpr u32 screenTextureWidth = 400; // Top screen is 400 pixels wide, bottom is 320
static constexpr u32 screenTextureHeight = 2 * 240; // Both screens are 240 pixels tall
@ -695,6 +747,11 @@ void RendererVK::displayTransfer(u32 inputAddr, u32 outputAddr, u32 inputSize, u
void RendererVK::textureCopy(u32 inputAddr, u32 outputAddr, u32 totalBytes, u32 inputSize, u32 outputSize, u32 flags) {}
void RendererVK::drawVertices(PICA::PrimType primType, std::span<const PICA::Vertex> vertices) {}
void RendererVK::drawVertices(PICA::PrimType primType, std::span<const PICA::Vertex> vertices) {
const u32 depthControl = regs[PICA::InternalRegs::DepthAndColorMask];
const bool depthEnable = depthControl & 1;
const vk::RenderPass curRenderPass = getRenderPass(colourBufferFormat, depthEnable ? std::make_optional(depthBufferFormat) : std::nullopt);
}
void RendererVK::screenshot(const std::string& name) {}