【问题标题】:OpenGL shader getting incorrect data even though it seems correctOpenGL着色器得到不正确的数据,即使它看起来是正确的
【发布时间】:2013-03-07 21:21:00
【问题描述】:

我刚开始学习如何使用 OpenGL 3.2,现在我正在尝试组合一个实用程序来加载几何数据。当我查看加载顶点的代码时,一切看起来都是正确的,我输入的调试输出甚至似乎给了我正确的值。代码应该生成一个三角形,每个点都变成完整的 r、g 或 b,但实际上变成了几乎纯红色/橙色。

这是包含几何数据的代码:

    final float[] vertexPositions3 = new float[] {
            0.0f,    0.5f, 0.0f, 1.0f,
            0.5f, -0.366f, 0.0f, 1.0f,
            -0.5f, -0.366f, 0.0f, 1.0f,
            1.0f,    0.0f, 0.0f, 1.0f,
            0.0f,    1.0f, 0.0f, 1.0f,
            0.0f,    0.0f, 1.0f, 1.0f,
    };

    geometry = new Geometry();
    geometry.addBufferOffset(vertexPositions3, GL_STATIC_DRAW, 0, Geometry.VERTEX_4F, Geometry.VERTEX_4F);

以及加载它的代码:

protected int getDataTypeSize(int dataType) {
    if(dataType == VERTEX_1F) {
        return 1;
    }
    else if(dataType == VERTEX_2F) {
        return 2;
    }
    else if(dataType == VERTEX_3F) {
        return 3;
    }
    else {
        return 4;
    }
}

public void addBufferOffset(float[] data, int usage, int vertexColumn, int... dataTypes) {
    if(numAttributes + dataTypes.length >= MAX_VBO) {
        throw new IllegalStateException("Can only have up to 16 attributes, requested: " + (numAttributes + dataTypes.length));
    }

    int rowSize = 0;
    for(int type : dataTypes) {
        rowSize += getDataTypeSize(type);
    }
    if(numRows == 0) {
        numRows = data.length / rowSize;
    }
    else if(numRows != (data.length / rowSize)) {
        throw new IllegalStateException("Number of rows in buffers don't match: " + numRows + " vs " + (data.length / rowSize));
    }

    if(numVertices == 0) {
        numVertices = getDataTypeSize(dataTypes[0]) * (data.length / rowSize) / getDataTypeSize(dataTypes[0]);
    }
    else {
        int addingVerts = getDataTypeSize(dataTypes[0]) * (data.length / rowSize);
        if(addingVerts != numVertices) {
            throw new IllegalStateException("Vertex count for buffers don't match: " + numVertices + " vs " + addingVerts);
        }
    }
    glBindVertexArray(vao);

    FloatBuffer buf = BufferUtils.createFloatBuffer(data.length);
    buf.put(data);
    buf.flip();
    int vbo = glGenBuffers();
    vbos[numBuffers++] = vbo;
    glBindBuffer(GL_ARRAY_BUFFER, vbo);
    glBufferData(GL_ARRAY_BUFFER, buf, usage);

    int startPos = 0;
    for(int i = 0; i < dataTypes.length; i++) {
        glEnableVertexAttribArray(numAttributes + i);
        glVertexAttribPointer(numAttributes + i, getDataTypeSize(dataTypes[i]), GL_FLOAT, false, 0, startPos);
        System.out.println((numAttributes + i) + ", " + getDataTypeSize(dataTypes[i]) + ", " + startPos);
        startPos += numRows * getDataTypeSize(dataTypes[i]);
    }
    System.out.println("Num verts: " + numVertices);
    if(usesIndices) {
        glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, indicesRef);
    }
    numAttributes += dataTypes.length;
    glBindVertexArray(0);
}

还有顶点着色器:

#version 330

layout(location = 0) in vec4 position;
layout(location = 1) in vec4 color;

smooth out vec4 theColor;
uniform float loopDuration;
uniform float time;
uniform vec4 camera;

void main() {
    float timeScale = 3.14159f * 2.0f / loopDuration;
    vec4 totalOffset = vec4(
        cos(time * 1.5f * timeScale) * 0.5f,
        sin(time * 1.5f * timeScale) * 0.5f,
        0.0f,
        0.0f);
    gl_Position = position + totalOffset;
    theColor = color;
}

最后是片段着色器:

#version 330

smooth in vec4 theColor;

out vec4 outputColor;

uniform float fragLoopDuration;
uniform float time;

const vec4 firstColor = vec4(1.0f, 1.0f, 1.0f, 1.0f);
const vec4 secondColor = vec4(0.0f, 1.0f, 0.0f, 1.0f);

void main() {
    float lerpTime = time * 1.5f / fragLoopDuration;
    outputColor = theColor;
}

当我运行程序时,addBufferOffset 方法会产生以下输出:

0, 4, 0
1, 4, 12
Num verts: 3

意味着它找到了两组三个顶点,每个顶点有四个浮点值。其中一个从数组偏移量 0 开始,另一个从数组偏移量 12 开始。这与发送给它的数组相对应:具有 x、y、z、w 值的 3 个顶点和具有 r、g、b 值的 3 个顶点,一个值。从输出来看,它似乎应该工作,但事实并非如此。我真的很感激能帮我解决这个问题。

【问题讨论】:

    标签: java opengl lwjgl opengl-3


    【解决方案1】:

    我对 lwjgl 不熟悉,但是 C API 中的 OpenGL glVertexAttribPointer 函数的最后一个参数是数据的字节偏移量 - 似乎您正在传递数组索引。这可以解释为什么您会看到一个红色/橙色三角形,因为从第 12 个字节(3 个浮点数之后)开始的数据如下所示:

            1.0f, 0.5f, -0.366f, 0.0f,
            1.0f, -0.5f, -0.366f, 0.0f,
            1.0f, 1.0f,  0.0f, 0.0f
    

    将这些值限制在 0 和 1 之间会给出

            1.0f, 0.5f, 0.0f, 0.0f,
            1.0f, 0.0f, 0.0f, 0.0f,
            1.0f, 1.0f,  0.0f, 0.0f
    

    假设您的混合设置忽略了零 alpha 分量,这些将给出橙色、红色和黄色顶点。

    【讨论】:

    • LWJGL 对 C API 进行了几乎 1 对 1 的翻译,只有很小的更改(例如不需要 FloatBuffer 的长度,因为长度存储在对象中)。字节偏移不正确是绝对正确的,非常感谢。我只需要将索引乘以 4,一切正常。我为此烦恼了一会儿。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-03-27
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多