Невозможно получить тривиальный 2D-шейдер для рисования красного треугольника в C ++ openGL (без ошибок компилятора) - PullRequest
1 голос
/ 26 апреля 2020

Я использовал openGL, GLEW и GLFW3. В openGl 3.0 mesa был использован шейдер по умолчанию и нарисован белый треугольник. В 4.2 экран остался черным. Сообщения об ошибках не генерировались.

Проблема не связана с неправильной маркировкой фрагментного шейдера как вершинного шейдера или наоборот.

Программа содержит 3 функции:

( 1) compileShader, который должен принимать тип шейдера и исходный код как std :: string и возвращать идентификатор шейдера с скомпилированным кодом.

(2) createProgram, принимает исходный код вершинного шейдера и фрагментный шейдер и возвращает идентификатор программы с скомпилированными и присоединенными шейдерами.

(3) И основной, где оба исходных кода шейдера определены как строки.

Извините и спасибо.

#include <GL/glew.h>
#include <GLFW/glfw3.h>

#include <iostream>
#include <string>

static unsigned int compileShader(unsigned int type, std::string const& sourceCode){
    unsigned int shaderId = glCreateShader(type);
    const char* source = sourceCode.c_str();
    glShaderSource(shaderId, 1, &source, nullptr);
    glCompileShader(shaderId);

    int result;
    glGetShaderiv(shaderId, GL_COMPILE_STATUS, &result);
    if (result == GL_FALSE){

        if (type == GL_FRAGMENT_SHADER) std::cout<<"Fragment\n";
        else std::cout<<"Vertex";
        std::cout<<"SHADER COMPILATION FAILED!"<<std::endl;
        int logLength;
        glGetShaderiv(shaderId, GL_INFO_LOG_LENGTH, &logLength);
        char* infoLog = new char[logLength];
        glGetShaderInfoLog(shaderId, logLength, NULL, infoLog);
        std::cout<<infoLog<<std::endl;
        delete[] infoLog;

        glDeleteShader(shaderId);
        return 0;
    }

    return shaderId;
}

static unsigned int createProgram(std::string const& vertexCode, std::string const& fragmentCode){
    unsigned int vsId = compileShader(GL_VERTEX_SHADER, vertexCode);
    unsigned int fsId = compileShader(GL_FRAGMENT_SHADER, fragmentCode);

    unsigned int programId = glCreateProgram();
    glAttachShader(programId, vsId);
    glAttachShader(programId, fsId);
    glLinkProgram(programId);
    glValidateProgram(programId);

    glDeleteShader(vsId);
    glDeleteShader(fsId);
    return programId;
}

int main()
{

    GLFWwindow* window;

    if (!glfwInit())
    {
        std::cout<<"GLFW initialization failed";
        return -1;
    }

    ///switches to opengl 4.2!
    glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 4);
    glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 2);
    glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);

    /* Create a windowed mode window and its OpenGL context */
    window = glfwCreateWindow(640, 480, "Hello World", NULL, NULL);
    if (!window)
    {
        glfwTerminate();
        return -1;
    }

    /* Make the window's context current */
    glfwMakeContextCurrent(window);
    glewInit();

    std::cout << "OpenGL " << glGetString(GL_VERSION) << "\n" << std::endl;
    float positions[6] = {
        0.5f, 0.5f,
        -0.5f, 0.5f,
        0.0f, 0.0f
    };

    unsigned int buffer = 1;
    glGenBuffers(1, &buffer);
    glBindBuffer(GL_ARRAY_BUFFER, buffer);
    glBufferData(GL_ARRAY_BUFFER, 6 * sizeof(float), positions, GL_STATIC_DRAW);

    glEnableVertexAttribArray(0);                                        //enables the xy attrib of position
    glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 2*sizeof(float), 0); //specifies layout of xy attrib

    std::string vertexSource =   R"(
#version 330 core
layout(location=0) in vec4 position;

void main(){
    gl_Position = position;
}
)";
    std::string fragmentSource = R"(
#version 330 core

layout(location=0) out vec4 color;

void main(){
    color = vec4(1.0, 0.0, 0.0, 1.0);
}
)";

    unsigned int programId = createProgram(vertexSource, fragmentSource);
    glUseProgram(programId);

    /* Loop until the user closes the window */
    while (!glfwWindowShouldClose(window))
    {
        /* Render here */
        glClear(GL_COLOR_BUFFER_BIT);

        glDrawArrays(GL_TRIANGLES, 0, 3);

        /* Swap front and back buffers */
        glfwSwapBuffers(window);

        /* Poll for and process events */
        glfwPollEvents();
    }

    glfwTerminate();
    return 0;
}

1 Ответ

2 голосов
/ 26 апреля 2020

Проблема не в шейдерной программе, а в использовании основного профиля Контекст OpenGL (GLFW_OPENGL_CORE_PROFILE). Таким образом, вы должны создать именованный Vertex Array Object . VAO (0) по умолчанию недопустимо в контексте основного профиля.

Создать имя объекта массива вершин по glGenVertexArrays, а также создать и связать объект с помощью glBindVertexArray:

unsigned int vao;
glGenVertexArrays(1, &vao);
glBindVertexArray(vao);

unsigned int buffer;
glGenBuffers(1, &buffer);
glBindBuffer(GL_ARRAY_BUFFER, buffer);
glBufferData(GL_ARRAY_BUFFER, 6 * sizeof(float), positions, GL_STATIC_DRAW);

glEnableVertexAttribArray(0);                                        //enables the xy attrib of position
glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 2*sizeof(float), 0); //specifies layout of xy attrib
...