/* cairo - a vector graphics library with display and print output * * Copyright © 2009 Intel Corporation * * This library is free software; you can redistribute it and/or * modify it either under the terms of the GNU Lesser General Public * License version 2.1 as published by the Free Software Foundation * (the "LGPL") or, at your option, under the terms of the Mozilla * Public License Version 1.1 (the "MPL"). If you do not alter this * notice, a recipient may use your version of this file under either * the MPL or the LGPL. * * You should have received a copy of the LGPL along with this library * in the file COPYING-LGPL-2.1; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Suite 500, Boston, MA 02110-1335, USA * You should have received a copy of the MPL along with this library * in the file COPYING-MPL-1.1 * * The contents of this file are subject to the Mozilla Public License * Version 1.1 (the "License"); you may not use this file except in * compliance with the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY * OF ANY KIND, either express or implied. See the LGPL or the MPL for * the specific language governing rights and limitations. * * Contributor(s): * Chris Wilson */ #include "cairoint.h" #include "cairo-xcb-private.h" #include xcb_pixmap_t _cairo_xcb_connection_create_pixmap (cairo_xcb_connection_t *connection, uint8_t depth, xcb_drawable_t drawable, uint16_t width, uint16_t height) { struct { uint8_t req; uint8_t depth; uint16_t len; uint32_t pixmap; uint32_t drawable; uint16_t width, height; } req; struct iovec vec[1]; req.req = 53; req.depth = depth; req.len = sizeof (req) >> 2; req.pixmap = _cairo_xcb_connection_get_xid (connection); req.drawable = drawable; req.width = width; req.height = height; vec[0].iov_base = &req; vec[0].iov_len = sizeof (req); _cairo_xcb_connection_write (connection, vec, 1); return req.pixmap; } void _cairo_xcb_connection_free_pixmap (cairo_xcb_connection_t *connection, xcb_pixmap_t pixmap) { struct { uint8_t req; uint8_t pad; uint16_t len; uint32_t pixmap; } req; struct iovec vec[1]; req.req = 54; req.len = sizeof (req) >> 2; req.pixmap = pixmap; vec[0].iov_base = &req; vec[0].iov_len = sizeof (req); _cairo_xcb_connection_write (connection, vec, 1); _cairo_xcb_connection_put_xid (connection, pixmap); } xcb_gcontext_t _cairo_xcb_connection_create_gc (cairo_xcb_connection_t *connection, xcb_drawable_t drawable, uint32_t value_mask, uint32_t *values) { struct { uint8_t req; uint8_t pad; uint16_t len; uint32_t gc; uint32_t drawable; uint32_t mask; } req; struct iovec vec[2]; int len = _cairo_popcount (value_mask) * 4; req.req = 55; req.len = (sizeof (req) + len) >> 2; req.gc = _cairo_xcb_connection_get_xid (connection); req.drawable = drawable; req.mask = value_mask; vec[0].iov_base = &req; vec[0].iov_len = sizeof (req); vec[1].iov_base = values; vec[1].iov_len = len; _cairo_xcb_connection_write (connection, vec, 2); return req.gc; } void _cairo_xcb_connection_free_gc (cairo_xcb_connection_t *connection, xcb_gcontext_t gc) { struct { uint8_t req; uint8_t pad; uint16_t len; uint32_t gc; } req; struct iovec vec[1]; req.req = 60; req.len = sizeof (req) >> 2; req.gc = gc; vec[0].iov_base = &req; vec[0].iov_len = sizeof (req); _cairo_xcb_connection_write (connection, vec, 1); _cairo_xcb_connection_put_xid (connection, gc); } void _cairo_xcb_connection_change_gc (cairo_xcb_connection_t *connection, xcb_gcontext_t gc, uint32_t value_mask, uint32_t *values) { struct { uint8_t req; uint8_t pad; uint16_t len; uint32_t gc; uint32_t mask; } req; struct iovec vec[2]; int len = _cairo_popcount (value_mask) * 4; req.req = 56; req.len = (sizeof (req) + len) >> 2; req.gc = gc; req.mask = value_mask; vec[0].iov_base = &req; vec[0].iov_len = sizeof (req); vec[1].iov_base = values; vec[1].iov_len = len; _cairo_xcb_connection_write (connection, vec, 2); } void _cairo_xcb_connection_copy_area (cairo_xcb_connection_t *connection, xcb_drawable_t src, xcb_drawable_t dst, xcb_gcontext_t gc, int16_t src_x, int16_t src_y, int16_t dst_x, int16_t dst_y, uint16_t width, uint16_t height) { struct { uint8_t req; uint8_t pad; uint16_t len; uint32_t src; uint32_t dst; uint32_t gc; int16_t src_x; int16_t src_y; int16_t dst_x; int16_t dst_y; uint16_t width; uint16_t height; } req; struct iovec vec[1]; req.req = 62; req.len = sizeof (req) >> 2; req.src = src; req.dst = dst; req.gc = gc; req.src_x = src_x; req.src_y = src_y; req.dst_x = dst_x; req.dst_y = dst_y; req.width = width; req.height = height; vec[0].iov_base = &req; vec[0].iov_len = sizeof (req); _cairo_xcb_connection_write (connection, vec, 1); } void _cairo_xcb_connection_poly_fill_rectangle (cairo_xcb_connection_t *connection, xcb_drawable_t dst, xcb_gcontext_t gc, uint32_t num_rectangles, xcb_rectangle_t *rectangles) { struct { uint8_t req; uint8_t pad; uint16_t len; uint32_t dst; uint32_t gc; } req; struct iovec vec[2]; req.req = 70; req.len = (sizeof (req) + num_rectangles * sizeof (xcb_rectangle_t)) >> 2; req.dst = dst; req.gc = gc; vec[0].iov_base = &req; vec[0].iov_len = sizeof (req); vec[1].iov_base = rectangles; vec[1].iov_len = num_rectangles * sizeof (xcb_rectangle_t); _cairo_xcb_connection_write (connection, vec, 2); } void _cairo_xcb_connection_put_image (cairo_xcb_connection_t *connection, xcb_drawable_t dst, xcb_gcontext_t gc, uint16_t width, uint16_t height, int16_t dst_x, int16_t dst_y, uint8_t depth, uint32_t stride, void *data) { struct { uint8_t req; uint8_t format; uint16_t len; uint32_t dst; uint32_t gc; uint16_t width; uint16_t height; int16_t dst_x; int16_t dst_y; uint8_t left; uint8_t depth; uint16_t pad; } req; struct iovec vec[3]; uint32_t prefix[2]; uint32_t length = height * stride; uint32_t len = (sizeof (req) + length) >> 2; req.req = 72; req.format = XCB_IMAGE_FORMAT_Z_PIXMAP; req.len = 0; req.dst = dst; req.gc = gc; req.width = width; req.height = height; req.dst_x = dst_x; req.dst_y = dst_y; req.left = 0; req.depth = depth; if (len < connection->root->maximum_request_length) { req.len = len; vec[0].iov_base = &req; vec[0].iov_len = sizeof (req); vec[1].iov_base = data; vec[1].iov_len = length; _cairo_xcb_connection_write (connection, vec, 2); } else if (len < connection->maximum_request_length) { prefix[0] = *(uint32_t *) &req; prefix[1] = len + 1; vec[0].iov_base = prefix; vec[0].iov_len = sizeof (prefix); vec[1].iov_base = (uint32_t *) &req + 1; vec[1].iov_len = sizeof (req) - 4; vec[2].iov_base = data; vec[2].iov_len = length; _cairo_xcb_connection_write (connection, vec, 3); } else { int rows; rows = (connection->maximum_request_length - sizeof (req) - 4) / stride; if (rows > 0) { do { if (rows > height) rows = height; length = rows * stride; len = (sizeof (req) + 4 + length) >> 2; req.height = rows; prefix[0] = *(uint32_t *) &req; prefix[1] = len; vec[0].iov_base = prefix; vec[0].iov_len = sizeof (prefix); vec[1].iov_base = (uint32_t *) &req + 1; vec[1].iov_len = sizeof (req) - 4; vec[2].iov_base = data; vec[2].iov_len = length; /* note may modify vec */ _cairo_xcb_connection_write (connection, vec, 3); height -= rows; req.dst_y += rows; data = (char *) data + length; } while (height); } else { ASSERT_NOT_REACHED; } } } void _cairo_xcb_connection_put_subimage (cairo_xcb_connection_t *connection, xcb_drawable_t dst, xcb_gcontext_t gc, int16_t src_x, int16_t src_y, uint16_t width, uint16_t height, uint16_t cpp, uint16_t stride, int16_t dst_x, int16_t dst_y, uint8_t depth, void *_data) { struct { uint8_t req; uint8_t format; uint16_t len; uint32_t dst; uint32_t gc; uint16_t width; uint16_t height; int16_t dst_x; int16_t dst_y; uint8_t left; uint8_t depth; uint16_t pad; } req; struct iovec vec_stack[CAIRO_STACK_ARRAY_LENGTH (struct iovec)]; struct iovec *vec = vec_stack; uint32_t prefix[2]; uint32_t len = (sizeof (req) + cpp*width*height) >> 2; uint8_t *data = _data; int n; req.req = 72; req.format = XCB_IMAGE_FORMAT_Z_PIXMAP; req.len = 0; req.dst = dst; req.gc = gc; req.width = width; req.height = height; req.dst_x = dst_x; req.dst_y = dst_y; req.left = 0; req.depth = depth; if (height + 2 > ARRAY_LENGTH (vec_stack)) { vec = _cairo_malloc_ab (height+2, sizeof (struct iovec)); if (unlikely (vec == NULL)) { /* XXX loop over ARRAY_LENGTH (vec_stack) */ return; } } data += src_y * stride + src_x * cpp; if (len < connection->root->maximum_request_length) { req.len = len; vec[0].iov_base = &req; vec[0].iov_len = sizeof (req); n = 1; } else if (len < connection->maximum_request_length) { prefix[0] = *(uint32_t *) &req; prefix[1] = len + 1; vec[0].iov_base = prefix; vec[0].iov_len = sizeof (prefix); vec[1].iov_base = (uint32_t *) &req + 1; vec[1].iov_len = sizeof (req) - 4; n = 2; } else { ASSERT_NOT_REACHED; } while (height--) { vec[n].iov_base = data; vec[n].iov_len = cpp * width; data += stride; n++; } _cairo_xcb_connection_write (connection, vec, n); if (vec != vec_stack) free (vec); } cairo_status_t _cairo_xcb_connection_get_image (cairo_xcb_connection_t *connection, xcb_drawable_t src, int16_t src_x, int16_t src_y, uint16_t width, uint16_t height, xcb_get_image_reply_t **reply) { xcb_generic_error_t *error; cairo_status_t status; *reply = xcb_get_image_reply (connection->xcb_connection, xcb_get_image (connection->xcb_connection, XCB_IMAGE_FORMAT_Z_PIXMAP, src, src_x, src_y, width, height, (uint32_t) -1), &error); if (error) { free (error); if (*reply) free (*reply); *reply = NULL; } status = _cairo_xcb_connection_take_socket (connection); if (unlikely (status)) { if (*reply) free (*reply); *reply = NULL; } return status; }