glGetProgramIv声称着色器链接失败,但glGetProgramInfoLog为空

问题描述 投票:1回答:1

我目前正在尝试使用OpenGl和C ++编写基本图形引擎。在这样做的过程中,OpenGl表现得非常奇怪,没有明显的解释。它声称着色器无法编译,但随后拒绝提供任何类型的错误来解释原因。具体来说,当我调用glGetProgramiv时,返回值为0,但glGetProgramInfoLog给出一个空字符串¹。

这是一个最小的完整可验证示例:

#include <SDL2/SDL.h>
#include <GL/glew.h>
#include <sstream>
#include <stdexcept>
#include <fstream>

GLuint loadShader(const std::string &path, GLenum type);

int main(int argc, char *argv[]){
    //show window & other housekeeping
    SDL_Init(SDL_INIT_EVERYTHING);
    SDL_GL_SetAttribute(SDL_GL_RED_SIZE, 8);
    SDL_GL_SetAttribute(SDL_GL_GREEN_SIZE, 8);
    SDL_GL_SetAttribute(SDL_GL_BLUE_SIZE, 8);
    SDL_GL_SetAttribute(SDL_GL_ALPHA_SIZE, 8);
    SDL_GL_SetAttribute(SDL_GL_BUFFER_SIZE, 32);
    SDL_GL_SetAttribute(SDL_GL_DOUBLEBUFFER, 1);
    SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 3);
    SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 3);
    SDL_Window *window = SDL_CreateWindow("test window", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, 800, 600, SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE);
    SDL_GLContext glContext = SDL_GL_CreateContext(window);
    SDL_GL_MakeCurrent(window, glContext);
    GLenum c = glewInit();
    if(c != GLEW_OK){
        std::ostringstream sout;
        sout << "Failed to initialize OpenGl: " << glewGetErrorString(c);
        throw std::runtime_error(sout.str());
    }

    //**** -- BEGIN IMPORTANT BIT -- ****//
    //This is where the shaders are loaded

    //load shaders
    GLuint vertexShader = loadShader("shader.vert", GL_VERTEX_SHADER);
    GLuint fragmentShader = loadShader("shader.frag", GL_FRAGMENT_SHADER);

    //create & link program
    GLuint program = glCreateProgram();
    glAttachShader(program, vertexShader);
    glAttachShader(program, fragmentShader);
    glLinkProgram(program);

    //check for errors
    GLint success;
    glGetProgramiv(program, GL_LINK_STATUS, &success);
    if(!success){
        GLchar infoLog[512];
        GLint size; //gives 0 when checked in debugger
        glGetProgramInfoLog(program, 512, &size, infoLog);
        throw std::runtime_error(std::string("Failed to link shader program: ") + infoLog);
    }

    //cleanup
    glDeleteShader(vertexShader);
    glDeleteShader(fragmentShader);

    //**** -- END IMPORTANT BIT -- ****//

    //update loop
    bool isClosed = false;
    while(!isClosed){
        SDL_GL_SwapWindow(window);

        SDL_Event e;
        while(SDL_PollEvent(&e)){
            switch(e.type){
                case SDL_QUIT:
                    isClosed = true;
                    break;
            }
        }
    }
}

GLuint loadShader(const std::string &path, GLenum type){
    //load file
    std::ifstream fin(path);
    std::string line, total = "";

    while(std::getline(fin, line)){
        total += (line + "\n");
    }

    //reformat source into OpenGl's ridiculous format
    const char *cArray = total.c_str();
    const char **pointerToCArray = &cArray;

    //create shader
    GLuint shader = glCreateShader(type);
    glShaderSource(shader, 1, pointerToCArray, nullptr);
    glCompileShader(shader);

    //check for erros
    GLint success;
    glGetShaderiv(shader, GL_COMPILE_STATUS, &success);
    if(!success){
        GLchar infoLog[512];
        glGetShaderInfoLog(shader, 512, nullptr, infoLog);
        throw std::runtime_error(std::string("Failed to compile shader: ") + infoLog);
    }
}

MCVE testshader.vert:

#version 330 core

layout (location = 0) in vec3 position;

void main()
{
    gl_Position = vec4(position.x, position.y, position.z, 1.0);
}

MCVE testshader.frag:

#version 330 core

out vec4 color;

void main()
{
    color = vec4(1.0f, 0.5f, 0.2f, 1.0f);
}

当我运行这个时,除了窗口的简短闪光之外,我还得到以下输出:

terminate called after throwing an instance of 'std::runtime_error'
  what():  Failed to link shader program: 
Aborted (core dumped)

额外说明:

  • 我使用的是Ubuntu 16.10 x64
  • 我使用的是带驱动程序版本375.39的Nvidia GTX 1070
  • SDL和glew版本分别是从libsdl2-devlibglew-dev软件包安装的版本。我正在使用pkg-config进行链接。

在我研究解决这个错误的过程中,我发现各种论坛和StackOverflow帖子在其他人的代码中发现了特定的错误。这些都不适用于我的代码,在发现此错误后,我将我的代码与this进行了比较,据说是工作示例。此示例还可以在我的计算机上正确编译和运行,从而排除任何类型的配置错误。

¹:glGetProgramInfoLog不可能对外部结果做任何事情,并且infoLog数组恰好在其第一个地址分配了一个空终止符,因为glGetProgramInfoLog的大小返回返回0,即使在glGetProgramInfoLog之前设置为不同的值叫做。

c++ opengl sdl-2 glew
1个回答
0
投票

我发现了这个问题:loadShader()方法中缺少return语句。然而,正如我在尝试解决这个问题时所经历的那样没有用。因此,我建议任何未来阅读本文的人都会找到一个工作示例(即this),它会执行类似于您尝试的操作,并将其代码的较小部分复制/粘贴到您的代码中(或签证 - 反之亦然)做一种二进制搜索来找到你的错误。

© www.soinside.com 2019 - 2024. All rights reserved.