Mercurial > MadButterfly
diff src/X_supp.c @ 986:c39d14139ca5 refine_backend_if
Implement IO manager interface for X
author | Thinker K.F. Li <thinker@codemud.net> |
---|---|
date | Mon, 22 Nov 2010 00:42:29 +0800 |
parents | 3fe8054457a8 |
children | bf0da8c7d03f |
line wrap: on
line diff
--- a/src/X_supp.c Mon Nov 22 00:42:29 2010 +0800 +++ b/src/X_supp.c Mon Nov 22 00:42:29 2010 +0800 @@ -10,6 +10,7 @@ #include "mb_redraw_man.h" #include "mb_timer.h" #include "mb_X_supp.h" +#include "mb_backend.h" #include "config.h" #ifdef XSHM @@ -36,13 +37,6 @@ }; /* @} */ -#define MAX_MONITORS 200 -typedef struct { - int type; - int fd; - mb_eventcb_t f; - void *arg; -} monitor_t; struct _X_MB_runtime { Display *display; @@ -57,8 +51,7 @@ int w, h; X_kb_info_t kbinfo; - monitor_t monitors[MAX_MONITORS]; - int n_monitor; + mb_IO_man_t *io_man; #ifndef ONLY_MOUSE_MOVE_RAW /* States */ @@ -81,6 +74,163 @@ int mbut_state; /* Button state of last motion event */ }; +/*! \defgroup x_mb_io IO manager for X. + * @{ + */ +#define MAX_MONITORS 200 + +typedef struct { + int type; + int fd; + mb_IO_cb_t cb; + void *data; +} monitor_t; + +struct _X_MB_IO_man { + mb_IO_man_t io_man; + monitor_t monitors[MAX_MONITORS]; + int n_monitor; +}; + +int _x_mb_io_man_reg(struct _mb_IO_man *io_man, + int fd, MB_IO_TYPE type, mb_IO_cb_t cb, void *data); +void _x_mb_io_man_unreg(struct _mb_IO_Man *io_man, int io_hdl); +mb_IO_man_t *_x_mb_io_man_new(void); +void _x_mb_io_man_free(mb_IO_man_t *io_man); + +static mb_IO_factory_t _X_supp_default_io_factory = { + _x_mb_io_man_new, + _x_mb_io_man_free +}; +static mb_IO_factory_t *_io_factory = _X_supp_default_io_factory; + +static struct _X_MB_IO_man _default_io_man = { + {_x_mb_io_man_reg, _x_mb_io_man_unreg}, + , /* monitors */ + 0 /* n_monitor */ +}; + +static mb_IO_man_t * +_x_mb_io_man_new(void) { + return (mb_IO_man_t *)&_default_io_man; +} + +static void +_x_mb_io_man_free(mb_IO_man_t *io_man) { +} + +static int +_x_mb_io_man_reg(struct _mb_IO_man *io_man, + int fd, MB_IO_TYPE type, mb_IO_cb_t cb, void *data) { + struct _x_mb_io_man *xmb_io_man = (struct _x_mb_io_man *)io_man; + int i; + + for(i = 0; i < xmb_io_man->n_monitor; i++) { + if (xmb_io_man->monitors[i].type == MB_IO_DUMMY) + break; + } + if (i == MAX_MONITORS) + return ERR; + + xmb_io_man->monitors[i].type = type; + xmb_io_man->monitors[i].fd = fd; + xmb_io_man->monitors[i].cb = cb; + xmb_io_man->monitors[i].data = data; + i++; + if(i > xmb_io_man->n_monitor) + xmb_io_man->n_monitor = i; + return i - 1; +} + +static void +_x_mb_io_man_unreg(struct _mb_IO_man *io_man, int io_hdl) { + struct _x_mb_io_man *xmb_io_man = (struct _x_mb_io_man *)io_man; + + ASSERT(io_hdl < xmb_io_man->n_monitor); + xmb_io_man->monitors[io_hdl].type = MB_IO_DUMMY; +} + +/*! \brief Handle connection coming data and timeout of timers. + * + * \param display is a Display returned by XOpenDisplay(). + * \param rdman is a redraw manager. + * \param tman is a timer manager. + * + * The display is managed by specified rdman and tman. rdman draws + * on the display, and tman trigger actions according timers. + */ +static void +_x_mb_handle_connection(struct _mb_IO_man *io_man) { + X_MB_runtime_t *rt = (X_MB_runtime_t *) be; + mb_tman_t *tman = rt->tman; + int fd; + mb_timeval_t now, tmo; + struct timeval tv; + fd_set rfds, wfds; + int nfds = 0; + int r, r1,i; + + handle_x_event(rt); + + while(1) { + FD_ZERO(&rfds); + FD_ZERO(&wfds); + for(i = 0; i < io_man->n_monitor; i++) { + if(io_man->monitors[i].type == MB_IO_R || + io_man->monitors[i].type == MB_IO_RW) { + FD_SET(io_man->monitors[i].fd, &rfds); + if(nfds <= io_man->monitors[i].fd) + nfds = io_man->monitors[i].fd + 1; + } + if(io_man->monitors[i].type == MB_IO_W || + io_man->monitors[i].type == MB_IO_RW) { + FD_SET(io_man->monitors[i].fd, &wfds); + if(nfds <= io_man->monitors[i].fd) + nfds = io_man->monitors[i].fd + 1; + } + } + + get_now(&now); + r = mb_tman_next_timeout(tman, &now, &tmo); + + if(r == 0) { + tv.tv_sec = MB_TIMEVAL_SEC(&tmo); + tv.tv_usec = MB_TIMEVAL_USEC(&tmo); + r1 = select(nfds, &rfds, NULL, NULL, &tv); + } else + r1 = select(nfds, &rfds, NULL, NULL, NULL); + + if(r1 == -1) { + perror("select"); + break; + } + + if(r1 == 0) { + get_now(&now); + mb_tman_handle_timeout(tman, &now); + } else { + for(i = 0; i < io_man->n_monitor; i++) { + if(io_man->monitors[i].type == MB_IO_R || + io_man->monitors[i].type == MB_IO_RW) { + if(FD_ISSET(io_man->monitors[i].fd, &rfds)) + ioman->monitors[i].cb(io_man->monitors[i].fd, + MB_IO_R, + rt->monitors[i].data); + } + if(io_man->monitors[i].type == MB_IO_W || + io_man->monitors[i].type == MB_IO_RW) { + if(FD_ISSET(io_man->monitors[i].fd, &wfds)) + io_man->monitors[i].cb(io_man->monitors[i].fd, + MB_IO_W, + io_man->monitors[i].data); + } + } + } + } +} + +/* @} */ + #ifdef XSHM static void XSHM_update(X_MB_runtime_t *xmb_rt) { @@ -167,7 +317,6 @@ subject_notify(kbinfo->kbevents, &event.event); } - /* @} */ static unsigned int get_button_state(unsigned int state) { @@ -498,9 +647,6 @@ } } -#define ERR -1 -#define OK 0 - static int X_init_connection(const char *display_name, int w, int h, Display **displayp, @@ -954,10 +1100,12 @@ } mb_backend_t backend = { X_MB_new, X_MB_new_with_window, + X_MB_free, X_MB_add_event, X_MB_remove_event, X_MB_handle_connection, + X_MB_kbevents, X_MB_rdman, X_MB_tman,