GLSL: Please help me debug - source included!

Hi Folks,

I am trying to make a very simple GLSL program. Attached is the current version. Note that the shader string fragments are at the very bottom. After tediously following the instructions in the Orange Book, I would expect to see a spinning red square. Instead, I get a spinning white square. The only function of importance should be GLEventListener.init(), starting on line 157.

Can you all run the program and do something for me:

  1. Tell me what color your square is.
  2. Give me any advice you have on what could be going wrong if you also get a white square.

As far as program logic goes, the fragment shader should just make the fragment red. As a sanity check, I tried putting all vertices at (0,0,0) but it looks like the fixed function pipeline still rules.

Thanks!


/*
 * Created on Dec 7, 2004
 */
import javax.swing.JFrame;
import net.java.games.jogl.*;
import net.java.games.jogl.GLDrawable;

/**
 * @author Mark Bastian (shaddam_IV on JOGL message boards)
 *
 */
public class TheSimplestVertexShader extends JFrame
{
    public TheSimplestVertexShader()
    {
        super("The Simplest Bare-Bones Shader Ever!");
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setSize(800, 600);
        initialize();
    }
    
    private static int loadShader(GL gl, String[] shaders, int shaderType)
    {
        //Following the steps in the orange book...
        
        //1. Create an empty shader object
        int shader = gl.glCreateShaderObjectARB(shaderType);
       
        //Sanity checks
        assert shader != 0; assert 0 == gl.glGetError();
        
        //Create an array consisting of shader lengths
        int[] lengths = new int[shaders.length];
        for(int i = 0; i < shaders.length; i++) lengths[i] = shaders[i].length();
        
        //2. Provide shader source code
        //Everything is simply copied over.  There is not compiling and such done at
        //this time.  Resources may be freed or modified after the call is made since 
        //a copy is performed.
        gl.glShaderSourceARB(shader, shaders.length, shaders, lengths);

        assert 0 == gl.glGetError();
        
        //3. Compile the shader
        gl.glCompileShaderARB(shader);
        
        assert 0 == gl.glGetError();
        
        validateShader(gl, shader);
        
        return shader;
    }
    
    private static void validateShader(GL gl, int shader)
    {
        //3a. Check the status of compilation
        int[] compileStatus = new int[1];
        gl.glGetObjectParameterivARB(shader, GL.GL_OBJECT_COMPILE_STATUS_ARB, compileStatus);
        if(compileStatus[0] == GL.GL_TRUE)
        {
            System.out.println("Shader compiled!");
        }
        else
        {
            System.out.println("Shader compilation failed!  Log dump:");
            int[] infologlength = new int[1];
            byte[] infolog = new byte[1024];
            gl.glGetInfoLogARB(shader, 1024, infologlength, infolog);
            String logstr = new String(infolog);
            System.out.println(logstr);
        }
    }
    
    private static void loadProgram(GL gl, String[] vertexShaders, String[] fragmentShaders)
    {
        int vertexShader = loadShader(gl, vertexShaders, GL.GL_VERTEX_SHADER_ARB);
        int fragmentShader = loadShader(gl, fragmentShaders, GL.GL_FRAGMENT_SHADER_ARB);
        
        //4. Create the program object
        //The object is initially empty, so you will need to link a shader to it.
        int programObject = gl.glCreateProgramObjectARB();
        
        assert 0 != programObject; assert 0 == gl.glGetError();
        
        //5. Attach the shader object to the program.  You can link as many shaders
        //as you want to a program object.  Once the linking is done, you no longer
        //need the shader object.
        gl.glAttachObjectARB(programObject, vertexShader);
        gl.glAttachObjectARB(programObject, fragmentShader);
        
        assert 0 == gl.glGetError();
        
        //6. Link the program object.
        gl.glLinkProgramARB(programObject);
        
        //6a. Check the status of compilation
        int[] linkStatus = new int[1];
        gl.glGetObjectParameterivARB(programObject, GL.GL_OBJECT_LINK_STATUS_ARB, linkStatus);
        
        assert 0 == gl.glGetError();
        
        if(linkStatus[0] == GL.GL_TRUE)
        {
            System.out.println("Program linked!");
        }
        else
        {
            System.out.println("Linking failed!  Log dump:");
            int[] infologlength = new int[1];
            byte[] infolog = new byte[1024];
            gl.glGetInfoLogARB(programObject, 1024, infologlength, infolog);
            String logstr = new String(infolog);
            System.out.println(logstr);
            System.exit(-1);
        }
        
        assert 0 == gl.glGetError();
        
        //7. Install the program into the current state machine
        gl.glUseProgramObjectARB(programObject);
        
        assert 0 == gl.glGetError();
    }
    
    private void initialize()
    {
        GLCapabilities glcaps = new GLCapabilities();
        GLCanvas canvas = GLDrawableFactory.getFactory().createGLCanvas(glcaps);
        final Animator animator = new Animator(canvas);
        
        canvas.addGLEventListener(new GLEventListener()
        {
            private float theta = 0.0f;
            public void display(GLDrawable drawable)
            {
                GL gl = drawable.getGL();
                
                assert 0 == gl.glGetError();
                
                gl.glClear(GL.GL_COLOR_BUFFER_BIT | GL.GL_DEPTH_BUFFER_BIT);
                gl.glLoadIdentity();
                gl.glTranslatef(0.0f, 0.0f, -5.0f);
                gl.glRotatef(theta += 1.0f, 1.0f, 1.0f, 0.0f);
                
                gl.glBegin(GL.GL_QUADS);
                gl.glVertex3f(-1.0f, -1.0f, 0.0f);
                gl.glVertex3f(+1.0f, -1.0f, 0.0f);
                gl.glVertex3f(+1.0f, +1.0f, 0.0f);
                gl.glVertex3f(-1.0f, +1.0f, 0.0f);
                gl.glEnd();
                
                assert 0 != gl.glGetHandleARB(GL.GL_PROGRAM_OBJECT_ARB);
            }

            public void displayChanged(GLDrawable arg0, boolean arg1, boolean arg2) {}

            public void init(GLDrawable drawable)
            {
                String[] vertexShaders = {vs};
                String[] fragmentShaders = {fs};
                GL gl = drawable.getGL();
                loadProgram(gl, vertexShaders, fragmentShaders);
            }

            public void reshape(GLDrawable drawable, int x, int y, int w, int h)
            {
                GL gl = drawable.getGL();
                GLU glu = drawable.getGLU();
                double aspect = (h != 0.0) ? (double)w / (double)h : 1.0;
                gl.glMatrixMode(GL.GL_PROJECTION);
                gl.glLoadIdentity();
                glu.gluPerspective(60.0, aspect, 0.01, 10.0);
                gl.glMatrixMode(GL.GL_MODELVIEW);
            }
        });
        getContentPane().add(canvas);
        show();
        animator.start();
    }
    
    public static void main(String[] args)
    {
        new TheSimplestVertexShader();
    }
    
    private static String vs = 
        "//\n"+
        "//The simplest vertex shader ever\n"+
        "void main()\n"+
        "{\n"+
        "    gl_Position = ftransform();\n" +
        "    //Sanity check: Smash everything to a point-doesn't work when enabled\n"+
        "    //gl_Position = vec4(0.0, 0.0, 0.0, 1.0);\n"+
        "}\n";
    
    private static String fs = 
        "//\n"+
        "//The simplest fragment shader ever\n" +
        "//Just color the fragment red\n"+
        "void main()\n"+
        "{\n"+
        "    gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);\n"+
        "}\n";
}

The square is red on my configuration: ATI Radeon 9800 Pro, Windows XP SP2, ATI Catalyst drivers version 4.9 (I had problems with 4.10 when I first installed this machine). The program didn’t run with the ATI Linux drivers I have installed or on my NVidia mobile chip which doesn’t support OpenGL 2.0.

Interesting. I just ran the code on my home machine (XP Home with ATI Radeon 9600 XT) and I now get the red square. This makes me :slight_smile: that my code seems right but :frowning: that it doesn’t work on my Macs.

So, let me solicit your advice on this issue with my Macs:
Both are OS X. One is a dual G5 PowerMac wth a GeForce FX 5200 Ultra and the other is a PowerBook G4 with an ATI Radeon Mobility 9700. Both have the requisite ARB_fragment_shader, ARB_vertex_shader, and ARB_shader_object extensions. Both state that the shaders compile and the program links. The program object has a nonzero index. After all this, the square is white!

I believe the problem is either 1)the Mac implementation of OpenGL or 2)the JOGL Mac bindings. I lean towards 1 because the problem occurs on the Mac but I lean towards 2 because the prerequisites are met and nothing is reported as an error.

Two tests to narrow things down:

  1. Do any of you know of any non-JOGL GLSL demos out there for the Mac?
  2. Are any of you other Mac users able to get the red square?

Any other ideas?

Thanks!

I decided to go the tedious route and write a C implemetation using GLUT and sure enough, I get only a white square. It looks like the Mac doesn’t properly implement the latest shader extensions (Even though both or my OS X cards claim 1.5+ compliance and list all of the required extension strings!).

Can any of you other Mac users out there run the above code and post your results, just so I can get a few more data points to further justify my results?

[quote]I decided to go the tedious route and write a C implemetation using GLUT and sure enough, I get only a white square. It looks like the Mac doesn’t properly implement the latest shader extensions (Even though both or my OS X cards claim 1.5+ compliance and list all of the required extension strings!).

Can any of you other Mac users out there run the above code and post your results, just so I can get a few more data points to further justify my results?
[/quote]
Could you post your C version of this test case? I would like to forward this to our OpenGL team at Apple and have them look at it.

btw. There were several updates to Mac OS X 10.3 recently, most of them addressing OpenGL in some way. Do you have the latest update installed?

cheers

Sure thing! I’ll put it up in a bit. In the mean time, gziemski, have you tried running my JOGL version? What was your result?

Edit: I am assuming you are running on a Mac, of course!

[quote]Sure thing! I’ll put it up in a bit. In the mean time, gziemski, have you tried running my JOGL version? What was your result?

Edit: I am assuming you are running on a Mac, of course!
[/quote]
I run on a Mac indeed. I work at Apple on Java for Mac OS X :wink:

I’ll try your test case later today, on both Panther and Tiger. From my experience I know our OpenGL guys will prefer a native test case, so please remember to post it.

cheers

Wow, some people have the greatest jobs!

Here’s the sample. Please read the comments inserted. I think there is some interesting stuff. I tried to be thorough. I also submitted a bug report earlier today (#3921753) to ADC but that version of the C source is now out of date. This one is much better. I will update that, too.

Thanks!


#include <stdlib.h>

#include <GLUT/glut.h>
//Note the name change here.  I am using the sgi glext.h file since it is more
//recent than the one in the OpenGL framework for Mac OS X.  It seems to work.
#include <OpenGL/glextsgi.h>

//Can a program get any simpler?
static char* vs = 
      "//\n"
      "//The simplest vertex shader ever\n"
    "void main()\n"
    "{\n"
    "  gl_Position = ftransform();\n"
      "  //Sanity check: Smash everything to a point-doesn't work when enabled\n"
    "  //gl_Position = vec4(0.0, 0.0, 0.0, 1.0);\n"
      "}\n";

static char* fs = 
      "//\n"
      "//The simplest fragment shader ever\n"
      "//Just color the fragment red\n"
      "void main()\n"
      "{\n"
      "gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);\n"
      "}\n";
      
void display(void)
{
    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

    glLoadIdentity();
      glTranslatef(0.0f, 0.0f, -5.0f);
      
    glBegin(GL_QUADS);
      glVertex3f(-1.0f, -1.0f, 0.0f);
      glVertex3f(+1.0f, -1.0f, 0.0f);
      glVertex3f(+1.0f, +1.0f, 0.0f);
      glVertex3f(-1.0f, +1.0f, 0.0f);
      glEnd();

    glutSwapBuffers();
}

void reshape(int width, int height)
{
    glViewport(0, 0, width, height);
      
      glMatrixMode(GL_PROJECTION);
      glLoadIdentity();
      gluPerspective(60.0, (double)width / (double)height, 0.01, 10.0);
      glMatrixMode(GL_MODELVIEW);
}

void idle(void)
{
    glutPostRedisplay();
}
      
//Note that this loading routine is a very heavily modified (simplified and printf commented)
//version of the routine found in the ogl2brick orange book example by Randi Rost.
//See chapter 7 of the orange book, esp. listing 7.2 for details.
void installShaders(const GLcharARB *simpleVertex, const GLcharARB *simpleFragment)
{
    GLhandleARB simpleVS, simpleFS, simpleProg;
    GLint vertCompiled, fragCompiled;
    GLint linked;

    printf("Attempting to create two shaders\n");

    // Create a vertex shader object and a fragment shader object
    simpleVS = glCreateShaderObjectARB(GL_VERTEX_SHADER_ARB);
    simpleFS = glCreateShaderObjectARB(GL_FRAGMENT_SHADER_ARB);

    printf("Shaders created...\nLoading shader source...\n");
      
    // Load source code strings into shaders
    glShaderSourceARB(simpleVS, 1, &simpleVertex, NULL);
    glShaderSourceARB(simpleFS, 1, &simpleFragment, NULL);

    printf("Source loaded...\nCompiling the vertex shader...\n");
    //Compile the shader
      glCompileShaderARB(simpleVS);

    printf("Shader compiling.  Awaiting result...\n");

    //Block until we know the compilation status
    glGetObjectParameterivARB(simpleVS, GL_OBJECT_COMPILE_STATUS_ARB, &vertCompiled);

    if(GL_TRUE == vertCompiled)
          printf("Vertex shader compiled successfully!\n");
    else
          printf("Vertex shader compilation failed!\n");

    printf("Compiling the fragment shader...\n");
    //Compile the shader
    glCompileShaderARB(simpleFS);
      
      printf("Shader compiling.  Awaiting result...\n");
      
    //Block until we know the compilation status
    glGetObjectParameterivARB(simpleFS, GL_OBJECT_COMPILE_STATUS_ARB, &fragCompiled);

    if(GL_TRUE == fragCompiled)
          printf("Fragment shader compiled successfully!\n");
    else
          printf("Fragment shader compilation failed!\n");

    //Return due to failure
    if((GL_TRUE != vertCompiled) || (GL_TRUE != fragCompiled)) return 0;

    printf("Create a GPU program object...");
      
    // Create a program object and attach the two compiled shaders
    simpleProg = glCreateProgramObjectARB();
      
      printf("Attaching shaders to the program...\n");
      
    glAttachObjectARB(simpleProg, simpleVS);
    //An interesting thing is that if you comment this line you have no fragment
      //shading element.  This should successfully link and just use the fixed 
      //function vertex processor.  That is what the orange book says.  Instead,
      //if you comment this line, the code won't link.
      glAttachObjectARB(simpleProg, simpleFS);

    printf("Shaders attached.\nLinking...");
      
    // Link the program object and print out the info log
    glLinkProgramARB(simpleProg);
      
      printf("Program linking.  Awaiting result...\n");
      
      //Block until we know the link status
    glGetObjectParameterivARB(simpleProg, GL_OBJECT_LINK_STATUS_ARB, &linked);

    if(GL_TRUE == linked)
          printf("Linked successfully!\n");
    else
          printf("Linking failed!\n");

    if(GL_TRUE != linked) return;

    // Install program object as part of current state
    printf("Using the program...\n");
    glUseProgramObjectARB(simpleProg);
      
      if(0 != glGetError()) printf("A GL Error Occurred!");
      printf("Successfully loaded the shader program!\n");
}

int main(int argc, char** argv)
{
    int success = 0;
      
    glutInit(&argc, argv);

    glutInitDisplayMode(GLUT_RGBA | GLUT_DOUBLE | GLUT_DEPTH);
    glutInitWindowSize(640, 480);
    glutCreateWindow("Simple GLSL Test Program");
    
      #ifdef GL_ARB_shader_objects
    printf("GL_ARB_shader_objects present\n");
    #endif
      
      #ifdef GL_ARB_vertex_shader
    printf("GL_ARB_vertex_shader present\n");
    #endif
      
      #ifdef GL_ARB_fragment_shader
    printf("GL_ARB_fragment_shader present\n");
    #endif
      
    glutDisplayFunc(display);
    glutReshapeFunc(reshape);
    glutIdleFunc(idle);
      
      installShaders(vs, fs);
      
    glutMainLoop();
    return EXIT_SUCCESS;
}

Ok, so I got a response from Apple (Talk about fast turnaround). Here’s the scoop.

I was incorrect that you need only the ARB_fragment_shader, ARB_vertex_shader, and ARB_shader_objects extensions to run GLSL programs. You also need GL_ARB_shading_language_100.

I verified that my home machine (Windows XP w/ ATI Radeon 9600XT) does indeed have all of the above, which explains why the sample works on that machine.

This leaves me with 2 questions:

  1. Do any of you have Windows machines with either a GeForce FX 5200 Ultra or an ATI Mobility Radeon 9700? If so, would you mind running the above sample and tell me if it works? Or, could you just dump the extension string and tell me if GL_ARB_shading_language_100 is supported? I think it should be as long as the drivers are updated, but I would like to be sure.

  2. What is the purpose of having the ARB_fragment_shader, ARB_vertex_shader, and ARB_shader_objects extensions without GL_ARB_shading_language_100? It seems like that is saying that the hardware supports high level shaders but doesn’t implement them. Is this interpretation correct?

Thanks!

got to love them apple dudes :slight_smile:

[quote]2. What is the purpose of having the ARB_fragment_shader, ARB_vertex_shader, and ARB_shader_objects extensions without GL_ARB_shading_language_100?
[/quote]
ARB_vertex_program and ARB_fragment_program are useful even without ARB_shader_objects. Withi them, you can write an idealized form of assembly code that gets uploaded onto the card and which is portable among graphics cards. GLSL provides a higher-level view of this kind of programmability. The JOGL demos were recently ported to use ARB_fragment_program instead of NVidia’s register combiners; see the sources for examples of using glGenProgramsARB, glBindProgramARB, glProgramStringARB, etc.

Yeah, I realize that I can write assembly style stuff with the ARB_program extensions. What I want is to use the ARB_shader extensions to do high level stuff.

I have in fact run and written stuff based on the newly revamped ARB based demos. I think these updates are great!

Even if I can’t do GLSL now, I am still very curious about why you would support the three shader object extensions and not support GL_ARB_shading_language_100. It just seems pointless, but surely there must be a reason. Any ideas?

Ah, I was confused and didn’t realize there were both an ARB_fragment_program and an ARB_fragment_shader. I agree, it does seem strange to support that extension but not GLSL in the drivers.