X-Git-Url: http://git.demorecorder.com/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=src%2Fxcb_in.c;h=fd6c2efd216491b6aa5f36a10949bed9e34c716f;hb=c7c5b710f2cc0782412c9e159986c96b52aa0d02;hp=b161e937527e32ba358b4c9ae062f2a0643e5917;hpb=7983bf0fbdc2725403f9db6154d0f5bc944040e5;p=free-sw%2Fxcb%2Flibxcb diff --git a/src/xcb_in.c b/src/xcb_in.c index b161e93..fd6c2ef 100644 --- a/src/xcb_in.c +++ b/src/xcb_in.c @@ -60,6 +60,23 @@ struct event_list { struct event_list *next; }; +struct xcb_special_event { + + struct xcb_special_event *next; + + /* Match XGE events for the specific extension and event ID (the + * first 32 bit word after evtype) + */ + uint8_t extension; + uint32_t eid; + uint32_t *stamp; + + struct event_list *events; + struct event_list **events_tail; + + pthread_cond_t special_event_cond; +}; + struct reply_list { void *reply; struct reply_list *next; @@ -104,6 +121,46 @@ static int read_fds(xcb_connection_t *c, int *fds, int nfd) } #endif +typedef struct xcb_ge_special_event_t { + uint8_t response_type; /**< */ + uint8_t extension; /**< */ + uint16_t sequence; /**< */ + uint32_t length; /**< */ + uint16_t evtype; /**< */ + uint8_t pad0[2]; /**< */ + uint32_t eid; /**< */ + uint8_t pad1[16]; /**< */ +} xcb_ge_special_event_t; + +static int event_special(xcb_connection_t *c, + struct event_list *event) +{ + struct xcb_special_event *special_event; + struct xcb_ge_special_event_t *ges = (void *) event->event; + + /* Special events are always XGE events */ + if ((ges->response_type & 0x7f) != XCB_XGE_EVENT) + return 0; + + for (special_event = c->in.special_events; + special_event; + special_event = special_event->next) + { + if (ges->extension == special_event->extension && + ges->eid == special_event->eid) + { + *special_event->events_tail = event; + special_event->events_tail = &event->next; + if (special_event->stamp) + ++(*special_event->stamp); + pthread_cond_signal(&special_event->special_event_cond); + return 1; + } + } + + return 0; +} + static int read_packet(xcb_connection_t *c) { xcb_generic_reply_t genrep; @@ -268,9 +325,12 @@ static int read_packet(xcb_connection_t *c) } event->event = buf; event->next = 0; - *c->in.events_tail = event; - c->in.events_tail = &event->next; - pthread_cond_signal(&c->in.event_cond); + + if (!event_special(c, event)) { + *c->in.events_tail = event; + c->in.events_tail = &event->next; + pthread_cond_signal(&c->in.event_cond); + } return 1; /* I have something for you... */ } @@ -661,17 +721,21 @@ xcb_generic_event_t *xcb_wait_for_special_event(xcb_connection_t *c, xcb_special_event_t * xcb_register_for_special_xge(xcb_connection_t *c, - uint8_t extension, + xcb_extension_t *ext, uint32_t eid, uint32_t *stamp) { xcb_special_event_t *se; + const xcb_query_extension_reply_t *ext_reply; if(c->has_error) return NULL; + ext_reply = xcb_get_extension_data(c, ext); + if (!ext_reply) + return NULL; pthread_mutex_lock(&c->iolock); for (se = c->in.special_events; se; se = se->next) { - if (se->extension == extension && + if (se->extension == ext_reply->major_opcode && se->eid == eid) { pthread_mutex_unlock(&c->iolock); return NULL; @@ -683,7 +747,7 @@ xcb_register_for_special_xge(xcb_connection_t *c, return NULL; } - se->extension = extension; + se->extension = ext_reply->major_opcode; se->eid = eid; se->events = NULL; @@ -694,7 +758,6 @@ xcb_register_for_special_xge(xcb_connection_t *c, se->next = c->in.special_events; c->in.special_events = se; - pthread_mutex_unlock(&c->iolock); return se; } @@ -706,6 +769,9 @@ xcb_unregister_for_special_event(xcb_connection_t *c, xcb_special_event_t *s, **prev; struct event_list *events, *next; + if (!se) + return; + if (c->has_error) return; @@ -822,17 +888,17 @@ int _xcb_in_read(xcb_connection_t *c) .iov_base = c->in.queue + c->in.queue_len, .iov_len = sizeof(c->in.queue) - c->in.queue_len, }; - struct { - struct cmsghdr cmsghdr; - int fd[XCB_MAX_PASS_FD]; - } fds; + union { + struct cmsghdr cmsghdr; + char buf[CMSG_SPACE(XCB_MAX_PASS_FD * sizeof(int))]; + } cmsgbuf; struct msghdr msg = { .msg_name = NULL, .msg_namelen = 0, .msg_iov = &iov, .msg_iovlen = 1, - .msg_control = &fds, - .msg_controllen = sizeof (struct cmsghdr) + sizeof(int) * (XCB_MAX_PASS_FD - c->in.in_fd.nfd), + .msg_control = cmsgbuf.buf, + .msg_controllen = CMSG_SPACE(sizeof(int) * (XCB_MAX_PASS_FD - c->in.in_fd.nfd)), }; n = recvmsg(c->fd, &msg, 0); @@ -850,15 +916,12 @@ int _xcb_in_read(xcb_connection_t *c) #endif if(n > 0) { #if HAVE_SENDMSG - if (msg.msg_controllen > sizeof (struct cmsghdr)) - { - if (fds.cmsghdr.cmsg_level == SOL_SOCKET && - fds.cmsghdr.cmsg_type == SCM_RIGHTS) - { - int nfd = (msg.msg_controllen - sizeof (struct cmsghdr)) / sizeof (int); - memmove(&c->in.in_fd.fd[c->in.in_fd.nfd], - fds.fd, - nfd); + struct cmsghdr *hdr; + + for (hdr = CMSG_FIRSTHDR(&msg); hdr; hdr = CMSG_NXTHDR(&msg, hdr)) { + if (hdr->cmsg_level == SOL_SOCKET && hdr->cmsg_type == SCM_RIGHTS) { + int nfd = (hdr->cmsg_len - CMSG_LEN(0)) / sizeof (int); + memcpy(&c->in.in_fd.fd[c->in.in_fd.nfd], CMSG_DATA(hdr), nfd * sizeof (int)); c->in.in_fd.nfd += nfd; } }