Coverage Report

Created: 2025-06-10 07:27

/src/ghostpdl/base/gxpdash.c
Line
Count
Source (jump to first uncovered line)
1
/* Copyright (C) 2001-2023 Artifex Software, Inc.
2
   All Rights Reserved.
3
4
   This software is provided AS-IS with no warranty, either express or
5
   implied.
6
7
   This software is distributed under license and may not be copied,
8
   modified or distributed except as expressly authorized under the terms
9
   of the license contained in the file LICENSE in this distribution.
10
11
   Refer to licensing information at http://www.artifex.com or contact
12
   Artifex Software, Inc.,  39 Mesa Street, Suite 108A, San Francisco,
13
   CA 94129, USA, for further information.
14
*/
15
16
17
/* Dash expansion for paths */
18
#include "math_.h"
19
#include "gx.h"
20
#include "gsmatrix.h"   /* for gscoord.h */
21
#include "gscoord.h"
22
#include "gxfixed.h"
23
#include "gxarith.h"
24
#include "gxgstate.h"
25
#include "gsline.h"
26
#include "gzline.h"
27
#include "gzpath.h"
28
29
/* Expand a dashed path into explicit segments. */
30
/* The path contains no curves. */
31
static int subpath_expand_dashes(const subpath *, gx_path *,
32
                                  const gs_gstate *,
33
                                  const gx_dash_params *);
34
int
35
gx_path_add_dash_expansion(const gx_path * ppath_old, gx_path * ppath,
36
                           const gs_gstate * pgs)
37
1.53k
{
38
1.53k
    const subpath *psub;
39
1.53k
    const gx_dash_params *dash = &gs_currentlineparams(pgs)->dash;
40
1.53k
    int code = 0;
41
42
1.53k
    if (dash->pattern_size == 0)
43
0
        return gx_path_copy(ppath_old, ppath);
44
3.13k
    for (psub = ppath_old->first_subpath; psub != 0 && code >= 0;
45
1.59k
         psub = (const subpath *)psub->last->next
46
1.53k
        )
47
1.59k
        code = subpath_expand_dashes(psub, ppath, pgs, dash);
48
1.53k
    return code;
49
1.53k
}
50
51
static int
52
subpath_expand_dashes(const subpath * psub, gx_path * ppath,
53
                   const gs_gstate * pgs, const gx_dash_params * dash)
54
1.59k
{
55
1.59k
    const float *pattern = dash->pattern;
56
1.59k
    int count, index;
57
1.59k
    bool ink_on;
58
1.59k
    double elt_length;
59
1.59k
    fixed x0 = psub->pt.x, y0 = psub->pt.y;
60
1.59k
    fixed x, y;
61
1.59k
    const segment *pseg;
62
1.59k
    int wrap = (dash->init_ink_on && psub->is_closed ? -1 : 0);
63
1.59k
    int drawing = wrap;
64
1.59k
    segment_notes notes = ~sn_not_first;
65
1.59k
    const gx_line_params *pgs_lp = gs_currentlineparams_inline(pgs);
66
1.59k
    bool zero_length = true;
67
1.59k
    int code;
68
1.59k
    gs_line_cap cap;
69
1.59k
    segment_notes start_notes, end_notes;
70
71
1.59k
    if (wrap) {
72
        /* If we are wrapping around, then we use dash caps throughout */
73
97
        cap         = pgs_lp->dash_cap;
74
97
        start_notes = sn_dash_head;
75
1.49k
    } else {
76
        /* Otherwise, start off with a start cap */
77
1.49k
        cap         = pgs_lp->start_cap;
78
1.49k
        start_notes = 0;
79
1.49k
    }
80
81
1.59k
    if ((code = gx_path_add_point(ppath, x0, y0)) < 0)
82
0
        return code;
83
    /*
84
     * To do the right thing at the beginning of a closed path, we have
85
     * to skip any initial line, and then redo it at the end of the
86
     * path.  Drawing = -1 while skipping, 0 while drawing normally, and
87
     * 1 on the second round.  Note that drawing != 0 implies ink_on.
88
     */
89
1.69k
  top:count = dash->pattern_size;
90
1.69k
    ink_on = dash->init_ink_on;
91
1.69k
    index = dash->init_index;
92
1.69k
    elt_length = dash->init_dist_left;
93
1.69k
    x = x0, y = y0;
94
1.69k
    pseg = (const segment *)psub;
95
20.2k
    while ((pseg = pseg->next) != 0 && pseg->type != s_start) {
96
18.6k
        fixed sx = pseg->pt.x, sy = pseg->pt.y;
97
18.6k
        fixed udx = sx - x, udy = sy - y;
98
18.6k
        double length, dx, dy;
99
18.6k
        double scale = 1;
100
18.6k
        double left;
101
18.6k
        int gap = pseg->type == s_gap;
102
103
18.6k
        if (!(udx | udy)) { /* degenerate */
104
26
            if (pgs_lp->dot_length == 0 &&
105
26
                cap != gs_cap_round) {
106
                /* From PLRM, stroke operator :
107
                   If a subpath is degenerate (consists of a single-point closed path
108
                   or of two or more points at the same coordinates),
109
                   stroke paints it only if round line caps have been specified */
110
26
                if (zero_length || pseg->type != s_line_close)
111
20
                    continue;
112
26
            }
113
6
            dx = 0, dy = 0, length = 0;
114
18.6k
        } else {
115
18.6k
            gs_point d;
116
117
18.6k
            zero_length = false;
118
18.6k
            dx = udx, dy = udy; /* scaled as fixed */
119
18.6k
            code = gs_gstate_idtransform(pgs, dx, dy, &d);
120
18.6k
            if (code < 0) {
121
0
                d.x = 0; d.y = 0;
122
                /* Swallow the error */
123
0
                code = 0;
124
0
            }
125
18.6k
            length = hypot(d.x, d.y) * (1.0 / fixed_1);
126
18.6k
            if (gs_gstate_currentdashadapt(pgs)) {
127
0
                double reps = length / dash->pattern_length;
128
129
0
                scale = reps / ceil(reps);
130
                /* Ensure we're starting at the start of a */
131
                /* repetition.  (This shouldn't be necessary, */
132
                /* but it is.) */
133
0
                count = dash->pattern_size;
134
0
                ink_on = dash->init_ink_on;
135
0
                index = dash->init_index;
136
0
                elt_length = dash->init_dist_left * scale;
137
0
            }
138
18.6k
        }
139
18.6k
        left = length;
140
1.93M
        while (left > elt_length) { /* We are using up the line segment. */
141
1.91M
            double fraction = elt_length / length;
142
1.91M
            fixed fx = (fixed) (dx * fraction);
143
1.91M
            fixed fy = (fixed) (dy * fraction);
144
1.91M
            fixed nx = x + fx;
145
1.91M
            fixed ny = y + fy;
146
147
1.91M
            if (ink_on && !gap) {
148
956k
                if (drawing >= 0) {
149
956k
                    if (left >= elt_length && any_abs(fx) + any_abs(fy) < fixed_half)
150
19.3k
                        code = gx_path_add_dash_notes(ppath, nx, ny, udx, udy,
151
19.3k
                                                      ((notes & pseg->notes)|
152
19.3k
                                                       start_notes|
153
19.3k
                                                       sn_dash_tail));
154
937k
                    else
155
937k
                        code = gx_path_add_line_notes(ppath, nx, ny,
156
937k
                                                      ((notes & pseg->notes)|
157
937k
                                                       start_notes|
158
937k
                                                       sn_dash_tail));
159
956k
                }
160
956k
                notes |= sn_not_first;
161
956k
            } else {
162
955k
                if (drawing > 0)  /* done */
163
93
                    return 0;
164
955k
                code = gx_path_add_point(ppath, nx, ny);
165
955k
                notes &= ~sn_not_first;
166
955k
                drawing = 0;
167
955k
            }
168
1.91M
            if (code < 0)
169
0
                return code;
170
1.91M
            left -= elt_length;
171
1.91M
            ink_on = !ink_on;
172
1.91M
            start_notes = sn_dash_head;
173
1.91M
            if (++index == count)
174
955k
                index = 0;
175
1.91M
            elt_length = pattern[index] * scale;
176
1.91M
            x = nx, y = ny;
177
1.91M
        }
178
18.5k
        elt_length -= left;
179
        /* Handle the last dash of a segment. */
180
18.5k
        if (wrap) {
181
            /* We are wrapping, therefore we always use the dash cap */
182
1.43k
            end_notes = sn_dash_tail;
183
17.1k
        } else {
184
            /* Look ahead to see if we have any more non-degenerate segments
185
             * before the next move or end of subpath. (i.e. should we use an
186
             * end cap or a dash cap?) */
187
17.1k
            const segment *pseg2 = pseg->next;
188
189
17.1k
            end_notes = 0;
190
17.1k
            while (pseg2 != 0 && pseg2->type != s_start)
191
15.6k
            {
192
15.6k
                if ((pseg2->pt.x != sx) || (pseg2->pt.x != sy)) {
193
                    /* Non degenerate. We aren't the last one */
194
15.6k
                    end_notes = sn_dash_tail;
195
15.6k
                    break;
196
15.6k
                }
197
0
                pseg2 = pseg2->next;
198
0
            }
199
17.1k
        }
200
18.5k
      on:if (ink_on && !gap) {
201
9.30k
            if (drawing >= 0) {
202
9.29k
                if (pseg->type == s_line_close && drawing > 0)
203
0
                    code = gx_path_close_subpath_notes(ppath,
204
0
                                                       ((notes & pseg->notes)|
205
0
                                                        start_notes |
206
0
                                                        end_notes));
207
9.29k
                else if ((any_abs(sx - x) + any_abs(sy - y) < fixed_half) &&
208
9.29k
                         (udx | udy))
209
                    /* If we only need to move a short distance, then output
210
                     * dash notes to ensure that the stroke tangent remains
211
                     * accurate. There is no point in outputting such dash
212
                     * notes if we don't have any useful information to put
213
                     * in the note though (if udx == 0 && udy == 0). */
214
715
                    code = gx_path_add_dash_notes(ppath, sx, sy, udx, udy,
215
715
                                                  ((notes & pseg->notes)|
216
715
                                                   start_notes | end_notes));
217
8.57k
                else
218
8.57k
                    code = gx_path_add_line_notes(ppath, sx, sy,
219
8.57k
                                                  ((notes & pseg->notes)|
220
8.57k
                                                   start_notes | end_notes));
221
9.29k
                notes |= sn_not_first;
222
9.29k
            }
223
9.30k
        } else {
224
9.29k
            code = gx_path_add_point(ppath, sx, sy);
225
9.29k
            notes &= ~sn_not_first;
226
9.29k
            if (elt_length < fixed2float(fixed_epsilon) &&
227
9.29k
                (pseg->next == 0 ||
228
38
                 pseg->next->type == s_start ||
229
38
                 pseg->next->type == s_gap ||
230
38
                 elt_length == 0)) {
231
                /*
232
                 * Ink is off, but we're within epsilon of the end
233
                 * of the dash element.
234
                 * "Stretch" a little so we get a dot.
235
                 * Also if the next dash pattern is zero length,
236
                 * use the last segment orientation.
237
                 */
238
38
                double elt_length1;
239
240
38
                if (code < 0)
241
0
                    return code;
242
38
                if (++index == count)
243
38
                    index = 0;
244
38
                elt_length1 = pattern[index] * scale;
245
38
                if (pseg->next == 0 ||
246
38
                    pseg->next->type == s_start ||
247
38
                    pseg->next->type == s_gap) {
248
37
                    elt_length = elt_length1;
249
37
                    left = 0;
250
37
                    ink_on = true;
251
37
                    goto on;
252
37
                }
253
                /* Looking ahead one dash pattern element.
254
                   If it is zero length, apply to the current segment
255
                   (at its end). */
256
1
                if (elt_length1 == 0) {
257
0
                    left = 0;
258
0
                    code = gx_path_add_dash_notes(ppath, sx, sy, udx, udy,
259
0
                                                  ((notes & pseg->notes)|
260
0
                                                  start_notes | end_notes));
261
0
                    if (++index == count)
262
0
                        index = 0;
263
0
                    elt_length = pattern[index] * scale;
264
0
                    ink_on = false;
265
1
                } else if (--index == 0) {
266
                    /* Revert lookahead. */
267
0
                    index = count - 1;
268
0
                }
269
1
            }
270
9.25k
            if (drawing > 0)  /* done */
271
0
                return code;
272
9.25k
            drawing = 0;
273
9.25k
        }
274
18.5k
        if (code < 0)
275
0
            return code;
276
18.5k
        x = sx, y = sy;
277
18.5k
        cap = pgs_lp->dash_cap;
278
18.5k
    }
279
    /* Check for wraparound. */
280
1.59k
    if (wrap && drawing <= 0) { /* We skipped some initial lines. */
281
        /* Go back and do them now. */
282
97
        drawing = 1;
283
97
        goto top;
284
97
    }
285
1.50k
    return 0;
286
1.59k
}