在 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, ©_warning, GL_FALSE);
我正在尝试向我的 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, ©_warning, GL_FALSE);