wok-current rev 13173
cairo: Forgot patch.
author | Christopher Rogers <slaxemulator@gmail.com> |
---|---|
date | Mon Jul 30 04:43:16 2012 +0000 (2012-07-30) |
parents | d8228805d4f0 |
children | 27727d07d680 |
files | cairo/stuff/cairo-1.12.2-reduce-broken-stopped-edge-continuation.patch |
line diff
1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 1.2 +++ b/cairo/stuff/cairo-1.12.2-reduce-broken-stopped-edge-continuation.patch Mon Jul 30 04:43:16 2012 +0000 1.3 @@ -0,0 +1,275 @@ 1.4 +From f228769dfe5a8b5d73c49a41e95e31ed73a77fb3 Mon Sep 17 00:00:00 2001 1.5 +From: Chris Wilson <chris@chris-wilson.co.uk> 1.6 +Date: Fri, 08 Jun 2012 16:22:41 +0000 1.7 +Subject: polygon-reduce: Reduce broken stopped-edge continuation 1.8 + 1.9 +This is hopefully a lesser used path and the attempted optimisation to 1.10 +continue a stopped edge with a colinear stopped edge highly unlikely and 1.11 +lost in the noise of the general inefficiency of the routine. As it was 1.12 +broken, rather than attempt to rectify the "optimisation" remove it. 1.13 + 1.14 +Reported-by: Evangelos Foutras <evangelos@foutrelis.com> 1.15 +Bugzilla: https://bugs.freedesktop.org/show_bug.cgi?id=50852 1.16 +Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk> 1.17 +--- 1.18 +diff --git a/src/cairo-polygon-reduce.c b/src/cairo-polygon-reduce.c 1.19 +index 8758070..ea457fe 100644 1.20 +--- a/src/cairo-polygon-reduce.c 1.21 ++++ b/src/cairo-polygon-reduce.c 1.22 +@@ -42,6 +42,8 @@ 1.23 + #include "cairo-freelist-private.h" 1.24 + #include "cairo-combsort-inline.h" 1.25 + 1.26 ++#define DEBUG_POLYGON 0 1.27 ++ 1.28 + typedef cairo_point_t cairo_bo_point32_t; 1.29 + 1.30 + typedef struct _cairo_bo_intersect_ordinate { 1.31 +@@ -114,7 +116,6 @@ typedef struct _cairo_bo_event_queue { 1.32 + 1.33 + typedef struct _cairo_bo_sweep_line { 1.34 + cairo_bo_edge_t *head; 1.35 +- cairo_bo_edge_t *stopped; 1.36 + int32_t current_y; 1.37 + cairo_bo_edge_t *current_edge; 1.38 + } cairo_bo_sweep_line_t; 1.39 +@@ -476,8 +477,8 @@ edges_compare_x_for_y (const cairo_bo_edge_t *a, 1.40 + static inline int 1.41 + _line_equal (const cairo_line_t *a, const cairo_line_t *b) 1.42 + { 1.43 +- return a->p1.x == b->p1.x && a->p1.y == b->p1.y && 1.44 +- a->p2.x == b->p2.x && a->p2.y == b->p2.y; 1.45 ++ return (a->p1.x == b->p1.x && a->p1.y == b->p1.y && 1.46 ++ a->p2.x == b->p2.x && a->p2.y == b->p2.y); 1.47 + } 1.48 + 1.49 + static int 1.50 +@@ -1024,7 +1025,6 @@ static void 1.51 + _cairo_bo_sweep_line_init (cairo_bo_sweep_line_t *sweep_line) 1.52 + { 1.53 + sweep_line->head = NULL; 1.54 +- sweep_line->stopped = NULL; 1.55 + sweep_line->current_y = INT32_MIN; 1.56 + sweep_line->current_edge = NULL; 1.57 + } 1.58 +@@ -1139,6 +1139,8 @@ edges_colinear (const cairo_bo_edge_t *a, const cairo_bo_edge_t *b) 1.59 + */ 1.60 + if (a->edge.line.p1.y == b->edge.line.p1.y) { 1.61 + return a->edge.line.p1.x == b->edge.line.p1.x; 1.62 ++ } else if (a->edge.line.p2.y == b->edge.line.p2.y) { 1.63 ++ return a->edge.line.p2.x == b->edge.line.p2.x; 1.64 + } else if (a->edge.line.p1.y < b->edge.line.p1.y) { 1.65 + return edge_compare_for_y_against_x (b, 1.66 + a->edge.line.p1.y, 1.67 +@@ -1205,82 +1207,48 @@ _active_edges_to_polygon (cairo_bo_edge_t *left, 1.68 + cairo_polygon_t *polygon) 1.69 + { 1.70 + cairo_bo_edge_t *right; 1.71 ++ unsigned int mask; 1.72 + 1.73 +- if (fill_rule == CAIRO_FILL_RULE_WINDING) { 1.74 +- while (left != NULL) { 1.75 +- int in_out = left->edge.dir; 1.76 +- 1.77 +- right = left->next; 1.78 +- if (left->deferred.right == NULL) { 1.79 +- while (right != NULL && right->deferred.right == NULL) 1.80 +- right = right->next; 1.81 +- 1.82 +- if (right != NULL && edges_colinear (left, right)) { 1.83 +- /* continuation on left */ 1.84 +- left->deferred = right->deferred; 1.85 +- right->deferred.right = NULL; 1.86 +- } 1.87 +- } 1.88 +- 1.89 +- right = left->next; 1.90 +- while (right != NULL) { 1.91 +- if (right->deferred.right != NULL) 1.92 +- _cairo_bo_edge_end (right, top, polygon); 1.93 +- 1.94 +- in_out += right->edge.dir; 1.95 +- if (in_out == 0) { 1.96 +- cairo_bo_edge_t *next; 1.97 +- cairo_bool_t skip = FALSE; 1.98 +- 1.99 +- /* skip co-linear edges */ 1.100 +- next = right->next; 1.101 +- if (next != NULL) 1.102 +- skip = edges_colinear (right, next); 1.103 ++ if (fill_rule == CAIRO_FILL_RULE_WINDING) 1.104 ++ mask = ~0; 1.105 ++ else 1.106 ++ mask = 1; 1.107 + 1.108 +- if (! skip) 1.109 +- break; 1.110 +- } 1.111 ++ while (left != NULL) { 1.112 ++ int in_out = left->edge.dir; 1.113 + 1.114 ++ right = left->next; 1.115 ++ if (left->deferred.right == NULL) { 1.116 ++ while (right != NULL && right->deferred.right == NULL) 1.117 + right = right->next; 1.118 +- } 1.119 +- 1.120 +- _cairo_bo_edge_start_or_continue (left, right, top, polygon); 1.121 + 1.122 +- left = right; 1.123 +- if (left != NULL) 1.124 +- left = left->next; 1.125 ++ if (right != NULL && edges_colinear (left, right)) { 1.126 ++ /* continuation on left */ 1.127 ++ left->deferred = right->deferred; 1.128 ++ right->deferred.right = NULL; 1.129 ++ } 1.130 + } 1.131 +- } else { 1.132 +- while (left != NULL) { 1.133 +- int in_out = 0; 1.134 + 1.135 +- right = left->next; 1.136 +- while (right != NULL) { 1.137 +- if (right->deferred.right != NULL) 1.138 +- _cairo_bo_edge_end (right, top, polygon); 1.139 ++ right = left->next; 1.140 ++ while (right != NULL) { 1.141 ++ if (right->deferred.right != NULL) 1.142 ++ _cairo_bo_edge_end (right, top, polygon); 1.143 + 1.144 +- if ((in_out++ & 1) == 0) { 1.145 +- cairo_bo_edge_t *next; 1.146 +- cairo_bool_t skip = FALSE; 1.147 +- 1.148 +- /* skip co-linear edges */ 1.149 +- next = right->next; 1.150 +- if (next != NULL) 1.151 +- skip = edges_colinear (right, next); 1.152 +- 1.153 +- if (! skip) 1.154 +- break; 1.155 +- } 1.156 +- 1.157 +- right = right->next; 1.158 ++ in_out += right->edge.dir; 1.159 ++ if ((in_out & mask) == 0) { 1.160 ++ /* skip co-linear edges */ 1.161 ++ if (right->next == NULL || !edges_colinear (right, right->next)) 1.162 ++ break; 1.163 + } 1.164 + 1.165 +- _cairo_bo_edge_start_or_continue (left, right, top, polygon); 1.166 +- 1.167 +- left = right; 1.168 +- if (left != NULL) 1.169 +- left = left->next; 1.170 ++ right = right->next; 1.171 + } 1.172 ++ 1.173 ++ _cairo_bo_edge_start_or_continue (left, right, top, polygon); 1.174 ++ 1.175 ++ left = right; 1.176 ++ if (left != NULL) 1.177 ++ left = left->next; 1.178 + } 1.179 + } 1.180 + 1.181 +@@ -1303,12 +1271,6 @@ _cairo_bentley_ottmann_tessellate_bo_edges (cairo_bo_event_t **start_events, 1.182 + 1.183 + while ((event = _cairo_bo_event_dequeue (&event_queue))) { 1.184 + if (event->point.y != sweep_line.current_y) { 1.185 +- for (e1 = sweep_line.stopped; e1; e1 = e1->next) { 1.186 +- if (e1->deferred.right != NULL) 1.187 +- _cairo_bo_edge_end (e1, e1->edge.bottom, polygon); 1.188 +- } 1.189 +- sweep_line.stopped = NULL; 1.190 +- 1.191 + _active_edges_to_polygon (sweep_line.head, 1.192 + sweep_line.current_y, 1.193 + fill_rule, polygon); 1.194 +@@ -1328,23 +1290,6 @@ _cairo_bentley_ottmann_tessellate_bo_edges (cairo_bo_event_t **start_events, 1.195 + if (unlikely (status)) 1.196 + goto unwind; 1.197 + 1.198 +- /* check to see if this is a continuation of a stopped edge */ 1.199 +- /* XXX change to an infinitesimal lengthening rule */ 1.200 +- for (left = sweep_line.stopped; left; left = left->next) { 1.201 +- if (e1->edge.top <= left->edge.bottom && 1.202 +- edges_colinear (e1, left)) 1.203 +- { 1.204 +- e1->deferred = left->deferred; 1.205 +- if (left->prev != NULL) 1.206 +- left->prev = left->next; 1.207 +- else 1.208 +- sweep_line.stopped = left->next; 1.209 +- if (left->next != NULL) 1.210 +- left->next->prev = left->prev; 1.211 +- break; 1.212 +- } 1.213 +- } 1.214 +- 1.215 + left = e1->prev; 1.216 + right = e1->next; 1.217 + 1.218 +@@ -1371,14 +1316,8 @@ _cairo_bentley_ottmann_tessellate_bo_edges (cairo_bo_event_t **start_events, 1.219 + 1.220 + _cairo_bo_sweep_line_delete (&sweep_line, e1); 1.221 + 1.222 +- /* first, check to see if we have a continuation via a fresh edge */ 1.223 +- if (e1->deferred.right != NULL) { 1.224 +- e1->next = sweep_line.stopped; 1.225 +- if (sweep_line.stopped != NULL) 1.226 +- sweep_line.stopped->prev = e1; 1.227 +- sweep_line.stopped = e1; 1.228 +- e1->prev = NULL; 1.229 +- } 1.230 ++ if (e1->deferred.right != NULL) 1.231 ++ _cairo_bo_edge_end (e1, e1->edge.bottom, polygon); 1.232 + 1.233 + if (left != NULL && right != NULL) { 1.234 + status = _cairo_bo_event_queue_insert_if_intersect_below_current_y (&event_queue, left, right); 1.235 +@@ -1420,10 +1359,6 @@ _cairo_bentley_ottmann_tessellate_bo_edges (cairo_bo_event_t **start_events, 1.236 + } 1.237 + } 1.238 + 1.239 +- for (e1 = sweep_line.stopped; e1; e1 = e1->next) { 1.240 +- if (e1->deferred.right != NULL) 1.241 +- _cairo_bo_edge_end (e1, e1->edge.bottom, polygon); 1.242 +- } 1.243 + unwind: 1.244 + _cairo_bo_event_queue_fini (&event_queue); 1.245 + 1.246 +@@ -1447,6 +1382,12 @@ _cairo_polygon_reduce (cairo_polygon_t *polygon, 1.247 + if (unlikely (0 == num_events)) 1.248 + return CAIRO_STATUS_SUCCESS; 1.249 + 1.250 ++ if (DEBUG_POLYGON) { 1.251 ++ FILE *file = fopen ("reduce_in.txt", "w"); 1.252 ++ _cairo_debug_print_polygon (file, polygon); 1.253 ++ fclose (file); 1.254 ++ } 1.255 ++ 1.256 + events = stack_events; 1.257 + event_ptrs = stack_event_ptrs; 1.258 + if (num_events > ARRAY_LENGTH (stack_events)) { 1.259 +@@ -1482,10 +1423,16 @@ _cairo_polygon_reduce (cairo_polygon_t *polygon, 1.260 + num_events, 1.261 + fill_rule, 1.262 + polygon); 1.263 +- polygon->num_limits = num_limits; 1.264 ++ polygon->num_limits = num_limits; 1.265 + 1.266 + if (events != stack_events) 1.267 + free (events); 1.268 + 1.269 ++ if (DEBUG_POLYGON) { 1.270 ++ FILE *file = fopen ("reduce_out.txt", "w"); 1.271 ++ _cairo_debug_print_polygon (file, polygon); 1.272 ++ fclose (file); 1.273 ++ } 1.274 ++ 1.275 + return status; 1.276 + } 1.277 +-- 1.278 +cgit v0.9.0.2-2-gbebe