我目前正在学习 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,而这正是您所得到的。
所以你有两个选择:
将着色器降级到
#version 130
。使用您发布的着色器,我没有看到任何不适用于该版本的内容,因此只需调整#version
指令就足够了。将您的上下文升级到现代 GL 3.2 核心配置文件。由于您使用的是 SDL2,相关函数是
SDL_GL_SetAttribute()
,另见 theSDL_GLprofile
documentation .使用核心配置文件意味着所有已弃用的功能都不再可用。您没有发布您的代码,因此不清楚是否依赖于已弃用的功能。
虽然备选方案 1 可能看起来更简单,但我强烈建议选择变体 2。如果您现在正在学习 GL,则应该学习新方法,而不是十多年前已弃用的方法。
关于无法链接程序,glLinkProgram() 给我 GL_INVALID_OPERATION,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31639760/