无法链接程序,glLinkProgram() 给我 GL_INVALID_OPERATION

标签 c opengl

我目前正在学习 OpenGL 3.X,正在使用 ubuntu 14.04。 我正在使用 C 语言创建我的应用程序和 SDL2 库。

命令

glxinfo | grep "OpenGL version"

给我以下输出

OpenGL version string: 3.0 Mesa 10.1.3

我的着色器创建有问题,这里是加载/编译/链接着色器的代码:

typedef struct  s_shader
{
   GLuint  vertex_id;
   GLuint  fragment_id;
   GLuint  program_id;
}               t_shader;

t_shader    *create_shader(const char *vertex_file, const char *fragment_file)
{
    t_shader    *new_shader;

    new_shader = (t_shader *)malloc(sizeof(t_shader));
    if (new_shader)
    {
        new_shader->vertex_id = glCreateShader(GL_VERTEX_SHADER);
        new_shader->fragment_id = glCreateShader(GL_FRAGMENT_SHADER);
        new_shader->program_id = glCreateProgram();
        load_shader(new_shader, vertex_file, fragment_file);
        glCompileShader(new_shader->vertex_id);
        glCompileShader(new_shader->fragment_id);
        glAttachShader(new_shader->program_id, new_shader->vertex_id);
        glAttachShader(new_shader->program_id, new_shader->fragment_id);
        glLinkProgram(new_shader->program_id);
    }
    return (new_shader);
}

void        load_shader(t_shader *shader,const char *vertex_file,const char *fragment_file)
{
    char    *vertex_file_content;
    char    *fragment_file_content;
    int     vertex_len;
    int     fragment_len;

    vertex_file_content = NULL;
    fragment_file_content = NULL;
    vertex_file_content = get_file_content(vertex_file);
    fragment_file_content = get_file_content(fragment_file);
    vertex_len = strlen(vertex_file_content);
    fragment_len = strlen(fragment_file_content);

    glShaderSource(shader->vertex_id, 1, (const GLchar **)&vertex_file_content, &vertex_len);
    glShaderSource(shader->fragment_id, 1, (const GLchar **)&fragment_file_content, &fragment_len);
}

我在每次 gl 调用后使用 glGetError(),我得到的第一个错误代码是 glLinkProgram() 调用,它给我一个 GL_INVALID_OPERATION(错误 1282)。 我尝试像这样输出日志错误:

    GLchar    buff[1024];
    GLsizei     length;
    glGetProgramInfoLog(new_shader->program_id, 1024, &length, buff);
    printf("length:%i\nlog:'%s'\n", length, buff);

但是日志是空的。

我对片段和顶点着色器都使用了 glIsShader(),它返回 GL_TRUE。 我还在 glCompileShader() 之后使用了 glGetShaderInfoLog() 并且片段和顶点着色器的日志都是空的。 我在文档中读到:

GL_INVALID_OPERATION is generated if program is not a program object.

GL_INVALID_OPERATION is generated if program is the currently active program object and transform feedback mode is active.

我的程序似乎是一个程序对象,因为 glCreateProgram 返回一个非空值。 此时在代码中,没有事件的程序对象。

窗口和 OpenGL 上下文:

SDL_Init(SDL_INIT_VIDEO);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 3);
SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 1);

SDL_GL_SetAttribute(SDL_GL_DOUBLEBUFFER, 1);
SDL_GL_SetAttribute(SDL_GL_DEPTH_SIZE, 24);

window = SDL_CreateWindow("My OpenGl", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, SCREEN_WIDTH, SCREEN_HEIGHT, SDL_WINDOW_OPENGL | SDL_WINDOW_SHOWN);
if (window)
{
  SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_CORE);
  gl_context = SDL_GL_CreateContext(window);

读取文件内容的函数如下:

char        *get_file_content(const char *file_name)
{
    FILE    *file;
    char    *file_content;
    long    file_size;

    file = NULL;
    file_content = NULL;
    file_size = 0;
    file = fopen(file_name, "r");
    if (file)
    {
        fseek(file, 0, SEEK_END);
        file_size = ftell(file);
        fseek(file, 0, SEEK_SET);
        file_content = (char *)malloc(sizeof(char) * file_size + 1);
        if (file_content)
        {
            memset(file_content, '\0', file_size);
            fread(file_content, sizeof(char), file_size, file);
            file_content[file_size + 1] = '\0';
        }
        fclose(file);
    }
    return (file_content);
}

片段着色器源代码:

#version 150 core

in vec3 color;
out vec4 out_Color;

void main()
{
    out_Color = vec4(1.0f, 1.0f, 1.0f, 1.0);
}

顶点着色器源代码:

#version 150 core

in vec3 in_Vertex;
in vec3 in_Color;

uniform mat4 projection;
uniform mat4 modelview;

out vec3 color;

void main()
{
    gl_Position = projection * modelview * vec4(in_Vertex, 1.0);
    color = in_Color;
}

还有其他调试glLinkProgram的方法吗?

最佳答案

您的着色器可能无法编译,因为您正尝试在 OpenGL 3.0 上下文中使用 GLSL #version 150(与 OpenGL 3.2 一起引入)。

由于您使用的是 mesa,您的 GPU 很可能实际上支持 3.2 或更高版本。但是在 mesa 上,您必须使用 core 配置文件才能访问 OpenGL >= 3.2 上下文。否则,mesa 将仅限于 GL 3.0,而这正是您所得到的。

所以你有两个选择:

  1. 将着色器降级到 #version 130。使用您发布的着色器,我没有看到任何不适用于该版本的内容,因此只需调整 #version 指令就足够了。

  2. 将您的上下文升级到现代 GL 3.2 核心配置文件。由于您使用的是 SDL2,相关函数是 SDL_GL_SetAttribute() ,另见 the SDL_GLprofile documentation .使用核心配置文件意味着所有已弃用的功能都不再可用。您没有发布您的代码,因此不清楚是否依赖于已弃用的功能。

虽然备选方案 1 可能看起来更简单,但我强烈建议选择变体 2。如果您现在正在学习 GL,则应该学习新方法,而不是十多年前已弃用的方法。

关于无法链接程序,glLinkProgram() 给我 GL_INVALID_OPERATION,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31639760/

相关文章:

opengl - OpenGL + Glut 的 OBJ 加载器

c++ - QGraphicsView 和背景中的复杂多边形

c - 是否有定义 IPv4/6 最大前缀长度的 POSIX header ?

c - 如何在 MinGW 上安装 C 库?

在 C 中连接 wchar_t Unicode 字符串?

c++ - 带有定向边界框的轴旋转

c++ - 使用 gluSphere() 和 gluCylinder() 渲染光滑的球体和圆柱体

c++ opengl我如何制作着色器文件并在主cpp中使用它

c - 未知类型名称 '__priority_which_t'

我们可以在不知道明文大小的情况下使用 C 中的 OpenSSL 解密(对称加密/解密)消息吗?