Android Java 版本的 MSAA OpenGL ES 多重采样

最近多次被小伙伴问到 OpenGL 多重采样,其实前面文章里多次讲过了,就是构建2个缓冲区,多重采样缓冲区和目标解析缓冲区。

代码流程

c 复制代码
// Framebuffer IDs
private int msaaFBO;
private int msaaColorBuffer;
private int msaaDepthBuffer;

private int resolveFBO;
private int resolveTexture;

public void initFramebuffers(int width, int height) {
    // Step 1: Create MSAA FBO
    int[] fbo = new int[1];
    GLES30.glGenFramebuffers(1, fbo, 0);
    msaaFBO = fbo[0];
    GLES30.glBindFramebuffer(GLES30.GL_FRAMEBUFFER, msaaFBO);

    // Create MSAA color buffer
    int[] renderbuffer = new int[1];
    GLES30.glGenRenderbuffers(1, renderbuffer, 0);
    msaaColorBuffer = renderbuffer[0];
    GLES30.glBindRenderbuffer(GLES30.GL_RENDERBUFFER, msaaColorBuffer);
    GLES30.glRenderbufferStorageMultisample(GLES30.GL_RENDERBUFFER, 4, GLES30.GL_RGBA8, width, height); // 4x MSAA
    GLES30.glFramebufferRenderbuffer(GLES30.GL_FRAMEBUFFER, GLES30.GL_COLOR_ATTACHMENT0, GLES30.GL_RENDERBUFFER, msaaColorBuffer);

    // Create MSAA depth buffer
    GLES30.glGenRenderbuffers(1, renderbuffer, 0);
    msaaDepthBuffer = renderbuffer[0];
    GLES30.glBindRenderbuffer(GLES30.GL_RENDERBUFFER, msaaDepthBuffer);
    GLES30.glRenderbufferStorageMultisample(GLES30.GL_RENDERBUFFER, 4, GLES30.GL_DEPTH_COMPONENT16, width, height);
    GLES30.glFramebufferRenderbuffer(GLES30.GL_FRAMEBUFFER, GLES30.GL_DEPTH_ATTACHMENT, GLES30.GL_RENDERBUFFER, msaaDepthBuffer);

    if (GLES30.glCheckFramebufferStatus(GLES30.GL_FRAMEBUFFER) != GLES30.GL_FRAMEBUFFER_COMPLETE) {
        throw new RuntimeException("MSAA Framebuffer is not complete!");
    }

    // Step 2: Create Resolve FBO
    GLES30.glGenFramebuffers(1, fbo, 0);
    resolveFBO = fbo[0];
    GLES30.glBindFramebuffer(GLES30.GL_FRAMEBUFFER, resolveFBO);

    // Create texture for resolved result
    int[] texture = new int[1];
    GLES30.glGenTextures(1, texture, 0);
    resolveTexture = texture[0];
    GLES30.glBindTexture(GLES30.GL_TEXTURE_2D, resolveTexture);
    GLES30.glTexImage2D(GLES30.GL_TEXTURE_2D, 0, GLES30.GL_RGBA, width, height, 0, GLES30.GL_RGBA, GLES30.GL_UNSIGNED_BYTE, null);
    GLES30.glTexParameteri(GLES30.GL_TEXTURE_2D, GLES30.GL_TEXTURE_MIN_FILTER, GLES30.GL_LINEAR);
    GLES30.glTexParameteri(GLES30.GL_TEXTURE_2D, GLES30.GL_TEXTURE_MAG_FILTER, GLES30.GL_LINEAR);
    GLES30.glFramebufferTexture2D(GLES30.GL_FRAMEBUFFER, GLES30.GL_COLOR_ATTACHMENT0, GLES30.GL_TEXTURE_2D, resolveTexture, 0);

    if (GLES30.glCheckFramebufferStatus(GLES30.GL_FRAMEBUFFER) != GLES30.GL_FRAMEBUFFER_COMPLETE) {
        throw new RuntimeException("Resolve Framebuffer is not complete!");
    }

    // Unbind framebuffer
    GLES30.glBindFramebuffer(GLES30.GL_FRAMEBUFFER, 0);
}

public void renderScene() {
    // Step 3: Render to MSAA FBO
    GLES30.glBindFramebuffer(GLES30.GL_FRAMEBUFFER, msaaFBO);
    GLES30.glClear(GLES30.GL_COLOR_BUFFER_BIT | GLES30.GL_DEPTH_BUFFER_BIT);

    // Render your scene here...

    // Step 4: Resolve to texture
    GLES30.glBindFramebuffer(GLES30.GL_READ_FRAMEBUFFER, msaaFBO);
    GLES30.glBindFramebuffer(GLES30.GL_DRAW_FRAMEBUFFER, resolveFBO);
    GLES30.glBlitFramebuffer(0, 0, width, height, 0, 0, width, height, GLES30.GL_COLOR_BUFFER_BIT, GLES30.GL_NEAREST);

    // Step 5: Unbind framebuffers
    GLES30.glBindFramebuffer(GLES30.GL_FRAMEBUFFER, 0);

    // Step 6: Use resolveTexture for further processing (e.g., render to screen)
    renderQuadWithTexture(resolveTexture);
}
相关推荐
pandarking11 小时前
[CTF]攻防世界:ez_curl
android·web安全·网络安全
艾莉丝努力练剑12 小时前
【C++:异常】C++ 异常处理完全指南:从理论到实践,深入理解栈展开与最佳实践
java·开发语言·c++·安全·c++11
武子康12 小时前
Java-184 缓存实战:本地缓存 vs 分布式缓存(含 Guava/Redis 7.2)
java·redis·分布式·缓存·微服务·guava·本地缓存
小马爱打代码18 小时前
Spring Boot:模块化实战 - 保持清晰架构
java·spring boot·架构
小坏讲微服务18 小时前
SpringBoot4.0整合knife4j 在线文档完整使用
java·spring cloud·在线文档·knife4j·文档·接口文档·swagger-ui
8***Z8918 小时前
springboot 异步操作
java·spring boot·mybatis
i***132419 小时前
Spring BOOT 启动参数
java·spring boot·后端
坚持不懈的大白19 小时前
后端:SpringMVC
java
IT_Octopus19 小时前
(旧)Spring Securit 实现JWT token认证(多平台登录&部分鉴权)
java·后端·spring
kk哥889919 小时前
Spring详解
java·后端·spring