1 /* cairo - a vector graphics library with display and print output
3 * Copyright © 2009 Intel Corporation
5 * This library is free software; you can redistribute it and/or
6 * modify it either under the terms of the GNU Lesser General Public
7 * License version 2.1 as published by the Free Software Foundation
8 * (the "LGPL") or, at your option, under the terms of the Mozilla
9 * Public License Version 1.1 (the "MPL"). If you do not alter this
10 * notice, a recipient may use your version of this file under either
11 * the MPL or the LGPL.
13 * You should have received a copy of the LGPL along with this library
14 * in the file COPYING-LGPL-2.1; if not, write to the Free Software
15 * Foundation, Inc., 51 Franklin Street, Suite 500, Boston, MA 02110-1335, USA
16 * You should have received a copy of the MPL along with this library
17 * in the file COPYING-MPL-1.1
19 * The contents of this file are subject to the Mozilla Public License
20 * Version 1.1 (the "License"); you may not use this file except in
21 * compliance with the License. You may obtain a copy of the License at
22 * http://www.mozilla.org/MPL/
24 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY
25 * OF ANY KIND, either express or implied. See the LGPL or the MPL for
26 * the specific language governing rights and limitations.
28 * The Original Code is the cairo graphics library.
30 * The Initial Developer of the Original Code is Red Hat, Inc.
33 * Chris Wilson <chris@chris-wilson.co.uk>
38 #include "cairo-clip-inline.h"
39 #include "cairo-error-private.h"
40 #include "cairo-composite-rectangles-private.h"
41 #include "cairo-pattern-private.h"
43 /* A collection of routines to facilitate writing compositors. */
45 void _cairo_composite_rectangles_fini (cairo_composite_rectangles_t
*extents
)
47 _cairo_clip_destroy (extents
->clip
);
51 _cairo_composite_reduce_pattern (const cairo_pattern_t
*src
,
52 cairo_pattern_union_t
*dst
)
56 _cairo_pattern_init_static_copy (&dst
->base
, src
);
57 if (dst
->base
.type
== CAIRO_PATTERN_TYPE_SOLID
)
60 dst
->base
.filter
= _cairo_pattern_analyze_filter (&dst
->base
);
63 if (_cairo_matrix_is_pixman_translation (&dst
->base
.matrix
,
67 dst
->base
.matrix
.x0
= tx
;
68 dst
->base
.matrix
.y0
= ty
;
72 static inline cairo_bool_t
73 _cairo_composite_rectangles_init (cairo_composite_rectangles_t
*extents
,
74 cairo_surface_t
*surface
,
76 const cairo_pattern_t
*source
,
77 const cairo_clip_t
*clip
)
79 if (_cairo_clip_is_all_clipped (clip
))
82 extents
->surface
= surface
;
85 _cairo_surface_get_extents (surface
, &extents
->destination
);
88 extents
->unbounded
= extents
->destination
;
89 if (clip
&& ! _cairo_rectangle_intersect (&extents
->unbounded
,
90 _cairo_clip_get_extents (clip
)))
93 extents
->bounded
= extents
->unbounded
;
94 extents
->is_bounded
= _cairo_operator_bounded_by_either (op
);
96 extents
->original_source_pattern
= source
;
97 _cairo_composite_reduce_pattern (source
, &extents
->source_pattern
);
99 _cairo_pattern_get_extents (&extents
->source_pattern
.base
,
101 if (extents
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_SOURCE
) {
102 if (! _cairo_rectangle_intersect (&extents
->bounded
, &extents
->source
))
106 extents
->original_mask_pattern
= NULL
;
107 extents
->mask_pattern
.base
.type
= CAIRO_PATTERN_TYPE_SOLID
;
108 extents
->mask_pattern
.solid
.color
.alpha
= 1.; /* XXX full initialisation? */
109 extents
->mask_pattern
.solid
.color
.alpha_short
= 0xffff;
115 _cairo_composite_rectangles_init_for_paint (cairo_composite_rectangles_t
*extents
,
116 cairo_surface_t
*surface
,
118 const cairo_pattern_t
*source
,
119 const cairo_clip_t
*clip
)
121 if (! _cairo_composite_rectangles_init (extents
,
122 surface
, op
, source
, clip
))
124 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
127 extents
->mask
= extents
->destination
;
129 extents
->clip
= _cairo_clip_reduce_for_composite (clip
, extents
);
130 if (_cairo_clip_is_all_clipped (extents
->clip
))
131 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
133 if (! _cairo_rectangle_intersect (&extents
->unbounded
,
134 _cairo_clip_get_extents (extents
->clip
)))
135 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
137 if (extents
->source_pattern
.base
.type
!= CAIRO_PATTERN_TYPE_SOLID
)
138 _cairo_pattern_sampled_area (&extents
->source_pattern
.base
,
140 &extents
->source_sample_area
);
142 return CAIRO_STATUS_SUCCESS
;
145 static cairo_int_status_t
146 _cairo_composite_rectangles_intersect (cairo_composite_rectangles_t
*extents
,
147 const cairo_clip_t
*clip
)
151 ret
= _cairo_rectangle_intersect (&extents
->bounded
, &extents
->mask
);
152 if (! ret
&& extents
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_MASK
)
153 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
155 if (extents
->is_bounded
== (CAIRO_OPERATOR_BOUND_BY_MASK
| CAIRO_OPERATOR_BOUND_BY_SOURCE
)) {
156 extents
->unbounded
= extents
->bounded
;
157 } else if (extents
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_MASK
) {
158 if (!_cairo_rectangle_intersect (&extents
->unbounded
, &extents
->mask
))
159 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
162 extents
->clip
= _cairo_clip_reduce_for_composite (clip
, extents
);
163 if (_cairo_clip_is_all_clipped (extents
->clip
))
164 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
166 if (! _cairo_rectangle_intersect (&extents
->unbounded
,
167 _cairo_clip_get_extents (extents
->clip
)))
168 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
170 if (! _cairo_rectangle_intersect (&extents
->bounded
,
171 _cairo_clip_get_extents (extents
->clip
)) &&
172 extents
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_MASK
)
174 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
177 if (extents
->source_pattern
.base
.type
!= CAIRO_PATTERN_TYPE_SOLID
)
178 _cairo_pattern_sampled_area (&extents
->source_pattern
.base
,
180 &extents
->source_sample_area
);
181 if (extents
->mask_pattern
.base
.type
!= CAIRO_PATTERN_TYPE_SOLID
) {
182 _cairo_pattern_sampled_area (&extents
->mask_pattern
.base
,
184 &extents
->mask_sample_area
);
185 if (extents
->mask_sample_area
.width
== 0 ||
186 extents
->mask_sample_area
.height
== 0) {
187 _cairo_composite_rectangles_fini (extents
);
188 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
192 return CAIRO_STATUS_SUCCESS
;
196 _cairo_composite_rectangles_intersect_source_extents (cairo_composite_rectangles_t
*extents
,
197 const cairo_box_t
*box
)
199 cairo_rectangle_int_t rect
;
202 _cairo_box_round_to_rectangle (box
, &rect
);
203 if (rect
.x
== extents
->source
.x
&&
204 rect
.y
== extents
->source
.y
&&
205 rect
.width
== extents
->source
.width
&&
206 rect
.height
== extents
->source
.height
)
208 return CAIRO_INT_STATUS_SUCCESS
;
211 _cairo_rectangle_intersect (&extents
->source
, &rect
);
213 rect
= extents
->bounded
;
214 if (! _cairo_rectangle_intersect (&extents
->bounded
, &extents
->source
) &&
215 extents
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_SOURCE
)
216 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
218 if (rect
.width
== extents
->bounded
.width
&&
219 rect
.height
== extents
->bounded
.height
)
220 return CAIRO_INT_STATUS_SUCCESS
;
222 if (extents
->is_bounded
== (CAIRO_OPERATOR_BOUND_BY_MASK
| CAIRO_OPERATOR_BOUND_BY_SOURCE
)) {
223 extents
->unbounded
= extents
->bounded
;
224 } else if (extents
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_MASK
) {
225 if (!_cairo_rectangle_intersect (&extents
->unbounded
, &extents
->mask
))
226 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
229 clip
= extents
->clip
;
230 extents
->clip
= _cairo_clip_reduce_for_composite (clip
, extents
);
231 if (clip
!= extents
->clip
)
232 _cairo_clip_destroy (clip
);
234 if (_cairo_clip_is_all_clipped (extents
->clip
))
235 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
237 if (! _cairo_rectangle_intersect (&extents
->unbounded
,
238 _cairo_clip_get_extents (extents
->clip
)))
239 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
241 if (extents
->source_pattern
.base
.type
!= CAIRO_PATTERN_TYPE_SOLID
)
242 _cairo_pattern_sampled_area (&extents
->source_pattern
.base
,
244 &extents
->source_sample_area
);
245 if (extents
->mask_pattern
.base
.type
!= CAIRO_PATTERN_TYPE_SOLID
) {
246 _cairo_pattern_sampled_area (&extents
->mask_pattern
.base
,
248 &extents
->mask_sample_area
);
249 if (extents
->mask_sample_area
.width
== 0 ||
250 extents
->mask_sample_area
.height
== 0)
251 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
254 return CAIRO_INT_STATUS_SUCCESS
;
258 _cairo_composite_rectangles_intersect_mask_extents (cairo_composite_rectangles_t
*extents
,
259 const cairo_box_t
*box
)
261 cairo_rectangle_int_t mask
;
264 _cairo_box_round_to_rectangle (box
, &mask
);
265 if (mask
.x
== extents
->mask
.x
&&
266 mask
.y
== extents
->mask
.y
&&
267 mask
.width
== extents
->mask
.width
&&
268 mask
.height
== extents
->mask
.height
)
270 return CAIRO_INT_STATUS_SUCCESS
;
273 _cairo_rectangle_intersect (&extents
->mask
, &mask
);
275 mask
= extents
->bounded
;
276 if (! _cairo_rectangle_intersect (&extents
->bounded
, &extents
->mask
) &&
277 extents
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_MASK
)
278 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
280 if (mask
.width
== extents
->bounded
.width
&&
281 mask
.height
== extents
->bounded
.height
)
282 return CAIRO_INT_STATUS_SUCCESS
;
284 if (extents
->is_bounded
== (CAIRO_OPERATOR_BOUND_BY_MASK
| CAIRO_OPERATOR_BOUND_BY_SOURCE
)) {
285 extents
->unbounded
= extents
->bounded
;
286 } else if (extents
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_MASK
) {
287 if (!_cairo_rectangle_intersect (&extents
->unbounded
, &extents
->mask
))
288 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
291 clip
= extents
->clip
;
292 extents
->clip
= _cairo_clip_reduce_for_composite (clip
, extents
);
293 if (clip
!= extents
->clip
)
294 _cairo_clip_destroy (clip
);
296 if (_cairo_clip_is_all_clipped (extents
->clip
))
297 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
299 if (! _cairo_rectangle_intersect (&extents
->unbounded
,
300 _cairo_clip_get_extents (extents
->clip
)))
301 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
303 if (extents
->source_pattern
.base
.type
!= CAIRO_PATTERN_TYPE_SOLID
)
304 _cairo_pattern_sampled_area (&extents
->source_pattern
.base
,
306 &extents
->source_sample_area
);
307 if (extents
->mask_pattern
.base
.type
!= CAIRO_PATTERN_TYPE_SOLID
) {
308 _cairo_pattern_sampled_area (&extents
->mask_pattern
.base
,
310 &extents
->mask_sample_area
);
311 if (extents
->mask_sample_area
.width
== 0 ||
312 extents
->mask_sample_area
.height
== 0)
313 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
316 return CAIRO_INT_STATUS_SUCCESS
;
320 _cairo_composite_rectangles_init_for_mask (cairo_composite_rectangles_t
*extents
,
321 cairo_surface_t
*surface
,
323 const cairo_pattern_t
*source
,
324 const cairo_pattern_t
*mask
,
325 const cairo_clip_t
*clip
)
327 if (! _cairo_composite_rectangles_init (extents
,
328 surface
, op
, source
, clip
))
330 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
333 extents
->original_mask_pattern
= mask
;
334 _cairo_composite_reduce_pattern (mask
, &extents
->mask_pattern
);
335 _cairo_pattern_get_extents (&extents
->mask_pattern
.base
, &extents
->mask
);
337 return _cairo_composite_rectangles_intersect (extents
, clip
);
341 _cairo_composite_rectangles_init_for_stroke (cairo_composite_rectangles_t
*extents
,
342 cairo_surface_t
*surface
,
344 const cairo_pattern_t
*source
,
345 const cairo_path_fixed_t
*path
,
346 const cairo_stroke_style_t
*style
,
347 const cairo_matrix_t
*ctm
,
348 const cairo_clip_t
*clip
)
350 if (! _cairo_composite_rectangles_init (extents
,
351 surface
, op
, source
, clip
))
353 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
356 _cairo_path_fixed_approximate_stroke_extents (path
, style
, ctm
, &extents
->mask
);
358 return _cairo_composite_rectangles_intersect (extents
, clip
);
362 _cairo_composite_rectangles_init_for_fill (cairo_composite_rectangles_t
*extents
,
363 cairo_surface_t
*surface
,
365 const cairo_pattern_t
*source
,
366 const cairo_path_fixed_t
*path
,
367 const cairo_clip_t
*clip
)
369 if (! _cairo_composite_rectangles_init (extents
,
370 surface
, op
, source
, clip
))
372 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
375 _cairo_path_fixed_approximate_fill_extents (path
, &extents
->mask
);
377 return _cairo_composite_rectangles_intersect (extents
, clip
);
381 _cairo_composite_rectangles_init_for_polygon (cairo_composite_rectangles_t
*extents
,
382 cairo_surface_t
*surface
,
384 const cairo_pattern_t
*source
,
385 const cairo_polygon_t
*polygon
,
386 const cairo_clip_t
*clip
)
388 if (! _cairo_composite_rectangles_init (extents
,
389 surface
, op
, source
, clip
))
391 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
394 _cairo_box_round_to_rectangle (&polygon
->extents
, &extents
->mask
);
395 return _cairo_composite_rectangles_intersect (extents
, clip
);
399 _cairo_composite_rectangles_init_for_boxes (cairo_composite_rectangles_t
*extents
,
400 cairo_surface_t
*surface
,
402 const cairo_pattern_t
*source
,
403 const cairo_boxes_t
*boxes
,
404 const cairo_clip_t
*clip
)
408 if (! _cairo_composite_rectangles_init (extents
,
409 surface
, op
, source
, clip
))
411 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
414 _cairo_boxes_extents (boxes
, &box
);
415 _cairo_box_round_to_rectangle (&box
, &extents
->mask
);
416 return _cairo_composite_rectangles_intersect (extents
, clip
);
420 _cairo_composite_rectangles_init_for_glyphs (cairo_composite_rectangles_t
*extents
,
421 cairo_surface_t
*surface
,
423 const cairo_pattern_t
*source
,
424 cairo_scaled_font_t
*scaled_font
,
425 cairo_glyph_t
*glyphs
,
427 const cairo_clip_t
*clip
,
428 cairo_bool_t
*overlap
)
430 cairo_status_t status
;
432 if (! _cairo_composite_rectangles_init (extents
, surface
, op
, source
, clip
))
433 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
435 /* Computing the exact bbox and the overlap is expensive.
436 * First perform a cheap test to see if the glyphs are all clipped out.
438 if (extents
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_MASK
&&
439 _cairo_scaled_font_glyph_approximate_extents (scaled_font
,
443 if (! _cairo_rectangle_intersect (&extents
->bounded
, &extents
->mask
))
444 return CAIRO_INT_STATUS_NOTHING_TO_DO
;
447 status
= _cairo_scaled_font_glyph_device_extents (scaled_font
,
451 if (unlikely (status
))
454 if (overlap
&& *overlap
&&
455 scaled_font
->options
.antialias
== CAIRO_ANTIALIAS_NONE
&&
456 _cairo_pattern_is_opaque_solid (&extents
->source_pattern
.base
))
461 return _cairo_composite_rectangles_intersect (extents
, clip
);
465 _cairo_composite_rectangles_can_reduce_clip (cairo_composite_rectangles_t
*composite
,
468 cairo_rectangle_int_t extents
;
474 extents
= composite
->destination
;
475 if (composite
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_SOURCE
)
476 _cairo_rectangle_intersect (&extents
, &composite
->source
);
477 if (composite
->is_bounded
& CAIRO_OPERATOR_BOUND_BY_MASK
)
478 _cairo_rectangle_intersect (&extents
, &composite
->mask
);
480 _cairo_box_from_rectangle (&box
, &extents
);
481 return _cairo_clip_contains_box (clip
, &box
);
485 _cairo_composite_rectangles_add_to_damage (cairo_composite_rectangles_t
*composite
,
486 cairo_boxes_t
*damage
)
488 cairo_int_status_t status
;
491 for (n
= 0; n
< composite
->clip
->num_boxes
; n
++) {
492 status
= _cairo_boxes_add (damage
,
493 CAIRO_ANTIALIAS_NONE
,
494 &composite
->clip
->boxes
[n
]);
495 if (unlikely (status
))
499 return CAIRO_INT_STATUS_SUCCESS
;