blob: 6149e12db259d0bd4d2d2697817fce38db75bb93 [file] [log] [blame]
Kristian Høgsbergffd710e2008-12-02 15:15:01 -05001/*
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -05002 * Copyright © 2008 Kristian Høgsberg
3 *
4 * Permission to use, copy, modify, distribute, and sell this software and its
5 * documentation for any purpose is hereby granted without fee, provided that
6 * the above copyright notice appear in all copies and that both that copyright
7 * notice and this permission notice appear in supporting documentation, and
8 * that the name of the copyright holders not be used in advertising or
9 * publicity pertaining to distribution of the software without specific,
10 * written prior permission. The copyright holders make no representations
11 * about the suitability of this software for any purpose. It is provided "as
12 * is" without express or implied warranty.
13 *
14 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
15 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
16 * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
17 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
18 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
19 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
20 * OF THIS SOFTWARE.
Kristian Høgsbergffd710e2008-12-02 15:15:01 -050021 */
22
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -050023#include <stdint.h>
24#include <stdio.h>
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -050025#include <stdlib.h>
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -050026#include <string.h>
27#include <fcntl.h>
28#include <unistd.h>
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -050029#include <math.h>
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -050030#include <time.h>
31#include <cairo.h>
32#include <glib.h>
33
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -050034#include <GL/gl.h>
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -050035#include <eagle.h>
36
37#include "wayland-client.h"
38#include "wayland-glib.h"
39
40#include "cairo-util.h"
41#include "window.h"
42
43static const char gem_device[] = "/dev/dri/card0";
44static const char socket_name[] = "\0wayland";
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -050045
46struct gears {
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -050047 struct window *window;
48
49 struct wl_display *wl_display;
50 struct rectangle rectangle;
51
52 EGLDisplay display;
53 EGLConfig config;
54 EGLSurface surface;
55 EGLContext context;
56 int resized;
57 int fd;
58 GLfloat angle;
59 struct buffer *buffer;
60
61 GLint gear_list[3];
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -050062};
63
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -050064struct gear_template {
65 GLfloat material[4];
66 GLfloat inner_radius;
67 GLfloat outer_radius;
68 GLfloat width;
69 GLint teeth;
70 GLfloat tooth_depth;
71};
72
73const static struct gear_template gear_templates[] = {
74 { { 0.8, 0.1, 0.0, 1.0 }, 1.0, 4.0, 1.0, 20, 0.7 },
75 { { 0.0, 0.8, 0.2, 1.0 }, 0.5, 2.0, 2.0, 10, 0.7 },
76 { { 0.2, 0.2, 1.0, 1.0 }, 1.3, 2.0, 0.5, 10, 0.7 },
77};
78
79static GLfloat light_pos[4] = {5.0, 5.0, 10.0, 0.0};
80
81static void die(const char *msg)
82{
83 fprintf(stderr, "%s", msg);
84 exit(EXIT_FAILURE);
85}
86
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -050087static void
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -050088make_gear(const struct gear_template *t)
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -050089{
90 GLint i;
91 GLfloat r0, r1, r2;
92 GLfloat angle, da;
93 GLfloat u, v, len;
94
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -050095 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, t->material);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -050096
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -050097 r0 = t->inner_radius;
98 r1 = t->outer_radius - t->tooth_depth / 2.0;
99 r2 = t->outer_radius + t->tooth_depth / 2.0;
100
101 da = 2.0 * M_PI / t->teeth / 4.0;
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500102
103 glShadeModel(GL_FLAT);
104
105 glNormal3f(0.0, 0.0, 1.0);
106
107 /* draw front face */
108 glBegin(GL_QUAD_STRIP);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500109 for (i = 0; i <= t->teeth; i++) {
110 angle = i * 2.0 * M_PI / t->teeth;
111 glVertex3f(r0 * cos(angle), r0 * sin(angle), t->width * 0.5);
112 glVertex3f(r1 * cos(angle), r1 * sin(angle), t->width * 0.5);
113 if (i < t->teeth) {
114 glVertex3f(r0 * cos(angle), r0 * sin(angle), t->width * 0.5);
115 glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500116 }
117 }
118 glEnd();
119
120 /* draw front sides of teeth */
121 glBegin(GL_QUADS);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500122 da = 2.0 * M_PI / t->teeth / 4.0;
123 for (i = 0; i < t->teeth; i++) {
124 angle = i * 2.0 * M_PI / t->teeth;
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500125
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500126 glVertex3f(r1 * cos(angle), r1 * sin(angle), t->width * 0.5);
127 glVertex3f(r2 * cos(angle + da), r2 * sin(angle + da), t->width * 0.5);
128 glVertex3f(r2 * cos(angle + 2 * da), r2 * sin(angle + 2 * da), t->width * 0.5);
129 glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500130 }
131 glEnd();
132
133 glNormal3f(0.0, 0.0, -1.0);
134
135 /* draw back face */
136 glBegin(GL_QUAD_STRIP);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500137 for (i = 0; i <= t->teeth; i++) {
138 angle = i * 2.0 * M_PI / t->teeth;
139 glVertex3f(r1 * cos(angle), r1 * sin(angle), -t->width * 0.5);
140 glVertex3f(r0 * cos(angle), r0 * sin(angle), -t->width * 0.5);
141 if (i < t->teeth) {
142 glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), -t->width * 0.5);
143 glVertex3f(r0 * cos(angle), r0 * sin(angle), -t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500144 }
145 }
146 glEnd();
147
148 /* draw back sides of teeth */
149 glBegin(GL_QUADS);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500150 da = 2.0 * M_PI / t->teeth / 4.0;
151 for (i = 0; i < t->teeth; i++) {
152 angle = i * 2.0 * M_PI / t->teeth;
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500153
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500154 glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), -t->width * 0.5);
155 glVertex3f(r2 * cos(angle + 2 * da), r2 * sin(angle + 2 * da), -t->width * 0.5);
156 glVertex3f(r2 * cos(angle + da), r2 * sin(angle + da), -t->width * 0.5);
157 glVertex3f(r1 * cos(angle), r1 * sin(angle), -t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500158 }
159 glEnd();
160
161 /* draw outward faces of teeth */
162 glBegin(GL_QUAD_STRIP);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500163 for (i = 0; i < t->teeth; i++) {
164 angle = i * 2.0 * M_PI / t->teeth;
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500165
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500166 glVertex3f(r1 * cos(angle), r1 * sin(angle), t->width * 0.5);
167 glVertex3f(r1 * cos(angle), r1 * sin(angle), -t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500168 u = r2 * cos(angle + da) - r1 * cos(angle);
169 v = r2 * sin(angle + da) - r1 * sin(angle);
170 len = sqrt(u * u + v * v);
171 u /= len;
172 v /= len;
173 glNormal3f(v, -u, 0.0);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500174 glVertex3f(r2 * cos(angle + da), r2 * sin(angle + da), t->width * 0.5);
175 glVertex3f(r2 * cos(angle + da), r2 * sin(angle + da), -t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500176 glNormal3f(cos(angle), sin(angle), 0.0);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500177 glVertex3f(r2 * cos(angle + 2 * da), r2 * sin(angle + 2 * da), t->width * 0.5);
178 glVertex3f(r2 * cos(angle + 2 * da), r2 * sin(angle + 2 * da), -t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500179 u = r1 * cos(angle + 3 * da) - r2 * cos(angle + 2 * da);
180 v = r1 * sin(angle + 3 * da) - r2 * sin(angle + 2 * da);
181 glNormal3f(v, -u, 0.0);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500182 glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), t->width * 0.5);
183 glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), -t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500184 glNormal3f(cos(angle), sin(angle), 0.0);
185 }
186
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500187 glVertex3f(r1 * cos(0), r1 * sin(0), t->width * 0.5);
188 glVertex3f(r1 * cos(0), r1 * sin(0), -t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500189
190 glEnd();
191
192 glShadeModel(GL_SMOOTH);
193
194 /* draw inside radius cylinder */
195 glBegin(GL_QUAD_STRIP);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500196 for (i = 0; i <= t->teeth; i++) {
197 angle = i * 2.0 * M_PI / t->teeth;
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500198 glNormal3f(-cos(angle), -sin(angle), 0.0);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500199 glVertex3f(r0 * cos(angle), r0 * sin(angle), -t->width * 0.5);
200 glVertex3f(r0 * cos(angle), r0 * sin(angle), t->width * 0.5);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500201 }
202 glEnd();
203}
204
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500205static void
206draw_gears(struct gears *gears)
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500207{
208 GLfloat view_rotx = 20.0, view_roty = 30.0, view_rotz = 0.0;
209
Kristian Høgsberg78231c82008-11-08 15:06:01 -0500210 glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
Kristian Høgsbergb8bf19b2008-11-05 07:38:46 -0500211
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500212 glPushMatrix();
213
214 glTranslatef(0.0, 0.0, -50);
215
216 glRotatef(view_rotx, 1.0, 0.0, 0.0);
217 glRotatef(view_roty, 0.0, 1.0, 0.0);
218 glRotatef(view_rotz, 0.0, 0.0, 1.0);
219
220 glPushMatrix();
221 glTranslatef(-3.0, -2.0, 0.0);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500222 glRotatef(gears->angle, 0.0, 0.0, 1.0);
223 glCallList(gears->gear_list[0]);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500224 glPopMatrix();
225
226 glPushMatrix();
227 glTranslatef(3.1, -2.0, 0.0);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500228 glRotatef(-2.0 * gears->angle - 9.0, 0.0, 0.0, 1.0);
229 glCallList(gears->gear_list[1]);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500230 glPopMatrix();
231
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500232 glPushMatrix();
233 glTranslatef(-3.1, 4.2, 0.0);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500234 glRotatef(-2.0 * gears->angle - 25.0, 0.0, 0.0, 1.0);
235 glCallList(gears->gear_list[2]);
Kristian Høgsberg8a9cda82008-11-03 15:31:30 -0500236 glPopMatrix();
237
238 glPopMatrix();
239
240 glFlush();
241}
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500242
243static void
244resize_window(struct gears *gears)
245{
Kristian Høgsberg22106762008-12-08 13:50:07 -0500246 /* Constrain child size to be square and at least 300x300 */
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500247 window_get_child_rectangle(gears->window, &gears->rectangle);
Kristian Høgsberg22106762008-12-08 13:50:07 -0500248 if (gears->rectangle.width > gears->rectangle.height)
249 gears->rectangle.height = gears->rectangle.width;
250 else
251 gears->rectangle.width = gears->rectangle.height;
252 if (gears->rectangle.width < 300) {
253 gears->rectangle.width = 300;
254 gears->rectangle.height = 300;
255 }
256 window_set_child_size(gears->window, &gears->rectangle);
257
258 window_draw(gears->window);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500259
260 if (gears->buffer != NULL)
261 buffer_destroy(gears->buffer, gears->fd);
262
263 gears->buffer = buffer_create(gears->fd,
264 gears->rectangle.width,
265 gears->rectangle.height,
266 (gears->rectangle.width * 4 + 15) & ~15);
267
268 gears->surface = eglCreateSurfaceForName(gears->display,
269 gears->config,
270 gears->buffer->name,
271 gears->buffer->width,
272 gears->buffer->height,
273 gears->buffer->stride, NULL);
274
275 eglMakeCurrent(gears->display,
276 gears->surface, gears->surface, gears->context);
277
278 glViewport(0, 0, gears->rectangle.width, gears->rectangle.height);
279 gears->resized = 0;
280}
281
282static void
Kristian Høgsberg22106762008-12-08 13:50:07 -0500283resize_handler(struct window *window, void *data)
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500284{
285 struct gears *gears = data;
286
287 /* Right now, resizing the window from the per-frame callback
288 * is fine, since the window drawing code is so slow that we
289 * can't draw more than one window per frame anyway. However,
290 * once we implement faster resizing, this will show lag
291 * between pointer motion and window size even if resizing is
292 * fast. We need to keep processing motion events and posting
293 * new frames as fast as possible so when the server
294 * composites the next frame it will have the most recent size
295 * possible, like what we do for window moves. */
296
297 gears->resized = 1;
298}
299
300static void
301acknowledge_handler(struct window *window, uint32_t key, void *data)
302{
303 struct gears *gears = data;
304
305 if (key != 0)
306 return;
307
308 if (gears->resized)
309 resize_window(gears);
310
311 draw_gears(gears);
312}
313
314static void
315frame_handler(struct window *window, uint32_t frame, uint32_t timestamp, void *data)
316{
317 struct gears *gears = data;
318
319 window_copy(gears->window,
320 &gears->rectangle,
321 gears->buffer->name, gears->buffer->stride);
322
323 wl_display_commit(gears->wl_display, 0);
324
325 gears->angle += 1;
326}
327
328static struct gears *
329gears_create(struct wl_display *display, int fd)
330{
331 const int x = 200, y = 200, width = 450, height = 500;
332 EGLint major, minor, count;
333 EGLConfig configs[64];
334 struct gears *gears;
335 int i;
336
337 gears = malloc(sizeof *gears);
338 memset(gears, 0, sizeof *gears);
339 gears->wl_display = display;
340 gears->fd = fd;
341 gears->window = window_create(display, fd, "Wayland Gears",
342 x, y, width, height);
343
344 gears->display = eglCreateDisplayNative("/dev/dri/card0", "i965");
345 if (gears->display == NULL)
346 die("failed to create egl display\n");
347
348 if (!eglInitialize(gears->display, &major, &minor))
349 die("failed to initialize display\n");
350
351 if (!eglGetConfigs(gears->display, configs, 64, &count))
352 die("failed to get configs\n");
353
354 gears->config = configs[24];
355 gears->context = eglCreateContext(gears->display, gears->config, NULL, NULL);
356 if (gears->context == NULL)
357 die("failed to create context\n");
358
359 resize_window(gears);
360
361 for (i = 0; i < 3; i++) {
362 gears->gear_list[i] = glGenLists(1);
363 glNewList(gears->gear_list[i], GL_COMPILE);
364 make_gear(&gear_templates[i]);
365 glEndList();
366 }
367
368 glEnable(GL_NORMALIZE);
369
370 glMatrixMode(GL_PROJECTION);
371 glLoadIdentity();
372 glFrustum(-1.0, 1.0, -1.0, 1.0, 5.0, 200.0);
373 glMatrixMode(GL_MODELVIEW);
374
375 glLightfv(GL_LIGHT0, GL_POSITION, light_pos);
376 glEnable(GL_CULL_FACE);
377 glEnable(GL_LIGHTING);
378 glEnable(GL_LIGHT0);
379 glEnable(GL_DEPTH_TEST);
380 glClearColor(0, 0, 0, 0.92);
381
382 draw_gears(gears);
Kristian Høgsberg0c4457f2008-12-07 19:59:11 -0500383
384 frame_handler(gears->window, 0, 0, gears);
385
386 window_set_resize_handler(gears->window, resize_handler, gears);
387 window_set_frame_handler(gears->window, frame_handler, gears);
388 window_set_acknowledge_handler(gears->window, acknowledge_handler, gears);
389
390 return gears;
391}
392
393int main(int argc, char *argv[])
394{
395 struct wl_display *display;
396 int fd;
397 GMainLoop *loop;
398 GSource *source;
399 struct gears *gears;
400
401 fd = open(gem_device, O_RDWR);
402 if (fd < 0) {
403 fprintf(stderr, "drm open failed: %m\n");
404 return -1;
405 }
406
407 display = wl_display_create(socket_name, sizeof socket_name);
408 if (display == NULL) {
409 fprintf(stderr, "failed to create display: %m\n");
410 return -1;
411 }
412
413 loop = g_main_loop_new(NULL, FALSE);
414 source = wl_glib_source_new(display);
415 g_source_attach(source, NULL);
416
417 gears = gears_create(display, fd);
418
419 g_main_loop_run(loop);
420
421 return 0;
422}