1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
|
/* Copyright (C) 2001-2006 Artifex Software, Inc.
All Rights Reserved.
This software is provided AS-IS with no warranty, either express or
implied.
This software is distributed under license and may not be copied, modified
or distributed except as expressly authorized under the terms of that
license. Refer to licensing information at http://www.artifex.com/
or contact Artifex Software, Inc., 7 Mt. Lassen Drive - Suite A-134,
San Rafael, CA 94903, U.S.A., +1(415)492-9861, for further information.
*/
/* $Id$ */
/* Common structures for the filling algorithm and dropout prevention. */
#ifndef gxfill_INCLUDED
# define gxfill_INCLUDED
/* Define the structure for keeping track of active lines. */
#ifndef active_line_DEFINED
# define active_line_DEFINED
typedef struct active_line_s active_line;
#endif
struct active_line_s {
gs_fixed_point start; /* x,y where line starts */
gs_fixed_point end; /* x,y where line ends */
gs_fixed_point diff; /* end - start */
fixed y_fast_max; /* can do x_at_y in fixed point */
/* if y <= y_fast_max */
fixed num_adjust; /* 0 if diff.x >= 0, -diff.y + epsilon if */
/* diff.x < 0 and division truncates */
#if ARCH_DIV_NEG_POS_TRUNCATES
/* neg/pos truncates, we must bias the numberator. */
# define SET_NUM_ADJUST(alp) \
(alp)->num_adjust =\
((alp)->diff.x >= 0 ? 0 : -(alp)->diff.y + fixed_epsilon)
# define ADD_NUM_ADJUST(num, alp) ((num) + (alp)->num_adjust)
# define MAX_MINUS_NUM_ADJUST(alp) ADD_NUM_ADJUST(max_fixed, alp)
#else
/* neg/pos takes the floor, no special action is needed. */
# define SET_NUM_ADJUST(alp) DO_NOTHING
# define ADD_NUM_ADJUST(num, alp) (num)
# define MAX_MINUS_NUM_ADJUST(alp) max_fixed
#endif
#define SET_AL_POINTS(alp, startp, endp)\
BEGIN\
(alp)->diff.y = (endp).y - (startp).y;\
(alp)->diff.x = (endp).x - (startp).x;\
SET_NUM_ADJUST(alp);\
(alp)->y_fast_max = MAX_MINUS_NUM_ADJUST(alp) /\
(((alp)->diff.x >= 0 ? (alp)->diff.x : -(alp)->diff.x) | 1) +\
(startp).y;\
(alp)->start = startp, (alp)->end = endp;\
END
/*
* We know that alp->start.y <= yv <= alp->end.y, because the fill loop
* guarantees that the only lines being considered are those with this
* property.
*/
#define AL_X_AT_Y(alp, yv)\
((yv) == (alp)->end.y ? (alp)->end.x :\
((yv) <= (alp)->y_fast_max ?\
ADD_NUM_ADJUST(((yv) - (alp)->start.y) * (alp)->diff.x, alp) / (alp)->diff.y :\
(INCR_EXPR(slow_x),\
fixed_mult_quo((alp)->diff.x, (yv) - (alp)->start.y, (alp)->diff.y))) +\
(alp)->start.x)
fixed x_current; /* current x position */
fixed x_next; /* x position at end of band */
const segment *pseg; /* endpoint of this line */
int direction; /* direction of line segment */
#define DIR_UP 1
#define DIR_HORIZONTAL 0 /* (these are handled specially) */
#define DIR_DOWN (-1)
bool monotonic_x; /* "false" means "don't know"; only for scanline. */
bool monotonic_y; /* "false" means "don't know"; only for scanline. */
gx_flattened_iterator fi;
bool more_flattened;
int contour_count;
/*
* "Pending" lines (not reached in the Y ordering yet) use next and prev
* to order lines by increasing starting Y. "Active" lines (being scanned)
* use next and prev to order lines by increasing current X, or if the
* current Xs are equal, by increasing final X.
*/
active_line *prev, *next;
/* Link together active_lines allocated individually */
active_line *alloc_next;
};
typedef struct fill_options_s {
bool pseudo_rasterization; /* See comment about "pseudo-rasterization". */
fixed ymin, ymax;
const gx_device_color * pdevc;
gs_logical_operation_t lop;
bool fill_direct;
fixed fixed_flat;
bool fill_by_trapezoids;
fixed adjust_left, adjust_right;
fixed adjust_below, adjust_above;
gx_device *dev;
const gs_fixed_rect * pbox;
bool is_spotan;
int rule;
dev_proc_fill_rectangle((*fill_rect));
dev_proc_fill_trapezoid((*fill_trap));
} fill_options;
/* Line list structure */
#ifndef line_list_DEFINED
# define line_list_DEFINED
typedef struct line_list_s line_list;
#endif
struct line_list_s {
gs_memory_t *memory;
active_line *active_area; /* allocated active_line list */
active_line *next_active; /* next allocation slot */
active_line *limit; /* limit of local allocation */
int close_count; /* # of added closing lines */
active_line *y_list; /* Y-sorted list of pending lines */
active_line *y_line; /* most recently inserted line */
active_line x_head; /* X-sorted list of active lines */
#define x_list x_head.next
active_line *h_list0, *h_list1; /* lists of horizontal lines for y, y1 */
margin_set margin_set0, margin_set1;
margin *free_margin_list;
int *windings;
int local_margin_alloc_count;
int bbox_left, bbox_width;
int main_dir;
fixed y_break;
const fill_options * const fo;
int contour_count;
/* Put the arrays last so the scalars will have */
/* small displacements. */
/* Allocate a few active_lines locally */
/* to avoid round trips through the allocator. */
#if arch_small_memory
# define MAX_LOCAL_ACTIVE 6 /* don't overburden the stack */
# define MAX_LOCAL_SECTION 50
#else
# define MAX_LOCAL_ACTIVE 20
# define MAX_LOCAL_SECTION 100
#endif
active_line local_active[MAX_LOCAL_ACTIVE];
margin local_margins[MAX_LOCAL_ACTIVE];
section local_section0[MAX_LOCAL_SECTION];
section local_section1[MAX_LOCAL_SECTION];
int local_windings[MAX_LOCAL_ACTIVE];
};
#define LOOP_FILL_RECTANGLE_DIRECT(fo, x, y, w, h)\
(/*fo->fill_direct*/FILL_DIRECT ?\
(fo)->fill_rect((fo)->dev, x, y, w, h, (fo)->pdevc->colors.pure) :\
gx_fill_rectangle_device_rop(x, y, w, h, (fo)->pdevc, (fo)->dev, (fo)->lop))
/* ---------------- Statistics ---------------- */
#ifdef DEBUG
struct stats_fill_s {
long
fill, fill_alloc, y_up, y_down, horiz, x_step, slow_x, iter, find_y,
band, band_step, band_fill, afill, slant, slant_shallow, sfill,
mq_cross, cross_slow, cross_low, order, slow_order;
};
typedef struct stats_fill_s stats_fill_t;
extern stats_fill_t stats_fill;
# define INCR(x) (++(stats_fill.x))
# define INCR_EXPR(x) INCR(x)
# define INCR_BY(x,n) (stats_fill.x += (n))
#else
# define INCR(x) DO_NOTHING
# define INCR_EXPR(x) discard(0)
# define INCR_BY(x,n) DO_NOTHING
#endif
#endif /* gxfill_INCLUDED */
|