/** * Test using a geometry and fragment shaders to implement stippled lines. * * Brian Paul * April 2011 */ #include <assert.h> #include <string.h> #include <stdio.h> #include <stdlib.h> #include <math.h> #include <GL/glew.h> #include "glut_wrap.h" #include "shaderutil.h" static GLint WinWidth = 500, WinHeight = 500; static GLint Win = 0; static GLuint VertShader, GeomShader, FragShader, Program; static GLboolean Anim = GL_TRUE; static GLboolean UseGeomShader = GL_TRUE; static GLfloat Xrot = 0, Yrot = 0; static int uViewportSize = -1, uStippleFactor = -1, uStipplePattern = -1; static int NumPoints = 50; static const GLushort StipplePattern = 0x10ff; static GLuint StippleFactor = 2; static void CheckError(int line) { GLenum err = glGetError(); if (err) { printf("GL Error %s (0x%x) at line %d\n", gluErrorString(err), (int) err, line); } } /** * Set stipple factor and pattern for geometry shader. * * We convert the 16-bit stipple pattern into an array of 16 float values * then pass the array as a uniform variable. * * Note: With GLSL 1.30 or later the stipple pattern could be implemented * as an ordinary integer since GLSL 1.30 has true integer types and bit * shifts and bit masks. * */ static void SetStippleUniform(GLint factor, GLushort pattern) { GLfloat p[16]; int i; for (i = 0; i < 16; i++) { p[i] = (pattern & (1 << i)) ? 1.0f : 0.0f; } glUniform1fv(uStipplePattern, 16, p); glUniform1f(uStippleFactor, factor); } static void Redisplay(void) { glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT ); glPushMatrix(); glRotatef(Xrot, 1, 0, 0); glRotatef(Yrot, 0, 0, 1); if (UseGeomShader) { glUseProgram(Program); glDisable(GL_LINE_STIPPLE); } else { glUseProgram(0); glEnable(GL_LINE_STIPPLE); } glDrawArrays(GL_LINES, 0, NumPoints / 2); glPopMatrix(); glutSwapBuffers(); } static void Idle(void) { int curTime = glutGet(GLUT_ELAPSED_TIME); Xrot = curTime * 0.02; Yrot = curTime * 0.05; glutPostRedisplay(); } static void Reshape(int width, int height) { float ar = (float) width / height; glViewport(0, 0, width, height); glMatrixMode(GL_PROJECTION); glLoadIdentity(); #if 1 glFrustum(-ar, ar, -1, 1, 3, 25); #else glOrtho(-3.0*ar, 3.0*ar, -3.0, 3.0, 3, 25); #endif glMatrixMode(GL_MODELVIEW); glLoadIdentity(); glTranslatef(0, 0, -10); { GLfloat viewport[4]; glGetFloatv(GL_VIEWPORT, viewport); glUniform2f(uViewportSize, viewport[2], viewport[3]); } } static void CleanUp(void) { glDeleteShader(FragShader); glDeleteShader(VertShader); glDeleteShader(GeomShader); glDeleteProgram(Program); glutDestroyWindow(Win); } static void Key(unsigned char key, int x, int y) { (void) x; (void) y; switch(key) { case ' ': case 'a': Anim = !Anim; if (Anim) { glutIdleFunc(Idle); } else glutIdleFunc(NULL); break; case 'g': UseGeomShader = !UseGeomShader; printf("Use geometry shader? %d\n", UseGeomShader); break; case 'x': Xrot ++; break; case 27: CleanUp(); exit(0); break; } glutPostRedisplay(); } static void MakePointsVBO(void) { struct vert { GLfloat pos[3]; GLfloat color[3]; }; struct vert *v; GLuint vbo; int i; glGenBuffers(1, &vbo); glBindBuffer(GL_ARRAY_BUFFER, vbo); glBufferData(GL_ARRAY_BUFFER, NumPoints * sizeof(struct vert), NULL, GL_STATIC_DRAW); v = glMapBuffer(GL_ARRAY_BUFFER, GL_WRITE_ONLY); for (i = 0; i < NumPoints; i++) { v[i].color[0] = (rand() % 1000) / 1000.0; v[i].color[1] = (rand() % 1000) / 1000.0; v[i].color[2] = (rand() % 1000) / 1000.0; v[i].pos[0] = ((rand() % 2000) - 1000.0) / 500.0; v[i].pos[1] = ((rand() % 2000) - 1000.0) / 500.0; v[i].pos[2] = ((rand() % 2000) - 1000.0) / 500.0; } glUnmapBuffer(GL_ARRAY_BUFFER); glVertexPointer(3, GL_FLOAT, sizeof(struct vert), (void *) 0); glEnable(GL_VERTEX_ARRAY); glColorPointer(3, GL_FLOAT, sizeof(struct vert), (void *) sizeof(float[3])); glEnable(GL_COLOR_ARRAY); } static void Init(void) { static const char *fragShaderText = "uniform float StipplePattern[16]; \n" "varying float stippleCoord; \n" "void main() \n" "{ \n" " // check the stipple pattern and discard if value is zero \n" " // TODO: we should really undo the perspective interpolation here \n" " // so that it's linear. \n" " float stip = StipplePattern[int(fract(stippleCoord) * 16.0)]; \n" " if (stip == 0.0) \n" " discard; \n" " gl_FragColor = gl_Color; \n" "} \n"; static const char *vertShaderText = "void main() \n" "{ \n" " gl_FrontColor = gl_Color; \n" " gl_Position = ftransform(); \n" "} \n"; static const char *geomShaderText = "#version 120 \n" "#extension GL_ARB_geometry_shader4: enable \n" "uniform vec2 ViewportSize; \n" "uniform float StippleFactor; \n" "varying float stippleCoord; \n" "void main() \n" "{ \n" " vec4 pos0 = gl_PositionIn[0]; \n" " vec4 pos1 = gl_PositionIn[1]; \n" " // Convert eye coords to window coords \n" " // Note: we're off by a factor of two here, make up for that below \n" " vec2 p0 = pos0.xy / pos0.w * ViewportSize; \n" " vec2 p1 = pos1.xy / pos1.w * ViewportSize; \n" " float len = length(p0.xy - p1.xy); \n" " // Emit first vertex \n" " gl_FrontColor = gl_FrontColorIn[0]; \n" " gl_Position = pos0; \n" " stippleCoord = 0.0; \n" " EmitVertex(); \n" " // Emit second vertex \n" " gl_FrontColor = gl_FrontColorIn[1]; \n" " gl_Position = pos1; \n" " stippleCoord = len / StippleFactor / 32.0; // Note: not 16, see above \n" " EmitVertex(); \n" "} \n"; if (!ShadersSupported()) exit(1); if (!glutExtensionSupported("GL_ARB_geometry_shader4")) { fprintf(stderr, "Sorry, GL_ARB_geometry_shader4 is not supported.\n"); exit(1); } VertShader = CompileShaderText(GL_VERTEX_SHADER, vertShaderText); FragShader = CompileShaderText(GL_FRAGMENT_SHADER, fragShaderText); GeomShader = CompileShaderText(GL_GEOMETRY_SHADER_ARB, geomShaderText); assert(GeomShader); Program = LinkShaders3(VertShader, GeomShader, FragShader); assert(Program); CheckError(__LINE__); /* * The geometry shader accepts lines and produces lines. */ glProgramParameteriARB(Program, GL_GEOMETRY_INPUT_TYPE_ARB, GL_LINES); glProgramParameteriARB(Program, GL_GEOMETRY_OUTPUT_TYPE_ARB, GL_LINE_STRIP); glProgramParameteriARB(Program, GL_GEOMETRY_VERTICES_OUT_ARB, 4); CheckError(__LINE__); glLinkProgramARB(Program); /* check link */ { GLint stat; GetProgramiv(Program, GL_LINK_STATUS, &stat); if (!stat) { GLchar log[1000]; GLsizei len; GetProgramInfoLog(Program, 1000, &len, log); fprintf(stderr, "Shader link error:\n%s\n", log); } } glUseProgram(Program); uViewportSize = glGetUniformLocation(Program, "ViewportSize"); uStippleFactor = glGetUniformLocation(Program, "StippleFactor"); uStipplePattern = glGetUniformLocation(Program, "StipplePattern"); glUniform1f(uStippleFactor, StippleFactor); glClearColor(0.3f, 0.3f, 0.3f, 0.0f); printf("GL_RENDERER = %s\n",(const char *) glGetString(GL_RENDERER)); assert(glIsProgram(Program)); assert(glIsShader(FragShader)); assert(glIsShader(VertShader)); assert(glIsShader(GeomShader)); glLineStipple(StippleFactor, StipplePattern); SetStippleUniform(StippleFactor, StipplePattern); MakePointsVBO(); } int main(int argc, char *argv[]) { glutInit(&argc, argv); if (argc > 1) { int n = atoi(argv[1]); if (n > 0) { NumPoints = n; } else { printf("Invalid number of points\n"); return 1; } } glutInitWindowSize(WinWidth, WinHeight); glutInitDisplayMode(GLUT_RGB | GLUT_DEPTH | GLUT_DOUBLE); Win = glutCreateWindow(argv[0]); glewInit(); glutReshapeFunc(Reshape); glutKeyboardFunc(Key); glutDisplayFunc(Redisplay); if (Anim) glutIdleFunc(Idle); Init(); glutMainLoop(); return 0; }