以下GLSL片段着色器编译并按预期工作:
#version 330 core
out vec3 color;
in float U;
in vec4 vertexNormal_worldSpace;
uniform sampler1D TextureSampler;
uniform vec4 LightPos;
void main()
{
float cosT = dot(normalize(vertexNormal_worldSpace.xyz),normalize(LightPos.xyz));
color = cosT * texture(TextureSampler,U).rgb;
}
但是,当我更改第9行以将“cosT”的值钳位在0和1之间时:
float cosT = clamp(dot(normalize(vertexNormal_worldSpace.xyz),normalize(LightPos.xyz)),0.0,1.0);
我得到错误:
0(1) : error C0000: syntax error, unexpected integer constant, expecting "::" at token "<int-const>"
0(10) : error C7532: global function texture requires "#version 130" or later
这似乎说错误出现在第一行,但根本没有任何改变。此外,第二个错误表明我正在使用的GLSL版本存在问题,但是#version 330 core应该是#version 130以后的版本,正如错误消息所述。
编辑:这是我在着色器中加载的代码:
static GLuint LoadShaders(const char* vertex_file_path, const char* frag_file_path){
GLuint VertID = glCreateShader(GL_VERTEX_SHADER);
GLuint FragID = glCreateShader(GL_FRAGMENT_SHADER);
char const* VertPointer = ReadShaderFile(vertex_file_path);
char const* FragPointer = ReadShaderFile(frag_file_path);
glShaderSource(VertID,1,&VertPointer,NULL);
glCompileShader(VertID);
GLint Result = GL_FALSE;
int InfoLogLength;
glGetShaderiv(VertID, GL_COMPILE_STATUS, &Result);
glGetShaderiv(VertID, GL_INFO_LOG_LENGTH, &InfoLogLength);
if ( InfoLogLength > 0 ){
std::vector<char> VertexShaderErrorMessage(InfoLogLength+1);
glGetShaderInfoLog(VertID, InfoLogLength, NULL, &VertexShaderErrorMessage[0]);
printf("%s\n", &VertexShaderErrorMessage[0]);
}
glShaderSource(FragID,1,&FragPointer,NULL);
glCompileShader(FragID);
glGetShaderiv(FragID, GL_COMPILE_STATUS, &Result);
glGetShaderiv(FragID, GL_INFO_LOG_LENGTH, &InfoLogLength);
if ( InfoLogLength > 0 ){
std::vector<char> FragmentShaderErrorMessage(InfoLogLength+1);
glGetShaderInfoLog(FragID, InfoLogLength, NULL, &FragmentShaderErrorMessage[0]);
printf("%s\n", &FragmentShaderErrorMessage[0]);
}
GLuint ProgramID = glCreateProgram();
glAttachShader(ProgramID,VertID);
glAttachShader(ProgramID,FragID);
glLinkProgram(ProgramID);
glGetProgramiv(ProgramID, GL_LINK_STATUS, &Result);
glGetProgramiv(ProgramID, GL_INFO_LOG_LENGTH, &InfoLogLength);
if ( InfoLogLength > 0 ){
std::vector<char> ProgramErrorMessage(InfoLogLength+1);
glGetProgramInfoLog(ProgramID, InfoLogLength, NULL, &ProgramErrorMessage[0]);
printf("%s\n", &ProgramErrorMessage[0]);
}
return ProgramID;
}
static char const* ReadShaderFile(const char* path){
std::string ShaderCode;
std::ifstream ShaderStream(path,std::ios::in);
if(ShaderStream.is_open()){
std::string line = "";
while (std::getline(ShaderStream,line)){
ShaderCode +="\n"+line;
}
ShaderStream.close();
return ShaderCode.c_str();
}else{
return 0;}
}
这基本上是直接来自我遵循link的教程,只有更改是将文件读取放在ReadShaderFile函数中。
编辑2:我的OpenGL上下文是使用以下版本创建的:
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR,3);
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR,3);
问题确实是着色器加载器。您正在添加此声明
ShaderCode +="\n"+line;
每行之前的换行符(\ n),与输入相比,向下移动一行。
由于版本语句必须位于着色器的第一行,并且您将其移动到第二行,因此驱动程序似乎忽略了该语句。例如,我的NVIDIA驱动程序声明:
error C0204: version directive must be first statement and may not be repeated
一个简单的解决方法是在每行之后添加到换行符而不是之前,但我强烈建议你不要逐行读取整个文件,因为这会给你带来糟糕的性能。例如,ShaderCode
变量将针对每一行调整大小,这意味着您将获得整个字符串的内存分配和复制操作。看看this question on how to read complete files。
编辑:
另一个问题是你要返回局部变量的c_str()
指针。当ReadShaderFile
方法结束时,std::string ShaderCode
变量超出范围(从而释放它的内容),并且返回的指针指向无效的内存地址。
解决方案:将std :: string对象而不是const char指针返回到其内容。