From: Chris Wilson <chris@chris-wilson.co.uk> Date: Wed, 18 Apr 2007 17:34:44 +0000 (+0100) Subject: Cache freed GCs X-Git-Url: http://gitweb.freedesktop.org/?p=users/ickle/cairo;a=commitdiff;h=3126c3948691cf7a6c0fc0f9cc6a7eca43c3553b Cache freed GCs Maintain a cache of freed GCs, one for each used depth. --- --- a/src/Makefile.am +++ b/src/Makefile.am @@ -46,7 +46,11 @@ endif if CAIRO_HAS_XLIB_SURFACE libcairo_xlib_headers = cairo-xlib.h cairo-xlib-xrender.h -libcairo_xlib_sources = cairo-xlib-surface.c cairo-xlib-screen.c cairo-xlib-private.h cairo-xlib-test.h +libcairo_xlib_sources = cairo-xlib-surface.c \ + cairo-xlib-display.c \ + cairo-xlib-screen.c \ + cairo-xlib-private.h \ + cairo-xlib-test.h backend_pkgconfigs += cairo-xlib.pc endif --- a/src/cairo-clip.c +++ b/src/cairo-clip.c @@ -404,7 +404,8 @@ _cairo_clip_intersect_mask (cairo_clip_t CAIRO_CONTENT_ALPHA, surface_rect.width, surface_rect.height, - CAIRO_COLOR_WHITE); + CAIRO_COLOR_WHITE, + NULL); if (surface->status) return CAIRO_STATUS_NO_MEMORY; --- a/src/cairo-debug.c +++ b/src/cairo-debug.c @@ -59,10 +59,6 @@ void cairo_debug_reset_static_data (void) { -#if CAIRO_HAS_XLIB_SURFACE - _cairo_xlib_screen_reset_static_data (); -#endif - _cairo_font_reset_static_data (); #if CAIRO_HAS_FT_FONT --- a/src/cairo-directfb-surface.c +++ b/src/cairo-directfb-surface.c @@ -1511,6 +1511,16 @@ _cairo_directfb_surface_show_glyphs ( vo #endif /* DFB_SHOW_GLYPHS */ +static cairo_bool_t +_cairo_directfb_surface_is_compatible (void *surface_a, + void *surface_b) +{ + cairo_directfb_surface_t *a = (cairo_directfb_surface_t *) surface_a; + cairo_directfb_surface_t *b = (cairo_directfb_surface_t *) surface_b; + + return a->dfb == b->dfb; +} + static cairo_surface_backend_t cairo_directfb_surface_backend = { CAIRO_SURFACE_TYPE_DIRECTFB, /*type*/ _cairo_directfb_surface_create_similar,/*create_similar*/ @@ -1560,7 +1570,8 @@ static cairo_surface_backend_t cairo_dir #else NULL, /* show_glyphs */ #endif - NULL /* snapshot */ + NULL, /* snapshot */ + _cairo_directfb_is_compatible }; --- a/src/cairo-glitz-surface.c +++ b/src/cairo-glitz-surface.c @@ -1002,7 +1002,8 @@ _cairo_glitz_surface_fill_rectangles (vo _cairo_surface_create_similar_solid (&dst->base, CAIRO_CONTENT_COLOR_ALPHA, 1, 1, - (cairo_color_t *) color); + (cairo_color_t *) color, + NULL); if (src->base.status) return CAIRO_STATUS_NO_MEMORY; @@ -2171,6 +2172,19 @@ _cairo_glitz_surface_flush (void *abstra return CAIRO_STATUS_SUCCESS; } +static cairo_bool_t +_cairo_glitz_surface_is_compatible (void *surface_a, + void *surface_b) +{ + cairo_glitz_surface_t *a = (cairo_glitz_surface_t *) surface_a; + cairo_glitz_surface_t *b = (cairo_glitz_surface_t *) surface_b; + + glitz_drawable_t *drawable_a = glitz_surface_get_drawable (a->surface); + glitz_drawable_t *drawable_b = glitz_surface_get_drawable (b->surface); + + return drawable_a == drawable_b; +} + static const cairo_surface_backend_t cairo_glitz_surface_backend = { CAIRO_SURFACE_TYPE_GLITZ, _cairo_glitz_surface_create_similar, @@ -2193,7 +2207,16 @@ static const cairo_surface_backend_t cai _cairo_glitz_surface_flush, NULL, /* mark_dirty_rectangle */ _cairo_glitz_surface_scaled_font_fini, - _cairo_glitz_surface_scaled_glyph_fini + _cairo_glitz_surface_scaled_glyph_fini, + + NULL, /* paint */ + NULL, /* mask */ + NULL, /* stroke */ + NULL, /* fill */ + NULL, /* show_glyphs */ + + NULL, /* snapshot */ + _cairo_glitz_surface_is_compatible }; static const cairo_surface_backend_t * --- a/src/cairo-mutex-list-private.h +++ b/src/cairo-mutex-list-private.h @@ -34,7 +34,8 @@ #ifndef CAIRO_MUTEX_LIST_PRIVATE_H #define CAIRO_MUTEX_LIST_PRIVATE_H -CAIRO_MUTEX_DECLARE (_cairo_pattern_solid_cache_lock); +CAIRO_MUTEX_DECLARE (_cairo_pattern_solid_pattern_cache_lock); +CAIRO_MUTEX_DECLARE (_cairo_pattern_solid_surface_cache_lock); CAIRO_MUTEX_DECLARE (_cairo_font_face_mutex); CAIRO_MUTEX_DECLARE (_cairo_scaled_font_map_mutex); @@ -44,7 +45,7 @@ CAIRO_MUTEX_DECLARE (_cairo_ft_unscaled_ #endif #if CAIRO_HAS_XLIB_SURFACE -CAIRO_MUTEX_DECLARE (_cairo_xlib_screen_mutex); +CAIRO_MUTEX_DECLARE (_cairo_xlib_display_mutex); #endif #endif --- a/src/cairo-pattern.c +++ b/src/cairo-pattern.c @@ -282,7 +282,7 @@ _cairo_pattern_create_solid (const cairo { cairo_solid_pattern_t *pattern = NULL; - CAIRO_MUTEX_LOCK (_cairo_pattern_solid_cache_lock); + CAIRO_MUTEX_LOCK (_cairo_pattern_solid_pattern_cache_lock); if (solid_pattern_cache.size) { int i = --solid_pattern_cache.size % @@ -291,7 +291,7 @@ _cairo_pattern_create_solid (const cairo solid_pattern_cache.patterns[i] = NULL; } - CAIRO_MUTEX_UNLOCK (_cairo_pattern_solid_cache_lock); + CAIRO_MUTEX_UNLOCK (_cairo_pattern_solid_pattern_cache_lock); if (pattern == NULL) { /* None cached, need to create a new pattern. */ @@ -306,12 +306,12 @@ _cairo_pattern_create_solid (const cairo return &pattern->base; } -void -_cairo_pattern_reset_static_data (void) +static void +_cairo_pattern_reset_solid_pattern_cache (void) { int i; - CAIRO_MUTEX_LOCK (_cairo_pattern_solid_cache_lock); + CAIRO_MUTEX_LOCK (_cairo_pattern_solid_pattern_cache_lock); for (i = 0; i < MIN (ARRAY_LENGTH (solid_pattern_cache.patterns), solid_pattern_cache.size); i++) { free (solid_pattern_cache.patterns[i]); @@ -319,7 +319,7 @@ _cairo_pattern_reset_static_data (void) } solid_pattern_cache.size = 0; - CAIRO_MUTEX_UNLOCK (_cairo_pattern_solid_cache_lock); + CAIRO_MUTEX_UNLOCK (_cairo_pattern_solid_pattern_cache_lock); } static const cairo_pattern_t * @@ -630,7 +630,7 @@ cairo_pattern_destroy (cairo_pattern_t * if (pattern->type == CAIRO_PATTERN_TYPE_SOLID) { int i; - CAIRO_MUTEX_LOCK (_cairo_pattern_solid_cache_lock); + CAIRO_MUTEX_LOCK (_cairo_pattern_solid_pattern_cache_lock); i = solid_pattern_cache.size++ % ARRAY_LENGTH (solid_pattern_cache.patterns); @@ -640,7 +640,7 @@ cairo_pattern_destroy (cairo_pattern_t * solid_pattern_cache.patterns[i] = (cairo_solid_pattern_t *) pattern; - CAIRO_MUTEX_UNLOCK (_cairo_pattern_solid_cache_lock); + CAIRO_MUTEX_UNLOCK (_cairo_pattern_solid_pattern_cache_lock); } else { free (pattern); } @@ -1235,6 +1235,17 @@ _cairo_pattern_acquire_surface_for_gradi return status; } +/* We maintain a small cache here, because we don't want to constantly + * recreate surfaces for simple solid colors. */ +#define MAX_SURFACE_CACHE_SIZE 16 +static struct { + struct { + cairo_color_t color; + cairo_surface_t *surface; + } cache[MAX_SURFACE_CACHE_SIZE]; + int size; +} solid_surface_cache; + static cairo_int_status_t _cairo_pattern_acquire_surface_for_solid (cairo_solid_pattern_t *pattern, cairo_surface_t *dst, @@ -1245,12 +1256,54 @@ _cairo_pattern_acquire_surface_for_solid cairo_surface_t **out, cairo_surface_attributes_t *attribs) { - *out = _cairo_surface_create_similar_solid (dst, - CAIRO_CONTENT_COLOR_ALPHA, - 1, 1, - &pattern->color); - if ((*out)->status) - return CAIRO_STATUS_NO_MEMORY; + static int i; + + cairo_surface_t *surface; + cairo_status_t status; + + CAIRO_MUTEX_LOCK (_cairo_pattern_solid_surface_cache_lock); + + /* Check cache first */ + if (i < solid_surface_cache.size && + _cairo_color_equal (&solid_surface_cache.cache[i].color, + &pattern->color) && + _cairo_surface_is_compatible (solid_surface_cache.cache[i].surface, dst)) + goto DONE; + + for (i = 0 ; i < solid_surface_cache.size; i++) { + if (_cairo_color_equal (&solid_surface_cache.cache[i].color, + &pattern->color) && + _cairo_surface_is_compatible (solid_surface_cache.cache[i].surface, dst)) + goto DONE; + } + + /* Not cached, need to create new */ + surface = _cairo_surface_create_similar_solid (dst, + CAIRO_CONTENT_COLOR_ALPHA, + 1, 1, + &pattern->color, + pattern); + assert (_cairo_surface_is_compatible (surface, dst)); + if (surface->status) { + status = surface->status; + goto UNLOCK; + } + + /* Cache new */ + if (solid_surface_cache.size < MAX_SURFACE_CACHE_SIZE) { + solid_surface_cache.size++; + } else { + i = rand () % MAX_SURFACE_CACHE_SIZE; + + /* Evict old */ + cairo_surface_destroy (solid_surface_cache.cache[i].surface); + } + + solid_surface_cache.cache[i].color = pattern->color; + solid_surface_cache.cache[i].surface = surface; + +DONE: + *out = cairo_surface_reference (solid_surface_cache.cache[i].surface); attribs->x_offset = attribs->y_offset = 0; cairo_matrix_init_identity (&attribs->matrix); @@ -1258,7 +1311,26 @@ _cairo_pattern_acquire_surface_for_solid attribs->filter = CAIRO_FILTER_NEAREST; attribs->acquired = FALSE; - return CAIRO_STATUS_SUCCESS; + status = CAIRO_STATUS_SUCCESS; + +UNLOCK: + CAIRO_MUTEX_UNLOCK (_cairo_pattern_solid_surface_cache_lock); + + return status; +} + +static void +_cairo_pattern_reset_solid_surface_cache (void) +{ + int i; + + CAIRO_MUTEX_LOCK (_cairo_pattern_solid_surface_cache_lock); + + for (i = 0; i < solid_surface_cache.size; i++) + cairo_surface_destroy (solid_surface_cache.cache[i].surface); + solid_surface_cache.size = 0; + + CAIRO_MUTEX_UNLOCK (_cairo_pattern_solid_surface_cache_lock); } /** @@ -2049,3 +2121,10 @@ cairo_pattern_get_radial_circles (cairo_ return CAIRO_STATUS_SUCCESS; } + +void +_cairo_pattern_reset_static_data (void) +{ + _cairo_pattern_reset_solid_pattern_cache (); + _cairo_pattern_reset_solid_surface_cache (); +} --- a/src/cairo-surface.c +++ b/src/cairo-surface.c @@ -288,7 +288,8 @@ cairo_surface_create_similar (cairo_surf return _cairo_surface_create_similar_solid (other, content, width, height, - CAIRO_COLOR_TRANSPARENT); + CAIRO_COLOR_TRANSPARENT, + NULL); } slim_hidden_def (cairo_surface_create_similar); @@ -297,7 +298,8 @@ _cairo_surface_create_similar_solid (cai cairo_content_t content, int width, int height, - const cairo_color_t *color) + const cairo_color_t *color, + cairo_pattern_t *pattern) { cairo_status_t status; cairo_surface_t *surface; @@ -310,19 +312,23 @@ _cairo_surface_create_similar_solid (cai return (cairo_surface_t*) &_cairo_surface_nil; } - source = _cairo_pattern_create_solid (color); - if (source->status) { - cairo_surface_destroy (surface); - _cairo_error (CAIRO_STATUS_NO_MEMORY); - return (cairo_surface_t*) &_cairo_surface_nil; - } + if (pattern == NULL) { + source = _cairo_pattern_create_solid (color); + if (source->status) { + cairo_surface_destroy (surface); + _cairo_error (CAIRO_STATUS_NO_MEMORY); + return (cairo_surface_t*) &_cairo_surface_nil; + } + } else + source = pattern; status = _cairo_surface_paint (surface, color == CAIRO_COLOR_TRANSPARENT ? CAIRO_OPERATOR_CLEAR : CAIRO_OPERATOR_SOURCE, source); - cairo_pattern_destroy (source); + if (source != pattern) + cairo_pattern_destroy (source); if (status) { cairo_surface_destroy (surface); @@ -1056,6 +1062,32 @@ _cairo_surface_snapshot (cairo_surface_t return _cairo_surface_fallback_snapshot (surface); } +/** + * _cairo_surface_is_compatible + * @surface_a: a #cairo_surface_t + * @surface_b: a #cairo_surface_t + * + * Find out whether the given surfaces share the same backend, + * and if so, whether they can be considered compatible. + * + * The definition of "compatible" depends on the backend. In the + * xlib case, it means the surface share the same display. + * + * Return value: TRUE if the surfaces are compatible. + **/ +cairo_bool_t +_cairo_surface_is_compatible (cairo_surface_t *surface_a, + cairo_surface_t *surface_b) +{ + if (surface_a->backend != surface_b->backend) + return FALSE; + + if (surface_a->backend->is_compatible != NULL) + return surface_a->backend->is_compatible (surface_a, surface_b); + + return TRUE; +} + cairo_status_t _cairo_surface_composite (cairo_operator_t op, cairo_pattern_t *src, --- a/src/cairo-win32-surface.c +++ b/src/cairo-win32-surface.c @@ -1854,6 +1854,16 @@ cairo_win32_surface_get_image (cairo_sur return ((cairo_win32_surface_t*)surface)->image; } +static cairo_bool_t +_cairo_win32_surface_is_compatible (void *surface_a, + void *surface_b) +{ + cairo_win32_surface_t *a = (cairo_win32_surface_t *) surface_a; + cairo_win32_surface_t *b = (cairo_win32_surface_t *) surface_b; + + return a->dc == b->dc; +} + static const cairo_surface_backend_t cairo_win32_surface_backend = { CAIRO_SURFACE_TYPE_WIN32, _cairo_win32_surface_create_similar, @@ -1884,7 +1894,8 @@ static const cairo_surface_backend_t cai NULL, /* fill */ _cairo_win32_surface_show_glyphs, - NULL /* snapshot */ + NULL, /* snapshot */ + _cairo_win32_surface_is_compatible }; /* Notes: --- a/src/cairo-xcb-surface.c +++ b/src/cairo-xcb-surface.c @@ -1577,6 +1577,16 @@ _cairo_xcb_surface_show_glyphs (void int num_glyphs, cairo_scaled_font_t *scaled_font); +static cairo_bool_t +_cairo_xcb_surface_is_compatible (void *surface_a, + void *surface_b) +{ + cairo_xcb_surface_t *a = (cairo_xcb_surface_t *) surface_a; + cairo_xcb_surface_t *b = (cairo_xcb_surface_t *) surface_b; + + return _cairo_xcb_surface_same_screen (dst, src); +} + /* XXX: move this to the bottom of the file, XCB and Xlib */ static const cairo_surface_backend_t cairo_xcb_surface_backend = { @@ -1608,7 +1618,8 @@ static const cairo_surface_backend_t cai NULL, /* stroke */ NULL, /* fill */ _cairo_xcb_surface_show_glyphs, - NULL /* snapshot */ + NULL, /* snapshot */ + _cairo_xcb_surface_is_compatible }; /** --- /dev/null +++ b/src/cairo-xlib-display.c @@ -0,0 +1,490 @@ +/* Cairo - a vector graphics library with display and print output + * + * Copyright © 2007 Chris Wilson + * + * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 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. + * + * The Original Code is the cairo graphics library. + * + * The Initial Developer of the Original Code is Chris Wilson. + * + */ + +#include "cairoint.h" + +#include "cairo-xlib-private.h" + +#include <fontconfig/fontconfig.h> + +#include <X11/Xlibint.h> /* For XESetCloseDisplay */ +#include <X11/extensions/Xrender.h> + +typedef int (*cairo_xlib_error_func_t) (Display *display, + XErrorEvent *event); + +struct _cairo_xlib_job { + cairo_xlib_job_t *next; + enum { + RESOURCE, + WORK + } type; + union { + struct { + cairo_xlib_notify_resource_func notify; + XID xid; + } resource; + struct { + cairo_xlib_notify_func notify; + void *data; + void (*destroy) (void *); + } work; + } func; +}; + +static cairo_xlib_display_t *_cairo_xlib_display_list = NULL; + +static void +_cairo_xlib_call_close_display_hooks (cairo_xlib_display_t *info) +{ + cairo_xlib_screen_info_t *screen; + cairo_xlib_hook_t *hooks; + + /* call all registered shutdown routines */ + CAIRO_MUTEX_LOCK (info->mutex); + + for (screen = info->screens; screen != NULL; screen = screen->next) + _cairo_xlib_screen_info_close_display (screen); + + hooks = info->close_display_hooks; + while (hooks != NULL) { + info->close_display_hooks = NULL; + CAIRO_MUTEX_UNLOCK (info->mutex); + + do { + cairo_xlib_hook_t *hook = hooks; + hooks = hook->next; + + hook->func (info->display, hook->data); + + free (hook); + } while (hooks != NULL); + + CAIRO_MUTEX_LOCK (info->mutex); + hooks = info->close_display_hooks; + } + info->closed = TRUE; + + CAIRO_MUTEX_UNLOCK (info->mutex); +} + +cairo_xlib_display_t * +_cairo_xlib_display_reference (cairo_xlib_display_t *info) +{ + if (info == NULL) + return NULL; + + /* use our mutex until we get a real atomic inc */ + CAIRO_MUTEX_LOCK (info->mutex); + + assert (info->ref_count > 0); + info->ref_count++; + + CAIRO_MUTEX_UNLOCK (info->mutex); + + return info; +} + +void +_cairo_xlib_display_destroy (cairo_xlib_display_t *info) +{ + if (info == NULL) + return; + + CAIRO_MUTEX_LOCK (info->mutex); + assert (info->ref_count > 0); + if (--info->ref_count == 0) { + assert (info->closed == TRUE); + assert (info->screens == NULL); + + /* destroy all outstanding notifies */ + while (info->workqueue != NULL) { + cairo_xlib_job_t *job = info->workqueue; + info->workqueue = job->next; + + if (job->type == WORK && job->func.work.destroy != NULL) + job->func.work.destroy (job->func.work.data); + + _cairo_freelist_free (&info->wq_freelist, job); + } + _cairo_freelist_fini (&info->wq_freelist); + + CAIRO_MUTEX_UNLOCK (info->mutex); + + free (info); + } else + CAIRO_MUTEX_UNLOCK (info->mutex); +} + +static int +_noop_error_handler (Display *display, + XErrorEvent *event) +{ + return False; /* return value is ignored */ +} +static int +_cairo_xlib_close_display (Display *dpy, XExtCodes *codes) +{ + cairo_xlib_display_t *info, **prev, *next; + + /* + * Unhook from the global list + */ + CAIRO_MUTEX_LOCK (_cairo_xlib_display_mutex); + prev = &_cairo_xlib_display_list; + for (info = _cairo_xlib_display_list; info; info = next) { + next = info->next; + if (info->display == dpy) { + cairo_xlib_error_func_t old_handler; + + /* drop the list mutex whilst triggering the hooks */ + CAIRO_MUTEX_UNLOCK (_cairo_xlib_display_mutex); + + /* protect the notifies from triggering XErrors */ + old_handler = XSetErrorHandler (_noop_error_handler); + + _cairo_xlib_display_notify (info); + _cairo_xlib_call_close_display_hooks (info); + + /* catch any that arrived before marking the display as closed */ + _cairo_xlib_display_notify (info); + + XSync (dpy, False); + XSetErrorHandler (old_handler); + + CAIRO_MUTEX_LOCK (_cairo_xlib_display_mutex); + _cairo_xlib_display_destroy (info); + *prev = next; + break; + } else + prev = &info->next; + } + CAIRO_MUTEX_UNLOCK (_cairo_xlib_display_mutex); + + /* Return value in accordance with requirements of + * XESetCloseDisplay */ + return 0; +} + +cairo_xlib_display_t * +_cairo_xlib_display_get (Display *dpy) +{ + cairo_xlib_display_t *info; + cairo_xlib_display_t **prev; + XExtCodes *codes; + + /* There is an apparent deadlock between this mutex and the + * mutex for the display, but it's actually safe. For the + * app to call XCloseDisplay() while any other thread is + * inside this function would be an error in the logic + * app, and the CloseDisplay hook is the only other place we + * acquire this mutex. + */ + CAIRO_MUTEX_LOCK (_cairo_xlib_display_mutex); + + for (prev = &_cairo_xlib_display_list; (info = *prev); prev = &(*prev)->next) + { + if (info->display == dpy) { + /* + * MRU the list + */ + if (prev != &_cairo_xlib_display_list) { + *prev = info->next; + info->next = _cairo_xlib_display_list; + _cairo_xlib_display_list = info; + } + break; + } + } + + if (info != NULL) { + info = _cairo_xlib_display_reference (info); + goto UNLOCK; + } + + info = malloc (sizeof (cairo_xlib_display_t)); + if (info == NULL) + goto UNLOCK; + + codes = XAddExtension (dpy); + if (codes == NULL) { + free (info); + info = NULL; + goto UNLOCK; + } + + XESetCloseDisplay (dpy, codes->extension, _cairo_xlib_close_display); + + _cairo_freelist_init (&info->wq_freelist, sizeof (cairo_xlib_job_t)); + info->ref_count = 2; /* add one for the CloseDisplay */ + CAIRO_MUTEX_INIT (&info->mutex); + info->display = dpy; + info->screens = NULL; + info->workqueue = NULL; + info->close_display_hooks = NULL; + info->closed = FALSE; + + info->next = _cairo_xlib_display_list; + _cairo_xlib_display_list = info; + +UNLOCK: + CAIRO_MUTEX_UNLOCK (_cairo_xlib_display_mutex); + return info; +} + +cairo_bool_t +_cairo_xlib_add_close_display_hook (Display *dpy, void (*func) (Display *, void *), void *data, const void *key) +{ + cairo_xlib_display_t *info; + cairo_xlib_hook_t *hook; + cairo_bool_t ret = FALSE; + + info = _cairo_xlib_display_get (dpy); + if (info == NULL) + return FALSE; + + hook = malloc (sizeof (cairo_xlib_hook_t)); + if (hook != NULL) { + hook->func = func; + hook->data = data; + hook->key = key; + + CAIRO_MUTEX_LOCK (info->mutex); + if (info->closed == FALSE) { + hook->next = info->close_display_hooks; + info->close_display_hooks = hook; + ret = TRUE; + } + CAIRO_MUTEX_UNLOCK (info->mutex); + } + + _cairo_xlib_display_destroy (info); + + return ret; +} + +void +_cairo_xlib_remove_close_display_hooks (Display *dpy, const void *key) +{ + cairo_xlib_display_t *info; + cairo_xlib_hook_t *hook, *next, **prev; + + info = _cairo_xlib_display_get (dpy); + if (info == NULL) + return; + + CAIRO_MUTEX_LOCK (info->mutex); + prev = &info->close_display_hooks; + for (hook = info->close_display_hooks; hook != NULL; hook = next) { + next = hook->next; + if (hook->key == key) { + *prev = hook->next; + free (hook); + } else + prev = &hook->next; + } + *prev = NULL; + CAIRO_MUTEX_UNLOCK (info->mutex); + + _cairo_xlib_display_destroy (info); +} + +void +_cairo_xlib_display_cancel_resource (cairo_xlib_display_t *display, + XID xid) +{ + cairo_xlib_job_t *job, *next, **prev; + + CAIRO_MUTEX_LOCK (display->mutex); + prev = &display->workqueue; + for (job = display->workqueue; job != NULL; job = next) { + next = job->next; + if (job->type == RESOURCE && job->func.resource.xid == xid) { + _cairo_freelist_free (&display->wq_freelist, job); + + *prev = next; + break; + } + prev = &job->next; + } + CAIRO_MUTEX_UNLOCK (display->mutex); +} + +static cairo_xlib_job_t * +_cairo_xlib_display_find_resource (cairo_xlib_display_t *display, XID xid) +{ + cairo_xlib_job_t *job; + + CAIRO_MUTEX_LOCK (display->mutex); + for (job = display->workqueue; job != NULL; job = job->next) { + if (job->type == RESOURCE && job->func.resource.xid == xid) + break; + } + CAIRO_MUTEX_UNLOCK (display->mutex); + + return job; +} + +cairo_status_t +_cairo_xlib_display_queue_resource (cairo_xlib_display_t *display, + cairo_xlib_notify_resource_func notify, + XID xid) +{ + cairo_xlib_job_t *job; + cairo_status_t status = CAIRO_STATUS_SUCCESS; + + assert (_cairo_xlib_display_find_resource (display, xid) == NULL); + + job = _cairo_freelist_alloc (&display->wq_freelist); + if (job == NULL) + return CAIRO_STATUS_NO_MEMORY; + + job->type = RESOURCE; + job->func.resource.xid = xid; + job->func.resource.notify = notify; + + CAIRO_MUTEX_LOCK (display->mutex); + if (display->closed == FALSE) { + job->next = display->workqueue; + display->workqueue = job; + } else { + _cairo_freelist_free (&display->wq_freelist, job); + job = NULL; + status = CAIRO_STATUS_NO_MEMORY; + } + CAIRO_MUTEX_UNLOCK (display->mutex); + + return status; +} + +void +_cairo_xlib_display_cancel_work (cairo_xlib_display_t *display, + void *data) +{ + cairo_xlib_job_t *job, *next, **prev; + + CAIRO_MUTEX_LOCK (display->mutex); + prev = &display->workqueue; + for (job = display->workqueue; job != NULL; job = next) { + next = job->next; + if (job->type == WORK && job->func.work.data == data) { + if (job->func.work.destroy != NULL) + job->func.work.destroy (job->func.work.data); + _cairo_freelist_free (&display->wq_freelist, job); + + *prev = next; + break; + } + prev = &job->next; + } + CAIRO_MUTEX_UNLOCK (display->mutex); +} + +cairo_status_t +_cairo_xlib_display_queue_work (cairo_xlib_display_t *display, + cairo_xlib_notify_func notify, + void *data, + void (*destroy) (void *)) +{ + cairo_xlib_job_t *job; + cairo_status_t status = CAIRO_STATUS_SUCCESS; + + job = _cairo_freelist_alloc (&display->wq_freelist); + if (job == NULL) + return CAIRO_STATUS_NO_MEMORY; + + job->type = WORK; + job->func.work.data = data; + job->func.work.notify = notify; + job->func.work.destroy = destroy; + + CAIRO_MUTEX_LOCK (display->mutex); + if (display->closed == FALSE) { + job->next = display->workqueue; + display->workqueue = job; + } else { + _cairo_freelist_free (&display->wq_freelist, job); + job = NULL; + status = CAIRO_STATUS_NO_MEMORY; + } + CAIRO_MUTEX_UNLOCK (display->mutex); + + return status; +} + +void +_cairo_xlib_display_notify (cairo_xlib_display_t *display) +{ + cairo_xlib_job_t *jobs, *job; + + CAIRO_MUTEX_LOCK (display->mutex); + jobs = display->workqueue; + while (jobs != NULL) { + display->workqueue = NULL; + CAIRO_MUTEX_UNLOCK (display->mutex); + + /* reverse the list to obtain FIFO order */ + job = NULL; + do { + cairo_xlib_job_t *next = jobs->next; + jobs->next = job; + job = jobs; + jobs = next; + } while (jobs != NULL); + jobs = job; + + do { + job = jobs; + jobs = job->next; + + switch (job->type){ + case WORK: + job->func.work.notify (display->display, job->func.work.data); + if (job->func.work.destroy != NULL) + job->func.work.destroy (job->func.work.data); + break; + + case RESOURCE: + job->func.resource.notify (display->display, + job->func.resource.xid); + break; + } + + _cairo_freelist_free (&display->wq_freelist, job); + } while (jobs != NULL); + + CAIRO_MUTEX_LOCK (display->mutex); + jobs = display->workqueue; + } + CAIRO_MUTEX_UNLOCK (display->mutex); +} --- a/src/cairo-xlib-private.h +++ b/src/cairo-xlib-private.h @@ -35,30 +35,82 @@ #include "cairoint.h" #include "cairo-xlib.h" +#include "cairo-freelist-private.h" +typedef struct _cairo_xlib_display cairo_xlib_display_t; typedef struct _cairo_xlib_screen_info cairo_xlib_screen_info_t; typedef struct _cairo_xlib_hook cairo_xlib_hook_t; +typedef struct _cairo_xlib_job cairo_xlib_job_t; +typedef void (*cairo_xlib_notify_func) (Display *, void *); +typedef void (*cairo_xlib_notify_resource_func) (Display *, XID); struct _cairo_xlib_hook { cairo_xlib_hook_t *next; void (*func) (Display *display, void *data); void *data; - void *key; + const void *key; +}; + +struct _cairo_xlib_display { + cairo_xlib_display_t *next; + unsigned int ref_count; + cairo_mutex_t mutex; + + Display *display; + cairo_xlib_screen_info_t *screens; + + cairo_xlib_job_t *workqueue; + cairo_freelist_t wq_freelist; + + cairo_xlib_hook_t *close_display_hooks; + unsigned int closed :1; }; struct _cairo_xlib_screen_info { cairo_xlib_screen_info_t *next; unsigned int ref_count; - Display *display; + cairo_xlib_display_t *display; Screen *screen; cairo_bool_t has_render; cairo_font_options_t font_options; - cairo_xlib_hook_t *close_display_hooks; + GC gc[6]; }; +cairo_private cairo_xlib_display_t * +_cairo_xlib_display_get (Display *display); + +cairo_private cairo_xlib_display_t * +_cairo_xlib_display_reference (cairo_xlib_display_t *info); +cairo_private void +_cairo_xlib_display_destroy (cairo_xlib_display_t *info); + +cairo_private cairo_bool_t +_cairo_xlib_add_close_display_hook (Display *display, void (*func) (Display *, void *), void *data, const void *key); +cairo_private void +_cairo_xlib_remove_close_display_hooks (Display *display, const void *key); + +cairo_private cairo_status_t +_cairo_xlib_display_queue_work (cairo_xlib_display_t *display, + cairo_xlib_notify_func notify, + void *data, + void (*destroy)(void *)); +cairo_private cairo_status_t +_cairo_xlib_display_queue_resource (cairo_xlib_display_t *display, + cairo_xlib_notify_resource_func notify, + XID resource); +cairo_private void +_cairo_xlib_display_cancel_work (cairo_xlib_display_t *display, + void *data); +cairo_private void +_cairo_xlib_display_cancel_resource (cairo_xlib_display_t *display, + XID resource); + +cairo_private void +_cairo_xlib_display_notify (cairo_xlib_display_t *display); + cairo_private cairo_xlib_screen_info_t * _cairo_xlib_screen_info_get (Display *display, Screen *screen); @@ -67,10 +119,14 @@ _cairo_xlib_screen_info_reference (cairo cairo_private void _cairo_xlib_screen_info_destroy (cairo_xlib_screen_info_t *info); -cairo_private cairo_bool_t -_cairo_xlib_add_close_display_hook (Display *display, void (*func) (Display *, void *), void *data, void *key); cairo_private void -_cairo_xlib_remove_close_display_hook (Display *display, void *key); +_cairo_xlib_screen_info_close_display (cairo_xlib_screen_info_t *info); + + +cairo_private GC +_cairo_xlib_screen_get_gc (cairo_xlib_screen_info_t *info, int depth); +cairo_private void +_cairo_xlib_screen_put_gc (cairo_xlib_screen_info_t *info, int depth, GC gc); #if CAIRO_HAS_XLIB_XRENDER_SURFACE --- a/src/cairo-xlib-screen.c +++ b/src/cairo-xlib-screen.c @@ -58,7 +58,6 @@ #include <fontconfig/fontconfig.h> -#include <X11/Xlibint.h> /* For XESetCloseDisplay */ #include <X11/extensions/Xrender.h> static int @@ -133,7 +132,7 @@ get_integer_default (Display *dpy, #endif static void -_cairo_xlib_init_screen_font_options (cairo_xlib_screen_info_t *info) +_cairo_xlib_init_screen_font_options (Display *dpy, cairo_xlib_screen_info_t *info) { cairo_bool_t xft_hinting; cairo_bool_t xft_antialias; @@ -143,23 +142,23 @@ _cairo_xlib_init_screen_font_options (ca cairo_subpixel_order_t subpixel_order; cairo_hint_style_t hint_style; - if (!get_boolean_default (info->display, "antialias", &xft_antialias)) + if (!get_boolean_default (dpy, "antialias", &xft_antialias)) xft_antialias = TRUE; - if (!get_boolean_default (info->display, "hinting", &xft_hinting)) + if (!get_boolean_default (dpy, "hinting", &xft_hinting)) xft_hinting = TRUE; - if (!get_integer_default (info->display, "hintstyle", &xft_hintstyle)) + if (!get_integer_default (dpy, "hintstyle", &xft_hintstyle)) xft_hintstyle = FC_HINT_FULL; - if (!get_integer_default (info->display, "rgba", &xft_rgba)) + if (!get_integer_default (dpy, "rgba", &xft_rgba)) { xft_rgba = FC_RGBA_UNKNOWN; #if RENDER_MAJOR > 0 || RENDER_MINOR >= 6 if (info->has_render) { - int render_order = XRenderQuerySubpixelOrder (info->display, + int render_order = XRenderQuerySubpixelOrder (dpy, XScreenNumberOfScreen (info->screen)); switch (render_order) @@ -243,285 +242,165 @@ _cairo_xlib_init_screen_font_options (ca cairo_font_options_set_hint_metrics (&info->font_options, CAIRO_HINT_METRICS_ON); } -static cairo_xlib_screen_info_t *_cairo_xlib_screen_list = NULL; - -/* NOTE: This function must be called with _cairo_xlib_screen_mutex held. */ -static void -_cairo_xlib_call_close_display_hooks (cairo_xlib_screen_info_t *info) -{ - /* call all registered shutdown routines */ - while (info->close_display_hooks != NULL) { - cairo_xlib_hook_t *hooks = info->close_display_hooks; - info->close_display_hooks = NULL; - - /* drop the list mutex whilst calling the hooks */ - CAIRO_MUTEX_UNLOCK (_cairo_xlib_screen_mutex); - do { - cairo_xlib_hook_t *hook = hooks; - hooks = hook->next; - - hook->func (info->display, hook->data); - - free (hook); - } while (hooks != NULL); - CAIRO_MUTEX_LOCK (_cairo_xlib_screen_mutex); - } -} - -static void -_cairo_xlib_screen_info_reference_lock_held (cairo_xlib_screen_info_t *info) -{ - assert (info->ref_count > 0); - info->ref_count++; -} - cairo_xlib_screen_info_t * _cairo_xlib_screen_info_reference (cairo_xlib_screen_info_t *info) { if (info == NULL) return NULL; - /* use our global mutex until we get a real atomic inc */ - CAIRO_MUTEX_LOCK (_cairo_xlib_screen_mutex); - - _cairo_xlib_screen_info_reference_lock_held (info); - - CAIRO_MUTEX_UNLOCK (_cairo_xlib_screen_mutex); + assert (info->ref_count > 0); + info->ref_count++; return info; } -static void -_cairo_xlib_screen_info_destroy_lock_held (cairo_xlib_screen_info_t *info) +void +_cairo_xlib_screen_info_close_display (cairo_xlib_screen_info_t *info) { - assert (info->ref_count > 0); - if (--info->ref_count) - return; + int i; - _cairo_xlib_call_close_display_hooks (info); - free (info); + for (i = 0; i < ARRAY_LENGTH (info->gc); i++) { + if (info->gc[i] != NULL) { + XFreeGC (info->display->display, info->gc[i]); + info->gc[i] = NULL; + } + } } void _cairo_xlib_screen_info_destroy (cairo_xlib_screen_info_t *info) { + cairo_xlib_screen_info_t **prev; + cairo_xlib_screen_info_t *list; + if (info == NULL) return; - CAIRO_MUTEX_LOCK (_cairo_xlib_screen_mutex); - - _cairo_xlib_screen_info_destroy_lock_held (info); - - CAIRO_MUTEX_UNLOCK (_cairo_xlib_screen_mutex); -} - -static int -_cairo_xlib_close_display (Display *dpy, XExtCodes *codes) -{ - cairo_xlib_screen_info_t *info, **prev, *next; + assert (info->ref_count > 0); + if (--info->ref_count) + return; - /* - * Unhook from the global list - */ - CAIRO_MUTEX_LOCK (_cairo_xlib_screen_mutex); - - prev = &_cairo_xlib_screen_list; - for (info = _cairo_xlib_screen_list; info; info = next) { - next = info->next; - if (info->display == dpy) { - /* trigger the hooks explicitly as we know the display is closing */ - _cairo_xlib_call_close_display_hooks (info); - _cairo_xlib_screen_info_destroy_lock_held (info); - *prev = next; - } else { - prev = &info->next; + CAIRO_MUTEX_LOCK (info->display->mutex); + for (prev = &info->display->screens; (list = *prev); prev = &list->next) { + if (list == info) { + *prev = info->next; + break; } } - *prev = NULL; - CAIRO_MUTEX_UNLOCK (_cairo_xlib_screen_mutex); - - /* Return value in accordance with requirements of - * XESetCloseDisplay */ - return 0; -} - -static void -_cairo_xlib_screen_info_reset (void) -{ - /* - * Delete everything in the list. - */ - CAIRO_MUTEX_LOCK (_cairo_xlib_screen_mutex); + assert (list != NULL); + CAIRO_MUTEX_UNLOCK (info->display->mutex); - while (_cairo_xlib_screen_list != NULL) { - cairo_xlib_screen_info_t *info = _cairo_xlib_screen_list; - _cairo_xlib_screen_list = info->next; - _cairo_xlib_screen_info_destroy_lock_held (info); - } + _cairo_xlib_screen_info_close_display (info); - CAIRO_MUTEX_UNLOCK (_cairo_xlib_screen_mutex); + _cairo_xlib_display_destroy (info->display); + free (info); } -static cairo_xlib_screen_info_t * -_cairo_xlib_screen_info_get_lock_held (Display *dpy, Screen *screen) +cairo_xlib_screen_info_t * +_cairo_xlib_screen_info_get (Display *dpy, Screen *screen) { - cairo_xlib_screen_info_t *info; - cairo_xlib_screen_info_t **prev; - int event_base, error_base; - XExtCodes *codes; - cairo_bool_t seen_display = FALSE; + cairo_xlib_display_t *display; + cairo_xlib_screen_info_t *info = NULL, **prev; - for (prev = &_cairo_xlib_screen_list; (info = *prev); prev = &(*prev)->next) - { - if (info->display == dpy) { - seen_display = TRUE; - if (info->screen == screen) { - /* - * MRU the list - */ - if (prev != &_cairo_xlib_screen_list) { - *prev = info->next; - info->next = _cairo_xlib_screen_list; - _cairo_xlib_screen_list = info; - } - break; - } - } - } - - if (info) - return info; - - info = malloc (sizeof (cairo_xlib_screen_info_t)); - if (!info) + display = _cairo_xlib_display_get (dpy); + if (display == NULL) return NULL; - if (!seen_display) { - codes = XAddExtension (dpy); - if (!codes) { - free (info); - return NULL; - } + CAIRO_MUTEX_LOCK (display->mutex); + if (display->closed) { + CAIRO_MUTEX_UNLOCK (display->mutex); + goto DONE; + } - XESetCloseDisplay (dpy, codes->extension, _cairo_xlib_close_display); + for (prev = &display->screens; (info = *prev); prev = &(*prev)->next) { + if (info->screen == screen) { + /* + * MRU the list + */ + if (prev != &display->screens) { + *prev = info->next; + info->next = display->screens; + display->screens = info; + } + break; + } } + CAIRO_MUTEX_UNLOCK (display->mutex); - info->ref_count = 1; - info->display = dpy; - info->screen = screen; - info->close_display_hooks = NULL; - info->has_render = FALSE; - _cairo_font_options_init_default (&info->font_options); + if (info != NULL) { + info = _cairo_xlib_screen_info_reference (info); + } else { + info = malloc (sizeof (cairo_xlib_screen_info_t)); + if (info != NULL) { + info->ref_count = 1; + info->display = _cairo_xlib_display_reference (display); + info->screen = screen; + info->has_render = FALSE; + _cairo_font_options_init_default (&info->font_options); + memset (info->gc, 0, sizeof (info->gc)); + + if (screen) { + int event_base, error_base; + info->has_render = (XRenderQueryExtension (dpy, &event_base, &error_base) && + (XRenderFindVisualFormat (dpy, DefaultVisual (dpy, DefaultScreen (dpy))) != 0)); + _cairo_xlib_init_screen_font_options (dpy, info); + } - if (screen) { - info->has_render = (XRenderQueryExtension (dpy, &event_base, &error_base) && - (XRenderFindVisualFormat (dpy, DefaultVisual (dpy, DefaultScreen (dpy))) != 0)); - _cairo_xlib_init_screen_font_options (info); + CAIRO_MUTEX_LOCK (display->mutex); + info->next = display->screens; + display->screens = info; + CAIRO_MUTEX_UNLOCK (display->mutex); + } } - info->next = _cairo_xlib_screen_list; - _cairo_xlib_screen_list = info; +DONE: + _cairo_xlib_display_destroy (display); return info; } -cairo_xlib_screen_info_t * -_cairo_xlib_screen_info_get (Display *dpy, Screen *screen) -{ - cairo_xlib_screen_info_t *info; - /* There is an apparent deadlock between this mutex and the - * mutex for the display, but it's actually safe. For the - * app to call XCloseDisplay() while any other thread is - * inside this function would be an error in the logic - * app, and the CloseDisplay hook is the only other place we - * acquire this mutex. - */ - CAIRO_MUTEX_LOCK (_cairo_xlib_screen_mutex); - - info = _cairo_xlib_screen_info_get_lock_held (dpy, screen); - if (info != NULL) - _cairo_xlib_screen_info_reference_lock_held (info); - - CAIRO_MUTEX_UNLOCK (_cairo_xlib_screen_mutex); - - return info; +static int +depth_to_index (int depth) +{ + switch(depth){ + case 1: return 0; + case 8: return 1; + case 15: return 2; + case 16: return 3; + case 24: return 4; + case 32: return 5; + } + return 0; } -cairo_bool_t -_cairo_xlib_add_close_display_hook (Display *dpy, void (*func) (Display *, void *), void *data, void *key) +GC +_cairo_xlib_screen_get_gc (cairo_xlib_screen_info_t *info, int depth) { - cairo_xlib_screen_info_t *info; - cairo_xlib_hook_t *hook; - cairo_xlib_hook_t **prev; - cairo_bool_t success = FALSE; - CAIRO_MUTEX_LOCK (_cairo_xlib_screen_mutex); + GC gc; - info = _cairo_xlib_screen_info_get_lock_held (dpy, NULL); - if (!info) - goto unlock; + depth = depth_to_index (depth); - for (prev = &info->close_display_hooks; (hook = *prev); prev = &hook->next) - { - if (hook->key == key) { - /* - * MRU the list - */ - if (prev != &info->close_display_hooks) { - *prev = hook->next; - hook->next = info->close_display_hooks; - info->close_display_hooks = hook; - } - break; - } - } + gc = info->gc[depth]; + info->gc[depth] = NULL; - if (!hook) { - hook = malloc (sizeof (cairo_xlib_hook_t)); - if (!hook) - goto unlock; - hook->func = func; - hook->data = data; - hook->key = key; - hook->next = info->close_display_hooks; - info->close_display_hooks = hook; - } - - success = TRUE; - unlock: - CAIRO_MUTEX_UNLOCK (_cairo_xlib_screen_mutex); - return success; + return gc; } void -_cairo_xlib_remove_close_display_hook (Display *dpy, void *key) +_cairo_xlib_screen_put_gc (cairo_xlib_screen_info_t *info, int depth, GC gc) { - cairo_xlib_screen_info_t *info; - cairo_xlib_hook_t *hook; - cairo_xlib_hook_t **prev; - - CAIRO_MUTEX_LOCK (_cairo_xlib_screen_mutex); + depth = depth_to_index (depth); - info = _cairo_xlib_screen_info_get_lock_held (dpy, NULL); - if (!info) - goto unlock; - - for (prev = &info->close_display_hooks; (hook = *prev); prev = &hook->next) - { - if (hook->key == key) { - *prev = hook->next; - free (hook); - break; - } + if (info->gc[depth] != NULL) { + cairo_status_t status; + status = _cairo_xlib_display_queue_work (info->display, + (cairo_xlib_notify_func) XFreeGC, + info->gc[depth], + NULL); + (void) status; } -unlock: - CAIRO_MUTEX_UNLOCK (_cairo_xlib_screen_mutex); -} - -void -_cairo_xlib_screen_reset_static_data (void) -{ - _cairo_xlib_screen_info_reset (); + info->gc[depth] = gc; } --- a/src/cairo-xlib-surface.c +++ b/src/cairo-xlib-surface.c @@ -127,8 +127,17 @@ struct _cairo_xlib_surface { int num_clip_rects; XRenderPictFormat *xrender_format; + cairo_filter_t filter; + int repeat; + XTransform xtransform; }; +static const XTransform identity = { { + { 1 << 16, 0x00000, 0x00000 }, + { 0x00000, 1 << 16, 0x00000 }, + { 0x00000, 0x00000, 1 << 16 }, +} }; + #define CAIRO_SURFACE_RENDER_AT_LEAST(surface, major, minor) \ (((surface)->render_major > major) || \ (((surface)->render_major == major) && ((surface)->render_minor >= minor))) @@ -281,6 +290,8 @@ _cairo_xlib_surface_create_similar (void cairo_xlib_surface_t *surface; Pixmap pix; + _cairo_xlib_display_notify (src->screen_info->display); + /* Start by examining the surface's XRenderFormat, or if it * doesn't have one, then look one up through its visual (in the * case of a bitmap, it won't even have that). */ @@ -322,21 +333,75 @@ _cairo_xlib_surface_create_similar (void return &surface->base; } +static void +_cairo_xlib_reset_clip_mask (Display *dpy, GC gc) +{ + XSetClipMask(dpy, gc, None); +} + static cairo_status_t _cairo_xlib_surface_finish (void *abstract_surface) { cairo_xlib_surface_t *surface = abstract_surface; - if (surface->dst_picture != None) - XRenderFreePicture (surface->dpy, surface->dst_picture); - - if (surface->src_picture != None) - XRenderFreePicture (surface->dpy, surface->src_picture); - - if (surface->owns_pixmap) - XFreePixmap (surface->dpy, surface->drawable); - - if (surface->gc != NULL) - XFreeGC (surface->dpy, surface->gc); + cairo_xlib_display_t *display = surface->screen_info ? + surface->screen_info->display : + NULL; + cairo_status_t status = CAIRO_STATUS_SUCCESS; + + if (surface->dst_picture != None) { + cairo_status_t status2; + assert (display != NULL); + status2 = _cairo_xlib_display_queue_resource (display, + XRenderFreePicture, + surface->dst_picture); + if (status2 == CAIRO_STATUS_SUCCESS) + surface->dst_picture = None; + else if (status == CAIRO_STATUS_SUCCESS) + status = status2; + } + + if (surface->src_picture != None) { + cairo_status_t status2; + assert (display != NULL); + status2 = _cairo_xlib_display_queue_resource (display, + XRenderFreePicture, + surface->src_picture); + if (status2 == CAIRO_STATUS_SUCCESS) + surface->src_picture = None; + else if (status == CAIRO_STATUS_SUCCESS) + status = status2; + } + + if (surface->owns_pixmap) { + cairo_status_t status2; + assert (display != NULL); + status2 = _cairo_xlib_display_queue_resource (display, + (cairo_xlib_notify_resource_func) XFreePixmap, + surface->drawable); + if (status2 == CAIRO_STATUS_SUCCESS) { + surface->owns_pixmap = FALSE; + surface->drawable = None; + } else if (status == CAIRO_STATUS_SUCCESS) + status = status2; + } + + if (surface->gc != NULL) { + _cairo_xlib_screen_put_gc (surface->screen_info, + surface->depth, + surface->gc); + if (surface->have_clip_rects) { + cairo_status_t status2; + assert (display != NULL); + status2 = _cairo_xlib_display_queue_work (display, + (cairo_xlib_notify_func) _cairo_xlib_reset_clip_mask, + surface->gc, + NULL); + if (status2 == CAIRO_STATUS_SUCCESS) + surface->gc = NULL; + else if (status == CAIRO_STATUS_SUCCESS) + status = status2; + } + } if (surface->clip_rects != NULL) free (surface->clip_rects); @@ -344,9 +409,12 @@ _cairo_xlib_surface_finish (void *abstra if (surface->screen_info != NULL) _cairo_xlib_screen_info_destroy (surface->screen_info); - surface->dpy = NULL; + if (surface->dpy != NULL) { + _cairo_xlib_remove_close_display_hooks (surface->dpy, surface); + surface->dpy = NULL; + } - return CAIRO_STATUS_SUCCESS; + return status; } static int @@ -747,11 +815,15 @@ _cairo_xlib_surface_ensure_gc (cairo_xli if (surface->gc) return CAIRO_STATUS_SUCCESS; - gcv.graphics_exposures = False; - surface->gc = XCreateGC (surface->dpy, surface->drawable, - GCGraphicsExposures, &gcv); - if (!surface->gc) - return CAIRO_STATUS_NO_MEMORY; + surface->gc = _cairo_xlib_screen_get_gc (surface->screen_info, + surface->depth); + if (surface->gc == NULL) { + gcv.graphics_exposures = False; + surface->gc = XCreateGC (surface->dpy, surface->drawable, + GCGraphicsExposures, &gcv); + if (!surface->gc) + return CAIRO_STATUS_NO_MEMORY; + } _cairo_xlib_surface_set_gc_clip_rects (surface); @@ -814,6 +886,8 @@ _cairo_xlib_surface_acquire_source_image cairo_image_surface_t *image; cairo_status_t status; + _cairo_xlib_display_notify (surface->screen_info->display); + status = _get_image_surface (surface, NULL, &image, NULL); if (status) return status; @@ -843,6 +917,8 @@ _cairo_xlib_surface_acquire_dest_image ( cairo_image_surface_t *image; cairo_status_t status; + _cairo_xlib_display_notify (surface->screen_info->display); + status = _get_image_surface (surface, interest_rect, &image, image_rect_out); if (status) return status; @@ -894,6 +970,8 @@ _cairo_xlib_surface_clone_similar (void cairo_xlib_surface_t *clone; cairo_status_t status; + _cairo_xlib_display_notify (surface->screen_info->display); + if (src->backend == surface->base.backend ) { cairo_xlib_surface_t *xlib_src = (cairo_xlib_surface_t *)src; @@ -952,19 +1030,17 @@ _cairo_xlib_surface_set_matrix (cairo_xl if (!CAIRO_SURFACE_RENDER_HAS_PICTURE_TRANSFORM (surface)) { - static const XTransform identity = { { - { 1 << 16, 0x00000, 0x00000 }, - { 0x00000, 1 << 16, 0x00000 }, - { 0x00000, 0x00000, 1 << 16 }, - } }; - if (memcmp (&xtransform, &identity, sizeof (XTransform)) == 0) return CAIRO_STATUS_SUCCESS; return CAIRO_INT_STATUS_UNSUPPORTED; } + if (memcmp (&xtransform, &surface->xtransform, sizeof (XTransform)) == 0) + return CAIRO_STATUS_SUCCESS; + XRenderSetPictureTransform (surface->dpy, surface->src_picture, &xtransform); + surface->xtransform = xtransform; return CAIRO_STATUS_SUCCESS; } @@ -978,6 +1054,9 @@ _cairo_xlib_surface_set_filter (cairo_xl if (!surface->src_picture) return CAIRO_STATUS_SUCCESS; + if (surface->filter == filter) + return CAIRO_STATUS_SUCCESS; + if (!CAIRO_SURFACE_RENDER_HAS_FILTERS (surface)) { if (filter == CAIRO_FILTER_FAST || filter == CAIRO_FILTER_NEAREST) @@ -1015,6 +1094,7 @@ _cairo_xlib_surface_set_filter (cairo_xl XRenderSetPictureFilter (surface->dpy, surface->src_picture, (char *) render_filter, NULL, 0); + surface->filter = filter; return CAIRO_STATUS_SUCCESS; } @@ -1028,10 +1108,14 @@ _cairo_xlib_surface_set_repeat (cairo_xl if (!surface->src_picture) return CAIRO_STATUS_SUCCESS; + if (surface->repeat == repeat) + return CAIRO_STATUS_SUCCESS; + mask = CPRepeat; pa.repeat = repeat; XRenderChangePicture (surface->dpy, surface->src_picture, mask, &pa); + surface->repeat = repeat; return CAIRO_STATUS_SUCCESS; } @@ -1327,6 +1411,8 @@ _cairo_xlib_surface_composite (cairo_ope int itx, ity; cairo_bool_t is_integer_translation; + _cairo_xlib_display_notify (dst->screen_info->display); + if (!CAIRO_SURFACE_RENDER_HAS_COMPOSITE (dst)) return CAIRO_INT_STATUS_UNSUPPORTED; @@ -1466,6 +1552,8 @@ _cairo_xlib_surface_fill_rectangles (voi cairo_xlib_surface_t *surface = abstract_surface; XRenderColor render_color; + _cairo_xlib_display_notify (surface->screen_info->display); + if (!CAIRO_SURFACE_RENDER_HAS_FILL_RECTANGLE (surface)) return CAIRO_INT_STATUS_UNSUPPORTED; @@ -1598,6 +1686,8 @@ _cairo_xlib_surface_composite_trapezoids int render_src_x, render_src_y; XRenderPictFormat *pict_format; + _cairo_xlib_display_notify (dst->screen_info->display); + if (!CAIRO_SURFACE_RENDER_HAS_TRAPEZOIDS (dst)) return CAIRO_INT_STATUS_UNSUPPORTED; @@ -1707,6 +1797,8 @@ _cairo_xlib_surface_set_clip_region (voi { cairo_xlib_surface_t *surface = (cairo_xlib_surface_t *) abstract_surface; + _cairo_xlib_display_notify (surface->screen_info->display); + if (surface->clip_rects) { free (surface->clip_rects); surface->clip_rects = NULL; @@ -1793,6 +1885,16 @@ static void _cairo_xlib_surface_scaled_glyph_fini (cairo_scaled_glyph_t *scaled_glyph, cairo_scaled_font_t *scaled_font); +static cairo_bool_t +_cairo_xlib_surface_is_compatible (void *surface_a, + void *surface_b) +{ + cairo_xlib_surface_t *a = (cairo_xlib_surface_t *) surface_a; + cairo_xlib_surface_t *b = (cairo_xlib_surface_t *) surface_b; + + return _cairo_xlib_surface_same_screen (a, b); +} + static const cairo_surface_backend_t cairo_xlib_surface_backend = { CAIRO_SURFACE_TYPE_XLIB, _cairo_xlib_surface_create_similar, @@ -1822,7 +1924,8 @@ static const cairo_surface_backend_t cai NULL, /* stroke */ NULL, /* fill */ _cairo_xlib_surface_show_glyphs, - NULL /* snapshot */ + NULL, /* snapshot */ + _cairo_xlib_surface_is_compatible, }; /** @@ -1839,6 +1942,44 @@ _cairo_surface_is_xlib (cairo_surface_t return surface->backend == &cairo_xlib_surface_backend; } +static void +_cairo_xlib_surface_detach_display (Display *dpy, void *data) +{ + cairo_xlib_surface_t *surface = data; + + surface->dpy = NULL; + + if (surface->dst_picture != None) { + XRenderFreePicture (dpy, surface->dst_picture); + _cairo_xlib_display_cancel_resource (surface->screen_info->display, + surface->dst_picture); + surface->dst_picture = None; + } + + if (surface->src_picture != None) { + XRenderFreePicture (dpy, surface->src_picture); + _cairo_xlib_display_cancel_resource (surface->screen_info->display, + surface->src_picture); + surface->src_picture = None; + } + + if (surface->owns_pixmap) { + XFreePixmap (dpy, surface->drawable); + _cairo_xlib_display_cancel_resource (surface->screen_info->display, + surface->drawable); + surface->drawable = None; + surface->owns_pixmap = FALSE; + } + + if (surface->gc != NULL) { + assert (surface->screen_info != NULL); + XFreeGC (dpy, surface->gc); + _cairo_xlib_display_cancel_work (surface->screen_info->display, + surface->gc); + surface->gc = NULL; + } +} + static cairo_surface_t * _cairo_xlib_surface_create_internal (Display *dpy, Drawable drawable, @@ -1865,6 +2006,14 @@ _cairo_xlib_surface_create_internal (Dis return (cairo_surface_t*) &_cairo_surface_nil; } + if (! _cairo_xlib_add_close_display_hook (dpy, + _cairo_xlib_surface_detach_display, surface, surface)) { + free (surface); + _cairo_xlib_screen_info_destroy (screen_info); + _cairo_error (CAIRO_STATUS_NO_MEMORY); + return (cairo_surface_t*) &_cairo_surface_nil; + } + if (xrender_format) { depth = xrender_format->depth; } else if (visual) { @@ -1935,6 +2084,9 @@ _cairo_xlib_surface_create_internal (Dis surface->visual = visual; surface->xrender_format = xrender_format; surface->depth = depth; + surface->filter = (cairo_filter_t) -1; /* XXX XRender default? */ + surface->repeat = FALSE; + surface->xtransform = identity; surface->have_clip_rects = FALSE; surface->clip_rects = NULL; @@ -2125,14 +2277,26 @@ cairo_xlib_surface_set_drawable (cairo_s return; if (surface->drawable != drawable) { - if (surface->dst_picture) - XRenderFreePicture (surface->dpy, surface->dst_picture); + if (surface->dst_picture != None) { + cairo_status_t status; + status = _cairo_xlib_display_queue_resource ( + surface->screen_info->display, + XRenderFreePicture, + surface->dst_picture); + (void) status; + surface->dst_picture = None; + } + + if (surface->src_picture != None) { + cairo_status_t status; + status = _cairo_xlib_display_queue_resource ( + surface->screen_info->display, + XRenderFreePicture, + surface->src_picture); + (void) status; - if (surface->src_picture) - XRenderFreePicture (surface->dpy, surface->src_picture); - - surface->dst_picture = None; - surface->src_picture = None; + surface->src_picture = None; + } surface->drawable = drawable; } @@ -2310,7 +2474,7 @@ typedef struct _cairo_xlib_surface_font_ static void _cairo_xlib_surface_remove_scaled_font (Display *dpy, - void *data) + void *data) { cairo_scaled_font_t *scaled_font = data; cairo_xlib_surface_font_private_t *font_private; @@ -2322,8 +2486,18 @@ _cairo_xlib_surface_remove_scaled_font ( _cairo_scaled_font_reset_cache (scaled_font); CAIRO_MUTEX_UNLOCK (scaled_font->mutex); - if (font_private) { + if (font_private != NULL) { + cairo_xlib_display_t *display; + XRenderFreeGlyphSet (font_private->dpy, font_private->glyphset); + + display = _cairo_xlib_display_get (dpy); + if (display != NULL) { + _cairo_xlib_display_cancel_resource (display, + font_private->glyphset); + _cairo_xlib_display_destroy (display); + } + free (font_private); } } @@ -2363,12 +2537,34 @@ _cairo_xlib_surface_scaled_font_fini (ca cairo_xlib_surface_font_private_t *font_private = scaled_font->surface_private; if (font_private) { - _cairo_xlib_remove_close_display_hook (font_private->dpy, scaled_font); - XRenderFreeGlyphSet (font_private->dpy, font_private->glyphset); + cairo_xlib_display_t *display; + + _cairo_xlib_remove_close_display_hooks (font_private->dpy, scaled_font); + + display = _cairo_xlib_display_get (font_private->dpy); + if (display != NULL) { + cairo_status_t status = _cairo_xlib_display_queue_resource (display, + XRenderFreeGlyphSet, + font_private->glyphset); + (void) status; /* cannot propagate failure */ + _cairo_xlib_display_destroy (display); + } + free (font_private); } } +struct _cairo_xlib_render_free_glyphs { + GlyphSet glyphset; + unsigned long glyph_index; +}; +static void _cairo_xlib_render_free_glyphs (Display *dpy, struct _cairo_xlib_render_free_glyphs *arg) +{ + XRenderFreeGlyphs (dpy, + arg->glyphset, + &arg->glyph_index, 1); +} + static void _cairo_xlib_surface_scaled_glyph_fini (cairo_scaled_glyph_t *scaled_glyph, cairo_scaled_font_t *scaled_font) @@ -2376,10 +2572,23 @@ _cairo_xlib_surface_scaled_glyph_fini (c cairo_xlib_surface_font_private_t *font_private = scaled_font->surface_private; if (font_private != NULL && scaled_glyph->surface_private != NULL) { - unsigned long glyph_index = _cairo_scaled_glyph_index(scaled_glyph); - XRenderFreeGlyphs (font_private->dpy, - font_private->glyphset, - &glyph_index, 1); + cairo_xlib_display_t *display = _cairo_xlib_display_get (font_private->dpy); + if (display != NULL) { + struct _cairo_xlib_render_free_glyphs *arg = malloc (sizeof (*arg)); + if (arg != NULL) { + cairo_status_t status; + arg->glyphset = font_private->glyphset; + arg->glyph_index = _cairo_scaled_glyph_index (scaled_glyph); + status = _cairo_xlib_display_queue_work (display, + (cairo_xlib_notify_func) _cairo_xlib_render_free_glyphs, + arg, + free); + (void) status; /* cannot propagate failure */ + } + + _cairo_xlib_display_destroy (display); + + } } } @@ -2990,6 +3199,7 @@ _cairo_xlib_surface_show_glyphs (void _cairo_pattern_fini (&solid_pattern.base); BAIL0: _cairo_scaled_font_thaw_cache (scaled_font); + _cairo_xlib_display_notify (dst->screen_info->display); return status; } --- a/src/cairoint.h +++ b/src/cairoint.h @@ -172,6 +172,11 @@ do { \ assert (NOT_REACHED); \ } while (0) +/* Performs a compile time assert */ +#define CAIRO_STATIC_ASSERT(condition) _CAIRO_STATIC_ASSERT_IMPL(condition, __LINE__) +#define _CAIRO_STATIC_ASSERT_IMPL(condition, line) _CAIRO_STATIC_ASSERT_IMPL2(condition, line) +#define _CAIRO_STATIC_ASSERT_IMPL2(condition, line) typedef int _cairo_static_assert_line_##line[(condition) ? 1 : -1] + #define CAIRO_REF_COUNT_INVALID ((unsigned int) -1) #include "cairo-mutex-private.h" @@ -591,12 +596,6 @@ _cairo_font_reset_static_data (void); cairo_private void _cairo_ft_font_reset_static_data (void); -cairo_private void -_cairo_xlib_surface_reset_static_data (void); - -cairo_private void -_cairo_xlib_screen_reset_static_data (void); - /* the font backend interface */ struct _cairo_unscaled_font_backend { @@ -974,6 +973,10 @@ struct _cairo_surface_backend { cairo_surface_t * (*snapshot) (void *surface); + + cairo_bool_t + (*is_compatible) (void *surface_a, + void *surface_b); }; typedef struct _cairo_format_masks { @@ -1848,7 +1851,8 @@ _cairo_surface_create_similar_solid (cai cairo_content_t content, int width, int height, - const cairo_color_t *color); + const cairo_color_t *color, + cairo_pattern_t *pattern); cairo_private void _cairo_surface_init (cairo_surface_t *surface, @@ -1993,6 +1997,10 @@ _cairo_surface_clone_similar (cairo_surf cairo_private cairo_surface_t * _cairo_surface_snapshot (cairo_surface_t *surface); +cairo_private cairo_bool_t +_cairo_surface_is_compatible (cairo_surface_t *surface_a, + cairo_surface_t *surface_b); + cairo_private unsigned int _cairo_surface_get_current_clip_serial (cairo_surface_t *surface); --- a/test/create-for-stream.c +++ b/test/create-for-stream.c @@ -177,12 +177,14 @@ test_surface (const char *filename, if (fread (file_contents, 1, wc.index, fp) != wc.index) { cairo_test_log ("Failed to read %s: %s\n", filename, strerror (errno)); + fclose (fp); return CAIRO_TEST_FAILURE; } if (memcmp (file_contents, wc.buffer, wc.index) != 0) { cairo_test_log ("Stream based output differ from file output for %s\n", filename); + fclose (fp); return CAIRO_TEST_FAILURE; } @@ -195,35 +197,36 @@ test_surface (const char *filename, int main (void) { - cairo_test_status_t status; + cairo_test_status_t status = CAIRO_TEST_SUCCESS; + cairo_test_status_t test_status; cairo_test_init ("create-for-stream"); #if CAIRO_HAS_PS_SURFACE - status = test_surface ("create-for-stream.ps", - cairo_ps_surface_create, - cairo_ps_surface_create_for_stream); - if (status != CAIRO_TEST_SUCCESS) - return status; + test_status = test_surface ("create-for-stream.ps", + cairo_ps_surface_create, + cairo_ps_surface_create_for_stream); + if (status == CAIRO_TEST_SUCCESS) + status = test_status; #endif #if CAIRO_HAS_PDF_SURFACE - status = test_surface ("create-for-stream.pdf", - cairo_pdf_surface_create, - cairo_pdf_surface_create_for_stream); - if (status != CAIRO_TEST_SUCCESS) - return status; + test_status = test_surface ("create-for-stream.pdf", + cairo_pdf_surface_create, + cairo_pdf_surface_create_for_stream); + if (status == CAIRO_TEST_SUCCESS) + status = test_status; #endif #if CAIRO_HAS_SVG_SURFACE - status = test_surface ("create-for-stream.svg", - cairo_svg_surface_create, - cairo_svg_surface_create_for_stream); - if (status != CAIRO_TEST_SUCCESS) - return status; + test_status = test_surface ("create-for-stream.svg", + cairo_svg_surface_create, + cairo_svg_surface_create_for_stream); + if (status == CAIRO_TEST_SUCCESS) + status = test_status; #endif cairo_test_fini (); - return CAIRO_TEST_SUCCESS; + return status; }