OpenGL GLSL在令牌处期 待“::” “将clamp()添加到片段着色器时

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

以下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);
c++ opengl glsl
1个回答
2
投票

问题确实是着色器加载器。您正在添加此声明

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指针返回到其内容。

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