parent
cf3c9c5182
commit
dbafcb7b55
@ -0,0 +1,20 @@ |
||||
#!/bin/bash |
||||
|
||||
# When running Debian's Xwayland and fontconfig, we hit memory leaks which |
||||
# aren't visible on other setups. We do have suppressions for these tests, but |
||||
# regrettably ASan can't see through omitted frame pointers in Expat, so for |
||||
# Xwayland specifically, we disable fast-unwind. |
||||
# |
||||
# Doing it globally makes the other tests far, far, too slow to run. |
||||
case "$1" in |
||||
*xwayland*) |
||||
export ASAN_OPTIONS="detect_leaks=0,fast_unwind_on_malloc=0" |
||||
;; |
||||
*) |
||||
export ASAN_OPTIONS="detect_leaks=0" |
||||
;; |
||||
esac |
||||
|
||||
export ASAN_OPTIONS |
||||
|
||||
exec "$@" |
@ -1,504 +0,0 @@ |
||||
/*
|
||||
* Copyright ยฉ 2008 Kristian Hรธgsberg |
||||
* |
||||
* Permission is hereby granted, free of charge, to any person obtaining a |
||||
* copy of this software and associated documentation files (the "Software"), |
||||
* to deal in the Software without restriction, including without limitation |
||||
* the rights to use, copy, modify, merge, publish, distribute, sublicense, |
||||
* and/or sell copies of the Software, and to permit persons to whom the |
||||
* Software is furnished to do so, subject to the following conditions: |
||||
* |
||||
* The above copyright notice and this permission notice (including the next |
||||
* paragraph) shall be included in all copies or substantial portions of the |
||||
* Software. |
||||
* |
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
||||
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
||||
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
||||
* DEALINGS IN THE SOFTWARE. |
||||
*/ |
||||
|
||||
#include "config.h" |
||||
|
||||
#include <stdint.h> |
||||
#include <stdbool.h> |
||||
#include <stdio.h> |
||||
#include <stdlib.h> |
||||
#include <string.h> |
||||
#include <math.h> |
||||
#include <time.h> |
||||
#include <errno.h> |
||||
|
||||
#include <GL/gl.h> |
||||
#include <EGL/egl.h> |
||||
#include <EGL/eglext.h> |
||||
|
||||
#include <linux/input.h> |
||||
#include <wayland-client.h> |
||||
|
||||
#include "window.h" |
||||
|
||||
struct gears { |
||||
struct window *window; |
||||
struct widget *widget; |
||||
|
||||
struct display *d; |
||||
|
||||
EGLDisplay display; |
||||
EGLDisplay config; |
||||
EGLContext context; |
||||
GLfloat angle; |
||||
|
||||
struct { |
||||
GLfloat rotx; |
||||
GLfloat roty; |
||||
} view; |
||||
|
||||
int button_down; |
||||
int last_x, last_y; |
||||
|
||||
GLint gear_list[3]; |
||||
int fullscreen; |
||||
int frames; |
||||
uint32_t last_fps; |
||||
}; |
||||
|
||||
struct gear_template { |
||||
GLfloat material[4]; |
||||
GLfloat inner_radius; |
||||
GLfloat outer_radius; |
||||
GLfloat width; |
||||
GLint teeth; |
||||
GLfloat tooth_depth; |
||||
}; |
||||
|
||||
static const struct gear_template gear_templates[] = { |
||||
{ { 0.8, 0.1, 0.0, 1.0 }, 1.0, 4.0, 1.0, 20, 0.7 }, |
||||
{ { 0.0, 0.8, 0.2, 1.0 }, 0.5, 2.0, 2.0, 10, 0.7 }, |
||||
{ { 0.2, 0.2, 1.0, 1.0 }, 1.3, 2.0, 0.5, 10, 0.7 }, |
||||
}; |
||||
|
||||
static GLfloat light_pos[4] = {5.0, 5.0, 10.0, 0.0}; |
||||
|
||||
static void die(const char *msg) |
||||
{ |
||||
fprintf(stderr, "%s", msg); |
||||
exit(EXIT_FAILURE); |
||||
} |
||||
|
||||
static void |
||||
make_gear(const struct gear_template *t) |
||||
{ |
||||
GLint i; |
||||
GLfloat r0, r1, r2; |
||||
GLfloat angle, da; |
||||
GLfloat u, v, len; |
||||
|
||||
glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, t->material); |
||||
|
||||
r0 = t->inner_radius; |
||||
r1 = t->outer_radius - t->tooth_depth / 2.0; |
||||
r2 = t->outer_radius + t->tooth_depth / 2.0; |
||||
|
||||
da = 2.0 * M_PI / t->teeth / 4.0; |
||||
|
||||
glShadeModel(GL_FLAT); |
||||
|
||||
glNormal3f(0.0, 0.0, 1.0); |
||||
|
||||
/* draw front face */ |
||||
glBegin(GL_QUAD_STRIP); |
||||
for (i = 0; i <= t->teeth; i++) { |
||||
angle = i * 2.0 * M_PI / t->teeth; |
||||
glVertex3f(r0 * cos(angle), r0 * sin(angle), t->width * 0.5); |
||||
glVertex3f(r1 * cos(angle), r1 * sin(angle), t->width * 0.5); |
||||
if (i < t->teeth) { |
||||
glVertex3f(r0 * cos(angle), r0 * sin(angle), t->width * 0.5); |
||||
glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), t->width * 0.5); |
||||
} |
||||
} |
||||
glEnd(); |
||||
|
||||
/* draw front sides of teeth */ |
||||
glBegin(GL_QUADS); |
||||
da = 2.0 * M_PI / t->teeth / 4.0; |
||||
for (i = 0; i < t->teeth; i++) { |
||||
angle = i * 2.0 * M_PI / t->teeth; |
||||
|
||||
glVertex3f(r1 * cos(angle), r1 * sin(angle), t->width * 0.5); |
||||
glVertex3f(r2 * cos(angle + da), r2 * sin(angle + da), t->width * 0.5); |
||||
glVertex3f(r2 * cos(angle + 2 * da), r2 * sin(angle + 2 * da), t->width * 0.5); |
||||
glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), t->width * 0.5); |
||||
} |
||||
glEnd(); |
||||
|
||||
glNormal3f(0.0, 0.0, -1.0); |
||||
|
||||
/* draw back face */ |
||||
glBegin(GL_QUAD_STRIP); |
||||
for (i = 0; i <= t->teeth; i++) { |
||||
angle = i * 2.0 * M_PI / t->teeth; |
||||
glVertex3f(r1 * cos(angle), r1 * sin(angle), -t->width * 0.5); |
||||
glVertex3f(r0 * cos(angle), r0 * sin(angle), -t->width * 0.5); |
||||
if (i < t->teeth) { |
||||
glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), -t->width * 0.5); |
||||
glVertex3f(r0 * cos(angle), r0 * sin(angle), -t->width * 0.5); |
||||
} |
||||
} |
||||
glEnd(); |
||||
|
||||
/* draw back sides of teeth */ |
||||
glBegin(GL_QUADS); |
||||
da = 2.0 * M_PI / t->teeth / 4.0; |
||||
for (i = 0; i < t->teeth; i++) { |
||||
angle = i * 2.0 * M_PI / t->teeth; |
||||
|
||||
glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), -t->width * 0.5); |
||||
glVertex3f(r2 * cos(angle + 2 * da), r2 * sin(angle + 2 * da), -t->width * 0.5); |
||||
glVertex3f(r2 * cos(angle + da), r2 * sin(angle + da), -t->width * 0.5); |
||||
glVertex3f(r1 * cos(angle), r1 * sin(angle), -t->width * 0.5); |
||||
} |
||||
glEnd(); |
||||
|
||||
/* draw outward faces of teeth */ |
||||
glBegin(GL_QUAD_STRIP); |
||||
for (i = 0; i < t->teeth; i++) { |
||||
angle = i * 2.0 * M_PI / t->teeth; |
||||
|
||||
glVertex3f(r1 * cos(angle), r1 * sin(angle), t->width * 0.5); |
||||
glVertex3f(r1 * cos(angle), r1 * sin(angle), -t->width * 0.5); |
||||
u = r2 * cos(angle + da) - r1 * cos(angle); |
||||
v = r2 * sin(angle + da) - r1 * sin(angle); |
||||
len = sqrt(u * u + v * v); |
||||
u /= len; |
||||
v /= len; |
||||
glNormal3f(v, -u, 0.0); |
||||
glVertex3f(r2 * cos(angle + da), r2 * sin(angle + da), t->width * 0.5); |
||||
glVertex3f(r2 * cos(angle + da), r2 * sin(angle + da), -t->width * 0.5); |
||||
glNormal3f(cos(angle), sin(angle), 0.0); |
||||
glVertex3f(r2 * cos(angle + 2 * da), r2 * sin(angle + 2 * da), t->width * 0.5); |
||||
glVertex3f(r2 * cos(angle + 2 * da), r2 * sin(angle + 2 * da), -t->width * 0.5); |
||||
u = r1 * cos(angle + 3 * da) - r2 * cos(angle + 2 * da); |
||||
v = r1 * sin(angle + 3 * da) - r2 * sin(angle + 2 * da); |
||||
glNormal3f(v, -u, 0.0); |
||||
glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), t->width * 0.5); |
||||
glVertex3f(r1 * cos(angle + 3 * da), r1 * sin(angle + 3 * da), -t->width * 0.5); |
||||
glNormal3f(cos(angle), sin(angle), 0.0); |
||||
} |
||||
|
||||
glVertex3f(r1 * cos(0), r1 * sin(0), t->width * 0.5); |
||||
glVertex3f(r1 * cos(0), r1 * sin(0), -t->width * 0.5); |
||||
|
||||
glEnd(); |
||||
|
||||
glShadeModel(GL_SMOOTH); |
||||
|
||||
/* draw inside radius cylinder */ |
||||
glBegin(GL_QUAD_STRIP); |
||||
for (i = 0; i <= t->teeth; i++) { |
||||
angle = i * 2.0 * M_PI / t->teeth; |
||||
glNormal3f(-cos(angle), -sin(angle), 0.0); |
||||
glVertex3f(r0 * cos(angle), r0 * sin(angle), -t->width * 0.5); |
||||
glVertex3f(r0 * cos(angle), r0 * sin(angle), t->width * 0.5); |
||||
} |
||||
glEnd(); |
||||
} |
||||
|
||||
static void |
||||
update_fps(struct gears *gears, uint32_t time) |
||||
{ |
||||
long diff_ms; |
||||
static bool first_call = true; |
||||
|
||||
if (first_call) { |
||||
gears->last_fps = time; |
||||
first_call = false; |
||||
} else |
||||
gears->frames++; |
||||
|
||||
diff_ms = time - gears->last_fps; |
||||
|
||||
if (diff_ms > 5000) { |
||||
float seconds = diff_ms / 1000.0; |
||||
float fps = gears->frames / seconds; |
||||
|
||||
printf("%d frames in %6.3f seconds = %6.3f FPS\n", gears->frames, seconds, fps); |
||||
fflush(stdout); |
||||
|
||||
gears->frames = 0; |
||||
gears->last_fps = time; |
||||
} |
||||
} |
||||
|
||||
static void |
||||
frame_callback(void *data, struct wl_callback *callback, uint32_t time) |
||||
{ |
||||
struct gears *gears = data; |
||||
|
||||
update_fps(gears, time); |
||||
|
||||
gears->angle = (GLfloat) (time % 8192) * 360 / 8192.0; |
||||
|
||||
window_schedule_redraw(gears->window); |
||||
|
||||
if (callback) |
||||
wl_callback_destroy(callback); |
||||
} |
||||
|
||||
static const struct wl_callback_listener listener = { |
||||
frame_callback |
||||
}; |
||||
|
||||
static int |
||||
motion_handler(struct widget *widget, struct input *input, |
||||
uint32_t time, float x, float y, void *data) |
||||
{ |
||||
struct gears *gears = data; |
||||
int offset_x, offset_y; |
||||
float step = 0.5; |
||||
|
||||
if (gears->button_down) { |
||||
offset_x = x - gears->last_x; |
||||
offset_y = y - gears->last_y; |
||||
gears->last_x = x; |
||||
gears->last_y = y; |
||||
gears->view.roty += offset_x * step; |
||||
gears->view.rotx += offset_y * step; |
||||
if (gears->view.roty >= 360) |
||||
gears->view.roty = gears->view.roty - 360; |
||||
if (gears->view.roty <= 0) |
||||
gears->view.roty = gears->view.roty + 360; |
||||
if (gears->view.rotx >= 360) |
||||
gears->view.rotx = gears->view.rotx - 360; |
||||
if (gears->view.rotx <= 0) |
||||
gears->view.rotx = gears->view.rotx + 360; |
||||
} |
||||
|
||||
return CURSOR_LEFT_PTR; |
||||
} |
||||
|
||||
static void |
||||
button_handler(struct widget *widget, struct input *input, |
||||
uint32_t time, uint32_t button, |
||||
enum wl_pointer_button_state state, void *data) |
||||
{ |
||||
struct gears *gears = data; |
||||
|
||||
if (button == BTN_LEFT) { |
||||
if (state == WL_POINTER_BUTTON_STATE_PRESSED) { |
||||
gears->button_down = 1; |
||||
input_get_position(input, |
||||
&gears->last_x, &gears->last_y); |
||||
} else { |
||||
gears->button_down = 0; |
||||
} |
||||
} |
||||
} |
||||
|
||||
static void |
||||
redraw_handler(struct widget *widget, void *data) |
||||
{ |
||||
struct rectangle window_allocation; |
||||
struct rectangle allocation; |
||||
struct wl_callback *callback; |
||||
struct gears *gears = data; |
||||
|
||||
widget_get_allocation(gears->widget, &allocation); |
||||
window_get_allocation(gears->window, &window_allocation); |
||||
|
||||
if (display_acquire_window_surface(gears->d, |
||||
gears->window, |
||||
gears->context) < 0) { |
||||
die("Unable to acquire window surface, " |
||||
"compiled without cairo-egl?\n"); |
||||
} |
||||
|
||||
glViewport(allocation.x, |
||||
window_allocation.height - allocation.height - allocation.y, |
||||
allocation.width, allocation.height); |
||||
glScissor(allocation.x, |
||||
window_allocation.height - allocation.height - allocation.y, |
||||
allocation.width, allocation.height); |
||||
|
||||
glEnable(GL_SCISSOR_TEST); |
||||
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); |
||||
|
||||
glPushMatrix(); |
||||
|
||||
glTranslatef(0.0, 0.0, -50); |
||||
|
||||
glRotatef(gears->view.rotx, 1.0, 0.0, 0.0); |
||||
glRotatef(gears->view.roty, 0.0, 1.0, 0.0); |
||||
|
||||
glPushMatrix(); |
||||
glTranslatef(-3.0, -2.0, 0.0); |
||||
glRotatef(gears->angle, 0.0, 0.0, 1.0); |
||||
glCallList(gears->gear_list[0]); |
||||
glPopMatrix(); |
||||
|
||||
glPushMatrix(); |
||||
glTranslatef(3.1, -2.0, 0.0); |
||||
glRotatef(-2.0 * gears->angle - 9.0, 0.0, 0.0, 1.0); |
||||
glCallList(gears->gear_list[1]); |
||||
glPopMatrix(); |
||||
|
||||
glPushMatrix(); |
||||
glTranslatef(-3.1, 4.2, 0.0); |
||||
glRotatef(-2.0 * gears->angle - 25.0, 0.0, 0.0, 1.0); |
||||
glCallList(gears->gear_list[2]); |
||||
glPopMatrix(); |
||||
|
||||
glPopMatrix(); |
||||
|
||||
glFlush(); |
||||
|
||||
display_release_window_surface(gears->d, gears->window); |
||||
|
||||
callback = wl_surface_frame(window_get_wl_surface(gears->window)); |
||||
wl_callback_add_listener(callback, &listener, gears); |
||||
} |
||||
|
||||
static void |
||||
resize_handler(struct widget *widget, |
||||
int32_t width, int32_t height, void *data) |
||||
{ |
||||
struct gears *gears = data; |
||||
int32_t size, big, small; |
||||
|
||||
/* Constrain child size to be square and at least 300x300 */ |
||||
if (width < height) { |
||||
small = width; |
||||
big = height; |
||||
} else { |
||||
small = height; |
||||
big = width; |
||||
} |
||||
|
||||
if (gears->fullscreen) |
||||
size = small; |
||||
else |
||||
size = big; |
||||
|
||||
widget_set_size(gears->widget, size, size); |
||||
} |
||||
|
||||
static void |
||||
keyboard_focus_handler(struct window *window, |
||||
struct input *device, void *data) |
||||
{ |
||||
window_schedule_redraw(window); |
||||
} |
||||
|
||||
static void |
||||
fullscreen_handler(struct window *window, void *data) |
||||
{ |
||||
struct gears *gears = data; |
||||
|
||||
gears->fullscreen ^= 1; |
||||
window_set_fullscreen(window, gears->fullscreen); |
||||
} |
||||
|
||||
static struct gears * |
||||
gears_create(struct display *display) |
||||
{ |
||||
const int width = 450, height = 500; |
||||
struct gears *gears; |
||||
int i; |
||||
|
||||
gears = zalloc(sizeof *gears); |
||||
gears->d = display; |
||||
gears->window = window_create(display); |
||||
gears->widget = window_frame_create(gears->window, gears); |
||||
window_set_title(gears->window, "Wayland Gears"); |
||||
window_set_appid(gears->window, "org.freedesktop.weston.wayland-gears"); |
||||
|
||||
gears->display = display_get_egl_display(gears->d); |
||||
if (gears->display == NULL) |
||||
die("failed to create egl display\n"); |
||||
|
||||
eglBindAPI(EGL_OPENGL_API); |
||||
|
||||
gears->config = display_get_argb_egl_config(gears->d); |
||||
|
||||
gears->context = eglCreateContext(gears->display, gears->config, |
||||
EGL_NO_CONTEXT, NULL); |
||||
if (gears->context == NULL) |
||||
die("failed to create context\n"); |
||||
|
||||
if (!eglMakeCurrent(gears->display, NULL, NULL, gears->context)) |
||||
die("failed to make context current\n"); |
||||
|
||||
for (i = 0; i < 3; i++) { |
||||
gears->gear_list[i] = glGenLists(1); |
||||
glNewList(gears->gear_list[i], GL_COMPILE); |
||||
make_gear(&gear_templates[i]); |
||||
glEndList(); |
||||
} |
||||
|
||||
gears->button_down = 0; |
||||
gears->last_x = 0; |
||||
gears->last_y = 0; |
||||
|
||||
gears->view.rotx = 20.0; |
||||
gears->view.roty = 30.0; |
||||
|
||||
printf("Warning: FPS count is limited by the wayland compositor or monitor refresh rate\n"); |
||||
|
||||
glEnable(GL_NORMALIZE); |
||||
|
||||
glMatrixMode(GL_PROJECTION); |
||||
glLoadIdentity(); |
||||
glFrustum(-1.0, 1.0, -1.0, 1.0, 5.0, 200.0); |
||||
glMatrixMode(GL_MODELVIEW); |
||||
|
||||
glLightfv(GL_LIGHT0, GL_POSITION, light_pos); |
||||
glEnable(GL_CULL_FACE); |
||||
glEnable(GL_LIGHTING); |
||||
glEnable(GL_LIGHT0); |
||||
glEnable(GL_DEPTH_TEST); |
||||
glClearColor(0, 0, 0, 0.92); |
||||
|
||||
window_set_user_data(gears->window, gears); |
||||
widget_set_resize_handler(gears->widget, resize_handler); |
||||
widget_set_redraw_handler(gears->widget, redraw_handler); |
||||
widget_set_button_handler(gears->widget, button_handler); |
||||
widget_set_motion_handler(gears->widget, motion_handler); |
||||
window_set_keyboard_focus_handler(gears->window, |
||||
keyboard_focus_handler); |
||||
window_set_fullscreen_handler(gears->window, fullscreen_handler); |
||||
|
||||
window_schedule_resize(gears->window, width, height); |
||||
|
||||
return gears; |
||||
} |
||||
|
||||
static void |
||||
gears_destroy(struct gears *gears) |
||||
{ |
||||
widget_destroy(gears->widget); |
||||
window_destroy(gears->window); |
||||
free(gears); |
||||
} |
||||
|
||||
int main(int argc, char *argv[]) |
||||
{ |
||||
struct display *d; |
||||
struct gears *gears; |
||||
|
||||
d = display_create(&argc, argv); |
||||
if (d == NULL) { |
||||
fprintf(stderr, "failed to create display: %s\n", |
||||
strerror(errno)); |
||||
return -1; |
||||
} |
||||
gears = gears_create(d); |
||||
display_run(d); |
||||
|
||||
gears_destroy(gears); |
||||
display_destroy(d); |
||||
|
||||
return 0; |
||||
} |
@ -1,374 +0,0 @@ |
||||
/*
|
||||
* Copyright ยฉ 2013 Intel Corporation |
||||
* |
||||
* Permission is hereby granted, free of charge, to any person obtaining a |
||||
* copy of this software and associated documentation files (the "Software"), |
||||
* to deal in the Software without restriction, including without limitation |
||||
* the rights to use, copy, modify, merge, publish, distribute, sublicense, |
||||
* and/or sell copies of the Software, and to permit persons to whom the |
||||
* Software is furnished to do so, subject to the following conditions: |
||||
* |
||||
* The above copyright notice and this permission notice (including the next |
||||
* paragraph) shall be included in all copies or substantial portions of the |
||||
* Software. |
||||
* |
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
||||
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
||||
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
||||
* DEALINGS IN THE SOFTWARE. |
||||
*/ |
||||
|
||||
#include "config.h" |
||||
|
||||
#include <stdint.h> |
||||
#include <stdio.h> |
||||
#include <stdlib.h> |
||||
#include <string.h> |
||||
|
||||
#include <wayland-egl.h> |
||||
#include <wayland-cursor.h> |
||||
|
||||
#include <GLES2/gl2.h> |
||||
#include <EGL/egl.h> |
||||
|
||||
#include "shared/platform.h" |
||||
|
||||
struct window; |
||||
struct seat; |
||||
|
||||
struct nested_client { |
||||
struct wl_display *display; |
||||
struct wl_registry *registry; |
||||
struct wl_compositor *compositor; |
||||
|
||||
EGLDisplay egl_display; |
||||
EGLContext egl_context; |
||||
EGLConfig egl_config; |
||||
EGLSurface egl_surface; |
||||
struct program *color_program; |
||||
|
||||
GLuint vert, frag, program; |
||||
GLuint rotation; |
||||
GLuint pos; |
||||
GLuint col; |
||||
|
||||
struct wl_surface *surface; |
||||
struct wl_egl_window *native; |
||||
int width, height; |
||||
}; |
||||
|
||||
#define POS 0 |
||||
#define COL 1 |
||||
|
||||
static GLuint |
||||
create_shader(const char *source, GLenum shader_type) |
||||
{ |
||||
GLuint shader; |
||||
GLint status; |
||||
|
||||
shader = glCreateShader(shader_type); |
||||
if (shader == 0) |
||||
return 0; |
||||
|
||||
glShaderSource(shader, 1, (const char **) &source, NULL); |
||||
glCompileShader(shader); |
||||
|
||||
glGetShaderiv(shader, GL_COMPILE_STATUS, &status); |
||||
if (!status) { |
||||
char log[1000]; |
||||
GLsizei len; |
||||
glGetShaderInfoLog(shader, 1000, &len, log); |
||||
fprintf(stderr, "Error: compiling %s: %.*s\n", |
||||
shader_type == GL_VERTEX_SHADER ? "vertex" : "fragment", |
||||
len, log); |
||||
return 0; |
||||
} |
||||
|
||||
return shader; |
||||
} |
||||
|
||||
static void |
||||
create_program(struct nested_client *client, |
||||
const char *vert, const char *frag) |
||||
{ |
||||
GLint status; |
||||
|
||||
client->vert = create_shader(vert, GL_VERTEX_SHADER); |
||||
client->frag = create_shader(frag, GL_FRAGMENT_SHADER); |
||||
|
||||
client->program = glCreateProgram(); |
||||
glAttachShader(client->program, client->frag); |
||||
glAttachShader(client->program, client->vert); |
||||
glBindAttribLocation(client->program, POS, "pos"); |
||||
glBindAttribLocation(client->program, COL, "color"); |
||||
glLinkProgram(client->program); |
||||
|
||||
glGetProgramiv(client->program, GL_LINK_STATUS, &status); |
||||
if (!status) { |
||||
char log[1000]; |
||||
GLsizei len; |
||||
glGetProgramInfoLog(client->program, 1000, &len, log); |
||||
fprintf(stderr, "Error: linking:\n%.*s\n", len, log); |
||||
exit(1); |
||||
} |
||||
|
||||
client->rotation = |
||||
glGetUniformLocation(client->program, "rotation"); |
||||
} |
||||
|
||||
static const char vertex_shader_text[] = |
||||
"uniform mat4 rotation;\n" |
||||
"attribute vec4 pos;\n" |
||||
"attribute vec4 color;\n" |
||||
"varying vec4 v_color;\n" |
||||
"void main() {\n" |
||||
" gl_Position = rotation * pos;\n" |
||||
" v_color = color;\n" |
||||
"}\n"; |
||||
|
||||
static const char color_fragment_shader_text[] = |
||||
"precision mediump float;\n" |
||||
"varying vec4 v_color;\n" |
||||
"void main() {\n" |
||||
" gl_FragColor = v_color;\n" |
||||
"}\n"; |
||||
|
||||
static void |
||||
render_triangle(struct nested_client *client, uint32_t time) |
||||
{ |
||||
static const GLfloat verts[3][2] = { |
||||
{ -0.5, -0.5 }, |
||||
{ 0.5, -0.5 }, |
||||
{ 0, 0.5 } |
||||
}; |
||||
static const GLfloat colors[3][3] = { |
||||
{ 1, 0, 0 }, |
||||
{ 0, 1, 0 }, |
||||
{ 0, 0, 1 } |
||||
}; |
||||
GLfloat angle; |
||||
GLfloat rotation[4][4] = { |
||||
{ 1, 0, 0, 0 }, |
||||
{ 0, 1, 0, 0 }, |
||||
{ 0, 0, 1, 0 }, |
||||
{ 0, 0, 0, 1 } |
||||
}; |
||||
static const int32_t speed_div = 5; |
||||
static uint32_t start_time = 0; |
||||
|
||||
if (client->program == 0) |
||||
create_program(client, vertex_shader_text, |
||||
color_fragment_shader_text); |
||||
|
||||
if (start_time == 0) |
||||
start_time = time; |
||||
|
||||
angle = ((time - start_time) / speed_div) % 360 * M_PI / 180.0; |
||||
rotation[0][0] = cos(angle); |
||||
rotation[0][2] = sin(angle); |
||||
rotation[2][0] = -sin(angle); |
||||
rotation[2][2] = cos(angle); |
||||
|
||||
glClearColor(0.4, 0.4, 0.4, 1.0); |
||||
glClear(GL_COLOR_BUFFER_BIT); |
||||
|
||||
glUseProgram(client->program); |
||||
|
||||
glViewport(0, 0, client->width, client->height); |
||||
|
||||
glUniformMatrix4fv(client->rotation, 1, GL_FALSE, |
||||
(GLfloat *) rotation); |
||||
|
||||
glVertexAttribPointer(POS, 2, GL_FLOAT, GL_FALSE, 0, verts); |
||||
glVertexAttribPointer(COL, 3, GL_FLOAT, GL_FALSE, 0, colors); |
||||
glEnableVertexAttribArray(POS); |
||||
glEnableVertexAttribArray(COL); |
||||
|
||||
glDrawArrays(GL_TRIANGLES, 0, 3); |
||||
|
||||
glDisableVertexAttribArray(POS); |
||||
glDisableVertexAttribArray(COL); |
||||
|
||||
glFlush(); |
||||
} |
||||
|
||||
static void |
||||
frame_callback(void *data, struct wl_callback *callback, uint32_t time); |
||||
|
||||