sgl
changeset 4:648f8604d2b2
cont. x11 module
author | John Tsiombikas <nuclear@siggraph.org> |
---|---|
date | Thu, 12 May 2011 11:04:10 +0300 (2011-05-12) |
parents | 1b6c5dadb460 |
children | 0570e27e5ebc |
files | include/sgl.h src/cb.c src/log.c src/log.h src/sgl.c src/wsys.h src/wsys_x11.c |
diffstat | 7 files changed, 447 insertions(+), 24 deletions(-) [+] |
line diff
1.1 --- a/include/sgl.h Wed May 11 09:09:43 2011 +0300 1.2 +++ b/include/sgl.h Thu May 12 11:04:10 2011 +0300 1.3 @@ -5,6 +5,7 @@ 1.4 #define SGL_DEPTH 2 1.5 #define SGL_STENCIL 4 1.6 #define SGL_STEREO 8 1.7 +#define SGL_MULTISAMPLE 16 1.8 1.9 enum { 1.10 SGL_CREATE, 1.11 @@ -23,15 +24,22 @@ 1.12 int sgl_init(void); 1.13 void sgl_shutdown(void); 1.14 1.15 -int sgl_set_video_mode(int xsz, int ysz, int bpp); 1.16 -int sgl_get_video_mode(int *xsz, int *ysz, int *bpp); 1.17 +int sgl_set_video_mode(int xsz, int ysz); 1.18 +int sgl_get_video_mode(int *xsz, int *ysz); 1.19 1.20 -int sgl_window(int xsz, int ysz, unsigned int mode); 1.21 +int sgl_create_window(int xsz, int ysz, unsigned int mode); 1.22 void sgl_close_window(int win); 1.23 1.24 +int sgl_set_active(int id); 1.25 +int sgl_set_title(const char *str); 1.26 + 1.27 +int sgl_process_events(void); 1.28 +int sgl_event_loop(void); 1.29 + 1.30 int sgl_push_callbacks(void); 1.31 int sgl_pop_callbacks(void); 1.32 void sgl_clear_callbacks(void); 1.33 -void (*sgl_callback(int idx, void (*func)()))(); 1.34 +void sgl_set_callback(int idx, void (*func)()); 1.35 +void (*sgl_get_callback(int idx))(); 1.36 1.37 #endif /* SGL_H_ */
2.1 --- a/src/cb.c Wed May 11 09:09:43 2011 +0300 2.2 +++ b/src/cb.c Thu May 12 11:04:10 2011 +0300 2.3 @@ -1,12 +1,15 @@ 2.4 #include <stdlib.h> 2.5 #include <string.h> 2.6 #include "sgl.h" 2.7 +#include "wsys.h" 2.8 2.9 struct cbnode { 2.10 void (*func[SGL_NUM_CALLBACKS])(); 2.11 struct cbnode *next; 2.12 }; 2.13 2.14 +static void notify_wsys(void); 2.15 + 2.16 struct cbnode first_cbnode; 2.17 struct cbnode *cb = &first_cbnode; 2.18 2.19 @@ -34,17 +37,36 @@ 2.20 node = cb; 2.21 cb = cb->next; 2.22 free(node); 2.23 + 2.24 + notify_wsys(); 2.25 return 0; 2.26 } 2.27 2.28 void sgl_clear_callbacks(void) 2.29 { 2.30 memset(cb->func, 0, sizeof cb->func); 2.31 + notify_wsys(); 2.32 } 2.33 2.34 -void (*sgl_callback(int idx, void (*func)()))() 2.35 +void sgl_callback(int idx, void (*func)()) 2.36 { 2.37 - void (*prev)() = cb->func[idx]; 2.38 cb->func[idx] = func; 2.39 - return prev; 2.40 } 2.41 + 2.42 +void (*sgl_get_callback(int idx))() 2.43 +{ 2.44 + return cb->func[idx]; 2.45 +} 2.46 + 2.47 +/* notify the window system module as to which events are active */ 2.48 +static void notify_wsys(void) 2.49 +{ 2.50 + int i; 2.51 + struct wsys_module *ws; 2.52 + 2.53 + if((ws = sgl_wsys_module())) { 2.54 + for(i=0; i<SGL_NUM_CALLBACKS; i++) { 2.55 + ws->set_event(i, cb->func[i] != 0); 2.56 + } 2.57 + } 2.58 +}
3.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 3.2 +++ b/src/log.c Thu May 12 11:04:10 2011 +0300 3.3 @@ -0,0 +1,21 @@ 3.4 +#include <stdio.h> 3.5 +#include <stdlib.h> 3.6 +#include <stdarg.h> 3.7 +#include "log.h" 3.8 + 3.9 +void sgl_log(const char *fmt, ...) 3.10 +{ 3.11 + va_list ap; 3.12 + const char *logfile; 3.13 + FILE *fp; 3.14 + 3.15 + if(!(logfile = getenv("SGL_LOG")) || !(fp = fopen(logfile, "a"))) { 3.16 + fp = stderr; 3.17 + } 3.18 + 3.19 + va_start(ap, fmt); 3.20 + vfprintf(fp, fmt, ap); 3.21 + va_end(ap); 3.22 + 3.23 + fclose(fp); 3.24 +}
4.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 4.2 +++ b/src/log.h Thu May 12 11:04:10 2011 +0300 4.3 @@ -0,0 +1,6 @@ 4.4 +#ifndef LOG_H_ 4.5 +#define LOG_H_ 4.6 + 4.7 +void sgl_log(const char *fmt, ...); 4.8 + 4.9 +#endif /* LOG_H_ */
5.1 --- a/src/sgl.c Wed May 11 09:09:43 2011 +0300 5.2 +++ b/src/sgl.c Thu May 12 11:04:10 2011 +0300 5.3 @@ -1,29 +1,57 @@ 5.4 #include "sgl.h" 5.5 +#include "wsys.h" 5.6 5.7 void sgl_register_modules(void); 5.8 5.9 +static struct wsys_module *ws; 5.10 + 5.11 int sgl_init(void) 5.12 { 5.13 sgl_register_modules(); 5.14 sgl_sort_modules(); 5.15 - return 0; 5.16 + 5.17 + if(!(ws = sgl_wsys_module())) { 5.18 + return -1; 5.19 + } 5.20 + return ws->init(); 5.21 } 5.22 5.23 -int sgl_set_video_mode(int xsz, int ysz, int bpp) 5.24 +void sgl_shutdown(void) 5.25 { 5.26 - return 0; 5.27 + ws->shutdown(); 5.28 } 5.29 5.30 -int sgl_get_video_mode(int *xsz, int *ysz, int *bpp) 5.31 +int sgl_set_video_mode(int xsz, int ysz) 5.32 { 5.33 - return 0; 5.34 + return ws->set_vidmode(xsz, ysz); 5.35 } 5.36 5.37 -int sgl_window(int x, int y, unsigned int mode) 5.38 +int sgl_get_video_mode(int *xsz, int *ysz) 5.39 { 5.40 - return 0; 5.41 + return ws->get_vidmode(xsz, ysz); 5.42 } 5.43 5.44 -void sgl_close(int win) 5.45 +int sgl_create_window(int xsz, int ysz, unsigned int mode) 5.46 { 5.47 + return ws->create_window(xsz, ysz, mode); 5.48 } 5.49 + 5.50 +void sgl_close_window(int win) 5.51 +{ 5.52 + ws->close_window(win); 5.53 +} 5.54 + 5.55 +int sgl_set_active(int id) 5.56 +{ 5.57 + return ws->set_active(id); 5.58 +} 5.59 + 5.60 +int sgl_set_title(const char *str) 5.61 +{ 5.62 + return ws->set_title(str); 5.63 +} 5.64 + 5.65 +int sgl_process_events(void) 5.66 +{ 5.67 + return ws->process_events(); 5.68 +}
6.1 --- a/src/wsys.h Wed May 11 09:09:43 2011 +0300 6.2 +++ b/src/wsys.h Thu May 12 11:04:10 2011 +0300 6.3 @@ -8,11 +8,21 @@ 6.4 int (*init)(void); 6.5 void (*shutdown)(void); 6.6 6.7 - int (*set_vidmode)(int, int, int); 6.8 - int (*get_vidmode)(int*, int*, int*); 6.9 + int (*set_vidmode)(int, int); 6.10 + int (*get_vidmode)(int*, int*); 6.11 + 6.12 int (*create_window)(int, int, unsigned int); 6.13 void (*close_window)(int); 6.14 6.15 + int (*set_active)(int); 6.16 + int (*set_title)(const char*); 6.17 + 6.18 + void (*set_event)(int idx, int enable); 6.19 + int (*process_events)(void); 6.20 + 6.21 + void (*redisplay)(void); 6.22 + void (*swap_buffers)(void); 6.23 + 6.24 struct wsys_module *next; 6.25 }; 6.26
7.1 --- a/src/wsys_x11.c Wed May 11 09:09:43 2011 +0300 7.2 +++ b/src/wsys_x11.c Thu May 12 11:04:10 2011 +0300 7.3 @@ -1,20 +1,43 @@ 7.4 +#include <stdlib.h> 7.5 #include <X11/Xlib.h> 7.6 #include <GL/glx.h> 7.7 +#include "sgl.h" 7.8 #include "wsys.h" 7.9 +#include "log.h" 7.10 7.11 struct window { 7.12 Window win; 7.13 GLXContext ctx; 7.14 + long evmask; 7.15 struct window *next; 7.16 }; 7.17 7.18 static int init(void); 7.19 static void shutdown(void); 7.20 -static int set_vidmode(int xsz, int ysz, int bpp); 7.21 -static int get_vidmode(int *xsz, int *ysz, int *bpp); 7.22 + 7.23 +/* video mode switching */ 7.24 +static int set_vidmode(int xsz, int ysz); 7.25 +static int get_vidmode(int *xsz, int *ysz); 7.26 + 7.27 +/* create/destroy windows */ 7.28 static int create_window(int xsz, int ysz, unsigned int flags); 7.29 +static void fill_attr(int *attr, unsigned int flags); 7.30 +static void print_visual_info(XVisualInfo *vis); 7.31 static void close_window(int id); 7.32 7.33 +/* window management */ 7.34 +static int set_active(int id); 7.35 +static int set_title(const char *str); 7.36 +static void redisplay(void); 7.37 +static void swap_buffers(void); 7.38 + 7.39 +/* event handling and friends */ 7.40 +static void set_bits(long *mask, long bits); 7.41 +static void clear_bits(long *mask, long bits); 7.42 +static void set_event(int idx, int enable); 7.43 +static int process_events(void); 7.44 + 7.45 + 7.46 static struct wsys_module ws = { 7.47 "x11-glx", 0, 7.48 init, 7.49 @@ -23,14 +46,23 @@ 7.50 get_vidmode, 7.51 create_window, 7.52 close_window, 7.53 + set_active, 7.54 + set_title, 7.55 + redisplay, 7.56 + swap_buffers, 7.57 + set_event, 7.58 + process_events, 7.59 0 7.60 }; 7.61 7.62 static Display *dpy; 7.63 +static Window root; 7.64 static int scr; 7.65 +static Atom xa_wm_prot, xa_wm_del_win; 7.66 static struct window *winlist; 7.67 +static struct window *active_win; 7.68 7.69 - 7.70 +/* this is the only exported function, everything else should be static */ 7.71 void sgl_register_x11(void) 7.72 { 7.73 sgl_register_module(&ws); 7.74 @@ -41,9 +73,14 @@ 7.75 winlist = 0; 7.76 7.77 if(!(dpy = XOpenDisplay(0))) { 7.78 + sgl_log("failed to open X display: %s\n", XDisplayName(0)); 7.79 return -1; 7.80 } 7.81 scr = DefaultScreen(dpy); 7.82 + root = RootWindow(dpy, scr); 7.83 + 7.84 + xa_wm_prot = XInternAtom(dpy, "WM_PROTOCOLS", False); 7.85 + xa_wm_del_win = XInternAtom(dpy, "WM_DELETE_WINDOW", False); 7.86 7.87 return 0; 7.88 } 7.89 @@ -55,25 +92,151 @@ 7.90 winlist = winlist->next; 7.91 7.92 glXDestroyContext(dpy, win->ctx); 7.93 - XCloseWindow(dpy, win->win); 7.94 + XDestroyWindow(dpy, win->win); 7.95 free(win); 7.96 } 7.97 XCloseDisplay(dpy); 7.98 dpy = 0; 7.99 } 7.100 7.101 -static int set_vidmode(int xsz, int ysz, int bpp) 7.102 +static int set_vidmode(int xsz, int ysz) 7.103 { 7.104 /* TODO */ 7.105 + return 0; 7.106 } 7.107 7.108 -static int get_vidmode(int *xsz, int *ysz, int *bpp) 7.109 +static int get_vidmode(int *xsz, int *ysz) 7.110 { 7.111 /* TODO */ 7.112 + return 0; 7.113 } 7.114 7.115 static int create_window(int xsz, int ysz, unsigned int flags) 7.116 { 7.117 + int attr[32]; 7.118 + Window win; 7.119 + GLXContext ctx; 7.120 + XVisualInfo *vis; 7.121 + XClassHint chint; 7.122 + XSetWindowAttributes xattr; 7.123 + unsigned int attr_valid; 7.124 + long evmask; 7.125 + struct window *wnode; 7.126 + 7.127 + if(!(wnode = malloc(sizeof *wnode))) { 7.128 + return -1; 7.129 + } 7.130 + 7.131 + fill_attr(attr, flags); 7.132 + 7.133 + if(!(vis = glXChooseVisual(dpy, scr, attr))) { 7.134 + sgl_log("failed to find a suitable visual\n"); 7.135 + free(wnode); 7.136 + return -1; 7.137 + } 7.138 + print_visual_info(vis); 7.139 + 7.140 + if(!(ctx = glXCreateContext(dpy, vis, 0, True))) { 7.141 + sgl_log("failed to create OpenGL context\n"); 7.142 + XFree(vis); 7.143 + free(wnode); 7.144 + return -1; 7.145 + } 7.146 + 7.147 + xattr.background_pixel = xattr.border_pixel = BlackPixel(dpy, scr); 7.148 + xattr.colormap = XCreateColormap(dpy, root, vis->visual, AllocNone); 7.149 + attr_valid = CWColormap | CWBackPixel | CWBorderPixel; 7.150 + 7.151 + if(!(win = XCreateWindow(dpy, root, 0, 0, xsz, ysz, 0, vis->depth, InputOutput, 7.152 + vis->visual, attr_valid, &xattr))) { 7.153 + sgl_log("failed to create window\n"); 7.154 + glXDestroyContext(dpy, ctx); 7.155 + XFree(vis); 7.156 + free(wnode); 7.157 + return -1; 7.158 + } 7.159 + XFree(vis); 7.160 + 7.161 + evmask = StructureNotifyMask | VisibilityChangeMask; 7.162 + XSelectInput(dpy, win, evmask); 7.163 + 7.164 + XSetWMProtocols(dpy, win, &xa_wm_del_win, 1); 7.165 + 7.166 + set_title("OpenGL/X11"); 7.167 + 7.168 + chint.res_name = chint.res_class = "simplygl"; 7.169 + XSetClassHint(dpy, win, &chint); 7.170 + 7.171 + XMapWindow(dpy, win); 7.172 + 7.173 + wnode->win = win; 7.174 + wnode->ctx = ctx; 7.175 + wnode->evmask = evmask; 7.176 + wnode->next = winlist; 7.177 + winlist->next = wnode; 7.178 + 7.179 + if(!active_win) { 7.180 + set_active(win); 7.181 + } 7.182 + return win; 7.183 +} 7.184 + 7.185 +static void fill_attr(int *attr, unsigned int flags) 7.186 +{ 7.187 + int i = 0; 7.188 + 7.189 + attr[i++] = GLX_RGBA; 7.190 + attr[i++] = GLX_RED_SIZE; attr[i++] = 1; 7.191 + attr[i++] = GLX_GREEN_SIZE; attr[i++] = 1; 7.192 + attr[i++] = GLX_BLUE_SIZE; attr[i++] = 1; 7.193 + 7.194 + if(flags & SGL_DOUBLE) { 7.195 + attr[i++] = GLX_DOUBLEBUFFER; 7.196 + } 7.197 + if(flags & SGL_DEPTH) { 7.198 + attr[i++] = GLX_DEPTH_SIZE; 7.199 + attr[i++] = 1; 7.200 + } 7.201 + if(flags & SGL_STENCIL) { 7.202 + attr[i++] = GLX_STENCIL_SIZE; 7.203 + attr[i++] = 1; 7.204 + } 7.205 + if(flags & SGL_STEREO) { 7.206 + attr[i++] = GLX_STEREO; 7.207 + } 7.208 +#if defined(GLX_VERSION_1_4) || defined(GLX_ARB_multisample) 7.209 + if(flags & SGL_MULTISAMPLE) { 7.210 + attr[i++] = GLX_SAMPLE_BUFFERS_ARB; 7.211 + attr[i++] = 1; 7.212 + attr[i++] = GLX_SAMPLES_ARB; 7.213 + attr[i++] = 1; 7.214 + } 7.215 +#endif /* defined GLX_VERSION_1_4 || GLX_ARB_multisample */ 7.216 + attr[i] = None; 7.217 +} 7.218 + 7.219 +static void print_visual_info(XVisualInfo *vis) 7.220 +{ 7.221 + int rbits, gbits, bbits, zbits, sbits, stereo, aa, samples; 7.222 + 7.223 + glXGetConfig(dpy, vis, GLX_RED_SIZE, &rbits); 7.224 + glXGetConfig(dpy, vis, GLX_GREEN_SIZE, &gbits); 7.225 + glXGetConfig(dpy, vis, GLX_BLUE_SIZE, &bbits); 7.226 + glXGetConfig(dpy, vis, GLX_DEPTH_SIZE, &zbits); 7.227 + glXGetConfig(dpy, vis, GLX_STENCIL_SIZE, &sbits); 7.228 + glXGetConfig(dpy, vis, GLX_STEREO, &stereo); 7.229 +#if defined(GLX_VERSION_1_4) || defined(GLX_ARB_multisample) 7.230 + glXGetConfig(dpy, vis, GLX_SAMPLE_BUFFERS_ARB, &aa); 7.231 + if(aa) { 7.232 + glXGetConfig(dpy, vis, GLX_SAMPLES_ARB, &samples); 7.233 + } else { 7.234 + samples = 1; 7.235 + } 7.236 +#endif /* defined GLX_VERSION_1_4 || GLX_ARB_multisample */ 7.237 + 7.238 + sgl_log("got visual: %d%d%d d:%d s:%d", rbits, gbits, bbits, zbits, sbits); 7.239 + sgl_log(" %s", stereo ? "stereo" : "mono"); 7.240 + sgl_log(" samples/pixel: %d\n", samples); 7.241 } 7.242 7.243 static void close_window(int id) 7.244 @@ -87,7 +250,7 @@ 7.245 while(win) { 7.246 if(win->win == id) { 7.247 glXDestroyContext(dpy, win->ctx); 7.248 - XCloseWindow(dpy, win->win); 7.249 + XDestroyWindow(dpy, win->win); 7.250 prev->next = win->next; 7.251 free(win); 7.252 return; 7.253 @@ -95,3 +258,168 @@ 7.254 win = win->next; 7.255 } 7.256 } 7.257 + 7.258 +static int set_active(int id) 7.259 +{ 7.260 + struct window *win = winlist; 7.261 + 7.262 + while(win) { 7.263 + if(win->win == id) { 7.264 + if(glXMakeCurrent(dpy, win->win, win->ctx) == False) { 7.265 + sgl_log("failed to activate window %d\n", id); 7.266 + return -1; 7.267 + } 7.268 + active_win = win; 7.269 + return 0; 7.270 + } 7.271 + win = win->next; 7.272 + } 7.273 + 7.274 + sgl_log("no such window: %d\n", id); 7.275 + return -1; 7.276 +} 7.277 + 7.278 +static int set_title(const char *str) 7.279 +{ 7.280 + XTextProperty wm_name; 7.281 + 7.282 + if(!str || !active_win) { 7.283 + return -1; 7.284 + } 7.285 + XStringListToTextProperty((char**)&str, 1, &wm_name); 7.286 + XSetWMName(dpy, active_win->win, &wm_name); 7.287 + XSetWMIconName(dpy, active_win->win, &wm_name); 7.288 + XFree(wm_name.value); 7.289 + return 0; 7.290 +} 7.291 + 7.292 +static void redisplay(void) 7.293 +{ 7.294 + active_win->redisp_pending = 1; 7.295 +} 7.296 + 7.297 +static void swap_buffers(void) 7.298 +{ 7.299 + glXSwapBuffers(dpy, active_win->ctx); 7.300 +} 7.301 + 7.302 +static void set_bits(long *mask, long bits) 7.303 +{ 7.304 + *mask |= bits; 7.305 +} 7.306 + 7.307 +static void clear_bits(long *mask, long bits) 7.308 +{ 7.309 + *mask &= ~bits; 7.310 +} 7.311 + 7.312 +static void set_event(int idx, int enable) 7.313 +{ 7.314 + void (*op)(long*, long); 7.315 + op = enable ? set_bits : clear_bits; 7.316 + 7.317 + switch(idx) { 7.318 + case SGL_DISPLAY: 7.319 + op(&active_win->evmask, ExposureMask); 7.320 + break; 7.321 + 7.322 + case SGL_KEYBOARD: 7.323 + op(&active_win->evmask, KeyPressMask | KeyReleaseMask); 7.324 + break; 7.325 + 7.326 + case SGL_MOUSE: 7.327 + op(&active_win->evmask, ButtonPressMask | ButtonReleaseMask); 7.328 + break; 7.329 + 7.330 + case SGL_MOTION: 7.331 + op(&active_win->evmask, ButtonMotionMask); 7.332 + break; 7.333 + 7.334 + case SGL_PASSIVE: 7.335 + op(&active_win->evmask, PointerMotionMask); 7.336 + break; 7.337 + 7.338 + default: 7.339 + return; 7.340 + } 7.341 + 7.342 + XSelectInput(dpy, active_win->win, active_win->evmask); 7.343 +} 7.344 + 7.345 +static int process_events(void) 7.346 +{ 7.347 + XEvent xev; 7.348 + void (*func)(); 7.349 + int state; 7.350 + 7.351 + while(XPending(dpy)) { 7.352 + XNextEvent(dpy, &xev); 7.353 + 7.354 + switch(xev.type) { 7.355 + case Expose: 7.356 + if(xev.xexpose.count == 0) { 7.357 + if((func = sgl_get_callback(SGL_DISPLAY))) { 7.358 + func(); 7.359 + active_win->redisp_pending = 0; 7.360 + } 7.361 + } 7.362 + break; 7.363 + 7.364 + case MotionNotify: 7.365 + if(xev.xmotion.state) { 7.366 + func = sgl_get_callback(SGL_MOTION); 7.367 + } else { 7.368 + func = sgl_get_callback(SGL_PASSIVE); 7.369 + } 7.370 + if(func) { 7.371 + func(xev.xmotion.x, xev.xmotion.y); 7.372 + } 7.373 + break; 7.374 + 7.375 + case ButtonPress: 7.376 + if(1) { 7.377 + state = 1; 7.378 + } else { 7.379 + case ButtonRelease: 7.380 + state = 0; 7.381 + } 7.382 + if((func = sgl_get_callback(SGL_MOUSE))) { 7.383 + int bn = xev.xbutton.button - 1; 7.384 + func(bn, state, xev.xbutton.x, xev.xbutton.y); 7.385 + } 7.386 + break; 7.387 + 7.388 + case KeyPress: 7.389 + if(1) { 7.390 + state = 1; 7.391 + } else { 7.392 + case KeyRelease: 7.393 + state = 0; 7.394 + } 7.395 + if((func = sgl_get_callback(SGL_KEYBOARD))) { 7.396 + KeySym sym = XLookupKeysym(xev.xkey, 0); 7.397 + func(sym, state); 7.398 + /* XXX */ 7.399 + } 7.400 + break; 7.401 + 7.402 + case ConfigureNotify: 7.403 + if((func = sgl_get_callback(SGL_RESHAPE))) { 7.404 + func(xev.xconfigure.width, xev.xconfigure.height); 7.405 + } 7.406 + break; 7.407 + } 7.408 + 7.409 + case ClientMessage: 7.410 + if(xev.xclient.message_type == xa_wm_prot) { 7.411 + if(xev.xclient.data.l[0] == xa_wm_del_win) { 7.412 + close_window(active_win->win); 7.413 + if(!active_win) { 7.414 + return 1; 7.415 + } 7.416 + } 7.417 + } 7.418 + break; 7.419 + } 7.420 + return 0; 7.421 +}