在 SSBO 中发送 uvec4 数组

Sending uvec4 array in SSBO

我正在尝试向我的 SSBO 发送一组 glm::uvec4。有可能还是我应该使用其他方法?

我正在使用实例化绘图,我告诉 OpenGL 绘制 X 个实例(取决于 glm::uvec4 可以变化的大小)

存储的数据:

std::vector<glm::uvec4> m_blocksToRender;

SSBO 设置:

// SSBO
glGenBuffers(1, &m_ssbo);
glBindBuffer(GL_SHADER_STORAGE_BUFFER, m_ssbo);
glBufferData(GL_SHADER_STORAGE_BUFFER, sizeof(m_blocksToRender), m_blocksToRender.data(), GL_STATIC_DRAW);
glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 0, m_ssbo);
glBindBuffer(GL_SHADER_STORAGE_BUFFER, 0);

着色器:

layout(std430, binding = 0) buffer blocksData
{
    uvec4 data[/* ??? can varry, depends on gl_InstanceID size */];
};

你绝对可以。直接的方法是像你所做的那样绑定缓冲区,在渲染调用之前设置数据(如果你想修改它)然后在你的着色器上读取它。

根据您的用途,您还需要直接在 SSBO 中或在制服上存储 arrya 的尺寸。

所以完整的解决方案是:

C++

//Create SSBO
glGenBuffers(1, &m_ssbo);
glBindBuffer(GL_SHADER_STORAGE_BUFFER, m_ssbo);
glBufferData(GL_SHADER_STORAGE_BUFFER, sizeof(m_blocksToRender), m_blocksToRender.data(), GL_STATIC_DRAW);
glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 0, m_ssbo);
glBindBuffer(GL_SHADER_STORAGE_BUFFER, 0);

void render()
{
  /* additional code */

  //Only needed if another buffer has been bound to the binding point 0
  glBindBuffer(GL_SHADER_STORAGE_BUFFER, m_ssbo);
  glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 0, m_ssbo);
  //Only needed if the buffer data needs to be changed
  glBufferData(GL_SHADER_STORAGE_BUFFER, sizeof(m_blocksToRender)

  //If you use the uniform option
  glUniform1i(glGetUniformLocation(programID, "size"), m_blocksToRender.size());
}

片段

如果使用统一方式:

uniform int size;
layout(std430, binding = 0) buffer blocksData
{
    uvec4 data[];
};

如果您使用元数据版本:

layout(std430, binding = 0) buffer blocksData
{
    uint size;
    uvec4 data[];
};

如果您选择第二个选项,您需要小心对齐问题,并且您还需要实现一种方法,将数据大小附加到发送到 SSBO 的缓冲区的开头。

编辑: 要禁用驱动程序警告(顺便说一下,这是 NVIDIA 特定的,AMD 卡似乎不会抛出此错误)

在绑定缓冲区之前在代码的某处添加以下行:

GLuint copy_warning = 0x20072;
glDebugMessageControl(GL_DEBUG_SOURCE_API, GL_DEBUG_TYPE_PERFORMANCE,
    GL_DONT_CARE, 1, &copy_warning, GL_FALSE);