blob: 6310479f46985718f8becea4fa490823da69f215 [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{
246 window_draw(gears->window);
247 window_get_child_rectangle(gears->window, &gears->rectangle);
248
249 if (gears->buffer != NULL)
250 buffer_destroy(gears->buffer, gears->fd);
251
252 gears->buffer = buffer_create(gears->fd,
253 gears->rectangle.width,
254 gears->rectangle.height,
255 (gears->rectangle.width * 4 + 15) & ~15);
256
257 gears->surface = eglCreateSurfaceForName(gears->display,
258 gears->config,
259 gears->buffer->name,
260 gears->buffer->width,
261 gears->buffer->height,
262 gears->buffer->stride, NULL);
263
264 eglMakeCurrent(gears->display,
265 gears->surface, gears->surface, gears->context);
266
267 glViewport(0, 0, gears->rectangle.width, gears->rectangle.height);
268 gears->resized = 0;
269}
270
271static void
272resize_handler(struct window *window, int32_t width, int32_t height, void *data)
273{
274 struct gears *gears = data;
275
276 /* Right now, resizing the window from the per-frame callback
277 * is fine, since the window drawing code is so slow that we
278 * can't draw more than one window per frame anyway. However,
279 * once we implement faster resizing, this will show lag
280 * between pointer motion and window size even if resizing is
281 * fast. We need to keep processing motion events and posting
282 * new frames as fast as possible so when the server
283 * composites the next frame it will have the most recent size
284 * possible, like what we do for window moves. */
285
286 gears->resized = 1;
287}
288
289static void
290acknowledge_handler(struct window *window, uint32_t key, void *data)
291{
292 struct gears *gears = data;
293
294 if (key != 0)
295 return;
296
297 if (gears->resized)
298 resize_window(gears);
299
300 draw_gears(gears);
301}
302
303static void
304frame_handler(struct window *window, uint32_t frame, uint32_t timestamp, void *data)
305{
306 struct gears *gears = data;
307
308 window_copy(gears->window,
309 &gears->rectangle,
310 gears->buffer->name, gears->buffer->stride);
311
312 wl_display_commit(gears->wl_display, 0);
313
314 gears->angle += 1;
315}
316
317static struct gears *
318gears_create(struct wl_display *display, int fd)
319{
320 const int x = 200, y = 200, width = 450, height = 500;
321 EGLint major, minor, count;
322 EGLConfig configs[64];
323 struct gears *gears;
324 int i;
325
326 gears = malloc(sizeof *gears);
327 memset(gears, 0, sizeof *gears);
328 gears->wl_display = display;
329 gears->fd = fd;
330 gears->window = window_create(display, fd, "Wayland Gears",
331 x, y, width, height);
332
333 gears->display = eglCreateDisplayNative("/dev/dri/card0", "i965");
334 if (gears->display == NULL)
335 die("failed to create egl display\n");
336
337 if (!eglInitialize(gears->display, &major, &minor))
338 die("failed to initialize display\n");
339
340 if (!eglGetConfigs(gears->display, configs, 64, &count))
341 die("failed to get configs\n");
342
343 gears->config = configs[24];
344 gears->context = eglCreateContext(gears->display, gears->config, NULL, NULL);
345 if (gears->context == NULL)
346 die("failed to create context\n");
347
348 resize_window(gears);
349
350 for (i = 0; i < 3; i++) {
351 gears->gear_list[i] = glGenLists(1);
352 glNewList(gears->gear_list[i], GL_COMPILE);
353 make_gear(&gear_templates[i]);
354 glEndList();
355 }
356
357 glEnable(GL_NORMALIZE);
358
359 glMatrixMode(GL_PROJECTION);
360 glLoadIdentity();
361 glFrustum(-1.0, 1.0, -1.0, 1.0, 5.0, 200.0);
362 glMatrixMode(GL_MODELVIEW);
363
364 glLightfv(GL_LIGHT0, GL_POSITION, light_pos);
365 glEnable(GL_CULL_FACE);
366 glEnable(GL_LIGHTING);
367 glEnable(GL_LIGHT0);
368 glEnable(GL_DEPTH_TEST);
369 glClearColor(0, 0, 0, 0.92);
370
371 draw_gears(gears);
372 gears->angle += 1;
373
374 frame_handler(gears->window, 0, 0, gears);
375
376 window_set_resize_handler(gears->window, resize_handler, gears);
377 window_set_frame_handler(gears->window, frame_handler, gears);
378 window_set_acknowledge_handler(gears->window, acknowledge_handler, gears);
379
380 return gears;
381}
382
383int main(int argc, char *argv[])
384{
385 struct wl_display *display;
386 int fd;
387 GMainLoop *loop;
388 GSource *source;
389 struct gears *gears;
390
391 fd = open(gem_device, O_RDWR);
392 if (fd < 0) {
393 fprintf(stderr, "drm open failed: %m\n");
394 return -1;
395 }
396
397 display = wl_display_create(socket_name, sizeof socket_name);
398 if (display == NULL) {
399 fprintf(stderr, "failed to create display: %m\n");
400 return -1;
401 }
402
403 loop = g_main_loop_new(NULL, FALSE);
404 source = wl_glib_source_new(display);
405 g_source_attach(source, NULL);
406
407 gears = gears_create(display, fd);
408
409 g_main_loop_run(loop);
410
411 return 0;
412}