rev |
line source |
nuclear@5
|
1 /* Very simple OculusSDK OpenGL usage example.
|
nuclear@6
|
2 *
|
nuclear@6
|
3 * Uses SDL2 (www.libsdl.org) for event handling and OpenGL context management.
|
nuclear@6
|
4 * Uses GLEW (glew.sourceforge.net) for OpenGL extension wrangling.
|
nuclear@5
|
5 *
|
nuclear@5
|
6 * Author: John Tsiombikas <nuclear@member.fsf.org>
|
nuclear@5
|
7 * This code is in the public domain. Do whatever you like with it.
|
nuclear@5
|
8 */
|
nuclear@0
|
9 #include <stdio.h>
|
nuclear@0
|
10 #include <stdlib.h>
|
nuclear@2
|
11 #include <assert.h>
|
nuclear@0
|
12 #include <SDL2/SDL.h>
|
nuclear@0
|
13 #include <GL/glew.h>
|
nuclear@0
|
14
|
nuclear@2
|
15 #ifdef WIN32
|
nuclear@2
|
16 #define OVR_OS_WIN32
|
nuclear@10
|
17 #elif defined(__APPLE__)
|
nuclear@2
|
18 #define OVR_OS_MAC
|
nuclear@10
|
19 #else
|
nuclear@10
|
20 #define OVR_OS_LINUX
|
nuclear@14
|
21 #include <X11/Xlib.h>
|
nuclear@14
|
22 #include <GL/glx.h>
|
nuclear@2
|
23 #endif
|
nuclear@2
|
24
|
nuclear@0
|
25 #include <OVR_CAPI.h>
|
nuclear@0
|
26 #include <OVR_CAPI_GL.h>
|
nuclear@0
|
27
|
nuclear@2
|
28 int init(void);
|
nuclear@2
|
29 void cleanup(void);
|
nuclear@4
|
30 void toggle_hmd_fullscreen(void);
|
nuclear@2
|
31 void display(void);
|
nuclear@2
|
32 void draw_scene(void);
|
nuclear@2
|
33 void draw_box(float xsz, float ysz, float zsz, float norm_sign);
|
nuclear@0
|
34 void update_rtarg(int width, int height);
|
nuclear@0
|
35 int handle_event(SDL_Event *ev);
|
nuclear@0
|
36 int key_event(int key, int state);
|
nuclear@0
|
37 void reshape(int x, int y);
|
nuclear@0
|
38 unsigned int next_pow2(unsigned int x);
|
nuclear@4
|
39 void quat_to_matrix(const float *quat, float *mat);
|
nuclear@4
|
40 unsigned int gen_chess_tex(float r0, float g0, float b0, float r1, float g1, float b1);
|
nuclear@0
|
41
|
nuclear@4
|
42 /* forward declaration to avoid including non-public headers of libovr */
|
nuclear@0
|
43 OVR_EXPORT void ovrhmd_EnableHSWDisplaySDKRender(ovrHmd hmd, ovrBool enable);
|
nuclear@0
|
44
|
nuclear@0
|
45 static SDL_Window *win;
|
nuclear@0
|
46 static SDL_GLContext ctx;
|
nuclear@2
|
47 static int win_width, win_height;
|
nuclear@0
|
48
|
nuclear@0
|
49 static unsigned int fbo, fb_tex, fb_depth;
|
nuclear@0
|
50 static int fb_width, fb_height;
|
nuclear@0
|
51 static int fb_tex_width, fb_tex_height;
|
nuclear@0
|
52
|
nuclear@0
|
53 static ovrHmd hmd;
|
nuclear@0
|
54 static ovrSizei eyeres[2];
|
nuclear@2
|
55 static ovrEyeRenderDesc eye_rdesc[2];
|
nuclear@2
|
56 static ovrGLTexture fb_ovr_tex[2];
|
nuclear@15
|
57 static union ovrGLConfig glcfg;
|
nuclear@15
|
58 static unsigned int distort_caps;
|
nuclear@15
|
59 static unsigned int hmd_caps;
|
nuclear@0
|
60
|
nuclear@4
|
61 static unsigned int chess_tex;
|
nuclear@0
|
62
|
nuclear@4
|
63
|
nuclear@4
|
64 int main(int argc, char **argv)
|
nuclear@0
|
65 {
|
nuclear@0
|
66 if(init() == -1) {
|
nuclear@0
|
67 return 1;
|
nuclear@0
|
68 }
|
nuclear@0
|
69
|
nuclear@0
|
70 for(;;) {
|
nuclear@0
|
71 SDL_Event ev;
|
nuclear@0
|
72 while(SDL_PollEvent(&ev)) {
|
nuclear@0
|
73 if(handle_event(&ev) == -1) {
|
nuclear@0
|
74 goto done;
|
nuclear@0
|
75 }
|
nuclear@0
|
76 }
|
nuclear@0
|
77 display();
|
nuclear@0
|
78 }
|
nuclear@0
|
79
|
nuclear@0
|
80 done:
|
nuclear@0
|
81 cleanup();
|
nuclear@0
|
82 return 0;
|
nuclear@0
|
83 }
|
nuclear@0
|
84
|
nuclear@0
|
85
|
nuclear@2
|
86 int init(void)
|
nuclear@0
|
87 {
|
nuclear@2
|
88 int i, x, y;
|
nuclear@15
|
89 unsigned int flags;
|
nuclear@0
|
90
|
nuclear@4
|
91 /* libovr must be initialized before we create the OpenGL context */
|
nuclear@0
|
92 ovr_Initialize();
|
nuclear@0
|
93
|
nuclear@0
|
94 SDL_Init(SDL_INIT_VIDEO | SDL_INIT_TIMER);
|
nuclear@0
|
95
|
nuclear@0
|
96 x = y = SDL_WINDOWPOS_UNDEFINED;
|
nuclear@2
|
97 flags = SDL_WINDOW_OPENGL;
|
nuclear@4
|
98 if(!(win = SDL_CreateWindow("press 'f' to move to the HMD", x, y, 1280, 800, flags))) {
|
nuclear@0
|
99 fprintf(stderr, "failed to create window\n");
|
nuclear@0
|
100 return -1;
|
nuclear@0
|
101 }
|
nuclear@0
|
102 if(!(ctx = SDL_GL_CreateContext(win))) {
|
nuclear@0
|
103 fprintf(stderr, "failed to create OpenGL context\n");
|
nuclear@0
|
104 return -1;
|
nuclear@0
|
105 }
|
nuclear@0
|
106
|
nuclear@0
|
107 glewInit();
|
nuclear@0
|
108
|
nuclear@0
|
109 if(!(hmd = ovrHmd_Create(0))) {
|
nuclear@0
|
110 fprintf(stderr, "failed to open Oculus HMD, falling back to virtual debug HMD\n");
|
nuclear@2
|
111 if(!(hmd = ovrHmd_CreateDebug(ovrHmd_DK2))) {
|
nuclear@0
|
112 fprintf(stderr, "failed to create virtual debug HMD\n");
|
nuclear@0
|
113 return -1;
|
nuclear@0
|
114 }
|
nuclear@0
|
115 }
|
nuclear@0
|
116 printf("initialized HMD: %s - %s\n", hmd->Manufacturer, hmd->ProductName);
|
nuclear@0
|
117
|
nuclear@5
|
118 /* resize our window to match the HMD resolution */
|
nuclear@2
|
119 SDL_SetWindowSize(win, hmd->Resolution.w, hmd->Resolution.h);
|
nuclear@4
|
120 SDL_SetWindowPosition(win, SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED);
|
nuclear@4
|
121 win_width = hmd->Resolution.w;
|
nuclear@4
|
122 win_height = hmd->Resolution.h;
|
nuclear@0
|
123
|
nuclear@11
|
124 /* enable position and rotation tracking */
|
nuclear@11
|
125 ovrHmd_ConfigureTracking(hmd, ovrTrackingCap_Orientation | ovrTrackingCap_MagYawCorrection | ovrTrackingCap_Position, 0);
|
nuclear@5
|
126 /* retrieve the optimal render target resolution for each eye */
|
nuclear@0
|
127 eyeres[0] = ovrHmd_GetFovTextureSize(hmd, ovrEye_Left, hmd->DefaultEyeFov[0], 1.0);
|
nuclear@0
|
128 eyeres[1] = ovrHmd_GetFovTextureSize(hmd, ovrEye_Right, hmd->DefaultEyeFov[1], 1.0);
|
nuclear@0
|
129
|
nuclear@5
|
130 /* and create a single render target texture to encompass both eyes */
|
nuclear@2
|
131 fb_width = eyeres[0].w + eyeres[1].w;
|
nuclear@0
|
132 fb_height = eyeres[0].h > eyeres[1].h ? eyeres[0].h : eyeres[1].h;
|
nuclear@0
|
133 update_rtarg(fb_width, fb_height);
|
nuclear@0
|
134
|
nuclear@5
|
135 /* fill in the ovrGLTexture structures that describe our render target texture */
|
nuclear@2
|
136 for(i=0; i<2; i++) {
|
nuclear@2
|
137 fb_ovr_tex[i].OGL.Header.API = ovrRenderAPI_OpenGL;
|
nuclear@2
|
138 fb_ovr_tex[i].OGL.Header.TextureSize.w = fb_tex_width;
|
nuclear@2
|
139 fb_ovr_tex[i].OGL.Header.TextureSize.h = fb_tex_height;
|
nuclear@5
|
140 /* this next field is the only one that differs between the two eyes */
|
nuclear@3
|
141 fb_ovr_tex[i].OGL.Header.RenderViewport.Pos.x = i == 0 ? 0 : fb_width / 2.0;
|
nuclear@15
|
142 fb_ovr_tex[i].OGL.Header.RenderViewport.Pos.y = 0;
|
nuclear@2
|
143 fb_ovr_tex[i].OGL.Header.RenderViewport.Size.w = fb_width / 2.0;
|
nuclear@2
|
144 fb_ovr_tex[i].OGL.Header.RenderViewport.Size.h = fb_height;
|
nuclear@5
|
145 fb_ovr_tex[i].OGL.TexId = fb_tex; /* both eyes will use the same texture id */
|
nuclear@2
|
146 }
|
nuclear@2
|
147
|
nuclear@5
|
148 /* fill in the ovrGLConfig structure needed by the SDK to draw our stereo pair
|
nuclear@5
|
149 * to the actual HMD display (SDK-distortion mode)
|
nuclear@5
|
150 */
|
nuclear@0
|
151 memset(&glcfg, 0, sizeof glcfg);
|
nuclear@0
|
152 glcfg.OGL.Header.API = ovrRenderAPI_OpenGL;
|
nuclear@0
|
153 glcfg.OGL.Header.RTSize = hmd->Resolution;
|
nuclear@0
|
154 glcfg.OGL.Header.Multisample = 1;
|
nuclear@0
|
155
|
nuclear@11
|
156 #ifdef WIN32
|
nuclear@11
|
157 glcfg.OGL.Window = GetActiveWindow();
|
nuclear@11
|
158 glcfg.OGL.DC = wglGetCurrentDC();
|
nuclear@11
|
159 #else
|
nuclear@11
|
160 glcfg.OGL.Disp = glXGetCurrentDisplay();
|
nuclear@11
|
161 glcfg.OGL.Win = glXGetCurrentDrawable();
|
nuclear@11
|
162 #endif
|
nuclear@11
|
163
|
nuclear@0
|
164 if(hmd->HmdCaps & ovrHmdCap_ExtendDesktop) {
|
nuclear@0
|
165 printf("running in \"extended desktop\" mode\n");
|
nuclear@0
|
166 } else {
|
nuclear@5
|
167 /* to sucessfully draw to the HMD display in "direct-hmd" mode, we have to
|
nuclear@5
|
168 * call ovrHmd_AttachToWindow
|
nuclear@5
|
169 * XXX: this doesn't work properly yet due to bugs in the oculus 0.4.1 sdk/driver
|
nuclear@5
|
170 */
|
nuclear@2
|
171 #ifdef WIN32
|
nuclear@11
|
172 ovrHmd_AttachToWindow(hmd, glcfg.OGL.Window, 0, 0);
|
nuclear@10
|
173 #else
|
nuclear@11
|
174 ovrHmd_AttachToWindow(hmd, (void*)glcfg.OGL.Win, 0, 0);
|
nuclear@2
|
175 #endif
|
nuclear@0
|
176 printf("running in \"direct-hmd\" mode\n");
|
nuclear@0
|
177 }
|
nuclear@5
|
178
|
nuclear@5
|
179 /* enable low-persistence display and dynamic prediction for lattency compensation */
|
nuclear@15
|
180 hmd_caps = ovrHmdCap_LowPersistence | ovrHmdCap_DynamicPrediction;
|
nuclear@15
|
181 ovrHmd_SetEnabledCaps(hmd, hmd_caps);
|
nuclear@0
|
182
|
nuclear@5
|
183 /* configure SDK-rendering and enable chromatic abberation correction, vignetting, and
|
nuclear@5
|
184 * timewrap, which shifts the image before drawing to counter any lattency between the call
|
nuclear@5
|
185 * to ovrHmd_GetEyePose and ovrHmd_EndFrame.
|
nuclear@5
|
186 */
|
nuclear@15
|
187 distort_caps = ovrDistortionCap_Chromatic | ovrDistortionCap_Vignette | ovrDistortionCap_TimeWarp |
|
nuclear@7
|
188 ovrDistortionCap_Overdrive;
|
nuclear@15
|
189 if(!ovrHmd_ConfigureRendering(hmd, &glcfg.Config, distort_caps, hmd->DefaultEyeFov, eye_rdesc)) {
|
nuclear@0
|
190 fprintf(stderr, "failed to configure distortion renderer\n");
|
nuclear@0
|
191 }
|
nuclear@0
|
192
|
nuclear@5
|
193 /* disable the retarded "health and safety warning" */
|
nuclear@3
|
194 ovrhmd_EnableHSWDisplaySDKRender(hmd, 0);
|
nuclear@0
|
195
|
nuclear@0
|
196 glEnable(GL_DEPTH_TEST);
|
nuclear@0
|
197 glEnable(GL_CULL_FACE);
|
nuclear@0
|
198 glEnable(GL_LIGHTING);
|
nuclear@0
|
199 glEnable(GL_LIGHT0);
|
nuclear@4
|
200 glEnable(GL_LIGHT1);
|
nuclear@4
|
201 glEnable(GL_NORMALIZE);
|
nuclear@0
|
202
|
nuclear@4
|
203 glClearColor(0.1, 0.1, 0.1, 1);
|
nuclear@0
|
204
|
nuclear@4
|
205 chess_tex = gen_chess_tex(1.0, 0.7, 0.4, 0.4, 0.7, 1.0);
|
nuclear@0
|
206 return 0;
|
nuclear@0
|
207 }
|
nuclear@0
|
208
|
nuclear@2
|
209 void cleanup(void)
|
nuclear@0
|
210 {
|
nuclear@0
|
211 if(hmd) {
|
nuclear@0
|
212 ovrHmd_Destroy(hmd);
|
nuclear@0
|
213 }
|
nuclear@0
|
214 ovr_Shutdown();
|
nuclear@0
|
215
|
nuclear@0
|
216 SDL_Quit();
|
nuclear@0
|
217 }
|
nuclear@0
|
218
|
nuclear@4
|
219 void toggle_hmd_fullscreen(void)
|
nuclear@4
|
220 {
|
nuclear@4
|
221 static int fullscr, prev_x, prev_y;
|
nuclear@4
|
222 fullscr = !fullscr;
|
nuclear@4
|
223
|
nuclear@4
|
224 if(fullscr) {
|
nuclear@5
|
225 /* going fullscreen on the rift. save current window position, and move it
|
nuclear@5
|
226 * to the rift's part of the desktop before going fullscreen
|
nuclear@5
|
227 */
|
nuclear@4
|
228 SDL_GetWindowPosition(win, &prev_x, &prev_y);
|
nuclear@4
|
229 SDL_SetWindowPosition(win, hmd->WindowsPos.x, hmd->WindowsPos.y);
|
nuclear@4
|
230 SDL_SetWindowFullscreen(win, SDL_WINDOW_FULLSCREEN_DESKTOP);
|
nuclear@15
|
231
|
nuclear@15
|
232 #ifdef OVR_OS_LINUX
|
nuclear@15
|
233 /* on linux for now we have to deal with screen rotation during rendering. The docs are promoting
|
nuclear@15
|
234 * not rotating the DK2 screen globally
|
nuclear@15
|
235 */
|
nuclear@15
|
236 glcfg.OGL.Header.RTSize.w = hmd->Resolution.h;
|
nuclear@15
|
237 glcfg.OGL.Header.RTSize.h = hmd->Resolution.w;
|
nuclear@15
|
238
|
nuclear@15
|
239 distort_caps |= ovrDistortionCap_LinuxDevFullscreen;
|
nuclear@15
|
240 ovrHmd_ConfigureRendering(hmd, &glcfg.Config, distort_caps, hmd->DefaultEyeFov, eye_rdesc);
|
nuclear@15
|
241 #endif
|
nuclear@4
|
242 } else {
|
nuclear@5
|
243 /* return to windowed mode and move the window back to its original position */
|
nuclear@4
|
244 SDL_SetWindowFullscreen(win, 0);
|
nuclear@4
|
245 SDL_SetWindowPosition(win, prev_x, prev_y);
|
nuclear@15
|
246
|
nuclear@15
|
247 #ifdef OVR_OS_LINUX
|
nuclear@15
|
248 glcfg.OGL.Header.RTSize = hmd->Resolution;
|
nuclear@15
|
249
|
nuclear@15
|
250 distort_caps &= ~ovrDistortionCap_LinuxDevFullscreen;
|
nuclear@15
|
251 ovrHmd_ConfigureRendering(hmd, &glcfg.Config, distort_caps, hmd->DefaultEyeFov, eye_rdesc);
|
nuclear@15
|
252 #endif
|
nuclear@4
|
253 }
|
nuclear@4
|
254 }
|
nuclear@4
|
255
|
nuclear@2
|
256 void display(void)
|
nuclear@0
|
257 {
|
nuclear@2
|
258 int i;
|
nuclear@2
|
259 ovrMatrix4f proj;
|
nuclear@2
|
260 ovrPosef pose[2];
|
nuclear@4
|
261 float rot_mat[16];
|
nuclear@2
|
262
|
nuclear@4
|
263 /* the drawing starts with a call to ovrHmd_BeginFrame */
|
nuclear@2
|
264 ovrHmd_BeginFrame(hmd, 0);
|
nuclear@2
|
265
|
nuclear@4
|
266 /* start drawing onto our texture render target */
|
nuclear@2
|
267 glBindFramebuffer(GL_FRAMEBUFFER, fbo);
|
nuclear@0
|
268 glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
|
nuclear@0
|
269
|
nuclear@2
|
270 /* for each eye ... */
|
nuclear@2
|
271 for(i=0; i<2; i++) {
|
nuclear@7
|
272 ovrEyeType eye = hmd->EyeRenderOrder[i];
|
nuclear@2
|
273
|
nuclear@4
|
274 /* -- viewport transformation --
|
nuclear@4
|
275 * setup the viewport to draw in the left half of the framebuffer when we're
|
nuclear@4
|
276 * rendering the left eye's view (0, 0, width/2, height), and in the right half
|
nuclear@4
|
277 * of the framebuffer for the right eye's view (width/2, 0, width/2, height)
|
nuclear@4
|
278 */
|
nuclear@7
|
279 glViewport(eye == ovrEye_Left ? 0 : fb_width / 2, 0, fb_width / 2, fb_height);
|
nuclear@2
|
280
|
nuclear@4
|
281 /* -- projection transformation --
|
nuclear@4
|
282 * we'll just have to use the projection matrix supplied by the oculus SDK for this eye
|
nuclear@4
|
283 * note that libovr matrices are the transpose of what OpenGL expects, so we have to
|
nuclear@4
|
284 * use glLoadTransposeMatrixf instead of glLoadMatrixf to load it.
|
nuclear@4
|
285 */
|
nuclear@2
|
286 proj = ovrMatrix4f_Projection(hmd->DefaultEyeFov[eye], 0.5, 500.0, 1);
|
nuclear@2
|
287 glMatrixMode(GL_PROJECTION);
|
nuclear@4
|
288 glLoadTransposeMatrixf(proj.M[0]);
|
nuclear@2
|
289
|
nuclear@4
|
290 /* -- view/camera transformation --
|
nuclear@4
|
291 * we need to construct a view matrix by combining all the information provided by the oculus
|
nuclear@4
|
292 * SDK, about the position and orientation of the user's head in the world.
|
nuclear@4
|
293 */
|
nuclear@8
|
294 /* TODO: use ovrHmd_GetEyePoses out of the loop instead */
|
nuclear@8
|
295 pose[eye] = ovrHmd_GetHmdPosePerEye(hmd, eye);
|
nuclear@2
|
296 glMatrixMode(GL_MODELVIEW);
|
nuclear@4
|
297 glLoadIdentity();
|
nuclear@8
|
298 glTranslatef(eye_rdesc[eye].HmdToEyeViewOffset.x,
|
nuclear@8
|
299 eye_rdesc[eye].HmdToEyeViewOffset.y,
|
nuclear@8
|
300 eye_rdesc[eye].HmdToEyeViewOffset.z);
|
nuclear@4
|
301 /* retrieve the orientation quaternion and convert it to a rotation matrix */
|
nuclear@4
|
302 quat_to_matrix(&pose[eye].Orientation.x, rot_mat);
|
nuclear@4
|
303 glMultMatrixf(rot_mat);
|
nuclear@4
|
304 /* translate the view matrix with the positional tracking */
|
nuclear@4
|
305 glTranslatef(-pose[eye].Position.x, -pose[eye].Position.y, -pose[eye].Position.z);
|
nuclear@4
|
306 /* move the camera to the eye level of the user */
|
nuclear@4
|
307 glTranslatef(0, -ovrHmd_GetFloat(hmd, OVR_KEY_EYE_HEIGHT, 1.65), 0);
|
nuclear@2
|
308
|
nuclear@4
|
309 /* finally draw the scene for this eye */
|
nuclear@2
|
310 draw_scene();
|
nuclear@2
|
311 }
|
nuclear@2
|
312
|
nuclear@4
|
313 /* after drawing both eyes into the texture render target, revert to drawing directly to the
|
nuclear@4
|
314 * display, and we call ovrHmd_EndFrame, to let the Oculus SDK draw both images properly
|
nuclear@4
|
315 * compensated for lens distortion and chromatic abberation onto the HMD screen.
|
nuclear@4
|
316 */
|
nuclear@2
|
317 glBindFramebuffer(GL_FRAMEBUFFER, 0);
|
nuclear@2
|
318
|
nuclear@2
|
319 ovrHmd_EndFrame(hmd, pose, &fb_ovr_tex[0].Texture);
|
nuclear@2
|
320
|
nuclear@17
|
321 /* workaround for the oculus sdk distortion renderer bug, which uses a shader
|
nuclear@17
|
322 * program, and doesn't restore the original binding when it's done.
|
nuclear@17
|
323 */
|
nuclear@17
|
324 glUseProgram(0);
|
nuclear@17
|
325
|
nuclear@2
|
326 assert(glGetError() == GL_NO_ERROR);
|
nuclear@2
|
327 }
|
nuclear@2
|
328
|
nuclear@15
|
329 void reshape(int x, int y)
|
nuclear@15
|
330 {
|
nuclear@15
|
331 win_width = x;
|
nuclear@15
|
332 win_height = y;
|
nuclear@15
|
333 }
|
nuclear@15
|
334
|
nuclear@2
|
335 void draw_scene(void)
|
nuclear@2
|
336 {
|
nuclear@2
|
337 int i;
|
nuclear@4
|
338 float grey[] = {0.8, 0.8, 0.8, 1};
|
nuclear@4
|
339 float col[] = {0, 0, 0, 1};
|
nuclear@4
|
340 float lpos[][4] = {
|
nuclear@4
|
341 {-8, 2, 10, 1},
|
nuclear@4
|
342 {0, 15, 0, 1}
|
nuclear@4
|
343 };
|
nuclear@4
|
344 float lcol[][4] = {
|
nuclear@4
|
345 {0.8, 0.8, 0.8, 1},
|
nuclear@4
|
346 {0.4, 0.3, 0.3, 1}
|
nuclear@4
|
347 };
|
nuclear@4
|
348
|
nuclear@4
|
349 for(i=0; i<2; i++) {
|
nuclear@4
|
350 glLightfv(GL_LIGHT0 + i, GL_POSITION, lpos[i]);
|
nuclear@4
|
351 glLightfv(GL_LIGHT0 + i, GL_DIFFUSE, lcol[i]);
|
nuclear@4
|
352 }
|
nuclear@4
|
353
|
nuclear@4
|
354 glMatrixMode(GL_MODELVIEW);
|
nuclear@2
|
355
|
nuclear@3
|
356 glPushMatrix();
|
nuclear@4
|
357 glTranslatef(0, 10, 0);
|
nuclear@4
|
358 glMaterialfv(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, grey);
|
nuclear@4
|
359 glBindTexture(GL_TEXTURE_2D, chess_tex);
|
nuclear@4
|
360 glEnable(GL_TEXTURE_2D);
|
nuclear@4
|
361 draw_box(30, 20, 30, -1.0);
|
nuclear@4
|
362 glDisable(GL_TEXTURE_2D);
|
nuclear@3
|
363 glPopMatrix();
|
nuclear@3
|
364
|
nuclear@4
|
365 for(i=0; i<4; i++) {
|
nuclear@4
|
366 glMaterialfv(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, grey);
|
nuclear@4
|
367 glPushMatrix();
|
nuclear@4
|
368 glTranslatef(i & 1 ? 5 : -5, 1, i & 2 ? -5 : 5);
|
nuclear@4
|
369 draw_box(0.5, 2, 0.5, 1.0);
|
nuclear@4
|
370 glPopMatrix();
|
nuclear@2
|
371
|
nuclear@4
|
372 col[0] = i & 1 ? 1.0 : 0.3;
|
nuclear@4
|
373 col[1] = i == 0 ? 1.0 : 0.3;
|
nuclear@4
|
374 col[2] = i & 2 ? 1.0 : 0.3;
|
nuclear@4
|
375 glMaterialfv(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, col);
|
nuclear@4
|
376
|
nuclear@2
|
377 glPushMatrix();
|
nuclear@2
|
378 if(i & 1) {
|
nuclear@4
|
379 glTranslatef(0, 0.25, i & 2 ? 2 : -2);
|
nuclear@2
|
380 } else {
|
nuclear@4
|
381 glTranslatef(i & 2 ? 2 : -2, 0.25, 0);
|
nuclear@2
|
382 }
|
nuclear@4
|
383 draw_box(0.5, 0.5, 0.5, 1.0);
|
nuclear@2
|
384 glPopMatrix();
|
nuclear@2
|
385 }
|
nuclear@12
|
386
|
nuclear@12
|
387 col[0] = 1;
|
nuclear@12
|
388 col[1] = 1;
|
nuclear@12
|
389 col[2] = 0.4;
|
nuclear@12
|
390 glMaterialfv(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE, col);
|
nuclear@12
|
391 draw_box(0.05, 1.2, 6, 1.0);
|
nuclear@12
|
392 draw_box(6, 1.2, 0.05, 1.0);
|
nuclear@2
|
393 }
|
nuclear@2
|
394
|
nuclear@2
|
395 void draw_box(float xsz, float ysz, float zsz, float norm_sign)
|
nuclear@2
|
396 {
|
nuclear@2
|
397 glMatrixMode(GL_MODELVIEW);
|
nuclear@12
|
398 glPushMatrix();
|
nuclear@2
|
399 glScalef(xsz * 0.5, ysz * 0.5, zsz * 0.5);
|
nuclear@2
|
400
|
nuclear@2
|
401 if(norm_sign < 0.0) {
|
nuclear@2
|
402 glFrontFace(GL_CW);
|
nuclear@2
|
403 }
|
nuclear@2
|
404
|
nuclear@2
|
405 glBegin(GL_QUADS);
|
nuclear@2
|
406 glNormal3f(0, 0, 1 * norm_sign);
|
nuclear@2
|
407 glTexCoord2f(0, 0); glVertex3f(-1, -1, 1);
|
nuclear@2
|
408 glTexCoord2f(1, 0); glVertex3f(1, -1, 1);
|
nuclear@2
|
409 glTexCoord2f(1, 1); glVertex3f(1, 1, 1);
|
nuclear@2
|
410 glTexCoord2f(0, 1); glVertex3f(-1, 1, 1);
|
nuclear@2
|
411 glNormal3f(1 * norm_sign, 0, 0);
|
nuclear@2
|
412 glTexCoord2f(0, 0); glVertex3f(1, -1, 1);
|
nuclear@2
|
413 glTexCoord2f(1, 0); glVertex3f(1, -1, -1);
|
nuclear@2
|
414 glTexCoord2f(1, 1); glVertex3f(1, 1, -1);
|
nuclear@2
|
415 glTexCoord2f(0, 1); glVertex3f(1, 1, 1);
|
nuclear@2
|
416 glNormal3f(0, 0, -1 * norm_sign);
|
nuclear@2
|
417 glTexCoord2f(0, 0); glVertex3f(1, -1, -1);
|
nuclear@2
|
418 glTexCoord2f(1, 0); glVertex3f(-1, -1, -1);
|
nuclear@2
|
419 glTexCoord2f(1, 1); glVertex3f(-1, 1, -1);
|
nuclear@2
|
420 glTexCoord2f(0, 1); glVertex3f(1, 1, -1);
|
nuclear@2
|
421 glNormal3f(-1 * norm_sign, 0, 0);
|
nuclear@2
|
422 glTexCoord2f(0, 0); glVertex3f(-1, -1, -1);
|
nuclear@2
|
423 glTexCoord2f(1, 0); glVertex3f(-1, -1, 1);
|
nuclear@2
|
424 glTexCoord2f(1, 1); glVertex3f(-1, 1, 1);
|
nuclear@2
|
425 glTexCoord2f(0, 1); glVertex3f(-1, 1, -1);
|
nuclear@4
|
426 glEnd();
|
nuclear@4
|
427 glBegin(GL_TRIANGLE_FAN);
|
nuclear@2
|
428 glNormal3f(0, 1 * norm_sign, 0);
|
nuclear@4
|
429 glTexCoord2f(0.5, 0.5); glVertex3f(0, 1, 0);
|
nuclear@2
|
430 glTexCoord2f(0, 0); glVertex3f(-1, 1, 1);
|
nuclear@2
|
431 glTexCoord2f(1, 0); glVertex3f(1, 1, 1);
|
nuclear@2
|
432 glTexCoord2f(1, 1); glVertex3f(1, 1, -1);
|
nuclear@2
|
433 glTexCoord2f(0, 1); glVertex3f(-1, 1, -1);
|
nuclear@4
|
434 glTexCoord2f(0, 0); glVertex3f(-1, 1, 1);
|
nuclear@4
|
435 glEnd();
|
nuclear@4
|
436 glBegin(GL_TRIANGLE_FAN);
|
nuclear@2
|
437 glNormal3f(0, -1 * norm_sign, 0);
|
nuclear@4
|
438 glTexCoord2f(0.5, 0.5); glVertex3f(0, -1, 0);
|
nuclear@2
|
439 glTexCoord2f(0, 0); glVertex3f(-1, -1, -1);
|
nuclear@2
|
440 glTexCoord2f(1, 0); glVertex3f(1, -1, -1);
|
nuclear@2
|
441 glTexCoord2f(1, 1); glVertex3f(1, -1, 1);
|
nuclear@2
|
442 glTexCoord2f(0, 1); glVertex3f(-1, -1, 1);
|
nuclear@4
|
443 glTexCoord2f(0, 0); glVertex3f(-1, -1, -1);
|
nuclear@2
|
444 glEnd();
|
nuclear@2
|
445
|
nuclear@2
|
446 glFrontFace(GL_CCW);
|
nuclear@12
|
447 glPopMatrix();
|
nuclear@0
|
448 }
|
nuclear@0
|
449
|
nuclear@5
|
450 /* update_rtarg creates (and/or resizes) the render target used to draw the two stero views */
|
nuclear@0
|
451 void update_rtarg(int width, int height)
|
nuclear@0
|
452 {
|
nuclear@0
|
453 if(!fbo) {
|
nuclear@5
|
454 /* if fbo does not exist, then nothing does... create every opengl object */
|
nuclear@0
|
455 glGenFramebuffers(1, &fbo);
|
nuclear@0
|
456 glGenTextures(1, &fb_tex);
|
nuclear@0
|
457 glGenRenderbuffers(1, &fb_depth);
|
nuclear@0
|
458
|
nuclear@0
|
459 glBindTexture(GL_TEXTURE_2D, fb_tex);
|
nuclear@0
|
460 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
nuclear@0
|
461 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
nuclear@0
|
462 }
|
nuclear@0
|
463
|
nuclear@0
|
464 glBindFramebuffer(GL_FRAMEBUFFER, fbo);
|
nuclear@0
|
465
|
nuclear@5
|
466 /* calculate the next power of two in both dimensions and use that as a texture size */
|
nuclear@0
|
467 fb_tex_width = next_pow2(width);
|
nuclear@0
|
468 fb_tex_height = next_pow2(height);
|
nuclear@0
|
469
|
nuclear@5
|
470 /* create and attach the texture that will be used as a color buffer */
|
nuclear@0
|
471 glBindTexture(GL_TEXTURE_2D, fb_tex);
|
nuclear@0
|
472 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, fb_tex_width, fb_tex_height, 0,
|
nuclear@0
|
473 GL_RGBA, GL_UNSIGNED_BYTE, 0);
|
nuclear@0
|
474 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, fb_tex, 0);
|
nuclear@0
|
475
|
nuclear@5
|
476 /* create and attach the renderbuffer that will serve as our z-buffer */
|
nuclear@0
|
477 glBindRenderbuffer(GL_RENDERBUFFER, fb_depth);
|
nuclear@0
|
478 glRenderbufferStorage(GL_RENDERBUFFER, GL_DEPTH_COMPONENT, fb_tex_width, fb_tex_height);
|
nuclear@0
|
479 glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_RENDERBUFFER, fb_depth);
|
nuclear@0
|
480
|
nuclear@0
|
481 if(glCheckFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE) {
|
nuclear@0
|
482 fprintf(stderr, "incomplete framebuffer!\n");
|
nuclear@0
|
483 }
|
nuclear@0
|
484
|
nuclear@0
|
485 glBindFramebuffer(GL_FRAMEBUFFER, 0);
|
nuclear@0
|
486 printf("created render target: %dx%d (texture size: %dx%d)\n", width, height, fb_tex_width, fb_tex_height);
|
nuclear@0
|
487 }
|
nuclear@0
|
488
|
nuclear@0
|
489 int handle_event(SDL_Event *ev)
|
nuclear@0
|
490 {
|
nuclear@0
|
491 switch(ev->type) {
|
nuclear@4
|
492 case SDL_QUIT:
|
nuclear@4
|
493 return -1;
|
nuclear@4
|
494
|
nuclear@0
|
495 case SDL_KEYDOWN:
|
nuclear@0
|
496 case SDL_KEYUP:
|
nuclear@0
|
497 if(key_event(ev->key.keysym.sym, ev->key.state == SDL_PRESSED) == -1) {
|
nuclear@0
|
498 return -1;
|
nuclear@0
|
499 }
|
nuclear@0
|
500 break;
|
nuclear@0
|
501
|
nuclear@15
|
502 case SDL_WINDOWEVENT:
|
nuclear@15
|
503 if(ev->window.event == SDL_WINDOWEVENT_RESIZED) {
|
nuclear@15
|
504 reshape(ev->window.data1, ev->window.data2);
|
nuclear@15
|
505 }
|
nuclear@15
|
506 break;
|
nuclear@15
|
507
|
nuclear@0
|
508 default:
|
nuclear@0
|
509 break;
|
nuclear@0
|
510 }
|
nuclear@0
|
511
|
nuclear@0
|
512 return 0;
|
nuclear@0
|
513 }
|
nuclear@0
|
514
|
nuclear@0
|
515 int key_event(int key, int state)
|
nuclear@0
|
516 {
|
nuclear@0
|
517 if(state) {
|
nuclear@5
|
518 /*
|
nuclear@4
|
519 ovrHSWDisplayState hsw;
|
nuclear@4
|
520 ovrHmd_GetHSWDisplayState(hmd, &hsw);
|
nuclear@4
|
521 if(hsw.Displayed) {
|
nuclear@4
|
522 ovrHmd_DismissHSWDisplay(hmd);
|
nuclear@4
|
523 }
|
nuclear@5
|
524 */
|
nuclear@4
|
525
|
nuclear@0
|
526 switch(key) {
|
nuclear@0
|
527 case 27:
|
nuclear@0
|
528 return -1;
|
nuclear@0
|
529
|
nuclear@4
|
530 case ' ':
|
nuclear@15
|
531 case 'r':
|
nuclear@4
|
532 /* allow the user to recenter by pressing space */
|
nuclear@4
|
533 ovrHmd_RecenterPose(hmd);
|
nuclear@4
|
534 break;
|
nuclear@4
|
535
|
nuclear@4
|
536 case 'f':
|
nuclear@4
|
537 /* press f to move the window to the HMD */
|
nuclear@4
|
538 toggle_hmd_fullscreen();
|
nuclear@4
|
539 break;
|
nuclear@4
|
540
|
nuclear@15
|
541 case 'v':
|
nuclear@15
|
542 distort_caps ^= ovrDistortionCap_Vignette;
|
nuclear@15
|
543 printf("Vignette: %s\n", distort_caps & ovrDistortionCap_Vignette ? "on" : "off");
|
nuclear@15
|
544 ovrHmd_ConfigureRendering(hmd, &glcfg.Config, distort_caps, hmd->DefaultEyeFov, eye_rdesc);
|
nuclear@15
|
545 break;
|
nuclear@15
|
546
|
nuclear@15
|
547 case 't':
|
nuclear@15
|
548 distort_caps ^= ovrDistortionCap_TimeWarp;
|
nuclear@15
|
549 printf("Time-warp: %s\n", distort_caps & ovrDistortionCap_TimeWarp ? "on" : "off");
|
nuclear@15
|
550 ovrHmd_ConfigureRendering(hmd, &glcfg.Config, distort_caps, hmd->DefaultEyeFov, eye_rdesc);
|
nuclear@15
|
551 break;
|
nuclear@15
|
552
|
nuclear@15
|
553 case 'o':
|
nuclear@15
|
554 distort_caps ^= ovrDistortionCap_Overdrive;
|
nuclear@15
|
555 printf("OLED over-drive: %s\n", distort_caps & ovrDistortionCap_Overdrive ? "on" : "off");
|
nuclear@15
|
556 ovrHmd_ConfigureRendering(hmd, &glcfg.Config, distort_caps, hmd->DefaultEyeFov, eye_rdesc);
|
nuclear@15
|
557 break;
|
nuclear@15
|
558
|
nuclear@15
|
559 case 'l':
|
nuclear@15
|
560 hmd_caps ^= ovrHmdCap_LowPersistence;
|
nuclear@15
|
561 printf("Low-persistence display: %s\n", hmd_caps & ovrHmdCap_LowPersistence ? "on" : "off");
|
nuclear@15
|
562 ovrHmd_SetEnabledCaps(hmd, hmd_caps);
|
nuclear@15
|
563 break;
|
nuclear@15
|
564
|
nuclear@0
|
565 default:
|
nuclear@0
|
566 break;
|
nuclear@0
|
567 }
|
nuclear@0
|
568 }
|
nuclear@0
|
569 return 0;
|
nuclear@0
|
570 }
|
nuclear@4
|
571
|
nuclear@4
|
572 unsigned int next_pow2(unsigned int x)
|
nuclear@4
|
573 {
|
nuclear@4
|
574 x -= 1;
|
nuclear@4
|
575 x |= x >> 1;
|
nuclear@4
|
576 x |= x >> 2;
|
nuclear@4
|
577 x |= x >> 4;
|
nuclear@4
|
578 x |= x >> 8;
|
nuclear@4
|
579 x |= x >> 16;
|
nuclear@4
|
580 return x + 1;
|
nuclear@4
|
581 }
|
nuclear@4
|
582
|
nuclear@5
|
583 /* convert a quaternion to a rotation matrix */
|
nuclear@4
|
584 void quat_to_matrix(const float *quat, float *mat)
|
nuclear@4
|
585 {
|
nuclear@4
|
586 mat[0] = 1.0 - 2.0 * quat[1] * quat[1] - 2.0 * quat[2] * quat[2];
|
nuclear@4
|
587 mat[4] = 2.0 * quat[0] * quat[1] + 2.0 * quat[3] * quat[2];
|
nuclear@4
|
588 mat[8] = 2.0 * quat[2] * quat[0] - 2.0 * quat[3] * quat[1];
|
nuclear@4
|
589 mat[12] = 0.0f;
|
nuclear@4
|
590
|
nuclear@4
|
591 mat[1] = 2.0 * quat[0] * quat[1] - 2.0 * quat[3] * quat[2];
|
nuclear@4
|
592 mat[5] = 1.0 - 2.0 * quat[0]*quat[0] - 2.0 * quat[2]*quat[2];
|
nuclear@4
|
593 mat[9] = 2.0 * quat[1] * quat[2] + 2.0 * quat[3] * quat[0];
|
nuclear@4
|
594 mat[13] = 0.0f;
|
nuclear@4
|
595
|
nuclear@4
|
596 mat[2] = 2.0 * quat[2] * quat[0] + 2.0 * quat[3] * quat[1];
|
nuclear@4
|
597 mat[6] = 2.0 * quat[1] * quat[2] - 2.0 * quat[3] * quat[0];
|
nuclear@4
|
598 mat[10] = 1.0 - 2.0 * quat[0]*quat[0] - 2.0 * quat[1]*quat[1];
|
nuclear@4
|
599 mat[14] = 0.0f;
|
nuclear@4
|
600
|
nuclear@4
|
601 mat[3] = mat[7] = mat[11] = 0.0f;
|
nuclear@4
|
602 mat[15] = 1.0f;
|
nuclear@4
|
603 }
|
nuclear@4
|
604
|
nuclear@5
|
605 /* generate a chessboard texture with tiles colored (r0, g0, b0) and (r1, g1, b1) */
|
nuclear@4
|
606 unsigned int gen_chess_tex(float r0, float g0, float b0, float r1, float g1, float b1)
|
nuclear@4
|
607 {
|
nuclear@4
|
608 int i, j;
|
nuclear@4
|
609 unsigned int tex;
|
nuclear@4
|
610 unsigned char img[8 * 8 * 3];
|
nuclear@4
|
611 unsigned char *pix = img;
|
nuclear@4
|
612
|
nuclear@4
|
613 for(i=0; i<8; i++) {
|
nuclear@4
|
614 for(j=0; j<8; j++) {
|
nuclear@4
|
615 int black = (i & 1) == (j & 1);
|
nuclear@4
|
616 pix[0] = (black ? r0 : r1) * 255;
|
nuclear@4
|
617 pix[1] = (black ? g0 : g1) * 255;
|
nuclear@4
|
618 pix[2] = (black ? b0 : b1) * 255;
|
nuclear@4
|
619 pix += 3;
|
nuclear@4
|
620 }
|
nuclear@4
|
621 }
|
nuclear@4
|
622
|
nuclear@4
|
623 glGenTextures(1, &tex);
|
nuclear@4
|
624 glBindTexture(GL_TEXTURE_2D, tex);
|
nuclear@4
|
625 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
nuclear@4
|
626 glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
nuclear@4
|
627 glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, 8, 8, 0, GL_RGB, GL_UNSIGNED_BYTE, img);
|
nuclear@4
|
628
|
nuclear@4
|
629 return tex;
|
nuclear@8
|
630 }
|