gpt4 book ai didi

c++ - Opengl vao 在绘图调用期间中断

转载 作者:行者123 更新时间:2023-11-30 02:22:13 25 4
gpt4 key购买 nike

目前我有一个包含 2 个缓冲区的系统,一个创建 CPU 端并设置为一个缓冲区。然后一个来自 ssbo 并从另一个着色器填充。

这是 ssbo 数据的结构:

struct ssbo_data_t
{
int maxcount = 1024;
float baz[1024];
} ssbo_data;

然后 vao 的构建方式如下:

//Build buffer
glGenBuffers(1, &buffer);
glBindBuffer(GL_ARRAY_BUFFER, buffer);
glBufferData(GL_ARRAY_BUFFER, arraysize * sizeof(DATATYPE), dataarr, GL_DYNAMIC_DRAW);

//Build vao
glGenVertexArrays(1, &vao);
glBindVertexArray(vao);

//index,size,type,norm,stride,pointerToFirst
glEnableVertexAttribArray(0); //Position of light
glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 0, 0);

glEnableVertexAttribArray(1); //Color
glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, 0, (void*)(sizeof(DATATYPE) * 2));

glEnableVertexAttribArray(2); //Intensity
glVertexAttribPointer(2, 1, GL_FLOAT, GL_FALSE, 0, (void*)(sizeof(DATATYPE) * 5));

glEnableVertexAttribArray(3); //Layer
glVertexAttribPointer(3, 1, GL_FLOAT, GL_FALSE, 0, (void*)(sizeof(DATATYPE) * 6));

//Try to bind ssbo for lighting data.
glBindBuffer(GL_ARRAY_BUFFER, ssbo);

glEnableVertexAttribArray(4); //MaxSize
glVertexAttribPointer(4, 1, GL_INT, GL_FALSE, 0, (void*)offsetof(ssbo_data_t,maxcount));

glEnableVertexAttribArray(5); //Corner Data
glVertexAttribPointer(5, 2, GL_FLOAT, GL_FALSE, sizeof(float) * 2, (void*)offsetof(ssbo_data_t, baz));

我的问题是,如果我没有为一个缓冲区设置步幅,然后为缓冲区的另一部分设置步幅,它应该可以正常工作吗?还是我遗漏了什么?

因为当我实际执行绘制调用时,绘制了第一条线,但所有其他线的位置为 -1,-1 就像绑定(bind)到 VAO 的第一个缓冲区被重置一样。

然而,我从另一个缓冲区收到了正确的药水,这表明它正在工作。就像第一个缓冲区在下一次绘制调用时被解除绑定(bind)。

由于项目太大,我无法真正发布一个工作示例。

Example

正如您在这里看到的,我将图元放在左侧,将另一个对象放在右侧。

图元被绘制,并将其角位置设置为 SSBO。

然后绘制第二个对象并从其位置创建 4 条线段。

第一个顶点按预期工作,并从其位置创建一条线段并终止于 VAO 指定的框的角。

接下来的抽奖没有正确读取位置。但是一定要从 ssbo 获取正确的信息。那么……?

如果我发布顶点着色器可能会有帮助:

#version 430 core
layout(location = 0) in vec2 aPos;
layout(location = 1) in vec3 aColor;
layout(location = 2) in float intensity;
layout(location = 3) in float layer;
layout(location = 4) in int maxcount;
layout(location = 5) in vec2 loc;

uniform vec2 screensize;

out VS_OUT{
vec3 color;
float intensity;
vec4 targ;
} vs_out;

void main()
{
vec2 npos = ((aPos - (screensize / 2.0)) / screensize) * 2; //Convert mouse chords to screen chords.
gl_Position = vec4(npos, layer, 1.0);
vs_out.targ = vec4(loc, layer, 1.0);
vs_out.color = aColor;
vs_out.intensity = intensity;
}

最佳答案

要回答您的问题,是的,您可以在同一个缓冲区中混合和匹配步幅和偏移量,并与 SSBO (Shader Storage Buffer Object) 共享顶点属性数据。 .

顶点步幅

我不确定您是如何尝试在那里使用 SSBO 的内容的。然而,顶点属性 #4 和 #5 的绑定(bind)看起来有问题。

glVertexAttribPointer(4, 1, GL_INT, GL_FALSE, 0, (void*)offsetof(ssbo_data_t,maxcount));

第一个顶点将具有 x 组件中预期的 maxcount。然而,步幅为 0 意味着连续的顶点属性被打包。因此,第二个顶点将从 baz[0] 读取 32 位并将其视为整数。第三个将读取 baz[1] 等等。简而言之,只有第一个顶点将具有正确的 maxcount 值,其余的将具有虚假值。

要解决此问题,请使用 instanced arrays (也称为顶点除数)。要使用的函数是 glVertexAttribDivisor()。另一种选择是直接绑定(bind)您的 SSBO 并将其作为 SSBO 读取(在 GLSL 中输入 buffer)。查看SSBO OpenGL wiki page举个例子。

最后:

glVertexAttribPointer(5, 2, GL_FLOAT, GL_FALSE, sizeof(float) * 2, (void*)offsetof(ssbo_data_t, baz));

你可以在这里使用 stride 0,因为你的属性值是紧密打包的。 sizeof(float) * 2 给出相同的结果。

(错误的)解决方案

我错误地指出 BindVertexArray() 重置了当前的 GL_ARRAY_BUFFER 绑定(bind)。不是这种情况。它确实重置了 indexed 顶点缓冲区绑定(bind),但你已经设置了那些。

关于c++ - Opengl vao 在绘图调用期间中断,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47539108/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com