Невозможно связать скомпилированные шейдеры (GLSL) - PullRequest
3 голосов
/ 21 декабря 2009

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

Vertex info
-----------
(0) : error C3001: no program defined
Fragment info
-------------
(0) : error C3001: no program defined

Приведенный ниже код выполняет связывание GLSL:

program_ = glCreateProgramObjectARB();

if (vertex_) {
   glAttachObjectARB(program_, vertex_);
}

if (fragment_) {
   glAttachObjectARB(program_, fragment_);
}

glLinkProgramARB(program_); 

vertex_ и фрагмент_ - вершинный и фрагментный шейдер, а program_ - это ProgramObjectARB.

Я бы хотел знать, какие необходимые модификации мне нужно сделать, чтобы запустить этот прогон.

Вершинный шейдер:

varying vec4 OCPosition;   // surface positon in world's coordinates
varying vec4 ECposition;   // surface position in eye coordinates
varying vec4 ECballCenter; // ball center in eye coordinates
uniform vec4 BallCenter;   // ball center in modelling coordinates

void main()
{
  OCPosition   = gl_Vertex;
  ECposition   = gl_ModelViewMatrix * OCPosition;
  ECballCenter = gl_ModelViewMatrix * BallCenter;

  gl_Position  = gl_ProjectionMatrix * ECposition;
}

Фрагмент шейдера:

varying vec4 OCPosition;   // surface position in world coordinates
varying vec4 ECposition;   // surface position in eye coordinates
varying vec4 ECballCenter; // ball center in eye coordinates

uniform vec4  LightDir;     // light direction, should be normalized
uniform vec4  HVector;      // reflection vector for infinite light source
uniform vec4  SpecularColor;
uniform vec4  BaseColor, StripeColor;

uniform float StripeWidth;  // = 0.3
uniform float FWidth;       // = 0.005

void main()
{
vec3  normal;              // Analytically computed normal
vec4  p;                   // Point in shader space
vec4  surfColor;           // Computed color of the surface
float intensity;           // Computed light intensity
vec4  distance;            // Computed distance values
float inorout;             // Counter for computing star pattern

p.xyz = normalize(OCPosition.xyz);    // Calculate p
p.w   = 1.0;

distance.y = StripeWidth - abs(p.z);
distance.xy = smoothstep(-FWidth, FWidth, distance.xy);

surfColor = mix(BaseColor, StripeColor, distance.y);

// normal = point on surface for sphere at (0,0,0)
normal = normalize(ECposition.xyz - ECballCenter.xyz);

// Per fragment diffuse lighting
intensity  = 0.2; // ambient
intensity += 0.8 * clamp(dot(LightDir.xyz, normal), 0.0, 1.0);
surfColor *= intensity;

// Per fragment specular lighting
intensity  = clamp(dot(HVector.xyz, normal), 0.0, 1.0);
intensity  = pow(intensity, SpecularColor.a);
surfColor += SpecularColor * intensity;

gl_FragColor = surfColor;
}

РЕДАКТИРОВАТЬ - РЕШЕНИЕ Кажется, метод, который я использовал для чтения исходного файла, был неверным. Шейдер будет компилироваться, но при компоновке он работает неправильно. Теперь я использую этот метод для чтения моего файла, и ссылки верны:

char *textFileRead(char *fn) {


FILE *fp;
char *content = NULL;

int count=0;

if (fn != NULL) {
    fp = fopen(fn,"rt");

    if (fp != NULL) {

        fseek(fp, 0, SEEK_END);
        count = ftell(fp);
        rewind(fp);

        if (count > 0) {
            content = (char *)malloc(sizeof(char) * (count+1));
            count = fread(content,sizeof(char),count,fp);
            content[count] = '\0';
        }
        fclose(fp);
    }
}
return content;
}

Ответы [ 3 ]

7 голосов
/ 21 декабря 2009

Этот специфический код ошибки из компилятора GLSL в драйвере Nvidia означает, что он не может найти функцию с именем 'main' в вашем шейдере (в данном случае это вершина или фрагмент)

Наиболее распространенная причина этого заключается в том, что вы фактически не передаёте программу драйверу, когда думаете, что это так - либо строка, которую вы передаете, не была инициализирована, либо вы передаете длину 0, и вы на самом деле ничего не компилируете.

Редактировать

Мое лучшее предположение - что-то не так с вызовами glCreateShader или glSetShaderSource. В большинстве случаев ошибки появляются только при попытке связать шейдеры (а не скомпилировать), поскольку драйвер впервые узнает, что у него есть полные шейдеры, а не только частичные, неполные шейдеры, которым для завершения нужен другой шейдер .

3 голосов
/ 21 декабря 2009

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

GLuint program = glCreateProgramObjectARB();
assert(program);

GLuint vertex_shader = glCreateShaderObjectARB(GL_VERTEX_SHADER_ARB);
assert(vertex_shader);
glShaderSourceARB(vertex_shader, 1, /*your vertex shader*/, 0);
glCompileShaderARB(vertex_shader);

GLint compile_status;
glGetObjectParameterivARB(vertex_shader, GL_OBJECT_COMPILE_STATUS_ARB, &compile_status);
assert(1 == compile_status);

GLuint fragment_shader = glCreateShaderObjectARB(GL_FRAGMENT_SHADER_ARB);
assert(fragment_shader);
glShaderSourceARB(fragment_shader, 1, /*your fragment shader*/, 0);
glCompileShaderARB(fragment_shader);

glGetObjectParameterivARB(vertex_shader, GL_OBJECT_COMPILE_STATUS_ARB, &compile_status);
assert(1 == compile_status);

glAttachObjectARB(program, vertex_shader);
assert(GL_NO_ERROR == glGetError());

glAttachObjectARB(program, fragment_shader);
assert(GL_NO_ERROR == glGetError());

glLinkProgramARB(program);  
assert(GL_NO_ERROR == glGetError());

GLint link_status;
glGetObjectParameterivARB(program, GL_OBJECT_LINK_STATUS_ARB, &link_status);
assert(1 == link_status);

Вы делаете то же самое?

2 голосов
/ 21 декабря 2009

Попробуйте использовать glGetError после ваших вызовов glAttachObjectARB, чтобы проверить, не работают ли они.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...