blob: 383d24046b27259f837e1aef0a822d7c026a36b1 [file] [log] [blame]
/*
* Copyright 2023 Rive
*/
#include "rive/pls/gl/pls_render_context_gl.hpp"
#include "../out/obj/generated/glsl.exports.h"
namespace rive::pls
{
constexpr static GLenum kPLSDrawBuffers[4] = {GL_COLOR_ATTACHMENT0,
GL_COLOR_ATTACHMENT1,
GL_COLOR_ATTACHMENT2,
GL_COLOR_ATTACHMENT3};
class PLSRenderContextGL::PLSImplRWTexture : public PLSRenderContextGL::PLSImpl
{
rcp<PLSRenderTargetGL> wrapGLRenderTarget(GLuint framebufferID,
size_t width,
size_t height,
const PlatformFeatures&) override
{
// For now, the main framebuffer also has to be an RW texture.
return nullptr;
}
rcp<PLSRenderTargetGL> makeOffscreenRenderTarget(
size_t width,
size_t height,
const PlatformFeatures& platformFeatures) override
{
auto renderTarget = rcp(new PLSRenderTargetGL(width, height, platformFeatures));
glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_WIDTH, width);
glFramebufferParameteri(GL_FRAMEBUFFER, GL_FRAMEBUFFER_DEFAULT_HEIGHT, height);
renderTarget->allocateCoverageBackingTextures();
renderTarget->createSideFramebuffer();
glDrawBuffers(4, kPLSDrawBuffers);
return renderTarget;
}
void activatePixelLocalStorage(PLSRenderContextGL* context,
const PLSRenderTargetGL* renderTarget,
LoadAction loadAction,
bool needsClipBuffer) override
{
// Clear the necessary textures.
constexpr static GLuint kZero[4]{};
glBindFramebuffer(GL_FRAMEBUFFER, renderTarget->sideFramebufferID());
if (loadAction == LoadAction::clear)
{
float clearColor4f[4];
UnpackColorToRGBA32F(context->frameDescriptor().clearColor, clearColor4f);
glClearBufferfv(GL_COLOR, kFramebufferPlaneIdx, clearColor4f);
}
glClearBufferuiv(GL_COLOR, kCoveragePlaneIdx, kZero);
if (needsClipBuffer)
{
glClearBufferuiv(GL_COLOR, kClipPlaneIdx, kZero);
}
// Bind the RW textures.
glBindImageTexture(kFramebufferPlaneIdx,
renderTarget->m_offscreenTextureID,
0,
GL_FALSE,
0,
GL_READ_WRITE,
GL_RGBA8);
glBindImageTexture(kCoveragePlaneIdx,
renderTarget->m_coverageTextureID,
0,
GL_FALSE,
0,
GL_READ_WRITE,
GL_R32UI);
glBindImageTexture(kOriginalDstColorPlaneIdx,
renderTarget->m_originalDstColorTextureID,
0,
GL_FALSE,
0,
GL_READ_WRITE,
GL_RGBA8);
if (needsClipBuffer)
{
glBindImageTexture(kClipPlaneIdx,
renderTarget->m_clipTextureID,
0,
GL_FALSE,
0,
GL_READ_WRITE,
GL_R32UI);
}
glBindFramebuffer(GL_FRAMEBUFFER, renderTarget->drawFramebufferID());
glMemoryBarrier(GL_SHADER_IMAGE_ACCESS_BARRIER_BIT);
}
void deactivatePixelLocalStorage(PLSRenderContextGL*) override
{
glMemoryBarrier(GL_ALL_BARRIER_BITS);
}
const char* shaderDefineName() const override { return GLSL_PLS_IMPL_RW_TEXTURE; }
void onBarrier() override { return glMemoryBarrier(GL_SHADER_IMAGE_ACCESS_BARRIER_BIT); }
};
std::unique_ptr<PLSRenderContextGL::PLSImpl> PLSRenderContextGL::MakePLSImplRWTexture()
{
return std::make_unique<PLSImplRWTexture>();
}
} // namespace rive::pls