尝试渲染高度图时,OpenGL在glDrawElements上崩溃

时间:2014-11-16 00:53:28

标签: c++ opengl sdl heightmap

我正在编写一个类,使用现代OpenGL技术从32位bmp渲染高度图。我让它使用立即模式,但想尝试将其转换为现代OpenGL上下文。我正在使用带有地图索引数据的元素数组缓冲区和带有顶点数据的vbo,其中y分量取自位图的高度(像素颜色)。

现在,我在绘制调用时遇到未处理的异常/访问冲突。这是我的代码:

void HeightMap::initHeightMap(const char* filePath, float size, float height){
SDL_Surface* image = SDL_LoadBMP(filePath);
if (!image)
{
    smodgine::fatalError("Image could not be loaded!");
}

std::vector<float> tmp;
for (int y = 0; y< image->h; y++)
{
    tmp.clear();
    for (int x = 0; x < image->w; x++)
    {
        Uint32 pixel = ((Uint32*)image->pixels)[y * image->pitch/4 + x];
        Uint8 r, g, b;
        SDL_GetRGB(pixel, image->format, &r, &g, &b);

        tmp.push_back((float)r / 255.0f);
    }
    heights.push_back(tmp);
}

for (int z = 0; z < heights.size(); z++)
{
    for (int x = 0; x < heights[0].size(); x++)
    {
        vertices.push_back(x * size);
        vertices.push_back(heights[z][x] * height);
        vertices.push_back(z * size);
    }
}

if (_vbo == 0) {
    glGenBuffers(1, &_vbo);
}

glBindBuffer(GL_ARRAY_BUFFER, _vbo);
glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices.data(), GL_STATIC_DRAW);
glBindBuffer(GL_ARRAY_BUFFER, 0);

_numRows = heights.size() - 1;
_numColumns = heights[0].size() - 1;

SDL_FreeSurface(image);

createElementBuffer();
}

void HeightMap::renderHeightMap()
{
glBindBuffer(GL_ARRAY_BUFFER, _vbo);
glEnableVertexAttribArray(0);
glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 0, 0);

glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, _eleBuffer);

glDrawElements(GL_TRIANGLES, indices.size(), GL_UNSIGNED_SHORT, 0);
}

void HeightMap::createElementBuffer()
{
for (int z = 0; z < _numRows; z++)
{
    for (int x = 0; x < _numColumns; x++)
    {
        //top left
        indices.push_back(z * _numRows + x);
        //bottom left
        indices.push_back((z + 1) * _numRows + x);
        //top right
        indices.push_back(z * _numRows + (x + 1));

        //top right
        indices.push_back(z * _numRows + (x + 1));
        //bottom left
        indices.push_back((z + 1) * _numRows + x);
        //bottom right
        indices.push_back((z + 1) * _numRows + (x + 1));
    }
}

if (_eleBuffer == 0){
    glGenBuffers(1, &_eleBuffer);
}

glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, _eleBuffer);
glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(indices), indices.data(), GL_STATIC_DRAW);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
}

我只使用现代OpenGL大约一个星期左右,感谢任何帮助:D

2 个答案:

答案 0 :(得分:3)

您正在使用sizeof运算符作为std::vector的实例,期望它会为您提供向量中数据的大小:

glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices.data(), GL_STATIC_DRAW);
...
glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(indices), indices.data(), GL_STATIC_DRAW);

这不会产生预期的结果。 sizeof为您提供vector对象的内存大小,而不是向量中存储的数据。该数据是动态分配的,不会存储在vector实例本身中。

要获得正确的尺寸,您可以使用:

glBufferData(GL_ARRAY_BUFFER, vertices.size() * sizeof(vertices[0]), vertices.data(), GL_STATIC_DRAW);
...
glBufferData(GL_ELEMENT_ARRAY_BUFFER, indices.size() * sizeof(indices[0]), indices.data(), GL_STATIC_DRAW);

答案 1 :(得分:0)

我想出来了,除了Reto Koradi帮助我的错误,在我的代码的另一部分(在主函数期间)我偶然地调用了glEnableVertexAttribArray(1)