blob: fa67a3c2cae9da9c2e93e2195e53bf21bc02b7a2 [file] [log] [blame]
Brian2ccd2642007-01-15 17:27:24 -07001/**
2 * "Mandelbrot" shader demo. Uses the example shaders from
3 * chapter 15 (or 18) of the OpenGL Shading Language "orange" book.
4 * 15 Jan 2007
5 */
6
7#include <assert.h>
8#include <string.h>
9#include <stdio.h>
10#include <stdlib.h>
11#include <math.h>
12#include <GL/gl.h>
13#include <GL/glut.h>
14#include <GL/glext.h>
15#include "extfuncs.h"
Brian2dca3372008-04-09 22:28:23 -060016#include "shaderutil.h"
Brian2ccd2642007-01-15 17:27:24 -070017
18
19static char *FragProgFile = "CH18-mandel.frag.txt";
20static char *VertProgFile = "CH18-mandel.vert.txt";
21
22/* program/shader objects */
23static GLuint fragShader;
24static GLuint vertShader;
25static GLuint program;
26
27
Brian2ccd2642007-01-15 17:27:24 -070028static struct uniform_info Uniforms[] = {
29 /* vert */
Brian2dca3372008-04-09 22:28:23 -060030 { "LightPosition", 3, GL_FLOAT, { 0.1, 0.1, 9.0, 0}, -1 },
31 { "SpecularContribution", 1, GL_FLOAT, { 0.5, 0, 0, 0 }, -1 },
32 { "DiffuseContribution", 1, GL_FLOAT, { 0.5, 0, 0, 0 }, -1 },
33 { "Shininess", 1, GL_FLOAT, { 20.0, 0, 0, 0 }, -1 },
Brian2ccd2642007-01-15 17:27:24 -070034 /* frag */
Brian2dca3372008-04-09 22:28:23 -060035 { "MaxIterations", 1, GL_FLOAT, { 12, 0, 0, 0 }, -1 },
36 { "Zoom", 1, GL_FLOAT, { 0.125, 0, 0, 0 }, -1 },
37 { "Xcenter", 1, GL_FLOAT, { -1.5, 0, 0, 0 }, -1 },
38 { "Ycenter", 1, GL_FLOAT, { .005, 0, 0, 0 }, -1 },
39 { "InnerColor", 3, GL_FLOAT, { 1, 0, 0, 0 }, -1 },
40 { "OuterColor1", 3, GL_FLOAT, { 0, 1, 0, 0 }, -1 },
41 { "OuterColor2", 3, GL_FLOAT, { 0, 0, 1, 0 }, -1 },
42 END_OF_UNIFORMS
Brian2ccd2642007-01-15 17:27:24 -070043};
44
45static GLint win = 0;
46
Brian2ccd2642007-01-15 17:27:24 -070047static GLfloat xRot = 0.0f, yRot = 0.0f, zRot = 0.0f;
48
Brian0d553462007-01-15 17:48:19 -070049static GLint uZoom, uXcenter, uYcenter;
50static GLfloat zoom = 1.0, xCenter = -1.5, yCenter = 0.0;
Brian2ccd2642007-01-15 17:27:24 -070051
52
53static void
54Redisplay(void)
55{
56 glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
57
Brian0d553462007-01-15 17:48:19 -070058 /* set interactive uniform parameters */
59 glUniform1fv_func(uZoom, 1, &zoom);
60 glUniform1fv_func(uXcenter, 1, &xCenter);
61 glUniform1fv_func(uYcenter, 1, &yCenter);
62
Brian2ccd2642007-01-15 17:27:24 -070063 glPushMatrix();
64 glRotatef(xRot, 1.0f, 0.0f, 0.0f);
65 glRotatef(yRot, 0.0f, 1.0f, 0.0f);
66 glRotatef(zRot, 0.0f, 0.0f, 1.0f);
67
68 glBegin(GL_POLYGON);
69 glTexCoord2f(0, 0); glVertex2f(-1, -1);
70 glTexCoord2f(1, 0); glVertex2f( 1, -1);
71 glTexCoord2f(1, 1); glVertex2f( 1, 1);
72 glTexCoord2f(0, 1); glVertex2f(-1, 1);
73 glEnd();
74
75 glPopMatrix();
76
Brian0d553462007-01-15 17:48:19 -070077 glFinish();
78 glFlush();
Brian2ccd2642007-01-15 17:27:24 -070079 glutSwapBuffers();
80}
81
82
83static void
84Reshape(int width, int height)
85{
86 glViewport(0, 0, width, height);
87 glMatrixMode(GL_PROJECTION);
88 glLoadIdentity();
89 glFrustum(-1.0, 1.0, -1.0, 1.0, 5.0, 25.0);
90 glMatrixMode(GL_MODELVIEW);
91 glLoadIdentity();
92 glTranslatef(0.0f, 0.0f, -6.0f);
93}
94
95
96static void
97CleanUp(void)
98{
99 glDeleteShader_func(fragShader);
100 glDeleteShader_func(vertShader);
101 glDeleteProgram_func(program);
102 glutDestroyWindow(win);
103}
104
105
106static void
107Key(unsigned char key, int x, int y)
108{
109 (void) x;
110 (void) y;
111
112 switch(key) {
113 case 'z':
Brian0d553462007-01-15 17:48:19 -0700114 zoom *= 0.9;
Brian2ccd2642007-01-15 17:27:24 -0700115 break;
116 case 'Z':
Brian0d553462007-01-15 17:48:19 -0700117 zoom /= 0.9;
Brian2ccd2642007-01-15 17:27:24 -0700118 break;
119 case 27:
120 CleanUp();
121 exit(0);
122 break;
123 }
124 glutPostRedisplay();
125}
126
127
128static void
129SpecialKey(int key, int x, int y)
130{
Brian0d553462007-01-15 17:48:19 -0700131 const GLfloat step = 0.1 * zoom;
Brian2ccd2642007-01-15 17:27:24 -0700132
133 (void) x;
134 (void) y;
135
136 switch(key) {
137 case GLUT_KEY_UP:
Brian0d553462007-01-15 17:48:19 -0700138 yCenter += step;
Brian2ccd2642007-01-15 17:27:24 -0700139 break;
140 case GLUT_KEY_DOWN:
Brian0d553462007-01-15 17:48:19 -0700141 yCenter -= step;
Brian2ccd2642007-01-15 17:27:24 -0700142 break;
143 case GLUT_KEY_LEFT:
Brian0d553462007-01-15 17:48:19 -0700144 xCenter -= step;
Brian2ccd2642007-01-15 17:27:24 -0700145 break;
146 case GLUT_KEY_RIGHT:
Brian0d553462007-01-15 17:48:19 -0700147 xCenter += step;
Brian2ccd2642007-01-15 17:27:24 -0700148 break;
149 }
150 glutPostRedisplay();
151}
152
153
Brian2ccd2642007-01-15 17:27:24 -0700154static void
155Init(void)
156{
Brian2dca3372008-04-09 22:28:23 -0600157 if (!ShadersSupported())
158 exit(1);
Brian2ccd2642007-01-15 17:27:24 -0700159
160 GetExtensionFuncs();
161
Brian2dca3372008-04-09 22:28:23 -0600162 vertShader = CompileShaderFile(GL_VERTEX_SHADER, VertProgFile);
163 fragShader = CompileShaderFile(GL_FRAGMENT_SHADER, FragProgFile);
164 program = LinkShaders(vertShader, fragShader);
Brian2ccd2642007-01-15 17:27:24 -0700165
Brian2ccd2642007-01-15 17:27:24 -0700166 glUseProgram_func(program);
167
Brian26f0b8f2008-04-09 22:29:33 -0600168 InitUniforms(program, Uniforms);
Brian2ccd2642007-01-15 17:27:24 -0700169
Brian0d553462007-01-15 17:48:19 -0700170 uZoom = glGetUniformLocation_func(program, "Zoom");
171 uXcenter = glGetUniformLocation_func(program, "Xcenter");
172 uYcenter = glGetUniformLocation_func(program, "Ycenter");
173
Brian2ccd2642007-01-15 17:27:24 -0700174 assert(glGetError() == 0);
175
176 glClearColor(0.4f, 0.4f, 0.8f, 0.0f);
177
178 printf("GL_RENDERER = %s\n",(const char *) glGetString(GL_RENDERER));
179
180 assert(glIsProgram_func(program));
181 assert(glIsShader_func(fragShader));
182 assert(glIsShader_func(vertShader));
183
184 glColor3f(1, 0, 0);
185}
186
187
188static void
189ParseOptions(int argc, char *argv[])
190{
191 int i;
192 for (i = 1; i < argc; i++) {
193 if (strcmp(argv[i], "-fs") == 0) {
194 FragProgFile = argv[i+1];
195 }
196 else if (strcmp(argv[i], "-vs") == 0) {
197 VertProgFile = argv[i+1];
198 }
199 }
200}
201
202
203int
204main(int argc, char *argv[])
205{
206 glutInit(&argc, argv);
207 glutInitWindowPosition( 0, 0);
208 glutInitWindowSize(400, 400);
209 glutInitDisplayMode(GLUT_RGB | GLUT_DOUBLE | GLUT_DEPTH);
210 win = glutCreateWindow(argv[0]);
211 glutReshapeFunc(Reshape);
212 glutKeyboardFunc(Key);
213 glutSpecialFunc(SpecialKey);
214 glutDisplayFunc(Redisplay);
215 ParseOptions(argc, argv);
216 Init();
217 glutMainLoop();
218 return 0;
219}
220