可以从 MSAA FBO 读取单个样本吗?

Possible to read a single sample from MSAA FBO?

使用 OpenGL 绘制对象并让我的片段着色器输出标量整数 ID。为了绘制对象,我使用多重采样来抗锯齿,所以当我为整数 ID 创建缓冲区时,我必须将其创建为 MSAA 缓冲区,以便完成 FBO:

  glBindRenderbuffer(GL_RENDERBUFFER, rboColorId);
  glRenderbufferStorageMultisample(GL_RENDERBUFFER, msaaSamples, GL_RGBA8,
                        cam.getWidth(), cam.getHeight());
  glBindRenderbuffer(GL_RENDERBUFFER, rboDepthId);
  glRenderbufferStorageMultisample(GL_RENDERBUFFER, msaaSamples, GL_DEPTH_COMPONENT,
                        cam.getWidth(), cam.getHeight());
  glBindRenderbuffer(GL_RENDERBUFFER, rboObjId);
  glRenderbufferStorageMultisample(GL_RENDERBUFFER, msaaSamples, GL_R32UI,
                        cam.getWidth(), cam.getHeight());
  glBindRenderbuffer(GL_RENDERBUFFER, rboColorNoMsaaId);
  glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA8,
                        cam.getWidth(), cam.getHeight());
  glBindRenderbuffer(GL_RENDERBUFFER, rboObjNoMsaaId);
  glRenderbufferStorage(GL_RENDERBUFFER, GL_R32UI,
                        cam.getWidth(), cam.getHeight());

  glBindRenderbuffer(GL_RENDERBUFFER, 0);
  glBindFramebuffer(GL_FRAMEBUFFER, fboId);
  glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, rboColorId);
  glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT,  GL_RENDERBUFFER, rboDepthId);
  glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT1, GL_RENDERBUFFER, rboObjId);
  glBindFramebuffer(GL_FRAMEBUFFER, fboNoMsaaId);
  glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, rboColorNoMsaaId);
  glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT1, GL_RENDERBUFFER, rboObjNoMsaaId);

正如您在上面的代码中看到的,我有 2 个 FBO。第一个是 MSAA,它有一个用于绘制场景的缓冲区、一个深度缓冲区和一个用于 ID 的整数缓冲区。第二个 FBO 是单采样(非 MSAA)并且只有绘制场景缓冲区和整数缓冲区。在绘制完所有内容后(片段着色器为每个像素设置索引),我首先将整数 ID 缓冲区 (GL_COLOR_ATTACHMENT1) 位图传送到单个采样 FBO,以便从中读取 glReadPixels。在这个特定的代码中,我只是读取鼠标指向的 1 个像素:

  glBindFramebuffer(GL_READ_FRAMEBUFFER, fboId);
  glReadBuffer(GL_COLOR_ATTACHMENT1);
  glBindFramebuffer(GL_DRAW_FRAMEBUFFER, fboNoMsaaId);
  glDrawBuffer(GL_COLOR_ATTACHMENT1);
  glBlitFramebuffer(mouse_x_pos, cam.getHeight() - mouse_y_pos, mouse_x_pos+1, cam.getHeight() - mouse_y_pos + 1,
                    0, 0, 1, 1,
                    GL_COLOR_BUFFER_BIT, GL_NEAREST);
  glBindFramebuffer(GL_READ_FRAMEBUFFER, fboNoMsaaId);
  glReadBuffer(GL_COLOR_ATTACHMENT1);
  GLuint objectId;
  glReadPixels(0, 0, 1, 1, GL_RED_INTEGER, GL_UNSIGNED_INT, &objectId);

我的问题是,当我 blit 时,我想要的像素的多样本被插入到我要读取的单个像素中。我希望它用于我用来绘制场景的颜色缓冲区,但我不希望它用于我读取的整数 ID。如果我正在读取一个包含 ID 为 50 和 ID 为 100 的片段的像素,我想读取 50 或 100(不关心哪个)。但我得到的是 50 到 100 之间的某个值,比如 75。75 实际上可能是一个完全不同的像素,所以我根本不想要那个。

我可以做些什么来读取整数 ID 的单个样本而不是多个样本的插值?

您可以在渲染到纹理通道中实现您自己的多重采样分辨率,而不是通过 blitting 解析多重采样纹理。您可以使用 sampler2DMS 类型的采样器,并使用 texelFetch( variant:

gvec4 texelFetch( gsampler2DMS sampler, ivec2 P, int sample);

所以P是2D非标准化纹素坐标,sample是样本的ID。如果你真的不关心你得到的是哪个值,你可以一直使用样本 0。但是您也可以例如遍历所有样本并选择出现次数最多的样本,或者任何适合您需要的样本。

为此,您必须从用于 ID 附件的渲染缓冲区切换到多重采样 2D 纹理。

所以基本上,您可以将非多重采样 FBO 绑定为绘制 FBO,对深度和颜色纹理执行标准 blit,并使用多重采样 ID 纹理执行全屏渲染通道,写入非多重采样 ID 颜色附件.