Ошибка компиляции шейдера OpenGL - PullRequest
1 голос
/ 27 мая 2010

У меня небольшая проблема с моим кодом для компиляции шейдеров, а именно они оба регистрируются как неудачные компиляции, и журнал не получен.

Это код компиляции шейдера:

/* Make the shader */

Uint size;
GLchar* file;

loadFileRaw(filePath, file, &size);

const char * pFile = file;
const GLint pSize = size;


newCashe.shader = glCreateShader(shaderType);
glShaderSource(newCashe.shader, 1, &pFile, &pSize); 
glCompileShader(newCashe.shader);

GLint shaderCompiled;

glGetShaderiv(newCashe.shader, GL_COMPILE_STATUS, &shaderCompiled);

if(shaderCompiled == GL_FALSE)
{
    ReportFiler->makeReport("ShaderCasher.cpp", "loadShader()", "Shader did not compile", "The shader " + filePath + " failed to compile, reporting the error - " + OpenGLServices::getShaderLog(newCashe.shader));

}

А это вспомогательные функции:

bool loadFileRaw(string fileName, char* data, Uint* size)
{    
    if (fileName != "") 
    {
        FILE *file = fopen(fileName.c_str(), "rt");

        if (file != NULL) 
        {
            fseek(file, 0, SEEK_END);
            *size = ftell(file);
            rewind(file);

            if (*size > 0) 
            {
                data = (char*)malloc(sizeof(char) * (*size + 1));
                *size = fread(data, sizeof(char), *size, file);
                data[*size] = '\0';
            }

            fclose(file);
        }
    }

    return data;
}

string OpenGLServices::getShaderLog(GLuint obj)
{
    int infologLength = 0;

    int charsWritten  = 0;
    char *infoLog;

    glGetShaderiv(obj, GL_INFO_LOG_LENGTH,&infologLength);

    if (infologLength > 0)
    {
        infoLog = (char *)malloc(infologLength);
        glGetShaderInfoLog(obj, infologLength, &charsWritten, infoLog);

        string log = infoLog;

        free(infoLog);

        return log;
    }

    return "<Blank Log>";
}

и загружаемые шейдеры:

void main(void)
{
    gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);
}

void main(void)
{
    gl_Position = ftransform();
}

Короче я получаю

From: ShaderCasher.cpp, In: loadShader(), Subject: Shader did not compile
Message: The shader Data/Shaders/Standard/standard.vs failed to compile, reporting the error - <Blank Log>

для каждого шейдера, который я компилирую.

Я пытался заменить чтение файла просто жестко закодированной строкой, но я получаю ту же ошибку, поэтому должно быть что-то не так с тем, как я их компилирую. Я запустил и скомпилировал примеры программ с шейдерами, поэтому сомневаюсь, что проблема в моих драйверах, но в любом случае я использую Nvidia 8600m GT.

Может кто-нибудь помочь?

Ответы [ 2 ]

4 голосов
/ 27 мая 2010

В loadFileRaw вы передаете char *data по значению. Затем вы назначаете ему:

data = (char*)malloc(sizeof(char) * (*size + 1));

так что ваш звонок

loadFileRaw(filePath, file, &size);

не не меняет значение file! Чтобы изменить file, передайте его указателем:

loadFileRaw(filePath, &file, &size);

, а затем измените свою функцию на

bool loadFileRaw(string fileName, char** data, Uint* size) { ... }

и в нем

*data = (char*)malloc(sizeof(char) * (*size + 1));

и т.д.. (для остальных ссылок на data).

Тем не менее, поскольку вы используете C ++, изучите использование std::vector<char> или std::string для управления памятью. Чтобы начать,

{
   std::vector<char> data;
   data.resize(100); // now it has 100 entries, initialized to zero
   silly_C_function_that_takes_a_char_pointer_and_a_size(&data[0], 100); // OK
} // presto, memory freed

Также вы можете использовать std::ifstream для чтения файлов вместо FILE *.

1 голос
/ 27 мая 2010

Попробуйте с:

glShaderSource (newCashe.shader, 1, & pFile, NULL);

Наиболее часто используемый вызов для загрузки источников шейдеров.

...