3

我正在修改别人的代码。他们使用通过 BufferedImage 加载的 PNG。我需要加载一个 TGA,它只是一个 18 字节的标头和 BGR 代码。我已加载并运行纹理,但我得到一个灰色框而不是纹理。我什至不知道如何调试这个。

纹理加载到 ByteBuffer 中:

final static int datasize = (WIDTH*HEIGHT*3) *2; // Double buffer size for OpenGL // not +18 no header
static ByteBuffer buffer = ByteBuffer.allocateDirect(datasize);

FileInputStream fin = new FileInputStream("/Volumes/RAMDisk/shot00021.tga");
FileChannel inc = fin.getChannel();

inc.position(18); // skip header

buffer.clear(); // prepare for read
int ret = inc.read(buffer);
fin.close();

我遵循了这个:[how-to-manage-memory-with-texture-in-opengl][1] ...因为我每帧更新一次纹理,比如视频。

调用一次:

GL11.glBindTexture(GL11.GL_TEXTURE_2D, textureID);
GL11.glTexParameteri(GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_S, GL11.GL_CLAMP);
GL11.glTexParameteri(GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_T, GL11.GL_CLAMP);
GL11.glTexParameteri(GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_MAG_FILTER, GL11.GL_NEAREST);
GL11.glTexParameteri(GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_MIN_FILTER, GL11.GL_NEAREST);

GL11.glTexImage2D(GL11.GL_TEXTURE_2D, 0, GL11.GL_RGB, width, height, 0, GL11.GL_RGB, GL11.GL_UNSIGNED_BYTE, (ByteBuffer) null);
assert(GL11.GL_NO_ERROR == GL11.glGetError());

反复调用:

GL11.glBindTexture(GL11.GL_TEXTURE_2D, textureID);
GL11.glTexSubImage2D(GL11.GL_TEXTURE_2D, 0, 0, 0, width, height, GL11.GL_RGB, GL11.GL_UNSIGNED_BYTE, byteBuffer);
assert(GL11.GL_NO_ERROR == GL11.glGetError());

return textureID;

渲染代码没有改变并且基于:

GL11.glDrawArrays(GL11.GL_TRIANGLES, 0, this.vertexCount);
4

4 回答 4

2

确保设置纹理采样模式。特别是最小过滤器:glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR)。默认设置为 mip 映射 (GL_NEAREST_MIPMAP_LINEAR),因此除非您上传 mip 映射,否则您将获得白色读取结果。

因此,要么将纹理设置为无 mip,要么生成它们。一种方法是在 tex img 调用之后调用 glGenerateMipmap。

(见https://www.khronos.org/opengles/sdk/docs/man/xhtml/glTexParameter.xml)。

这是一个非常常见的 gl 陷阱,人们在被它咬过几次之后就会知道这一点。

没有简单的方法来调试这样的东西。例如 xcode 中有很好的 gl 调试工具,但它们不会告诉你这种情况。

于 2015-05-22T07:29:20.703 回答
2

调试 GPU 代码总是很麻烦。随着越来越多的公司发现 GPU 的强大功能,我将赌注押在该领域的重大行业进步上。直到那时; 我将分享我的两个最好的 GPU 调试朋友:

1) 定义一个打印 OGL 错误的函数:

int printOglError(const char *file, int line)
{
    /* Returns 1 if an OpenGL error occurred, 0 otherwise. */
    GLenum glErr;
    int    retCode = 0;

    glErr = glGetError();
    while (glErr != GL_NO_ERROR) {
        printf("glError in file %s @ line %d: %s\n", file, line, gluErrorString(glErr));
        retCode = 1;
        glErr = glGetError();
    }
    return retCode;
}

#define printOpenGLError() printOglError(__FILE__, __LINE__)

并在您的渲染绘制调用之后调用它(可能会出现更早的错误):

GL11.glDrawArrays(GL11.GL_TRIANGLES, 0, this.vertexCount);
printOpenGLError();

如果您进行了一些无效操作(可能只是您的情况),这会发出警报,但您通常必须通过反复试验找到错误发生的位置。

2) 查看gDEBugger,这是一款包含大量 GPU 内存信息的免费软件。

[编辑]:我还建议使用开源库DevIL - 它非常适合加载各种图像格式。

于 2015-05-29T16:32:15.663 回答
1

可能导致问题的一些想法:

  • 你的纹理被放置在某个地方。我不知道整个代码,但我猜某处有一个glDeleteTextures,如果在错误的时间调用,这可能会导致一些问题。
  • 纹理宽度和高度是两个的幂吗?如果不是,这可能是一个问题,具体取决于您的硬件。旧硬件有时不支持两个图像的非电源
  • 纹理参数在其他点的绘制调用之间发生了变化(使用glGetTexParameter对参数进行调试检查)。
  • 加载下一张图片(编辑:甚至是第一张图片)时可能会出现加载问题。检查是否显示第一张图像而不加载下一张图像。如果是这样,它必须是上述情况之一。
于 2015-05-29T07:39:28.347 回答
1

感谢 Felix,通过不调用 glTexSubImage2D(使内存有效,但未初始化),我注意到默认内存留下了一个残余模式。这表明纹理正在显示,但负载很可能是问题所在。

**更新:

上面代码的问题本质上是缓冲区。缓冲区为 1024*1024,但它仅被读取部分填充,limit将 ByteBuffer 的标记留在 2359296(1024*768*3) 而不是 3145728(1024*1024*3)。这给出了错误:

Number of remaining buffer elements is must be ... at least ...

我认为OpenGL需要空间来返回数据,所以我将缓冲区的大小增加了一倍。缓冲区大小加倍以补偿错误。

final static int datasize = (WIDTH*HEIGHT*3) *2; // Double buffer size for OpenGL // not +18 no header

这是错误的,需要将ByteBuffer置于读取模式的flip()功能(非常感谢 Reto Koradi 对缓冲区倒带的小提示) 。由于缓冲区只是半满的,因此 OpenGL 缓冲区检查会出错。正确的做法是不要将缓冲区大小加倍;在执行.buffer.position(buffer.capacity())flip()

final static int datasize = (WIDTH*HEIGHT*3); // not +18 no header

buffer.clear(); // prepare for read
int ret = inc.read(buffer);
fin.close();
buffer.position(buffer.capacity()); // make sure buffer is completely FILLED!
buffer.flip(); // flip buffer to read mode

为了解决这个问题,对缓冲区的内存进行硬编码以确保 OpenGL 调用正常工作,从而隔离负载问题是有帮助的。然后当 OpenGL 调用正确时,专注于缓冲区的加载。glTexSubImage2D正如 Felix K 所建议的,在重复调用之前确保正确绘制了一个纹理是很好的。

于 2015-05-30T05:53:40.530 回答