Coverage Report

Created: 2026-02-04 06:14

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/xpdf-4.06/splash/SplashPath.cc
Line
Count
Source
1
//========================================================================
2
//
3
// SplashPath.cc
4
//
5
// Copyright 2003-2013 Glyph & Cog, LLC
6
//
7
//========================================================================
8
9
#include <aconf.h>
10
11
#include <string.h>
12
#include "gmem.h"
13
#include "gmempp.h"
14
#include "SplashErrorCodes.h"
15
#include "SplashPath.h"
16
17
//------------------------------------------------------------------------
18
// SplashPath
19
//------------------------------------------------------------------------
20
21
// A path can be in three possible states:
22
//
23
// 1. no current point -- zero or more finished subpaths
24
//    [curSubpath == length]
25
//
26
// 2. one point in subpath
27
//    [curSubpath == length - 1]
28
//
29
// 3. open subpath with two or more points
30
//    [curSubpath < length - 1]
31
32
1.22M
SplashPath::SplashPath() {
33
1.22M
  pts = NULL;
34
1.22M
  flags = NULL;
35
1.22M
  length = size = 0;
36
1.22M
  curSubpath = 0;
37
1.22M
  hints = NULL;
38
1.22M
  hintsLength = hintsSize = 0;
39
1.22M
}
40
41
0
SplashPath::SplashPath(SplashPath *path) {
42
0
  length = path->length;
43
0
  size = path->size;
44
0
  pts = (SplashPathPoint *)gmallocn(size, sizeof(SplashPathPoint));
45
0
  flags = (Guchar *)gmallocn(size, sizeof(Guchar));
46
0
  memcpy(pts, path->pts, length * sizeof(SplashPathPoint));
47
0
  memcpy(flags, path->flags, length * sizeof(Guchar));
48
0
  curSubpath = path->curSubpath;
49
0
  if (path->hints) {
50
0
    hintsLength = hintsSize = path->hintsLength;
51
0
    hints = (SplashPathHint *)gmallocn(hintsSize, sizeof(SplashPathHint));
52
0
    memcpy(hints, path->hints, hintsLength * sizeof(SplashPathHint));
53
0
  } else {
54
0
    hints = NULL;
55
0
    hintsLength = hintsSize = 0;
56
0
  }
57
0
}
58
59
1.22M
SplashPath::~SplashPath() {
60
1.22M
  gfree(pts);
61
1.22M
  gfree(flags);
62
1.22M
  gfree(hints);
63
1.22M
}
64
65
// Add space for <nPts> more points.
66
95.7M
void SplashPath::grow(int nPts) {
67
95.7M
  if (length + nPts > size) {
68
1.25M
    if (size == 0) {
69
1.16M
      size = 32;
70
1.16M
    }
71
1.34M
    while (size < length + nPts) {
72
90.6k
      size *= 2;
73
90.6k
    }
74
1.25M
    pts = (SplashPathPoint *)greallocn(pts, size, sizeof(SplashPathPoint));
75
1.25M
    flags = (Guchar *)greallocn(flags, size, sizeof(Guchar));
76
1.25M
  }
77
95.7M
}
78
79
285k
void SplashPath::append(SplashPath *path) {
80
285k
  int i;
81
82
285k
  curSubpath = length + path->curSubpath;
83
285k
  grow(path->length);
84
5.71M
  for (i = 0; i < path->length; ++i) {
85
5.43M
    pts[length] = path->pts[i];
86
5.43M
    flags[length] = path->flags[i];
87
5.43M
    ++length;
88
5.43M
  }
89
285k
}
90
91
20.1M
SplashError SplashPath::moveTo(SplashCoord x, SplashCoord y) {
92
20.1M
  if (onePointSubpath()) {
93
0
    return splashErrBogusPath;
94
0
  }
95
20.1M
  grow(1);
96
20.1M
  pts[length].x = x;
97
20.1M
  pts[length].y = y;
98
20.1M
  flags[length] = splashPathFirst | splashPathLast;
99
20.1M
  curSubpath = length++;
100
20.1M
  return splashOk;
101
20.1M
}
102
103
71.2M
SplashError SplashPath::lineTo(SplashCoord x, SplashCoord y) {
104
71.2M
  if (noCurrentPoint()) {
105
0
    return splashErrNoCurPt;
106
0
  }
107
71.2M
  flags[length-1] &= (Guchar)~splashPathLast;
108
71.2M
  grow(1);
109
71.2M
  pts[length].x = x;
110
71.2M
  pts[length].y = y;
111
71.2M
  flags[length] = splashPathLast;
112
71.2M
  ++length;
113
71.2M
  return splashOk;
114
71.2M
}
115
116
SplashError SplashPath::curveTo(SplashCoord x1, SplashCoord y1,
117
        SplashCoord x2, SplashCoord y2,
118
4.10M
        SplashCoord x3, SplashCoord y3) {
119
4.10M
  if (noCurrentPoint()) {
120
0
    return splashErrNoCurPt;
121
0
  }
122
4.10M
  flags[length-1] &= (Guchar)~splashPathLast;
123
4.10M
  grow(3);
124
4.10M
  pts[length].x = x1;
125
4.10M
  pts[length].y = y1;
126
4.10M
  flags[length] = splashPathCurve;
127
4.10M
  ++length;
128
4.10M
  pts[length].x = x2;
129
4.10M
  pts[length].y = y2;
130
4.10M
  flags[length] = splashPathCurve;
131
4.10M
  ++length;
132
4.10M
  pts[length].x = x3;
133
4.10M
  pts[length].y = y3;
134
4.10M
  flags[length] = splashPathLast;
135
4.10M
  ++length;
136
4.10M
  return splashOk;
137
4.10M
}
138
139
16.8M
SplashError SplashPath::close(GBool force) {
140
16.8M
  if (noCurrentPoint()) {
141
8
    return splashErrNoCurPt;
142
8
  }
143
16.8M
  if (force ||
144
8.30M
      curSubpath == length - 1 ||
145
8.28M
      pts[length - 1].x != pts[curSubpath].x ||
146
14.2M
      pts[length - 1].y != pts[curSubpath].y) {
147
14.2M
    lineTo(pts[curSubpath].x, pts[curSubpath].y);
148
14.2M
  }
149
16.8M
  flags[curSubpath] |= splashPathClosed;
150
16.8M
  flags[length - 1] |= splashPathClosed;
151
16.8M
  curSubpath = length;
152
16.8M
  return splashOk;
153
16.8M
}
154
155
void SplashPath::addStrokeAdjustHint(int ctrl0, int ctrl1,
156
             int firstPt, int lastPt,
157
23.3M
             GBool projectingCap) {
158
23.3M
  if (hintsLength == hintsSize) {
159
68.9k
    hintsSize = hintsLength ? 2 * hintsLength : 8;
160
68.9k
    hints = (SplashPathHint *)greallocn(hints, hintsSize,
161
68.9k
          sizeof(SplashPathHint));
162
68.9k
  }
163
23.3M
  hints[hintsLength].ctrl0 = ctrl0;
164
23.3M
  hints[hintsLength].ctrl1 = ctrl1;
165
23.3M
  hints[hintsLength].firstPt = firstPt;
166
23.3M
  hints[hintsLength].lastPt = lastPt;
167
23.3M
  hints[hintsLength].projectingCap = projectingCap;
168
23.3M
  ++hintsLength;
169
23.3M
}
170
171
293k
void SplashPath::offset(SplashCoord dx, SplashCoord dy) {
172
293k
  int i;
173
174
5.76M
  for (i = 0; i < length; ++i) {
175
5.46M
    pts[i].x += dx;
176
5.46M
    pts[i].y += dy;
177
5.46M
  }
178
293k
}
179
180
404k
GBool SplashPath::getCurPt(SplashCoord *x, SplashCoord *y) {
181
404k
  if (noCurrentPoint()) {
182
0
    return gFalse;
183
0
  }
184
404k
  *x = pts[length - 1].x;
185
404k
  *y = pts[length - 1].y;
186
404k
  return gTrue;
187
404k
}
188
189
0
GBool SplashPath::containsZeroLengthSubpaths() {
190
0
  GBool zeroLength;
191
0
  int i;
192
193
0
  zeroLength = gTrue;  // make gcc happy
194
0
  for (i = 0; i < length; ++i) {
195
0
    if (flags[i] & splashPathFirst) {
196
0
      zeroLength = gTrue;
197
0
    } else {
198
0
      if (pts[i].x != pts[i-1].x || pts[i].y != pts[i-1].y) {
199
0
  zeroLength = gFalse;
200
0
      }
201
0
      if (flags[i] & splashPathLast) {
202
0
  if (zeroLength) {
203
0
    return gTrue;
204
0
  }
205
0
      }
206
0
    }
207
0
  }
208
0
  return gFalse;
209
0
}