为了账号安全,请及时绑定邮箱和手机立即绑定

Blitting FBO 颜色附件

Blitting FBO 颜色附件

Go
翻过高山走不出你 2023-08-14 17:30:37
我有 2 个 FBO + MRT,它们有相同的附件(每个有 4 个颜色附件)。对于深度缓冲区和一个color_attachment,使用 glBlitFrameBuffer 可以按预期工作。然而,当我位块复制多个颜色附件时,事情就会变糟。我做了很多研究,尝试了很多不同的方法,但没有一个有效。我没有使用 renderBufferStorage,因为我的纹理具有不同的内部格式(RGBA 和 RGB16F)。这听起来像是一个类似的问题,但我不使用多重采样,只使用 MRT。OpenGL 版本 4.3原因:我想创建地形预照明计算的状态,这样我只需要在发生变化(即相机移动)时渲染地形,然后将这些颜色附件复制到下一个 FBO。这是为了延迟着色,FBO 与 gBuffer 非常相似。起初,我希望我应该使用在初始设置中使用的 glDrawBuffers,但是没有 glReadBuffers,因此我假设我无法链接它们。我可能错了,还不是专家=)原始代码是用 Golang 编写的,应该可以轻松转移到 C++(如果需要,将进行翻译)。更新/已解决:对于任何偶然发现此问题的人。当您使用 glDrawBuffers() 并使用对 glDrawBuffer() 的调用时,您将覆盖 glDrawBuffers() 状态。它需要重置为其原始 glDrawBuffers() 状态。更新2:对于那些对这样的方法感兴趣的人,如果你有一个非动画世界并且相机不经常移动,我可以确认它具有令人难以置信的性能结果。就我的目的而言,它很棒(RTS),但对于像 CS 这样的 FPS 游戏来说,这将是一个非常糟糕的方法。在位块传输后添加以下内容(具体情况而定):var attachments = [4]uint32{gl.COLOR_ATTACHMENT0, gl.COLOR_ATTACHMENT1, gl.COLOR_ATTACHMENT2, gl.COLOR_ATTACHMENT3}gl.DrawBuffers(4, &attachments[0])固定基地运营基地设置gl.GenFramebuffers(1, &fbo.ID)gl.BindFramebuffer(gl.FRAMEBUFFER, fbo.ID)//setting up color attachmentsgl.GenTextures(1, &fbo.Position)gl.BindTexture(gl.TEXTURE_2D, fbo.Position)gl.TexImage2D(gl.TEXTURE_2D, 0, gl.RGB16F, windowWidth, windowHeight, 0, gl.RGB, gl.FLOAT, nil)gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST)gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST)gl.FramebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, fbo.Position, 0)//repeated 3 times for the additional color attachmentsvar attachments = [4]uint32{gl.COLOR_ATTACHMENT0, gl.COLOR_ATTACHMENT1, gl.COLOR_ATTACHMENT2, gl.COLOR_ATTACHMENT3}gl.DrawBuffers(4, &attachments[0])gl.GenRenderbuffers(1, &fbo.DepthBuffer)gl.BindRenderbuffer(gl.RENDERBUFFER, fbo.DepthBuffer)gl.RenderbufferStorage(gl.RENDERBUFFER, gl.DEPTH_COMPONENT, windowWidth, windowHeight)gl.FramebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_ATTACHMENT, gl.RENDERBUFFER, fbo.DepthBuffer)gl.BindFramebuffer(gl.FRAMEBUFFER, 0)
查看完整描述

1 回答

?
回首忆惘然

TA贡献1847条经验 获得超11个赞

glDrawBuffer(x)在概念上等同于调用GLenum bufs[1]={x}; glDrawBuffers(1, bufs). 由于绘制缓冲区状态是 FBO 状态的一部分,因此您的位图传输代码会覆盖这些状态,如果不手动恢复这些状态,则之后的渲染将无法按预期进行。如果你在循环中调用它,这可能会导致错误的结论,即位图传输是问题所在,但实际上,位图传输工作正常,只是在错误的输入数据上。



查看完整回答
反对 回复 2023-08-14
  • 1 回答
  • 0 关注
  • 110 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信