#include #include #include "renderer.hpp" #include "mtl_texture.hpp" #include "mtl_render_target.hpp" #include "mtl_blit_pipeline_cache.hpp" #include "mtl_draw_pipeline_cache.hpp" #include "mtl_depth_stencil_cache.hpp" #include "mtl_vertex_buffer_cache.hpp" // HACK: use the OpenGL cache #include "../renderer_gl/surface_cache.hpp" class GPU; struct Color4 { float r, g, b, a; }; class RendererMTL final : public Renderer { public: RendererMTL(GPU& gpu, const std::array& internalRegs, const std::array& externalRegs); ~RendererMTL() override; void reset() override; void display() override; void initGraphicsContext(SDL_Window* window) override; void clearBuffer(u32 startAddress, u32 endAddress, u32 value, u32 control) override; void displayTransfer(u32 inputAddr, u32 outputAddr, u32 inputSize, u32 outputSize, u32 flags) override; void textureCopy(u32 inputAddr, u32 outputAddr, u32 totalBytes, u32 inputSize, u32 outputSize, u32 flags) override; void drawVertices(PICA::PrimType primType, std::span vertices) override; void screenshot(const std::string& name) override; void deinitGraphicsContext() override; #ifdef PANDA3DS_FRONTEND_QT virtual void initGraphicsContext([[maybe_unused]] GL::Context* context) override {} #endif private: CA::MetalLayer* metalLayer; MTL::Device* device; MTL::CommandQueue* commandQueue; // Libraries MTL::Library* library; // Caches SurfaceCache colorRenderTargetCache; SurfaceCache depthStencilRenderTargetCache; SurfaceCache textureCache; Metal::BlitPipelineCache blitPipelineCache; Metal::DrawPipelineCache drawPipelineCache; Metal::DepthStencilCache depthStencilCache; Metal::VertexBufferCache vertexBufferCache; // Objects MTL::SamplerState* nearestSampler; MTL::SamplerState* linearSampler; MTL::Texture* lutTexture; MTL::DepthStencilState* defaultDepthStencilState; // Pipelines MTL::RenderPipelineState* displayPipeline; MTL::RenderPipelineState* copyToLutTexturePipeline; // Clears std::map colorClearOps; std::map depthClearOps; std::map stencilClearOps; // Active state MTL::CommandBuffer* commandBuffer = nullptr; MTL::RenderCommandEncoder* renderCommandEncoder = nullptr; MTL::Texture* lastColorTexture = nullptr; MTL::Texture* lastDepthTexture = nullptr; // Debug std::string nextRenderPassName; void createCommandBufferIfNeeded() { if (!commandBuffer) { commandBuffer = commandQueue->commandBuffer(); } } void endRenderPass() { if (renderCommandEncoder) { renderCommandEncoder->endEncoding(); renderCommandEncoder = nullptr; } } void beginRenderPassIfNeeded(MTL::RenderPassDescriptor* renderPassDescriptor, bool doesClears, MTL::Texture* colorTexture, MTL::Texture* depthTexture = nullptr) { createCommandBufferIfNeeded(); if (doesClears || !renderCommandEncoder || colorTexture != lastColorTexture || (depthTexture != lastDepthTexture && !(lastDepthTexture && !depthTexture))) { endRenderPass(); renderCommandEncoder = commandBuffer->renderCommandEncoder(renderPassDescriptor); renderCommandEncoder->setLabel(toNSString(nextRenderPassName)); lastColorTexture = colorTexture; lastDepthTexture = depthTexture; } renderPassDescriptor->release(); } void commitCommandBuffer() { if (renderCommandEncoder) { renderCommandEncoder->endEncoding(); renderCommandEncoder->release(); renderCommandEncoder = nullptr; } if (commandBuffer) { commandBuffer->commit(); commandBuffer->release(); commandBuffer = nullptr; } } template inline void clearAttachment(MTL::RenderPassDescriptor* renderPassDescriptor, MTL::Texture* texture, ClearDataT clearData, GetAttachmentT getAttachment, SetClearDataT setClearData) { bool beginRenderPass = (renderPassDescriptor == nullptr); if (!renderPassDescriptor) { renderPassDescriptor = MTL::RenderPassDescriptor::alloc()->init(); } AttachmentT* attachment = getAttachment(renderPassDescriptor); attachment->setTexture(texture); setClearData(attachment, clearData); attachment->setLoadAction(MTL::LoadActionClear); attachment->setStoreAction(MTL::StoreActionStore); if (beginRenderPass) { if (std::is_same::value) beginRenderPassIfNeeded(renderPassDescriptor, true, texture); else beginRenderPassIfNeeded(renderPassDescriptor, true, nullptr, texture); } } template inline bool clearAttachment(MTL::RenderPassDescriptor* renderPassDescriptor, MTL::Texture* texture, std::map& clearOps, GetAttachmentT getAttachment, SetClearDataT setClearData) { auto it = clearOps.find(texture); if (it != clearOps.end()) { clearAttachment(renderPassDescriptor, texture, it->second, getAttachment, setClearData); clearOps.erase(it); return true; } if (renderPassDescriptor) { AttachmentT* attachment = getAttachment(renderPassDescriptor); attachment->setTexture(texture); attachment->setLoadAction(MTL::LoadActionLoad); attachment->setStoreAction(MTL::StoreActionStore); } return false; } bool clearColor(MTL::RenderPassDescriptor* renderPassDescriptor, MTL::Texture* texture) { return clearAttachment(renderPassDescriptor, texture, colorClearOps, [](MTL::RenderPassDescriptor* renderPassDescriptor) { return renderPassDescriptor->colorAttachments()->object(0); }, [](auto attachment, auto& color) { attachment->setClearColor(MTL::ClearColor(color.r, color.g, color.b, color.a)); }); } bool clearDepth(MTL::RenderPassDescriptor* renderPassDescriptor, MTL::Texture* texture) { return clearAttachment(renderPassDescriptor, texture, depthClearOps, [](MTL::RenderPassDescriptor* renderPassDescriptor) { return renderPassDescriptor->depthAttachment(); }, [](auto attachment, auto& depth) { attachment->setClearDepth(depth); }); } bool clearStencil(MTL::RenderPassDescriptor* renderPassDescriptor, MTL::Texture* texture) { return clearAttachment(renderPassDescriptor, texture, stencilClearOps, [](MTL::RenderPassDescriptor* renderPassDescriptor) { return renderPassDescriptor->stencilAttachment(); }, [](auto attachment, auto& stencil) { attachment->setClearStencil(stencil); }); } std::optional getColorRenderTarget(u32 addr, PICA::ColorFmt format, u32 width, u32 height, bool createIfnotFound = true); Metal::DepthStencilRenderTarget& getDepthRenderTarget(); Metal::Texture& getTexture(Metal::Texture& tex); void setupTextureEnvState(MTL::RenderCommandEncoder* encoder); void bindTexturesToSlots(MTL::RenderCommandEncoder* encoder); void updateLightingLUT(MTL::RenderCommandEncoder* encoder); void updateFogLUT(MTL::RenderCommandEncoder* encoder); void textureCopyImpl(Metal::ColorRenderTarget& srcFramebuffer, Metal::ColorRenderTarget& destFramebuffer, const Math::Rect& srcRect, const Math::Rect& destRect); };