#pragma once #include #include #include #include #include #include #include #include #include "PICA/float_types.hpp" #include "PICA/pica_frag_config.hpp" #include "PICA/pica_hash.hpp" #include "PICA/pica_vert_config.hpp" #include "PICA/pica_vertex.hpp" #include "PICA/regs.hpp" #include "PICA/shader_gen.hpp" #include "gl/stream_buffer.h" #include "gl_driver.hpp" #include "gl_state.hpp" #include "helpers.hpp" #include "logger.hpp" #include "renderer.hpp" #include "surface_cache.hpp" #include "textures.hpp" // More circular dependencies! class GPU; class RendererGL final : public Renderer { GLStateManager gl = {}; OpenGL::Program triangleProgram; OpenGL::Program displayProgram; // VAO for when not using accelerated vertex shaders. Contains attribute declarations matching to the PICA fixed function fragment attributes OpenGL::VertexArray defaultVAO; // VAO for when using accelerated vertex shaders. The PICA vertex shader inputs are passed as attributes without CPU processing. OpenGL::VertexArray hwShaderVAO; OpenGL::VertexBuffer vbo; // Data struct { // TEV configuration uniform locations GLint textureEnvSourceLoc = -1; GLint textureEnvOperandLoc = -1; GLint textureEnvCombinerLoc = -1; GLint textureEnvColorLoc = -1; GLint textureEnvScaleLoc = -1; // Uniform of PICA registers GLint picaRegLoc = -1; // Depth configuration uniform locations GLint depthOffsetLoc = -1; GLint depthScaleLoc = -1; GLint depthmapEnableLoc = -1; } ubershaderData; float oldDepthScale = -1.0; float oldDepthOffset = 0.0; bool oldDepthmapEnable = false; // Set by prepareForDraw, tells us whether the current draw is using hw-accelerated shader bool usingAcceleratedShader = false; bool performIndexedRender = false; bool usingShortIndices = false; // Set by prepareForDraw, metadata for indexed renders GLuint minimumIndex = 0; GLuint maximumIndex = 0; void* hwIndexBufferOffset = nullptr; // When doing hw shaders, we cache which attributes are enabled in our VAO to avoid having to enable/disable all attributes on each draw u32 previousAttributeMask = 0; // Cached pointer to the current vertex shader when using HW accelerated shaders OpenGL::Shader* generatedVertexShader = nullptr; SurfaceCache depthBufferCache; SurfaceCache colourBufferCache; SurfaceCache textureCache; // Dummy VAO/VBO for blitting the final output OpenGL::VertexArray dummyVAO; OpenGL::VertexBuffer dummyVBO; OpenGL::Texture screenTexture; OpenGL::Texture LUTTexture; OpenGL::Framebuffer screenFramebuffer; OpenGL::Texture blankTexture; // The "default" vertex shader to use when using specialized shaders but not PICA vertex shader -> GLSL recompilation // We can compile this once and then link it with all other generated fragment shaders OpenGL::Shader defaultShadergenVs; GLuint shadergenFragmentUBO; // UBO for uploading the PICA uniforms when using hw shaders GLuint hwShaderUniformUBO; using StreamBuffer = OpenGLStreamBuffer; std::unique_ptr hwVertexBuffer; std::unique_ptr hwIndexBuffer; // Cache of fixed attribute values so that we don't do any duplicate updates std::array, 16> fixedAttrValues; // Cached recompiled fragment shader struct CachedProgram { OpenGL::Program program; }; struct ShaderCache { std::unordered_map> vertexShaderCache; std::unordered_map fragmentShaderCache; // Program cache indexed by GLuints for the vertex and fragment shader to use // Top 32 bits are the vertex shader GLuint, bottom 32 bits are the fs GLuint std::unordered_map programCache; void clear() { for (auto& it : programCache) { CachedProgram& cachedProgram = it.second; cachedProgram.program.free(); } for (auto& it : vertexShaderCache) { if (it.second.has_value()) { it.second->free(); } } for (auto& it : fragmentShaderCache) { it.second.free(); } programCache.clear(); vertexShaderCache.clear(); fragmentShaderCache.clear(); } }; ShaderCache shaderCache; OpenGL::Framebuffer getColourFBO(); OpenGL::Texture getTexture(Texture& tex); OpenGL::Program& getSpecializedShader(); PICA::ShaderGen::FragmentGenerator fragShaderGen; OpenGL::Driver driverInfo; MAKE_LOG_FUNCTION(log, rendererLogger) void setupBlending(); void setupStencilTest(bool stencilEnable); void bindDepthBuffer(); void setupUbershaderTexEnv(); void bindTexturesToSlots(); void updateLightingLUT(); void updateFogLUT(); void initGraphicsContextInternal(); void accelerateVertexUpload(ShaderUnit& shaderUnit, PICA::DrawAcceleration* accel); public: RendererGL(GPU& gpu, const std::array& internalRegs, const std::array& externalRegs) : Renderer(gpu, internalRegs, externalRegs), fragShaderGen(PICA::ShaderGen::API::GL, PICA::ShaderGen::Language::GLSL) {} ~RendererGL() override; void reset() override; void display() override; // Display the 3DS screen contents to the window void initGraphicsContext(SDL_Window* window) override; // Initialize graphics context void clearBuffer(u32 startAddress, u32 endAddress, u32 value, u32 control) override; // Clear a GPU buffer in VRAM void displayTransfer(u32 inputAddr, u32 outputAddr, u32 inputSize, u32 outputSize, u32 flags) override; // Perform display transfer void textureCopy(u32 inputAddr, u32 outputAddr, u32 totalBytes, u32 inputSize, u32 outputSize, u32 flags) override; void drawVertices(PICA::PrimType primType, std::span vertices) override; // Draw the given vertices void deinitGraphicsContext() override; virtual bool supportsShaderReload() override { return true; } virtual std::string getUbershader() override; virtual void setUbershader(const std::string& shader) override; virtual bool prepareForDraw(ShaderUnit& shaderUnit, PICA::DrawAcceleration* accel) override; std::optional getColourBuffer(u32 addr, PICA::ColorFmt format, u32 width, u32 height, bool createIfnotFound = true); // Note: The caller is responsible for deleting the currently bound FBO before calling this void setFBO(uint handle) { screenFramebuffer.m_handle = handle; } void resetStateManager() { gl.reset(); } void initUbershader(OpenGL::Program& program); #ifdef PANDA3DS_FRONTEND_QT virtual void initGraphicsContext([[maybe_unused]] GL::Context* context) override { initGraphicsContextInternal(); } #endif // Take a screenshot of the screen and store it in a file void screenshot(const std::string& name) override; };