Почему Vertex Array Object вызывает ошибку?
У меня есть opengl-программа, которая работает... Пока я не попробую использовать объект Vertex Array. Я пытался изменить код назад и вперед в течение нескольких дней, но не могу найти, в чем проблема.
Это часть кода. glCall(...) - вспомогательная функция для проверки ошибок.
static struct {
GLint color;
GLuint vertices;
GLuint mvpMatrix;
const char * vertexCode =
R"___(
#version 330 core
in vec4 vPosition;
uniform vec4 uColor;
uniform mat4 mvp_matrix;
out vec4 fColor;
void main() {
gl_Position = mvp_matrix * vPosition;
fColor = uColor;
}
)___";
const char * fragmentCode =
R"__(
#version 330 core
in vec4 fColor;
out vec4 FragColor;
void main() {
gl_FragColor = fColor;
})__";
} program1;
static Matrix<float> mvpMatrix;
static double screenWidth, screenHeight;
static ShaderProgram squareShaderProgram;
static GLuint vertexArray;
static GLuint vertexBuffer;
//Square
static const vector<float> squareVertices = { 0.f, 0.f, 1.f, 0.f, 1.f, 1.f, 0.f, 1.f };
static const vector<float> squareColors = {.8, .8, 1., 1};
static void setDimensions(double width, double height){
screenWidth = width;
screenHeight = height;
}
static bool initDrawModule(double width, double height) {
//GLuint vao; <----------------------- //This seems to be the problem
//glCall(glGenVertexArrays(1, &vao));
//glCall(glBindVertexArray(vao));
squareShaderProgram.initProgram(program1.vertexCode, program1.fragmentCode);
if (!squareShaderProgram.getProgram()) {
cerr << "Could not create shader program in " << __FILE__ << ":" << __LINE__ << endl;
return false;
}
glCall(squareShaderProgram.use());
program1.vertices = squareShaderProgram.getAttribute("vPosition");
program1.color = squareShaderProgram.getUniform("uColor");
program1.mvpMatrix = squareShaderProgram.getUniform("mvp_matrix");
setDimensions(width, height);
return false;
}
static void drawSquare(Vec p, double a, double sx, double sy){
squareShaderProgram.use();
{
mvpMatrix = mvpMatrix.RotationZ(a / 180.);
mvpMatrix.scale(sx, sy, 1);
mvpMatrix.scale(1. / screenWidth, 1. / screenHeight, 1);
mvpMatrix.setTranslation(
p.x / screenWidth * 2 - 1.,
p.y / screenHeight * 2 - 1.,
p.z
);
glCall(glUniformMatrix4fv(program1.mvpMatrix, 1, GL_FALSE, mvpMatrix));
}
glCall(glEnableVertexAttribArray(program1.vertices));
glCall(glBindBuffer(GL_ARRAY_BUFFER, vertexBuffer))
glCall(glVertexAttribPointer( //<----- this fails
program1.vertices,
2,
GL_FLOAT,
GL_FALSE,
0,
&squareVertices[0]));
glCall(glUniform4fv(program1.color, 1, &squareColors[0]));
glCall(glDrawArrays(GL_TRIANGLE_FAN, 0, 4));
glDisableVertexAttribArray(program1.vertices);
}
static void render() {
static float x = 0;
drawSquare(Vec(.1 + x, 50, 1), 20 + x * 2, 100,100);
x += 20;
}
bool init() {
return initDrawModule(512, 512);
}
void die(string message);
void checkSDLError(int line = -1);
int main(int argc, char *argv[])
{
SDL_Window *mainwindow;
SDL_GLContext maincontext;
// Create our window centered at 512x512 resolution
mainwindow = SDL_CreateWindow("sdl-window", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED,
512, 512, SDL_WINDOW_OPENGL | SDL_WINDOW_SHOWN);
if (!mainwindow) {
die("Unable to create window");
}
checkSDLError(__LINE__);
maincontext = SDL_GL_CreateContext(mainwindow);
checkSDLError(__LINE__);
init();
SDL_GL_SetSwapInterval(1);
for (int i = 0; i < 10; ++i) {
glClearColor ( 0.1 * i, 0.0, 0.0, 1.0 );
glClear ( GL_COLOR_BUFFER_BIT );
render();
SDL_GL_SwapWindow(mainwindow);
SDL_Delay(200);
}
SDL_GL_DeleteContext(maincontext);
SDL_DestroyWindow(mainwindow);
SDL_Quit();
return 0;
}
Единственное, что я знаю, это то, что, когда я раскомментирую строки в начале initDrawModule(), glVertexAttribPointer (отмеченный в коде) завершается с ошибкой "GL_INVALID_OPERATION". Что здесь происходит?
1 ответ
GL_INVALID_OPERATION
вызвано последним параметром glVertexAttribPointer
, указатель на данные.
В профиле совместимости, если объект буфера вершин 0 связан, то последний параметр glVertexAttribPointer
рассматривается как указатель на данные массива вершин.
Если именованный объект буфера вершин связан, то последний параметр glVertexAttribPointer
смещение в этом буфере.
Вы смешали два случая, вы связали именованный буферный объект, но вы также передали указатель на данные массива вершин:
glBindBuffer(GL_ARRAY_BUFFER, vertexBuffer);
glVertexAttribPointer(program1.vertices, 2, GL_FLOAT, GL_FALSE, 0, &squareVertices[0]);
Если вы хотите использовать объект буфера вершин, то вы должны использовать glBufferData
создать и инициализировать хранилище данных буферного объекта:
glBindBuffer(GL_ARRAY_BUFFER, vertexBuffer);
GLsizeiptr size_in_bytes = ...; // size of the buffer data in bytes
glBufferData(GL_ARRAY_BUFFER, size_in_bytes, &squareVertices[0], GL_STATIC_DRAW);
Именованный буфер вершин может использоваться для определения массива общих данных атрибутов вершин:
glBindBuffer(GL_ARRAY_BUFFER, vertexBuffer);
glVertexAttribPointer(program1.vertices, 2, GL_FLOAT, GL_FALSE, 0, NULL);
Смотрите также объект Vertex Buffer.