X-Git-Url: http://git.demorecorder.com/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=src%2Fxcb_out.c;h=2494ef34d4f9c465319db7d60de147a0c1946390;hb=5437032c7dee85da99b612a707fa94d012d40282;hp=78ec8ca9cd70e59a5ff60d683a2c0ae72b2afc31;hpb=87905f0579e749ac6d92843780af246160318eca;p=free-sw%2Fxcb%2Flibxcb diff --git a/src/xcb_out.c b/src/xcb_out.c index 78ec8ca..2494ef3 100644 --- a/src/xcb_out.c +++ b/src/xcb_out.c @@ -36,18 +36,6 @@ #include "xcbint.h" #include "extensions/bigreq.h" -static int force_sequence_wrap(XCBConnection *c) -{ - int ret = 1; - if((c->out.request - c->in.request_read) > 65530) - { - pthread_mutex_unlock(&c->iolock); - ret = XCBSync(c, 0); - pthread_mutex_lock(&c->iolock); - } - return ret; -} - /* Public interface */ CARD32 XCBGetMaximumRequestLength(XCBConnection *c) @@ -69,15 +57,22 @@ CARD32 XCBGetMaximumRequestLength(XCBConnection *c) return c->out.maximum_request_length; } -int XCBSendRequest(XCBConnection *c, unsigned int *request, int flags, struct iovec *vector, const XCBProtocolRequest *req) +unsigned int XCBSendRequest(XCBConnection *c, int flags, struct iovec *vector, const XCBProtocolRequest *req) { - int ret; - CARD32 prefix[2]; + static const union { + struct { + CARD8 major; + CARD8 pad; + CARD16 len; + } fields; + CARD32 packet; + } sync = { { /* GetInputFocus */ 43, 0, 1 } }; + unsigned int request; + CARD32 prefix[3] = { 0 }; int veclen = req->count; enum workarounds workaround = WORKAROUND_NONE; assert(c != 0); - assert(request != 0); assert(vector != 0); assert(req->count > 0); @@ -92,8 +87,8 @@ int XCBSendRequest(XCBConnection *c, unsigned int *request, int flags, struct io if(req->ext) { const XCBQueryExtensionRep *extension = XCBGetExtensionData(c, req->ext); - /* TODO: better error handling here, please! */ - assert(extension && extension->present); + if(!(extension && extension->present)) + return 0; ((CARD8 *) vector[0].iov_base)[0] = extension->major_opcode; ((CARD8 *) vector[0].iov_base)[1] = req->opcode; } @@ -125,16 +120,7 @@ int XCBSendRequest(XCBConnection *c, unsigned int *request, int flags, struct io /* set the length field. */ ((CARD16 *) vector[0].iov_base)[1] = shortlen; if(!shortlen) - { - memmove(vector + 1, vector, veclen++ * sizeof(*vector)); - ++veclen; - vector[0].iov_base = prefix; - vector[0].iov_len = sizeof(prefix); - prefix[0] = ((CARD32 *) vector[0].iov_base)[0]; - prefix[1] = ++longlen; - vector[1].iov_base = ((char *) vector[1].iov_base) + sizeof(CARD32); - vector[1].iov_len -= sizeof(CARD32); - } + prefix[2] = ++longlen; } flags &= ~XCB_REQUEST_RAW; @@ -146,23 +132,41 @@ int XCBSendRequest(XCBConnection *c, unsigned int *request, int flags, struct io /* get a sequence number and arrange for delivery. */ pthread_mutex_lock(&c->iolock); - if(req->isvoid && !force_sequence_wrap(c)) - { - pthread_mutex_unlock(&c->iolock); - return -1; - } - /* wait for other writing threads to get out of my way. */ while(c->out.writing) pthread_cond_wait(&c->out.cond, &c->iolock); - *request = ++c->out.request; + if(req->isvoid && c->out.request == c->in.request_expected + (1 << 16) - 2) + { + prefix[0] = sync.packet; + request = ++c->out.request; + _xcb_in_expect_reply(c, request, WORKAROUND_NONE, XCB_REQUEST_DISCARD_REPLY); + c->in.request_expected = c->out.request; + } + + request = ++c->out.request; + assert(request != 0); + _xcb_in_expect_reply(c, request, workaround, flags); + if(!req->isvoid) + c->in.request_expected = c->out.request; - _xcb_in_expect_reply(c, *request, workaround, flags); + if(prefix[0] || prefix[2]) + { + --vector, ++veclen; + if(prefix[2]) + { + prefix[1] = ((CARD32 *) vector[1].iov_base)[0]; + vector[1].iov_base = (CARD32 *) vector[1].iov_base + 1; + vector[1].iov_len -= sizeof(CARD32); + } + vector[0].iov_len = sizeof(CARD32) * (prefix[0] ? 1 : 0 | prefix[2] ? 2 : 0); + vector[0].iov_base = prefix + !prefix[0]; + } - ret = _xcb_out_write_block(c, vector, veclen); + if(!_xcb_out_write_block(c, vector, veclen)) + request = 0; pthread_mutex_unlock(&c->iolock); - return ret; + return request; } int XCBFlush(XCBConnection *c) @@ -237,12 +241,14 @@ int _xcb_out_write_block(XCBConnection *c, struct iovec *vector, size_t count) { memcpy(c->out.queue + c->out.queue_len, vector[0].iov_base, vector[0].iov_len); c->out.queue_len += vector[0].iov_len; + vector[0].iov_base = (char *) vector[0].iov_base + vector[0].iov_len; + vector[0].iov_len = 0; ++vector, --count; } if(!count) return 1; - memmove(vector + 1, vector, count++ * sizeof(struct iovec)); + --vector, ++count; vector[0].iov_base = c->out.queue; vector[0].iov_len = c->out.queue_len; c->out.queue_len = 0;