Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(196)

Side by Side Diff: src/pathops/SkOpSegment.cpp

Issue 1002693002: pathops version two (Closed) Base URL: https://skia.googlesource.com/skia.git@master
Patch Set: fix arm 64 inspired coincident handling Created 5 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « src/pathops/SkOpSegment.h ('k') | src/pathops/SkOpSpan.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 /* 1 /*
2 * Copyright 2012 Google Inc. 2 * Copyright 2012 Google Inc.
3 * 3 *
4 * Use of this source code is governed by a BSD-style license that can be 4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file. 5 * found in the LICENSE file.
6 */ 6 */
7 #include "SkIntersections.h" 7 #include "SkOpCoincidence.h"
8 #include "SkOpContour.h" 8 #include "SkOpContour.h"
9 #include "SkOpSegment.h" 9 #include "SkOpSegment.h"
10 #include "SkPathWriter.h" 10 #include "SkPathWriter.h"
11 #include "SkTSort.h" 11
12 /*
13 After computing raw intersections, post process all segments to:
14 - find small collections of points that can be collapsed to a single point
15 - find missing intersections to resolve differences caused by different algorith ms
16
17 Consider segments containing tiny or small intervals. Consider coincident segmen ts
18 because coincidence finds intersections through distance measurement that non-co incident
19 intersection tests cannot.
20 */
12 21
13 #define F (false) // discard the edge 22 #define F (false) // discard the edge
14 #define T (true) // keep the edge 23 #define T (true) // keep the edge
15 24
16 static const bool gUnaryActiveEdge[2][2] = { 25 static const bool gUnaryActiveEdge[2][2] = {
17 // from=0 from=1 26 // from=0 from=1
18 // to=0,1 to=0,1 27 // to=0,1 to=0,1
19 {F, T}, {T, F}, 28 {F, T}, {T, F},
20 }; 29 };
21 30
22 static const bool gActiveEdge[kXOR_PathOp + 1][2][2][2][2] = { 31 static const bool gActiveEdge[kXOR_PathOp + 1][2][2][2][2] = {
23 // miFrom=0 miFrom=1 32 // miFrom=0 miFrom=1
24 // miTo=0 miTo=1 miTo=0 miTo=1 33 // miTo=0 miTo=1 miTo=0 miTo=1
25 // suFrom=0 1 suFrom=0 1 suFrom=0 1 suFrom=0 1 34 // suFrom=0 1 suFrom=0 1 suFrom=0 1 suFrom=0 1
26 // suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1 35 // suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1 suTo=0,1
27 {{{{F, F}, {F, F}}, {{T, F}, {T, F}}}, {{{T, T}, {F, F}}, {{F, T}, {T, F}}}} , // mi - su 36 {{{{F, F}, {F, F}}, {{T, F}, {T, F}}}, {{{T, T}, {F, F}}, {{F, T}, {T, F}}}} , // mi - su
28 {{{{F, F}, {F, F}}, {{F, T}, {F, T}}}, {{{F, F}, {T, T}}, {{F, T}, {T, F}}}} , // mi & su 37 {{{{F, F}, {F, F}}, {{F, T}, {F, T}}}, {{{F, F}, {T, T}}, {{F, T}, {T, F}}}} , // mi & su
29 {{{{F, T}, {T, F}}, {{T, T}, {F, F}}}, {{{T, F}, {T, F}}, {{F, F}, {F, F}}}} , // mi | su 38 {{{{F, T}, {T, F}}, {{T, T}, {F, F}}}, {{{T, F}, {T, F}}, {{F, F}, {F, F}}}} , // mi | su
30 {{{{F, T}, {T, F}}, {{T, F}, {F, T}}}, {{{T, F}, {F, T}}, {{F, T}, {T, F}}}} , // mi ^ su 39 {{{{F, T}, {T, F}}, {{T, F}, {F, T}}}, {{{T, F}, {F, T}}, {{F, T}, {T, F}}}} , // mi ^ su
31 }; 40 };
32 41
33 #undef F 42 #undef F
34 #undef T 43 #undef T
35 44
36 enum { 45 SkOpAngle* SkOpSegment::activeAngle(SkOpSpanBase* start, SkOpSpanBase** startPtr ,
37 kOutsideTrackedTCount = 16, // FIXME: determine what this should be 46 SkOpSpanBase** endPtr, bool* done, bool* sortable) {
38 kMissingSpanCount = 4, // FIXME: determine what this should be 47 if (SkOpAngle* result = activeAngleInner(start, startPtr, endPtr, done, sort able)) {
39 };
40
41 const SkOpAngle* SkOpSegment::activeAngle(int index, int* start, int* end, bool* done,
42 bool* sortable) const {
43 if (const SkOpAngle* result = activeAngleInner(index, start, end, done, sort able)) {
44 return result; 48 return result;
45 } 49 }
46 double referenceT = fTs[index].fT; 50 if (SkOpAngle* result = activeAngleOther(start, startPtr, endPtr, done, sort able)) {
47 int lesser = index; 51 return result;
48 while (--lesser >= 0
49 && (precisely_negative(referenceT - fTs[lesser].fT) || fTs[lesser].f Tiny)) {
50 if (const SkOpAngle* result = activeAngleOther(lesser, start, end, done, sortable)) {
51 return result;
52 }
53 } 52 }
54 do {
55 if (const SkOpAngle* result = activeAngleOther(index, start, end, done, sortable)) {
56 return result;
57 }
58 if (++index == fTs.count()) {
59 break;
60 }
61 if (fTs[index - 1].fTiny) {
62 referenceT = fTs[index].fT;
63 continue;
64 }
65 } while (precisely_negative(fTs[index].fT - referenceT));
66 return NULL; 53 return NULL;
67 } 54 }
68 55
69 const SkOpAngle* SkOpSegment::activeAngleInner(int index, int* start, int* end, bool* done, 56 SkOpAngle* SkOpSegment::activeAngleInner(SkOpSpanBase* start, SkOpSpanBase** sta rtPtr,
70 bool* sortable) const { 57 SkOpSpanBase** endPtr, bool* done, bool* sortable) {
71 int next = nextExactSpan(index, 1); 58 SkOpSpan* upSpan = start->upCastable();
72 if (next > 0) { 59 if (upSpan) {
73 const SkOpSpan& upSpan = fTs[index]; 60 if (upSpan->windValue() || upSpan->oppValue()) {
74 if (upSpan.fWindValue || upSpan.fOppValue) { 61 SkOpSpanBase* next = upSpan->next();
75 if (*end < 0) { 62 if (!*endPtr) {
76 *start = index; 63 *startPtr = start;
77 *end = next; 64 *endPtr = next;
78 } 65 }
79 if (!upSpan.fDone) { 66 if (!upSpan->done()) {
80 if (upSpan.fWindSum != SK_MinS32) { 67 if (upSpan->windSum() != SK_MinS32) {
81 return spanToAngle(index, next); 68 return spanToAngle(start, next);
82 } 69 }
83 *done = false; 70 *done = false;
84 } 71 }
85 } else { 72 } else {
86 SkASSERT(upSpan.fDone); 73 SkASSERT(upSpan->done());
87 } 74 }
88 } 75 }
89 int prev = nextExactSpan(index, -1); 76 SkOpSpan* downSpan = start->prev();
90 // edge leading into junction 77 // edge leading into junction
91 if (prev >= 0) { 78 if (downSpan) {
92 const SkOpSpan& downSpan = fTs[prev]; 79 if (downSpan->windValue() || downSpan->oppValue()) {
93 if (downSpan.fWindValue || downSpan.fOppValue) { 80 if (!*endPtr) {
94 if (*end < 0) { 81 *startPtr = start;
95 *start = index; 82 *endPtr = downSpan;
96 *end = prev;
97 } 83 }
98 if (!downSpan.fDone) { 84 if (!downSpan->done()) {
99 if (downSpan.fWindSum != SK_MinS32) { 85 if (downSpan->windSum() != SK_MinS32) {
100 return spanToAngle(index, prev); 86 return spanToAngle(start, downSpan);
101 } 87 }
102 *done = false; 88 *done = false;
103 } 89 }
104 } else { 90 } else {
105 SkASSERT(downSpan.fDone); 91 SkASSERT(downSpan->done());
106 } 92 }
107 } 93 }
108 return NULL; 94 return NULL;
109 } 95 }
110 96
111 const SkOpAngle* SkOpSegment::activeAngleOther(int index, int* start, int* end, bool* done, 97 SkOpAngle* SkOpSegment::activeAngleOther(SkOpSpanBase* start, SkOpSpanBase** sta rtPtr,
112 bool* sortable) const { 98 SkOpSpanBase** endPtr, bool* done, bool* sortable) {
113 const SkOpSpan* span = &fTs[index]; 99 SkOpPtT* oPtT = start->ptT()->next();
114 SkOpSegment* other = span->fOther; 100 SkOpSegment* other = oPtT->segment();
115 int oIndex = span->fOtherIndex; 101 SkOpSpanBase* oSpan = oPtT->span();
116 return other->activeAngleInner(oIndex, start, end, done, sortable); 102 return other->activeAngleInner(oSpan, startPtr, endPtr, done, sortable);
117 } 103 }
118 104
119 SkPoint SkOpSegment::activeLeftTop(int* firstT) const { 105 SkPoint SkOpSegment::activeLeftTop(SkOpSpanBase** firstSpan) {
120 SkASSERT(!done()); 106 SkASSERT(!done());
121 SkPoint topPt = {SK_ScalarMax, SK_ScalarMax}; 107 SkPoint topPt = {SK_ScalarMax, SK_ScalarMax};
122 int count = fTs.count();
123 // see if either end is not done since we want smaller Y of the pair 108 // see if either end is not done since we want smaller Y of the pair
124 bool lastDone = true; 109 bool lastDone = true;
125 double lastT = -1; 110 double lastT = -1;
126 for (int index = 0; index < count; ++index) { 111 SkOpSpanBase* span = &fHead;
127 const SkOpSpan& span = fTs[index]; 112 do {
128 if (span.fDone && lastDone) { 113 if (lastDone && (span->final() || span->upCast()->done())) {
129 goto next;
130 }
131 if (approximately_negative(span.fT - lastT)) {
132 goto next; 114 goto next;
133 } 115 }
134 { 116 {
135 const SkPoint& xy = xyAtT(&span); 117 const SkPoint& xy = span->pt();
136 if (topPt.fY > xy.fY || (topPt.fY == xy.fY && topPt.fX > xy.fX)) { 118 if (topPt.fY > xy.fY || (topPt.fY == xy.fY && topPt.fX > xy.fX)) {
137 topPt = xy; 119 topPt = xy;
138 if (firstT) { 120 if (firstSpan) {
139 *firstT = index; 121 *firstSpan = span;
140 } 122 }
141 } 123 }
142 if (fVerb != SkPath::kLine_Verb && !lastDone) { 124 if (fVerb != SkPath::kLine_Verb && !lastDone) {
143 SkPoint curveTop = (*CurveTop[SkPathOpsVerbToPoints(fVerb)])(fPt s, lastT, span.fT); 125 SkPoint curveTop = (*CurveTop[SkPathOpsVerbToPoints(fVerb)])(fPt s, lastT,
126 span->t());
144 if (topPt.fY > curveTop.fY || (topPt.fY == curveTop.fY 127 if (topPt.fY > curveTop.fY || (topPt.fY == curveTop.fY
145 && topPt.fX > curveTop.fX)) { 128 && topPt.fX > curveTop.fX)) {
146 topPt = curveTop; 129 topPt = curveTop;
147 if (firstT) { 130 if (firstSpan) {
148 *firstT = index; 131 *firstSpan = span;
149 } 132 }
150 } 133 }
151 } 134 }
152 lastT = span.fT; 135 lastT = span->t();
153 } 136 }
154 next: 137 next:
155 lastDone = span.fDone; 138 if (span->final()) {
156 } 139 break;
140 }
141 lastDone = span->upCast()->done();
142 } while ((span = span->upCast()->next()));
157 return topPt; 143 return topPt;
158 } 144 }
159 145
160 bool SkOpSegment::activeOp(int index, int endIndex, int xorMiMask, int xorSuMask , SkPathOp op) { 146 bool SkOpSegment::activeOp(SkOpSpanBase* start, SkOpSpanBase* end, int xorMiMask , int xorSuMask,
161 int sumMiWinding = updateWinding(endIndex, index); 147 SkPathOp op) {
162 int sumSuWinding = updateOppWinding(endIndex, index); 148 int sumMiWinding = this->updateWinding(end, start);
149 int sumSuWinding = this->updateOppWinding(end, start);
163 #if DEBUG_LIMIT_WIND_SUM 150 #if DEBUG_LIMIT_WIND_SUM
164 SkASSERT(abs(sumMiWinding) <= DEBUG_LIMIT_WIND_SUM); 151 SkASSERT(abs(sumMiWinding) <= DEBUG_LIMIT_WIND_SUM);
165 SkASSERT(abs(sumSuWinding) <= DEBUG_LIMIT_WIND_SUM); 152 SkASSERT(abs(sumSuWinding) <= DEBUG_LIMIT_WIND_SUM);
166 #endif 153 #endif
167 if (fOperand) { 154 if (this->operand()) {
168 SkTSwap<int>(sumMiWinding, sumSuWinding); 155 SkTSwap<int>(sumMiWinding, sumSuWinding);
169 } 156 }
170 return activeOp(xorMiMask, xorSuMask, index, endIndex, op, &sumMiWinding, &s umSuWinding); 157 return this->activeOp(xorMiMask, xorSuMask, start, end, op, &sumMiWinding, & sumSuWinding);
171 } 158 }
172 159
173 bool SkOpSegment::activeOp(int xorMiMask, int xorSuMask, int index, int endIndex , SkPathOp op, 160 bool SkOpSegment::activeOp(int xorMiMask, int xorSuMask, SkOpSpanBase* start, Sk OpSpanBase* end,
174 int* sumMiWinding, int* sumSuWinding) { 161 SkPathOp op, int* sumMiWinding, int* sumSuWinding) {
175 int maxWinding, sumWinding, oppMaxWinding, oppSumWinding; 162 int maxWinding, sumWinding, oppMaxWinding, oppSumWinding;
176 setUpWindings(index, endIndex, sumMiWinding, sumSuWinding, 163 this->setUpWindings(start, end, sumMiWinding, sumSuWinding,
177 &maxWinding, &sumWinding, &oppMaxWinding, &oppSumWinding); 164 &maxWinding, &sumWinding, &oppMaxWinding, &oppSumWinding);
178 bool miFrom; 165 bool miFrom;
179 bool miTo; 166 bool miTo;
180 bool suFrom; 167 bool suFrom;
181 bool suTo; 168 bool suTo;
182 if (operand()) { 169 if (operand()) {
183 miFrom = (oppMaxWinding & xorMiMask) != 0; 170 miFrom = (oppMaxWinding & xorMiMask) != 0;
184 miTo = (oppSumWinding & xorMiMask) != 0; 171 miTo = (oppSumWinding & xorMiMask) != 0;
185 suFrom = (maxWinding & xorSuMask) != 0; 172 suFrom = (maxWinding & xorSuMask) != 0;
186 suTo = (sumWinding & xorSuMask) != 0; 173 suTo = (sumWinding & xorSuMask) != 0;
187 } else { 174 } else {
188 miFrom = (maxWinding & xorMiMask) != 0; 175 miFrom = (maxWinding & xorMiMask) != 0;
189 miTo = (sumWinding & xorMiMask) != 0; 176 miTo = (sumWinding & xorMiMask) != 0;
190 suFrom = (oppMaxWinding & xorSuMask) != 0; 177 suFrom = (oppMaxWinding & xorSuMask) != 0;
191 suTo = (oppSumWinding & xorSuMask) != 0; 178 suTo = (oppSumWinding & xorSuMask) != 0;
192 } 179 }
193 bool result = gActiveEdge[op][miFrom][miTo][suFrom][suTo]; 180 bool result = gActiveEdge[op][miFrom][miTo][suFrom][suTo];
194 #if DEBUG_ACTIVE_OP 181 #if DEBUG_ACTIVE_OP
195 SkDebugf("%s id=%d t=%1.9g tEnd=%1.9g op=%s miFrom=%d miTo=%d suFrom=%d suTo =%d result=%d\n", 182 SkDebugf("%s id=%d t=%1.9g tEnd=%1.9g op=%s miFrom=%d miTo=%d suFrom=%d suTo =%d result=%d\n",
196 __FUNCTION__, debugID(), span(index).fT, span(endIndex).fT, 183 __FUNCTION__, debugID(), start->t(), end->t(),
197 SkPathOpsDebug::kPathOpStr[op], miFrom, miTo, suFrom, suTo, result); 184 SkPathOpsDebug::kPathOpStr[op], miFrom, miTo, suFrom, suTo, result);
198 #endif 185 #endif
199 return result; 186 return result;
200 } 187 }
201 188
202 bool SkOpSegment::activeWinding(int index, int endIndex) { 189 bool SkOpSegment::activeWinding(SkOpSpanBase* start, SkOpSpanBase* end) {
203 int sumWinding = updateWinding(endIndex, index); 190 int sumWinding = updateWinding(end, start);
204 return activeWinding(index, endIndex, &sumWinding); 191 return activeWinding(start, end, &sumWinding);
205 } 192 }
206 193
207 bool SkOpSegment::activeWinding(int index, int endIndex, int* sumWinding) { 194 bool SkOpSegment::activeWinding(SkOpSpanBase* start, SkOpSpanBase* end, int* sum Winding) {
208 int maxWinding; 195 int maxWinding;
209 setUpWinding(index, endIndex, &maxWinding, sumWinding); 196 setUpWinding(start, end, &maxWinding, sumWinding);
210 bool from = maxWinding != 0; 197 bool from = maxWinding != 0;
211 bool to = *sumWinding != 0; 198 bool to = *sumWinding != 0;
212 bool result = gUnaryActiveEdge[from][to]; 199 bool result = gUnaryActiveEdge[from][to];
213 return result; 200 return result;
214 } 201 }
215 202
216 void SkOpSegment::addCancelOutsides(const SkPoint& startPt, const SkPoint& endPt , 203 void SkOpSegment::addCurveTo(const SkOpSpanBase* start, const SkOpSpanBase* end,
217 SkOpSegment* other) { 204 SkPathWriter* path, bool active) const {
218 int tIndex = -1;
219 int tCount = fTs.count();
220 int oIndex = -1;
221 int oCount = other->fTs.count();
222 do {
223 ++tIndex;
224 } while (startPt != fTs[tIndex].fPt && tIndex < tCount);
225 int tIndexStart = tIndex;
226 do {
227 ++oIndex;
228 } while (endPt != other->fTs[oIndex].fPt && oIndex < oCount);
229 int oIndexStart = oIndex;
230 const SkPoint* nextPt;
231 do {
232 nextPt = &fTs[++tIndex].fPt;
233 SkASSERT(fTs[tIndex].fT < 1 || startPt != *nextPt);
234 } while (startPt == *nextPt);
235 double nextT = fTs[tIndex].fT;
236 const SkPoint* oNextPt;
237 do {
238 oNextPt = &other->fTs[++oIndex].fPt;
239 SkASSERT(other->fTs[oIndex].fT < 1 || endPt != *oNextPt);
240 } while (endPt == *oNextPt);
241 double oNextT = other->fTs[oIndex].fT;
242 // at this point, spans before and after are at:
243 // fTs[tIndexStart - 1], fTs[tIndexStart], fTs[tIndex]
244 // if tIndexStart == 0, no prior span
245 // if nextT == 1, no following span
246
247 // advance the span with zero winding
248 // if the following span exists (not past the end, non-zero winding)
249 // connect the two edges
250 if (!fTs[tIndexStart].fWindValue) {
251 if (tIndexStart > 0 && fTs[tIndexStart - 1].fWindValue) {
252 #if DEBUG_CONCIDENT
253 SkDebugf("%s 1 this=%d other=%d t [%d] %1.9g (%1.9g,%1.9g)\n",
254 __FUNCTION__, fID, other->fID, tIndexStart - 1,
255 fTs[tIndexStart].fT, xyAtT(tIndexStart).fX,
256 xyAtT(tIndexStart).fY);
257 #endif
258 SkPoint copy = fTs[tIndexStart].fPt; // add t pair may move the poi nt array
259 addTPair(fTs[tIndexStart].fT, other, other->fTs[oIndex].fT, false, c opy);
260 }
261 if (nextT < 1 && fTs[tIndex].fWindValue) {
262 #if DEBUG_CONCIDENT
263 SkDebugf("%s 2 this=%d other=%d t [%d] %1.9g (%1.9g,%1.9g)\n",
264 __FUNCTION__, fID, other->fID, tIndex,
265 fTs[tIndex].fT, xyAtT(tIndex).fX,
266 xyAtT(tIndex).fY);
267 #endif
268 SkPoint copy = fTs[tIndex].fPt; // add t pair may move the point ar ray
269 addTPair(fTs[tIndex].fT, other, other->fTs[oIndexStart].fT, false, c opy);
270 }
271 } else {
272 SkASSERT(!other->fTs[oIndexStart].fWindValue);
273 if (oIndexStart > 0 && other->fTs[oIndexStart - 1].fWindValue) {
274 #if DEBUG_CONCIDENT
275 SkDebugf("%s 3 this=%d other=%d t [%d] %1.9g (%1.9g,%1.9g)\n",
276 __FUNCTION__, fID, other->fID, oIndexStart - 1,
277 other->fTs[oIndexStart].fT, other->xyAtT(oIndexStart).fX,
278 other->xyAtT(oIndexStart).fY);
279 other->debugAddTPair(other->fTs[oIndexStart].fT, *this, fTs[tIndex]. fT);
280 #endif
281 }
282 if (oNextT < 1 && other->fTs[oIndex].fWindValue) {
283 #if DEBUG_CONCIDENT
284 SkDebugf("%s 4 this=%d other=%d t [%d] %1.9g (%1.9g,%1.9g)\n",
285 __FUNCTION__, fID, other->fID, oIndex,
286 other->fTs[oIndex].fT, other->xyAtT(oIndex).fX,
287 other->xyAtT(oIndex).fY);
288 other->debugAddTPair(other->fTs[oIndex].fT, *this, fTs[tIndexStart]. fT);
289 #endif
290 }
291 }
292 }
293
294 void SkOpSegment::addCoinOutsides(const SkPoint& startPt, const SkPoint& endPt,
295 SkOpSegment* other) {
296 // walk this to startPt
297 // walk other to startPt
298 // if either is > 0, add a pointer to the other, copying adjacent winding
299 int tIndex = -1;
300 int oIndex = -1;
301 do {
302 ++tIndex;
303 } while (startPt != fTs[tIndex].fPt);
304 int ttIndex = tIndex;
305 bool checkOtherTMatch = false;
306 do {
307 const SkOpSpan& span = fTs[ttIndex];
308 if (startPt != span.fPt) {
309 break;
310 }
311 if (span.fOther == other && span.fPt == startPt) {
312 checkOtherTMatch = true;
313 break;
314 }
315 } while (++ttIndex < count());
316 do {
317 ++oIndex;
318 } while (startPt != other->fTs[oIndex].fPt);
319 bool skipAdd = false;
320 if (checkOtherTMatch) {
321 int ooIndex = oIndex;
322 do {
323 const SkOpSpan& oSpan = other->fTs[ooIndex];
324 if (startPt != oSpan.fPt) {
325 break;
326 }
327 if (oSpan.fT == fTs[ttIndex].fOtherT) {
328 skipAdd = true;
329 break;
330 }
331 } while (++ooIndex < other->count());
332 }
333 if ((tIndex > 0 || oIndex > 0 || fOperand != other->fOperand) && !skipAdd) {
334 addTPair(fTs[tIndex].fT, other, other->fTs[oIndex].fT, false, startPt);
335 }
336 SkPoint nextPt = startPt;
337 do {
338 const SkPoint* workPt;
339 do {
340 workPt = &fTs[++tIndex].fPt;
341 } while (nextPt == *workPt);
342 const SkPoint* oWorkPt;
343 do {
344 oWorkPt = &other->fTs[++oIndex].fPt;
345 } while (nextPt == *oWorkPt);
346 nextPt = *workPt;
347 double tStart = fTs[tIndex].fT;
348 double oStart = other->fTs[oIndex].fT;
349 if (tStart == 1 && oStart == 1 && fOperand == other->fOperand) {
350 break;
351 }
352 if (*workPt == *oWorkPt) {
353 addTPair(tStart, other, oStart, false, nextPt);
354 }
355 } while (endPt != nextPt);
356 }
357
358 void SkOpSegment::addCubic(const SkPoint pts[4], bool operand, bool evenOdd) {
359 init(pts, SkPath::kCubic_Verb, operand, evenOdd);
360 fBounds.setCubicBounds(pts);
361 }
362
363 void SkOpSegment::addCurveTo(int start, int end, SkPathWriter* path, bool active ) const {
364 SkPoint edge[4]; 205 SkPoint edge[4];
365 const SkPoint* ePtr; 206 const SkPoint* ePtr;
366 int lastT = fTs.count() - 1; 207 if ((start == &fHead && end == &fTail) || (start == &fTail && end == &fHead) ) {
367 if (lastT < 0 || (start == 0 && end == lastT) || (start == lastT && end == 0 )) {
368 ePtr = fPts; 208 ePtr = fPts;
369 } else { 209 } else {
370 // OPTIMIZE? if not active, skip remainder and return xyAtT(end) 210 // OPTIMIZE? if not active, skip remainder and return xyAtT(end)
371 subDivide(start, end, edge); 211 subDivide(start, end, edge);
372 ePtr = edge; 212 ePtr = edge;
373 } 213 }
374 if (active) { 214 if (active) {
375 bool reverse = ePtr == fPts && start != 0; 215 bool reverse = ePtr == fPts && start != &fHead;
376 if (reverse) { 216 if (reverse) {
377 path->deferredMoveLine(ePtr[SkPathOpsVerbToPoints(fVerb)]); 217 path->deferredMoveLine(ePtr[SkPathOpsVerbToPoints(fVerb)]);
378 switch (fVerb) { 218 switch (fVerb) {
379 case SkPath::kLine_Verb: 219 case SkPath::kLine_Verb:
380 path->deferredLine(ePtr[0]); 220 path->deferredLine(ePtr[0]);
381 break; 221 break;
382 case SkPath::kQuad_Verb: 222 case SkPath::kQuad_Verb:
383 path->quadTo(ePtr[1], ePtr[0]); 223 path->quadTo(ePtr[1], ePtr[0]);
384 break; 224 break;
385 case SkPath::kCubic_Verb: 225 case SkPath::kCubic_Verb:
386 path->cubicTo(ePtr[2], ePtr[1], ePtr[0]); 226 path->cubicTo(ePtr[2], ePtr[1], ePtr[0]);
387 break; 227 break;
388 default: 228 default:
389 SkASSERT(0); 229 SkASSERT(0);
390 } 230 }
391 // return ePtr[0];
392 } else { 231 } else {
393 path->deferredMoveLine(ePtr[0]); 232 path->deferredMoveLine(ePtr[0]);
394 switch (fVerb) { 233 switch (fVerb) {
395 case SkPath::kLine_Verb: 234 case SkPath::kLine_Verb:
396 path->deferredLine(ePtr[1]); 235 path->deferredLine(ePtr[1]);
397 break; 236 break;
398 case SkPath::kQuad_Verb: 237 case SkPath::kQuad_Verb:
399 path->quadTo(ePtr[1], ePtr[2]); 238 path->quadTo(ePtr[1], ePtr[2]);
400 break; 239 break;
401 case SkPath::kCubic_Verb: 240 case SkPath::kCubic_Verb:
402 path->cubicTo(ePtr[1], ePtr[2], ePtr[3]); 241 path->cubicTo(ePtr[1], ePtr[2], ePtr[3]);
403 break; 242 break;
404 default: 243 default:
405 SkASSERT(0); 244 SkASSERT(0);
406 } 245 }
407 } 246 }
408 } 247 }
409 // return ePtr[SkPathOpsVerbToPoints(fVerb)]; 248 }
410 } 249
411 250 SkOpPtT* SkOpSegment::addMissing(double t, SkOpSegment* opp, SkChunkAlloc* alloc ator) {
412 void SkOpSegment::addEndSpan(int endIndex) { 251 SkOpSpanBase* existing = NULL;
413 SkASSERT(span(endIndex).fT == 1 || (span(endIndex).fTiny 252 SkOpSpanBase* test = &fHead;
414 // && approximately_greater_than_one(span(endIndex).fT) 253 double testT;
415 ));
416 int spanCount = fTs.count();
417 int startIndex = endIndex - 1;
418 while (fTs[startIndex].fT == 1 || fTs[startIndex].fTiny) {
419 --startIndex;
420 SkASSERT(startIndex > 0);
421 --endIndex;
422 }
423 SkOpAngle& angle = fAngles.push_back();
424 angle.set(this, spanCount - 1, startIndex);
425 #if DEBUG_ANGLE
426 debugCheckPointsEqualish(endIndex, spanCount);
427 #endif
428 setFromAngle(endIndex, &angle);
429 }
430
431 void SkOpSegment::setFromAngle(int endIndex, SkOpAngle* angle) {
432 int spanCount = fTs.count();
433 do { 254 do {
434 fTs[endIndex].fFromAngle = angle; 255 if ((testT = test->ptT()->fT) >= t) {
435 } while (++endIndex < spanCount); 256 if (testT == t) {
436 } 257 existing = test;
437 258 }
438 void SkOpSegment::addLine(const SkPoint pts[2], bool operand, bool evenOdd) { 259 break;
439 init(pts, SkPath::kLine_Verb, operand, evenOdd); 260 }
440 fBounds.set(pts, 2); 261 } while ((test = test->upCast()->next()));
441 } 262 SkOpPtT* result;
442 263 if (existing && existing->contains(opp)) {
443 // add 2 to edge or out of range values to get T extremes 264 result = existing->ptT();
444 void SkOpSegment::addOtherT(int index, double otherT, int otherIndex) { 265 } else {
445 SkOpSpan& span = fTs[index]; 266 result = this->addT(t, SkOpSegment::kNoAlias, allocator);
446 if (precisely_zero(otherT)) { 267 }
447 otherT = 0; 268 SkASSERT(result);
448 } else if (precisely_equal(otherT, 1)) { 269 return result;
449 otherT = 1; 270 }
450 } 271
451 span.fOtherT = otherT; 272 SkOpAngle* SkOpSegment::addSingletonAngleDown(SkOpSegment** otherPtr, SkOpAngle* * anglePtr,
452 span.fOtherIndex = otherIndex; 273 SkChunkAlloc* allocator) {
453 } 274 SkOpSpan* startSpan = fTail.prev();
454 275 SkASSERT(startSpan);
455 void SkOpSegment::addQuad(const SkPoint pts[3], bool operand, bool evenOdd) { 276 SkOpAngle* angle = SkOpTAllocator<SkOpAngle>::Allocate(allocator);
456 init(pts, SkPath::kQuad_Verb, operand, evenOdd); 277 *anglePtr = angle;
457 fBounds.setQuadBounds(pts); 278 angle->set(&fTail, startSpan);
458 } 279 fTail.setFromAngle(angle);
459 280 SkOpSegment* other = NULL; // these initializations silence a release build warning
460 SkOpAngle* SkOpSegment::addSingletonAngleDown(SkOpSegment** otherPtr, SkOpAngle* * anglePtr) { 281 SkOpSpan* oStartSpan = NULL;
461 int spanIndex = count() - 1; 282 SkOpSpanBase* oEndSpan = NULL;
462 int startIndex = nextExactSpan(spanIndex, -1); 283 SkOpPtT* ptT = fTail.ptT(), * startPtT = ptT;
463 SkASSERT(startIndex >= 0); 284 while ((ptT = ptT->next()) != startPtT) {
464 SkOpAngle& angle = fAngles.push_back(); 285 other = ptT->segment();
465 *anglePtr = &angle; 286 oStartSpan = ptT->span()->upCastable();
466 angle.set(this, spanIndex, startIndex); 287 if (oStartSpan && oStartSpan->windValue()) {
467 setFromAngle(spanIndex, &angle); 288 oEndSpan = oStartSpan->next();
468 SkOpSegment* other; 289 break;
469 int oStartIndex, oEndIndex; 290 }
470 do { 291 oEndSpan = ptT->span();
471 const SkOpSpan& span = fTs[spanIndex]; 292 oStartSpan = oEndSpan->prev();
472 SkASSERT(span.fT > 0); 293 if (oStartSpan && oStartSpan->windValue()) {
473 other = span.fOther; 294 break;
474 oStartIndex = span.fOtherIndex; 295 }
475 oEndIndex = other->nextExactSpan(oStartIndex, 1); 296 }
476 if (oEndIndex > 0 && other->span(oStartIndex).fWindValue) { 297 SkOpAngle* oAngle = SkOpTAllocator<SkOpAngle>::Allocate(allocator);
477 break; 298 oAngle->set(oStartSpan, oEndSpan);
478 } 299 oStartSpan->setToAngle(oAngle);
479 oEndIndex = oStartIndex;
480 oStartIndex = other->nextExactSpan(oEndIndex, -1);
481 --spanIndex;
482 } while (oStartIndex < 0 || !other->span(oStartIndex).fWindSum);
483 SkOpAngle& oAngle = other->fAngles.push_back();
484 oAngle.set(other, oStartIndex, oEndIndex);
485 other->setToAngle(oEndIndex, &oAngle);
486 *otherPtr = other; 300 *otherPtr = other;
487 return &oAngle; 301 return oAngle;
488 } 302 }
489 303
490 SkOpAngle* SkOpSegment::addSingletonAngleUp(SkOpSegment** otherPtr, SkOpAngle** anglePtr) { 304 SkOpAngle* SkOpSegment::addSingletonAngles(int step, SkChunkAlloc* allocator) {
491 int endIndex = nextExactSpan(0, 1);
492 SkASSERT(endIndex > 0);
493 SkOpAngle& angle = fAngles.push_back();
494 *anglePtr = &angle;
495 angle.set(this, 0, endIndex);
496 setToAngle(endIndex, &angle);
497 int spanIndex = 0;
498 SkOpSegment* other;
499 int oStartIndex, oEndIndex;
500 do {
501 const SkOpSpan& span = fTs[spanIndex];
502 SkASSERT(span.fT < 1);
503 other = span.fOther;
504 oEndIndex = span.fOtherIndex;
505 oStartIndex = other->nextExactSpan(oEndIndex, -1);
506 if (oStartIndex >= 0 && other->span(oStartIndex).fWindValue) {
507 break;
508 }
509 oStartIndex = oEndIndex;
510 oEndIndex = other->nextExactSpan(oStartIndex, 1);
511 ++spanIndex;
512 } while (oEndIndex < 0 || !other->span(oStartIndex).fWindValue);
513 SkOpAngle& oAngle = other->fAngles.push_back();
514 oAngle.set(other, oEndIndex, oStartIndex);
515 other->setFromAngle(oEndIndex, &oAngle);
516 *otherPtr = other;
517 return &oAngle;
518 }
519
520 SkOpAngle* SkOpSegment::addSingletonAngles(int step) {
521 SkOpSegment* other; 305 SkOpSegment* other;
522 SkOpAngle* angle, * otherAngle; 306 SkOpAngle* angle, * otherAngle;
523 if (step > 0) { 307 if (step > 0) {
524 otherAngle = addSingletonAngleUp(&other, &angle); 308 otherAngle = addSingletonAngleUp(&other, &angle, allocator);
525 } else { 309 } else {
526 otherAngle = addSingletonAngleDown(&other, &angle); 310 otherAngle = addSingletonAngleDown(&other, &angle, allocator);
527 } 311 }
528 angle->insert(otherAngle); 312 angle->insert(otherAngle);
529 return angle; 313 return angle;
530 } 314 }
531 315
532 void SkOpSegment::addStartSpan(int endIndex) { 316 SkOpAngle* SkOpSegment::addSingletonAngleUp(SkOpSegment** otherPtr, SkOpAngle** anglePtr,
533 int index = 0; 317 SkChunkAlloc* allocator) {
534 SkOpAngle& angle = fAngles.push_back(); 318 SkOpSpanBase* endSpan = fHead.next();
535 angle.set(this, index, endIndex); 319 SkASSERT(endSpan);
536 #if DEBUG_ANGLE 320 SkOpAngle* angle = SkOpTAllocator<SkOpAngle>::Allocate(allocator);
537 debugCheckPointsEqualish(index, endIndex); 321 *anglePtr = angle;
322 angle->set(&fHead, endSpan);
323 fHead.setToAngle(angle);
324 SkOpSegment* other = NULL; // these initializations silence a release build warning
325 SkOpSpan* oStartSpan = NULL;
326 SkOpSpanBase* oEndSpan = NULL;
327 SkOpPtT* ptT = fHead.ptT(), * startPtT = ptT;
328 while ((ptT = ptT->next()) != startPtT) {
329 other = ptT->segment();
330 oEndSpan = ptT->span();
331 oStartSpan = oEndSpan->prev();
332 if (oStartSpan && oStartSpan->windValue()) {
333 break;
334 }
335 oStartSpan = oEndSpan->upCastable();
336 if (oStartSpan && oStartSpan->windValue()) {
337 oEndSpan = oStartSpan->next();
338 break;
339 }
340 }
341 SkOpAngle* oAngle = SkOpTAllocator<SkOpAngle>::Allocate(allocator);
342 oAngle->set(oEndSpan, oStartSpan);
343 oEndSpan->setFromAngle(oAngle);
344 *otherPtr = other;
345 return oAngle;
346 }
347
348 SkOpPtT* SkOpSegment::addT(double t, AllowAlias allowAlias, SkChunkAlloc* alloca tor) {
349 debugValidate();
350 SkPoint pt = this->ptAtT(t);
351 SkOpSpanBase* span = &fHead;
352 do {
353 SkOpPtT* result = span->ptT();
354 if (t == result->fT) {
355 return result;
356 }
357 if (this->match(result, this, t, pt)) {
358 // see if any existing alias matches segment, pt, and t
359 SkOpPtT* loop = result->next();
360 bool duplicatePt = false;
361 while (loop != result) {
362 bool ptMatch = loop->fPt == pt;
363 if (loop->segment() == this && loop->fT == t && ptMatch) {
364 return result;
365 }
366 duplicatePt |= ptMatch;
367 loop = loop->next();
368 }
369 if (kNoAlias == allowAlias) {
370 return result;
371 }
372 SkOpPtT* alias = SkOpTAllocator<SkOpPtT>::Allocate(allocator);
373 alias->init(result->span(), t, pt, duplicatePt);
374 result->insert(alias);
375 result->span()->unaligned();
376 this->debugValidate();
377 #if DEBUG_ADD_T
378 SkDebugf("%s alias t=%1.9g segID=%d spanID=%d\n", __FUNCTION__, t,
379 alias->segment()->debugID(), alias->span()->debugID());
538 #endif 380 #endif
539 setToAngle(endIndex, &angle); 381 return alias;
540 } 382 }
541 383 if (t < result->fT) {
542 void SkOpSegment::setToAngle(int endIndex, SkOpAngle* angle) { 384 SkOpSpan* prev = result->span()->prev();
543 int index = 0; 385 SkOpSpan* span = insert(prev, allocator);
386 span->init(this, prev, t, pt);
387 this->debugValidate();
388 #if DEBUG_ADD_T
389 SkDebugf("%s insert t=%1.9g segID=%d spanID=%d\n", __FUNCTION__, t,
390 span->segment()->debugID(), span->debugID());
391 #endif
392 return span->ptT();
393 }
394 SkASSERT(span != &fTail);
395 } while ((span = span->upCast()->next()));
396 SkASSERT(0);
397 return NULL;
398 }
399
400 // choose a solitary t and pt value; remove aliases; align the opposite ends
401 void SkOpSegment::align() {
402 debugValidate();
403 SkOpSpanBase* span = &fHead;
404 if (!span->aligned()) {
405 span->alignEnd(0, fPts[0]);
406 }
407 while ((span = span->upCast()->next())) {
408 if (span == &fTail) {
409 break;
410 }
411 span->align();
412 }
413 if (!span->aligned()) {
414 span->alignEnd(1, fPts[SkPathOpsVerbToPoints(fVerb)]);
415 }
416 debugValidate();
417 }
418
419 bool SkOpSegment::BetweenTs(const SkOpSpanBase* lesser, double testT,
420 const SkOpSpanBase* greater) {
421 if (lesser->t() > greater->t()) {
422 SkTSwap<const SkOpSpanBase*>(lesser, greater);
423 }
424 return approximately_between(lesser->t(), testT, greater->t());
425 }
426
427 void SkOpSegment::calcAngles(SkChunkAlloc* allocator) {
428 bool activePrior = !fHead.isCanceled();
429 if (activePrior && !fHead.simple()) {
430 addStartSpan(allocator);
431 }
432 SkOpSpan* prior = &fHead;
433 SkOpSpanBase* spanBase = fHead.next();
434 while (spanBase != &fTail) {
435 if (activePrior) {
436 SkOpAngle* priorAngle = SkOpTAllocator<SkOpAngle>::Allocate(allocato r);
437 priorAngle->set(spanBase, prior);
438 spanBase->setFromAngle(priorAngle);
439 }
440 SkOpSpan* span = spanBase->upCast();
441 bool active = !span->isCanceled();
442 SkOpSpanBase* next = span->next();
443 if (active) {
444 SkOpAngle* angle = SkOpTAllocator<SkOpAngle>::Allocate(allocator);
445 angle->set(span, next);
446 span->setToAngle(angle);
447 }
448 activePrior = active;
449 prior = span;
450 spanBase = next;
451 }
452 if (activePrior && !fTail.simple()) {
453 addEndSpan(allocator);
454 }
455 }
456
457 void SkOpSegment::checkAngleCoin(SkOpCoincidence* coincidences, SkChunkAlloc* al locator) {
458 SkOpSpanBase* base = &fHead;
459 SkOpSpan* span;
544 do { 460 do {
545 fTs[index].fToAngle = angle; 461 SkOpAngle* angle = base->fromAngle();
546 } while (++index < endIndex); 462 if (angle && angle->fCheckCoincidence) {
547 } 463 angle->checkNearCoincidence();
548 464 }
549 // Defer all coincident edge processing until 465 if (base->final()) {
550 // after normal intersections have been computed 466 break;
551 467 }
552 // no need to be tricky; insert in normal T order 468 span = base->upCast();
553 // resolve overlapping ts when considering coincidence later 469 angle = span->toAngle();
554 470 if (angle && angle->fCheckCoincidence) {
555 // add non-coincident intersection. Resulting edges are sorted in T. 471 angle->checkNearCoincidence();
556 int SkOpSegment::addT(SkOpSegment* other, const SkPoint& pt, double newT) { 472 }
557 SkASSERT(this != other || fVerb == SkPath::kCubic_Verb); 473 } while ((base = span->next()));
558 #if 0 // this needs an even rougher association to be useful 474 }
559 SkASSERT(SkDPoint::RoughlyEqual(ptAtT(newT), pt)); 475
560 #endif 476 // from http://stackoverflow.com/questions/1165647/how-to-determine-if-a-list-of -polygon-points-are-in-clockwise-order
561 const SkPoint& firstPt = fPts[0]; 477 bool SkOpSegment::clockwise(const SkOpSpanBase* start, const SkOpSpanBase* end, bool* swap) const {
562 const SkPoint& lastPt = fPts[SkPathOpsVerbToPoints(fVerb)]; 478 SkASSERT(fVerb != SkPath::kLine_Verb);
563 SkASSERT(newT == 0 || !precisely_zero(newT)); 479 SkPoint edge[4];
564 SkASSERT(newT == 1 || !precisely_equal(newT, 1)); 480 if (fVerb == SkPath::kCubic_Verb) {
565 // FIXME: in the pathological case where there is a ton of intercepts, 481 double startT = start->t();
566 // binary search? 482 double endT = end->t();
567 int insertedAt = -1; 483 bool flip = startT > endT;
568 int tCount = fTs.count(); 484 SkDCubic cubic;
569 for (int index = 0; index < tCount; ++index) { 485 cubic.set(fPts);
570 // OPTIMIZATION: if there are three or more identical Ts, then 486 double inflectionTs[2];
571 // the fourth and following could be further insertion-sorted so 487 int inflections = cubic.findInflections(inflectionTs);
572 // that all the edges are clockwise or counterclockwise. 488 for (int index = 0; index < inflections; ++index) {
573 // This could later limit segment tests to the two adjacent 489 double inflectionT = inflectionTs[index];
574 // neighbors, although it doesn't help with determining which 490 if (between(startT, inflectionT, endT)) {
575 // circular direction to go in. 491 if (flip) {
576 const SkOpSpan& span = fTs[index]; 492 if (inflectionT != endT) {
577 if (newT < span.fT) { 493 startT = inflectionT;
578 insertedAt = index; 494 }
579 break;
580 }
581 if (newT == span.fT) {
582 if (pt == span.fPt) {
583 insertedAt = index;
584 break;
585 }
586 if ((pt == firstPt && newT == 0) || (span.fPt == lastPt && newT == 1 )) {
587 insertedAt = index;
588 break;
589 }
590 }
591 }
592 SkOpSpan* span;
593 if (insertedAt >= 0) {
594 span = fTs.insert(insertedAt);
595 } else {
596 insertedAt = tCount;
597 span = fTs.append();
598 }
599 span->fT = newT;
600 span->fOtherT = -1;
601 span->fOther = other;
602 span->fPt = pt;
603 #if 0
604 // cubics, for instance, may not be exact enough to satisfy this check (e.g. , cubicOp69d)
605 SkASSERT(approximately_equal(xyAtT(newT).fX, pt.fX)
606 && approximately_equal(xyAtT(newT).fY, pt.fY));
607 #endif
608 span->fFromAngle = NULL;
609 span->fToAngle = NULL;
610 span->fWindSum = SK_MinS32;
611 span->fOppSum = SK_MinS32;
612 span->fWindValue = 1;
613 span->fOppValue = 0;
614 span->fChased = false;
615 span->fCoincident = false;
616 span->fLoop = false;
617 span->fNear = false;
618 span->fMultiple = false;
619 span->fSmall = false;
620 span->fTiny = false;
621 if ((span->fDone = newT == 1)) {
622 ++fDoneSpans;
623 }
624 setSpanFlags(pt, newT, span);
625 return insertedAt;
626 }
627
628 void SkOpSegment::setSpanFlags(const SkPoint& pt, double newT, SkOpSpan* span) {
629 int less = -1;
630 // FIXME: note that this relies on spans being a continguous array
631 // find range of spans with nearly the same point as this one
632 // FIXME: SkDPoint::ApproximatelyEqual is better but breaks tests at the mom ent
633 while (&span[less + 1] - fTs.begin() > 0 && AlmostEqualUlps(span[less].fPt, pt)) {
634 if (fVerb == SkPath::kCubic_Verb) {
635 double tInterval = newT - span[less].fT;
636 double tMid = newT - tInterval / 2;
637 SkDPoint midPt = dcubic_xy_at_t(fPts, tMid);
638 if (!midPt.approximatelyEqual(xyAtT(span))) {
639 break;
640 }
641 }
642 --less;
643 }
644 int more = 1;
645 // FIXME: SkDPoint::ApproximatelyEqual is better but breaks tests at the mom ent
646 while (fTs.end() - &span[more - 1] > 1 && AlmostEqualUlps(span[more].fPt, pt )) {
647 if (fVerb == SkPath::kCubic_Verb) {
648 double tEndInterval = span[more].fT - newT;
649 double tMid = newT - tEndInterval / 2;
650 SkDPoint midEndPt = dcubic_xy_at_t(fPts, tMid);
651 if (!midEndPt.approximatelyEqual(xyAtT(span))) {
652 break;
653 }
654 }
655 ++more;
656 }
657 ++less;
658 --more;
659 while (more - 1 > less && span[more].fPt == span[more - 1].fPt
660 && span[more].fT == span[more - 1].fT) {
661 --more;
662 }
663 if (less == more) {
664 return;
665 }
666 if (precisely_negative(span[more].fT - span[less].fT)) {
667 return;
668 }
669 // if the total range of t values is big enough, mark all tiny
670 bool tiny = span[less].fPt == span[more].fPt;
671 int index = less;
672 do {
673 fSmall = span[index].fSmall = true;
674 fTiny |= span[index].fTiny = tiny;
675 if (!span[index].fDone) {
676 span[index].fDone = true;
677 ++fDoneSpans;
678 }
679 } while (++index < more);
680 return;
681 }
682
683 void SkOpSegment::resetSpanFlags() {
684 fSmall = fTiny = false;
685 fDoneSpans = 0;
686 int start = 0;
687 int last = this->count() - 1;
688 do {
689 SkOpSpan* startSpan = &this->fTs[start];
690 double startT = startSpan->fT;
691 startSpan->fSmall = startSpan->fTiny = false; // sets range initial
692 bool terminus = startT == 1;
693 if ((startSpan->fDone = !startSpan->fWindValue | terminus)) {
694 ++fDoneSpans;
695 }
696 ++start; // range initial + 1
697 if (terminus) {
698 continue;
699 }
700 const SkPoint& pt = startSpan->fPt;
701 int end = start; // range initial + 1
702 while (end <= last) {
703 const SkOpSpan& endSpan = this->span(end);
704 if (!AlmostEqualUlps(endSpan.fPt, pt)) {
705 break;
706 }
707 if (fVerb == SkPath::kCubic_Verb) {
708 double tMid = (startSpan->fT + endSpan.fT) / 2;
709 SkDPoint midEndPt = dcubic_xy_at_t(fPts, tMid);
710 if (!midEndPt.approximatelyEqual(xyAtT(startSpan))) {
711 break;
712 }
713 }
714 ++end;
715 }
716 if (start == end) { // end == range final + 1
717 continue;
718 }
719 while (--end >= start) { // end == range final
720 const SkOpSpan& endSpan = this->span(end);
721 const SkOpSpan& priorSpan = this->span(end - 1);
722 if (endSpan.fPt != priorSpan.fPt || endSpan.fT != priorSpan.fT) {
723 break; // end == range final + 1
724 }
725 }
726 if (end < start) { // end == range final + 1
727 continue;
728 }
729 int index = start - 1; // index == range initial
730 start = end; // start = range final + 1
731 const SkOpSpan& nextSpan = this->span(end);
732 if (precisely_negative(nextSpan.fT - startSpan->fT)) {
733 while (++index < end) {
734 startSpan = &this->fTs[index];
735 startSpan->fSmall = startSpan->fTiny = false; // sets range ini tial + 1
736 if ((startSpan->fDone = !startSpan->fWindValue)) {
737 ++fDoneSpans;
738 }
739 }
740 continue;
741 }
742 if (!startSpan->fWindValue) {
743 --fDoneSpans; // added back below
744 }
745 bool tiny = nextSpan.fPt == startSpan->fPt;
746 do {
747 fSmall = startSpan->fSmall = true; // sets range initial
748 fTiny |= startSpan->fTiny = tiny;
749 startSpan->fDone = true;
750 ++fDoneSpans;
751 startSpan = &this->fTs[++index];
752 } while (index < end); // loop through tiny small range end (last)
753 } while (start <= last);
754 }
755
756 // set spans from start to end to decrement by one
757 // note this walks other backwards
758 // FIXME: there's probably an edge case that can be constructed where
759 // two span in one segment are separated by float epsilon on one span but
760 // not the other, if one segment is very small. For this
761 // case the counts asserted below may or may not be enough to separate the
762 // spans. Even if the counts work out, what if the spans aren't correctly
763 // sorted? It feels better in such a case to match the span's other span
764 // pointer since both coincident segments must contain the same spans.
765 // FIXME? It seems that decrementing by one will fail for complex paths that
766 // have three or more coincident edges. Shouldn't this subtract the difference
767 // between the winding values?
768 /* |--> |-->
769 this 0>>>>1>>>>2>>>>3>>>4 0>>>>1>>>>2>>>>3>>>4 0>>>>1>>>>2>>>>3>>> 4
770 other 2<<<<1<<<<0 2<<<<1<<<<0 2<<<<1<<<<0
771 ^ ^ <--| <--|
772 startPt endPt test/oTest first pos test/oTest final po s
773 */
774 void SkOpSegment::addTCancel(const SkPoint& startPt, const SkPoint& endPt, SkOpS egment* other) {
775 bool binary = fOperand != other->fOperand;
776 int index = 0;
777 while (startPt != fTs[index].fPt) {
778 SkASSERT(index < fTs.count());
779 ++index;
780 }
781 while (index > 0 && precisely_equal(fTs[index].fT, fTs[index - 1].fT)) {
782 --index;
783 }
784 bool oFoundEnd = false;
785 int oIndex = other->fTs.count();
786 while (startPt != other->fTs[--oIndex].fPt) { // look for startPt match
787 SkASSERT(oIndex > 0);
788 }
789 double oStartT = other->fTs[oIndex].fT;
790 // look for first point beyond match
791 while (startPt == other->fTs[--oIndex].fPt || precisely_equal(oStartT, other ->fTs[oIndex].fT)) {
792 if (!oIndex) {
793 return; // tiny spans may move in the wrong direction
794 }
795 }
796 SkOpSpan* test = &fTs[index];
797 SkOpSpan* oTest = &other->fTs[oIndex];
798 SkSTArray<kOutsideTrackedTCount, SkPoint, true> outsidePts;
799 SkSTArray<kOutsideTrackedTCount, SkPoint, true> oOutsidePts;
800 bool decrement, track, bigger;
801 int originalWindValue;
802 const SkPoint* testPt;
803 const SkPoint* oTestPt;
804 do {
805 SkASSERT(test->fT < 1);
806 SkASSERT(oTest->fT < 1);
807 decrement = test->fWindValue && oTest->fWindValue;
808 track = test->fWindValue || oTest->fWindValue;
809 bigger = test->fWindValue >= oTest->fWindValue;
810 testPt = &test->fPt;
811 double testT = test->fT;
812 oTestPt = &oTest->fPt;
813 double oTestT = oTest->fT;
814 do {
815 if (decrement) {
816 if (binary && bigger) {
817 test->fOppValue--;
818 } else { 495 } else {
819 decrementSpan(test); 496 if (inflectionT != startT) {
820 } 497 endT = inflectionT;
821 } else if (track) {
822 TrackOutsidePair(&outsidePts, *testPt, *oTestPt);
823 }
824 SkASSERT(index < fTs.count() - 1);
825 test = &fTs[++index];
826 } while (*testPt == test->fPt || precisely_equal(testT, test->fT));
827 originalWindValue = oTest->fWindValue;
828 do {
829 SkASSERT(oTest->fT < 1);
830 SkASSERT(originalWindValue == oTest->fWindValue);
831 if (decrement) {
832 if (binary && !bigger) {
833 oTest->fOppValue--;
834 } else {
835 other->decrementSpan(oTest);
836 }
837 } else if (track) {
838 TrackOutsidePair(&oOutsidePts, *oTestPt, *testPt);
839 }
840 if (!oIndex) {
841 break;
842 }
843 oFoundEnd |= endPt == oTest->fPt;
844 oTest = &other->fTs[--oIndex];
845 } while (*oTestPt == oTest->fPt || precisely_equal(oTestT, oTest->fT));
846 } while (endPt != test->fPt && test->fT < 1);
847 // FIXME: determine if canceled edges need outside ts added
848 if (!oFoundEnd) {
849 for (int oIdx2 = oIndex; oIdx2 >= 0; --oIdx2) {
850 SkOpSpan* oTst2 = &other->fTs[oIdx2];
851 if (originalWindValue != oTst2->fWindValue) {
852 goto skipAdvanceOtherCancel;
853 }
854 if (!oTst2->fWindValue) {
855 goto skipAdvanceOtherCancel;
856 }
857 if (endPt == other->fTs[oIdx2].fPt) {
858 break;
859 }
860 }
861 oFoundEnd = endPt == oTest->fPt;
862 do {
863 SkASSERT(originalWindValue == oTest->fWindValue);
864 if (decrement) {
865 if (binary && !bigger) {
866 oTest->fOppValue--;
867 } else {
868 other->decrementSpan(oTest);
869 }
870 } else if (track) {
871 TrackOutsidePair(&oOutsidePts, *oTestPt, *testPt);
872 }
873 if (!oIndex) {
874 break;
875 }
876 oTest = &other->fTs[--oIndex];
877 oFoundEnd |= endPt == oTest->fPt;
878 } while (!oFoundEnd || endPt == oTest->fPt);
879 }
880 skipAdvanceOtherCancel:
881 int outCount = outsidePts.count();
882 if (!done() && outCount) {
883 addCancelOutsides(outsidePts[0], outsidePts[1], other);
884 if (outCount > 2) {
885 addCancelOutsides(outsidePts[outCount - 2], outsidePts[outCount - 1] , other);
886 }
887 }
888 if (!other->done() && oOutsidePts.count()) {
889 other->addCancelOutsides(oOutsidePts[0], oOutsidePts[1], this);
890 }
891 setCoincidentRange(startPt, endPt, other);
892 other->setCoincidentRange(startPt, endPt, this);
893 }
894
895 int SkOpSegment::addSelfT(const SkPoint& pt, double newT) {
896 // if the tail nearly intersects itself but not quite, the caller records th is separately
897 int result = addT(this, pt, newT);
898 SkOpSpan* span = &fTs[result];
899 fLoop = span->fLoop = true;
900 return result;
901 }
902
903 // find the starting or ending span with an existing loop of angles
904 // FIXME? replicate for all identical starting/ending spans?
905 // OPTIMIZE? remove the spans pointing to windValue==0 here or earlier?
906 // FIXME? assert that only one other span has a valid windValue or oppValue
907 void SkOpSegment::addSimpleAngle(int index) {
908 SkOpSpan* span = &fTs[index];
909 int idx;
910 int start, end;
911 if (span->fT == 0) {
912 idx = 0;
913 span = &fTs[0];
914 do {
915 if (span->fToAngle) {
916 SkASSERT(span->fToAngle->loopCount() == 2);
917 SkASSERT(!span->fFromAngle);
918 span->fFromAngle = span->fToAngle->next();
919 return;
920 }
921 span = &fTs[++idx];
922 } while (span->fT == 0);
923 SkASSERT(!fTs[0].fTiny && fTs[idx].fT > 0);
924 addStartSpan(idx);
925 start = 0;
926 end = idx;
927 } else {
928 idx = count() - 1;
929 span = &fTs[idx];
930 do {
931 if (span->fFromAngle) {
932 SkASSERT(span->fFromAngle->loopCount() == 2);
933 SkASSERT(!span->fToAngle);
934 span->fToAngle = span->fFromAngle->next();
935 return;
936 }
937 span = &fTs[--idx];
938 } while (span->fT == 1);
939 SkASSERT(!fTs[idx].fTiny && fTs[idx].fT < 1);
940 addEndSpan(++idx);
941 start = idx;
942 end = count();
943 }
944 SkOpSegment* other;
945 SkOpSpan* oSpan;
946 index = start;
947 do {
948 span = &fTs[index];
949 other = span->fOther;
950 int oFrom = span->fOtherIndex;
951 oSpan = &other->fTs[oFrom];
952 if (oSpan->fT < 1 && oSpan->fWindValue) {
953 break;
954 }
955 if (oSpan->fT == 0) {
956 continue;
957 }
958 oFrom = other->nextExactSpan(oFrom, -1);
959 SkOpSpan* oFromSpan = &other->fTs[oFrom];
960 SkASSERT(oFromSpan->fT < 1);
961 if (oFromSpan->fWindValue) {
962 break;
963 }
964 } while (++index < end);
965 SkOpAngle* angle, * oAngle;
966 if (span->fT == 0) {
967 SkASSERT(span->fOtherIndex - 1 >= 0);
968 SkASSERT(span->fOtherT == 1);
969 SkDEBUGCODE(int oPriorIndex = other->nextExactSpan(span->fOtherIndex, -1 ));
970 SkDEBUGCODE(const SkOpSpan& oPrior = other->span(oPriorIndex));
971 SkASSERT(!oPrior.fTiny && oPrior.fT < 1);
972 other->addEndSpan(span->fOtherIndex);
973 angle = span->fToAngle;
974 oAngle = oSpan->fFromAngle;
975 } else {
976 SkASSERT(span->fOtherIndex + 1 < other->count());
977 SkASSERT(span->fOtherT == 0);
978 SkASSERT(!oSpan->fTiny && (other->fTs[span->fOtherIndex + 1].fT > 0
979 || (other->fTs[span->fOtherIndex + 1].fFromAngle == NULL
980 && other->fTs[span->fOtherIndex + 1].fToAngle == NULL)));
981 int oIndex = 1;
982 do {
983 const SkOpSpan& osSpan = other->span(oIndex);
984 if (osSpan.fFromAngle || osSpan.fT > 0) {
985 break;
986 }
987 ++oIndex;
988 SkASSERT(oIndex < other->count());
989 } while (true);
990 other->addStartSpan(oIndex);
991 angle = span->fFromAngle;
992 oAngle = oSpan->fToAngle;
993 }
994 angle->insert(oAngle);
995 }
996
997 void SkOpSegment::alignMultiples(SkTDArray<AlignedSpan>* alignedArray) {
998 debugValidate();
999 int count = this->count();
1000 for (int index = 0; index < count; ++index) {
1001 SkOpSpan& span = fTs[index];
1002 if (!span.fMultiple) {
1003 continue;
1004 }
1005 int end = nextExactSpan(index, 1);
1006 SkASSERT(end > index + 1);
1007 const SkPoint& thisPt = span.fPt;
1008 while (index < end - 1) {
1009 SkOpSegment* other1 = span.fOther;
1010 int oCnt = other1->count();
1011 for (int idx2 = index + 1; idx2 < end; ++idx2) {
1012 SkOpSpan& span2 = fTs[idx2];
1013 SkOpSegment* other2 = span2.fOther;
1014 for (int oIdx = 0; oIdx < oCnt; ++oIdx) {
1015 SkOpSpan& oSpan = other1->fTs[oIdx];
1016 if (oSpan.fOther != other2) {
1017 continue;
1018 }
1019 if (oSpan.fPt == thisPt) {
1020 goto skipExactMatches;
1021 } 498 }
1022 } 499 }
1023 for (int oIdx = 0; oIdx < oCnt; ++oIdx) { 500 }
1024 SkOpSpan& oSpan = other1->fTs[oIdx]; 501 }
1025 if (oSpan.fOther != other2) { 502 SkDCubic part = cubic.subDivide(startT, endT);
1026 continue; 503 for (int index = 0; index < 4; ++index) {
1027 } 504 edge[index] = part[index].asSkPoint();
1028 if (SkDPoint::RoughlyEqual(oSpan.fPt, thisPt)) { 505 }
1029 SkOpSpan& oSpan2 = other2->fTs[oSpan.fOtherIndex]; 506 } else {
1030 if (zero_or_one(span.fOtherT) || zero_or_one(oSpan.fT) 507 subDivide(start, end, edge);
1031 || zero_or_one(span2.fOtherT) || zero_or_one(oSp an2.fT)) { 508 }
1032 return; 509 bool sumSet = false;
1033 } 510 int points = SkPathOpsVerbToPoints(fVerb);
1034 if (!way_roughly_equal(span.fOtherT, oSpan.fT) 511 double sum = (edge[0].fX - edge[points].fX) * (edge[0].fY + edge[points].fY) ;
1035 || !way_roughly_equal(span2.fOtherT, oSpan2.fT) 512 if (!sumSet) {
1036 || !way_roughly_equal(span2.fOtherT, oSpan.fOthe rT) 513 for (int idx = 0; idx < points; ++idx){
1037 || !way_roughly_equal(span.fOtherT, oSpan2.fOthe rT)) { 514 sum += (edge[idx + 1].fX - edge[idx].fX) * (edge[idx + 1].fY + edge[ idx].fY);
1038 return; 515 }
1039 } 516 }
1040 alignSpan(thisPt, span.fOtherT, other1, span2.fOtherT, 517 if (fVerb == SkPath::kCubic_Verb) {
1041 other2, &oSpan, alignedArray); 518 SkDCubic cubic;
1042 alignSpan(thisPt, span2.fOtherT, other2, span.fOtherT, 519 cubic.set(edge);
1043 other1, &oSpan2, alignedArray); 520 *swap = sum > 0 && !cubic.monotonicInY();
1044 break; 521 } else {
1045 } 522 SkDQuad quad;
1046 } 523 quad.set(edge);
1047 skipExactMatches: 524 *swap = sum > 0 && !quad.monotonicInY();
1048 ; 525 }
1049 } 526 return sum <= 0;
1050 ++index; 527 }
1051 } 528
1052 } 529 void SkOpSegment::ComputeOneSum(const SkOpAngle* baseAngle, SkOpAngle* nextAngle ,
1053 debugValidate(); 530 SkOpAngle::IncludeType includeType) {
1054 } 531 const SkOpSegment* baseSegment = baseAngle->segment();
1055 532 int sumMiWinding = baseSegment->updateWindingReverse(baseAngle);
1056 void SkOpSegment::alignRange(int lower, int upper, 533 int sumSuWinding;
1057 const SkOpSegment* other, int oLower, int oUpper) { 534 bool binary = includeType >= SkOpAngle::kBinarySingle;
1058 for (int oIndex = oLower; oIndex <= oUpper; ++oIndex) {
1059 const SkOpSpan& oSpan = other->span(oIndex);
1060 const SkOpSegment* oOther = oSpan.fOther;
1061 if (oOther == this) {
1062 continue;
1063 }
1064 SkOpSpan* matchSpan;
1065 int matchIndex;
1066 const SkOpSpan* refSpan;
1067 for (int iIndex = lower; iIndex <= upper; ++iIndex) {
1068 const SkOpSpan& iSpan = this->span(iIndex);
1069 const SkOpSegment* iOther = iSpan.fOther;
1070 if (iOther == other) {
1071 continue;
1072 }
1073 if (iOther == oOther) {
1074 goto nextI;
1075 }
1076 }
1077 {
1078 // oSpan does not have a match in this
1079 int iCount = this->count();
1080 const SkOpSpan* iMatch = NULL;
1081 double iMatchTDiff;
1082 matchIndex = -1;
1083 for (int iIndex = 0; iIndex < iCount; ++iIndex) {
1084 const SkOpSpan& iSpan = this->span(iIndex);
1085 const SkOpSegment* iOther = iSpan.fOther;
1086 if (iOther != oOther) {
1087 continue;
1088 }
1089 double testTDiff = fabs(iSpan.fOtherT - oSpan.fOtherT);
1090 if (!iMatch || testTDiff < iMatchTDiff) {
1091 matchIndex = iIndex;
1092 iMatch = &iSpan;
1093 iMatchTDiff = testTDiff;
1094 }
1095 }
1096 if (matchIndex < 0) {
1097 continue; // the entry is missing, & will be picked up later (F IXME: fix it here?)
1098 }
1099 matchSpan = &this->fTs[matchIndex];
1100 refSpan = &this->span(lower);
1101 if (!SkDPoint::ApproximatelyEqual(matchSpan->fPt, refSpan->fPt)) {
1102 goto nextI;
1103 }
1104 if (matchIndex != lower - 1 && matchIndex != upper + 1) {
1105 // the consecutive spans need to be rearranged to get the missin g one close
1106 continue; // FIXME: more work to do
1107 }
1108 }
1109 {
1110 this->fixOtherTIndex();
1111 SkScalar newT;
1112 if (matchSpan->fT != 0 && matchSpan->fT != 1) {
1113 newT = matchSpan->fT = refSpan->fT;
1114 matchSpan->fOther->fTs[matchSpan->fOtherIndex].fOtherT = refSpan ->fT;
1115 } else { // leave span at the start or end there and adjust the nei ghbors
1116 newT = matchSpan->fT;
1117 for (int iIndex = lower; iIndex <= upper; ++iIndex) {
1118 matchSpan = &this->fTs[iIndex];
1119 matchSpan->fT = newT;
1120 matchSpan->fOther->fTs[matchSpan->fOtherIndex].fOtherT = new T;
1121 }
1122 }
1123 this->resetSpanFlags(); // fix up small / tiny / done
1124 // align ts of other ranges with adjacent spans that match the align ed points
1125 lower = SkTMin(lower, matchIndex);
1126 while (lower > 0) {
1127 const SkOpSpan& span = this->span(lower - 1);
1128 if (span.fT != newT) {
1129 break;
1130 }
1131 --lower;
1132 }
1133 upper = SkTMax(upper, matchIndex);
1134 int last = this->count() - 1;
1135 while (upper < last) {
1136 const SkOpSpan& span = this->span(upper + 1);
1137 if (span.fT != newT) {
1138 break;
1139 }
1140 ++upper;
1141 }
1142 for (int iIndex = lower; iIndex <= upper; ++iIndex) {
1143 const SkOpSpan& span = this->span(iIndex);
1144 SkOpSegment* aOther = span.fOther;
1145 int aLower = span.fOtherIndex;
1146 SkScalar aT = span.fOtherT;
1147 bool aResetFlags = false;
1148 while (aLower > 0) {
1149 SkOpSpan* aSpan = &aOther->fTs[aLower - 1];
1150 for (int iIndex = lower; iIndex <= upper; ++iIndex) {
1151 if (aSpan->fPt == this->fTs[iIndex].fPt) {
1152 goto matchFound;
1153 }
1154 }
1155 break;
1156 matchFound:
1157 --aLower;
1158 }
1159 int aUpper = span.fOtherIndex;
1160 int aLast = aOther->count() - 1;
1161 while (aUpper < aLast) {
1162 SkOpSpan* aSpan = &aOther->fTs[aUpper + 1];
1163 for (int iIndex = lower; iIndex <= upper; ++iIndex) {
1164 if (aSpan->fPt == this->fTs[iIndex].fPt) {
1165 goto matchFound2;
1166 }
1167 }
1168 break;
1169 matchFound2:
1170 ++aUpper;
1171 }
1172 if (aOther->fTs[aLower].fT == 0) {
1173 aT = 0;
1174 } else if (aOther->fTs[aUpper].fT == 1) {
1175 aT = 1;
1176 }
1177 bool aFixed = false;
1178 for (int aIndex = aLower; aIndex <= aUpper; ++aIndex) {
1179 SkOpSpan* aSpan = &aOther->fTs[aIndex];
1180 if (aSpan->fT == aT) {
1181 continue;
1182 }
1183 SkASSERT(way_roughly_equal(aSpan->fT, aT));
1184 if (!aFixed) {
1185 aOther->fixOtherTIndex();
1186 aFixed = true;
1187 }
1188 aSpan->fT = aT;
1189 aSpan->fOther->fTs[aSpan->fOtherIndex].fOtherT = aT;
1190 aResetFlags = true;
1191 }
1192 if (aResetFlags) {
1193 aOther->resetSpanFlags();
1194 }
1195 }
1196 }
1197 nextI: ;
1198 }
1199 }
1200
1201 void SkOpSegment::alignSpan(const SkPoint& newPt, double newT, const SkOpSegment * other,
1202 double otherT, const SkOpSegment* other2, SkOpSpan* oSpan,
1203 SkTDArray<AlignedSpan>* alignedArray) {
1204 AlignedSpan* aligned = alignedArray->append();
1205 aligned->fOldPt = oSpan->fPt;
1206 aligned->fPt = newPt;
1207 aligned->fOldT = oSpan->fT;
1208 aligned->fT = newT;
1209 aligned->fSegment = this; // OPTIMIZE: may be unused, can remove
1210 aligned->fOther1 = other;
1211 aligned->fOther2 = other2;
1212 SkASSERT(SkDPoint::RoughlyEqual(oSpan->fPt, newPt));
1213 oSpan->fPt = newPt;
1214 // SkASSERT(way_roughly_equal(oSpan->fT, newT));
1215 oSpan->fT = newT;
1216 // SkASSERT(way_roughly_equal(oSpan->fOtherT, otherT));
1217 oSpan->fOtherT = otherT;
1218 }
1219
1220 bool SkOpSegment::alignSpan(int index, double thisT, const SkPoint& thisPt) {
1221 bool aligned = false;
1222 SkOpSpan* span = &fTs[index];
1223 SkOpSegment* other = span->fOther;
1224 int oIndex = span->fOtherIndex;
1225 SkOpSpan* oSpan = &other->fTs[oIndex];
1226 if (span->fT != thisT) {
1227 span->fT = thisT;
1228 oSpan->fOtherT = thisT;
1229 aligned = true;
1230 }
1231 if (span->fPt != thisPt) {
1232 span->fPt = thisPt;
1233 oSpan->fPt = thisPt;
1234 aligned = true;
1235 }
1236 double oT = oSpan->fT;
1237 if (oT == 0) {
1238 return aligned;
1239 }
1240 int oStart = other->nextSpan(oIndex, -1) + 1;
1241 oSpan = &other->fTs[oStart];
1242 int otherIndex = oStart;
1243 if (oT == 1) {
1244 if (aligned) {
1245 while (oSpan->fPt == thisPt && oSpan->fT != 1) {
1246 oSpan->fTiny = true;
1247 ++oSpan;
1248 }
1249 }
1250 return aligned;
1251 }
1252 oT = oSpan->fT;
1253 int oEnd = other->nextSpan(oIndex, 1);
1254 bool oAligned = false;
1255 if (oSpan->fPt != thisPt) {
1256 oAligned |= other->alignSpan(oStart, oT, thisPt);
1257 }
1258 while (++otherIndex < oEnd) {
1259 SkOpSpan* oNextSpan = &other->fTs[otherIndex];
1260 if (oNextSpan->fT != oT || oNextSpan->fPt != thisPt) {
1261 oAligned |= other->alignSpan(otherIndex, oT, thisPt);
1262 }
1263 }
1264 if (oAligned) {
1265 other->alignSpanState(oStart, oEnd);
1266 }
1267 return aligned;
1268 }
1269
1270 void SkOpSegment::alignSpanState(int start, int end) {
1271 SkOpSpan* lastSpan = &fTs[--end];
1272 bool allSmall = lastSpan->fSmall;
1273 bool allTiny = lastSpan->fTiny;
1274 bool allDone = lastSpan->fDone;
1275 SkDEBUGCODE(int winding = lastSpan->fWindValue);
1276 SkDEBUGCODE(int oppWinding = lastSpan->fOppValue);
1277 int index = start;
1278 while (index < end) {
1279 SkOpSpan* span = &fTs[index];
1280 span->fSmall = allSmall;
1281 span->fTiny = allTiny;
1282 if (span->fDone != allDone) {
1283 span->fDone = allDone;
1284 fDoneSpans += allDone ? 1 : -1;
1285 }
1286 SkASSERT(span->fWindValue == winding);
1287 SkASSERT(span->fOppValue == oppWinding);
1288 ++index;
1289 }
1290 }
1291
1292 void SkOpSegment::blindCancel(const SkCoincidence& coincidence, SkOpSegment* oth er) {
1293 bool binary = fOperand != other->fOperand;
1294 int index = 0;
1295 int last = this->count();
1296 do {
1297 SkOpSpan& span = this->fTs[--last];
1298 if (span.fT != 1 && !span.fSmall) {
1299 break;
1300 }
1301 span.fCoincident = true;
1302 } while (true);
1303 int oIndex = other->count();
1304 do {
1305 SkOpSpan& oSpan = other->fTs[--oIndex];
1306 if (oSpan.fT != 1 && !oSpan.fSmall) {
1307 break;
1308 }
1309 oSpan.fCoincident = true;
1310 } while (true);
1311 do {
1312 SkOpSpan* test = &this->fTs[index];
1313 int baseWind = test->fWindValue;
1314 int baseOpp = test->fOppValue;
1315 int endIndex = index;
1316 while (++endIndex <= last) {
1317 SkOpSpan* endSpan = &this->fTs[endIndex];
1318 SkASSERT(endSpan->fT < 1);
1319 if (endSpan->fWindValue != baseWind || endSpan->fOppValue != baseOpp ) {
1320 break;
1321 }
1322 endSpan->fCoincident = true;
1323 }
1324 SkOpSpan* oTest = &other->fTs[oIndex];
1325 int oBaseWind = oTest->fWindValue;
1326 int oBaseOpp = oTest->fOppValue;
1327 int oStartIndex = oIndex;
1328 while (--oStartIndex >= 0) {
1329 SkOpSpan* oStartSpan = &other->fTs[oStartIndex];
1330 if (oStartSpan->fWindValue != oBaseWind || oStartSpan->fOppValue != oBaseOpp) {
1331 break;
1332 }
1333 oStartSpan->fCoincident = true;
1334 }
1335 bool decrement = baseWind && oBaseWind;
1336 bool bigger = baseWind >= oBaseWind;
1337 do {
1338 SkASSERT(test->fT < 1);
1339 if (decrement) {
1340 if (binary && bigger) {
1341 test->fOppValue--;
1342 } else {
1343 decrementSpan(test);
1344 }
1345 }
1346 test->fCoincident = true;
1347 test = &fTs[++index];
1348 } while (index < endIndex);
1349 do {
1350 SkASSERT(oTest->fT < 1);
1351 if (decrement) {
1352 if (binary && !bigger) {
1353 oTest->fOppValue--;
1354 } else {
1355 other->decrementSpan(oTest);
1356 }
1357 }
1358 oTest->fCoincident = true;
1359 oTest = &other->fTs[--oIndex];
1360 } while (oIndex > oStartIndex);
1361 } while (index <= last && oIndex >= 0);
1362 SkASSERT(index > last);
1363 SkASSERT(oIndex < 0);
1364 }
1365
1366 void SkOpSegment::blindCoincident(const SkCoincidence& coincidence, SkOpSegment* other) {
1367 bool binary = fOperand != other->fOperand;
1368 int index = 0;
1369 int last = this->count();
1370 do {
1371 SkOpSpan& span = this->fTs[--last];
1372 if (span.fT != 1 && !span.fSmall) {
1373 break;
1374 }
1375 span.fCoincident = true;
1376 } while (true);
1377 int oIndex = 0;
1378 int oLast = other->count();
1379 do {
1380 SkOpSpan& oSpan = other->fTs[--oLast];
1381 if (oSpan.fT != 1 && !oSpan.fSmall) {
1382 break;
1383 }
1384 oSpan.fCoincident = true;
1385 } while (true);
1386 do {
1387 SkOpSpan* test = &this->fTs[index];
1388 int baseWind = test->fWindValue;
1389 int baseOpp = test->fOppValue;
1390 int endIndex = index;
1391 SkOpSpan* endSpan;
1392 while (++endIndex <= last) {
1393 endSpan = &this->fTs[endIndex];
1394 SkASSERT(endSpan->fT < 1);
1395 if (endSpan->fWindValue != baseWind || endSpan->fOppValue != baseOpp ) {
1396 break;
1397 }
1398 endSpan->fCoincident = true;
1399 }
1400 SkOpSpan* oTest = &other->fTs[oIndex];
1401 int oBaseWind = oTest->fWindValue;
1402 int oBaseOpp = oTest->fOppValue;
1403 int oEndIndex = oIndex;
1404 SkOpSpan* oEndSpan;
1405 while (++oEndIndex <= oLast) {
1406 oEndSpan = &this->fTs[oEndIndex];
1407 SkASSERT(oEndSpan->fT < 1);
1408 if (oEndSpan->fWindValue != oBaseWind || oEndSpan->fOppValue != oBas eOpp) {
1409 break;
1410 }
1411 oEndSpan->fCoincident = true;
1412 }
1413 // consolidate the winding count even if done
1414 if ((test->fWindValue || test->fOppValue) && (oTest->fWindValue || oTest ->fOppValue)) {
1415 if (!binary || test->fWindValue + oTest->fOppValue >= 0) {
1416 bumpCoincidentBlind(binary, index, endIndex);
1417 other->bumpCoincidentOBlind(oIndex, oEndIndex);
1418 } else {
1419 other->bumpCoincidentBlind(binary, oIndex, oEndIndex);
1420 bumpCoincidentOBlind(index, endIndex);
1421 }
1422 }
1423 index = endIndex;
1424 oIndex = oEndIndex;
1425 } while (index <= last && oIndex <= oLast);
1426 SkASSERT(index > last);
1427 SkASSERT(oIndex > oLast);
1428 }
1429
1430 void SkOpSegment::bumpCoincidentBlind(bool binary, int index, int endIndex) {
1431 const SkOpSpan& oTest = fTs[index];
1432 int oWindValue = oTest.fWindValue;
1433 int oOppValue = oTest.fOppValue;
1434 if (binary) { 535 if (binary) {
1435 SkTSwap<int>(oWindValue, oOppValue); 536 sumSuWinding = baseSegment->updateOppWindingReverse(baseAngle);
1436 } 537 if (baseSegment->operand()) {
1437 do { 538 SkTSwap<int>(sumMiWinding, sumSuWinding);
1438 (void) bumpSpan(&fTs[index], oWindValue, oOppValue); 539 }
1439 } while (++index < endIndex); 540 }
1440 } 541 SkOpSegment* nextSegment = nextAngle->segment();
1441 542 int maxWinding, sumWinding;
1442 bool SkOpSegment::bumpCoincidentThis(const SkOpSpan& oTest, bool binary, int* in dexPtr, 543 SkOpSpanBase* last;
1443 SkTArray<SkPoint, true>* outsideTs) {
1444 int index = *indexPtr;
1445 int oWindValue = oTest.fWindValue;
1446 int oOppValue = oTest.fOppValue;
1447 if (binary) { 544 if (binary) {
1448 SkTSwap<int>(oWindValue, oOppValue); 545 int oppMaxWinding, oppSumWinding;
1449 } 546 nextSegment->setUpWindings(nextAngle->start(), nextAngle->end(), &sumMiW inding,
1450 SkOpSpan* const test = &fTs[index]; 547 &sumSuWinding, &maxWinding, &sumWinding, &oppMaxWinding, &oppSum Winding);
1451 SkOpSpan* end = test; 548 last = nextSegment->markAngle(maxWinding, sumWinding, oppMaxWinding, opp SumWinding,
1452 const SkPoint& oStartPt = oTest.fPt; 549 nextAngle);
1453 do { 550 } else {
1454 if (end->fDone && !end->fTiny && !end->fSmall) { // extremely large pat hs trigger this 551 nextSegment->setUpWindings(nextAngle->start(), nextAngle->end(), &sumMiW inding,
1455 return false; 552 &maxWinding, &sumWinding);
1456 } 553 last = nextSegment->markAngle(maxWinding, sumWinding, nextAngle);
1457 if (bumpSpan(end, oWindValue, oOppValue)) { 554 }
1458 TrackOutside(outsideTs, oStartPt); 555 nextAngle->setLastMarked(last);
1459 } 556 }
1460 end = &fTs[++index]; 557
1461 } while ((end->fPt == test->fPt || precisely_equal(end->fT, test->fT)) && en d->fT < 1); 558 void SkOpSegment::ComputeOneSumReverse(const SkOpAngle* baseAngle, SkOpAngle* ne xtAngle,
1462 *indexPtr = index; 559 SkOpAngle::IncludeType includeType) {
1463 return true; 560 const SkOpSegment* baseSegment = baseAngle->segment();
1464 } 561 int sumMiWinding = baseSegment->updateWinding(baseAngle);
1465 562 int sumSuWinding;
1466 void SkOpSegment::bumpCoincidentOBlind(int index, int endIndex) { 563 bool binary = includeType >= SkOpAngle::kBinarySingle;
1467 do { 564 if (binary) {
1468 zeroSpan(&fTs[index]); 565 sumSuWinding = baseSegment->updateOppWinding(baseAngle);
1469 } while (++index < endIndex); 566 if (baseSegment->operand()) {
1470 } 567 SkTSwap<int>(sumMiWinding, sumSuWinding);
1471 568 }
1472 // because of the order in which coincidences are resolved, this and other 569 }
1473 // may not have the same intermediate points. Compute the corresponding 570 SkOpSegment* nextSegment = nextAngle->segment();
1474 // intermediate T values (using this as the master, other as the follower) 571 int maxWinding, sumWinding;
1475 // and walk other conditionally -- hoping that it catches up in the end 572 SkOpSpanBase* last;
1476 bool SkOpSegment::bumpCoincidentOther(const SkOpSpan& test, int* oIndexPtr, 573 if (binary) {
1477 SkTArray<SkPoint, true>* oOutsidePts, const SkPoint& oEndPt) { 574 int oppMaxWinding, oppSumWinding;
1478 int oIndex = *oIndexPtr; 575 nextSegment->setUpWindings(nextAngle->end(), nextAngle->start(), &sumMiW inding,
1479 SkOpSpan* const oTest = &fTs[oIndex]; 576 &sumSuWinding, &maxWinding, &sumWinding, &oppMaxWinding, &oppSum Winding);
1480 SkOpSpan* oEnd = oTest; 577 last = nextSegment->markAngle(maxWinding, sumWinding, oppMaxWinding, opp SumWinding,
1481 const SkPoint& oStartPt = oTest->fPt; 578 nextAngle);
1482 double oStartT = oTest->fT; 579 } else {
1483 #if 0 // FIXME : figure out what disabling this breaks 580 nextSegment->setUpWindings(nextAngle->end(), nextAngle->start(), &sumMiW inding,
1484 const SkPoint& startPt = test.fPt; 581 &maxWinding, &sumWinding);
1485 // this is always true since oEnd == oTest && oStartPt == oTest->fPt -- find proper condition 582 last = nextSegment->markAngle(maxWinding, sumWinding, nextAngle);
1486 if (oStartPt == oEnd->fPt || precisely_equal(oStartT, oEnd->fT)) { 583 }
1487 TrackOutside(oOutsidePts, startPt); 584 nextAngle->setLastMarked(last);
1488 }
1489 #endif
1490 bool foundEnd = false;
1491 while (oStartPt == oEnd->fPt || precisely_equal(oStartT, oEnd->fT)) {
1492 foundEnd |= oEndPt == oEnd->fPt;
1493 zeroSpan(oEnd);
1494 oEnd = &fTs[++oIndex];
1495 }
1496 *oIndexPtr = oIndex;
1497 return foundEnd;
1498 }
1499
1500 // FIXME: need to test this case:
1501 // contourA has two segments that are coincident
1502 // contourB has two segments that are coincident in the same place
1503 // each ends up with +2/0 pairs for winding count
1504 // since logic below doesn't transfer count (only increments/decrements) can thi s be
1505 // resolved to +4/0 ?
1506
1507 // set spans from start to end to increment the greater by one and decrement
1508 // the lesser
1509 bool SkOpSegment::addTCoincident(const SkPoint& startPt, const SkPoint& endPt, d ouble endT,
1510 SkOpSegment* other) {
1511 bool binary = fOperand != other->fOperand;
1512 int index = 0;
1513 while (startPt != fTs[index].fPt) {
1514 SkASSERT(index < fTs.count());
1515 ++index;
1516 }
1517 double startT = fTs[index].fT;
1518 while (index > 0 && precisely_equal(fTs[index - 1].fT, startT)) {
1519 --index;
1520 }
1521 int oIndex = 0;
1522 while (startPt != other->fTs[oIndex].fPt) {
1523 SkASSERT(oIndex < other->fTs.count());
1524 ++oIndex;
1525 }
1526 double oStartT = other->fTs[oIndex].fT;
1527 while (oIndex > 0 && precisely_equal(other->fTs[oIndex - 1].fT, oStartT)) {
1528 --oIndex;
1529 }
1530 SkSTArray<kOutsideTrackedTCount, SkPoint, true> outsidePts;
1531 SkSTArray<kOutsideTrackedTCount, SkPoint, true> oOutsidePts;
1532 SkOpSpan* test = &fTs[index];
1533 const SkPoint* testPt = &test->fPt;
1534 double testT = test->fT;
1535 SkOpSpan* oTest = &other->fTs[oIndex];
1536 const SkPoint* oTestPt = &oTest->fPt;
1537 // paths with extreme data will fail this test and eject out of pathops alto gether later on
1538 // SkASSERT(AlmostEqualUlps(*testPt, *oTestPt));
1539 do {
1540 SkASSERT(test->fT < 1);
1541 if (oTest->fT == 1) {
1542 // paths with extreme data may be so mismatched that we fail here
1543 return false;
1544 }
1545
1546 // consolidate the winding count even if done
1547 bool foundEnd = false;
1548 if ((test->fWindValue == 0 && test->fOppValue == 0)
1549 || (oTest->fWindValue == 0 && oTest->fOppValue == 0)) {
1550 SkDEBUGCODE(int firstWind = test->fWindValue);
1551 SkDEBUGCODE(int firstOpp = test->fOppValue);
1552 do {
1553 SkASSERT(firstWind == fTs[index].fWindValue);
1554 SkASSERT(firstOpp == fTs[index].fOppValue);
1555 ++index;
1556 SkASSERT(index < fTs.count());
1557 } while (*testPt == fTs[index].fPt);
1558 SkDEBUGCODE(firstWind = oTest->fWindValue);
1559 SkDEBUGCODE(firstOpp = oTest->fOppValue);
1560 do {
1561 SkASSERT(firstWind == other->fTs[oIndex].fWindValue);
1562 SkASSERT(firstOpp == other->fTs[oIndex].fOppValue);
1563 ++oIndex;
1564 SkASSERT(oIndex < other->fTs.count());
1565 } while (*oTestPt == other->fTs[oIndex].fPt);
1566 } else {
1567 if (!binary || test->fWindValue + oTest->fOppValue >= 0) {
1568 if (!bumpCoincidentThis(*oTest, binary, &index, &outsidePts)) {
1569 return false;
1570 }
1571 foundEnd = other->bumpCoincidentOther(*test, &oIndex, &oOutsideP ts, endPt);
1572 } else {
1573 if (!other->bumpCoincidentThis(*test, binary, &oIndex, &oOutside Pts)) {
1574 return false;
1575 }
1576 foundEnd = bumpCoincidentOther(*oTest, &index, &outsidePts, endP t);
1577 }
1578 }
1579 test = &fTs[index];
1580 testPt = &test->fPt;
1581 testT = test->fT;
1582 oTest = &other->fTs[oIndex];
1583 oTestPt = &oTest->fPt;
1584 if (endPt == *testPt || precisely_equal(endT, testT)) {
1585 break;
1586 }
1587 if (0 && foundEnd) { // FIXME: this is likely needed but wait until a t est case triggers it
1588 break;
1589 }
1590 // SkASSERT(AlmostEqualUlps(*testPt, *oTestPt));
1591 } while (endPt != *oTestPt);
1592 // in rare cases, one may have ended before the other
1593 if (endPt != *testPt && !precisely_equal(endT, testT)) {
1594 int lastWind = test[-1].fWindValue;
1595 int lastOpp = test[-1].fOppValue;
1596 bool zero = lastWind == 0 && lastOpp == 0;
1597 do {
1598 if (test->fWindValue || test->fOppValue) {
1599 test->fWindValue = lastWind;
1600 test->fOppValue = lastOpp;
1601 if (zero) {
1602 SkASSERT(!test->fDone);
1603 test->fDone = true;
1604 ++fDoneSpans;
1605 }
1606 }
1607 test = &fTs[++index];
1608 testPt = &test->fPt;
1609 } while (endPt != *testPt);
1610 }
1611 if (endPt != *oTestPt) {
1612 // look ahead to see if zeroing more spans will allows us to catch up
1613 int oPeekIndex = oIndex;
1614 bool success = true;
1615 SkOpSpan* oPeek;
1616 int oCount = other->count();
1617 do {
1618 oPeek = &other->fTs[oPeekIndex];
1619 if (++oPeekIndex == oCount) {
1620 success = false;
1621 break;
1622 }
1623 } while (endPt != oPeek->fPt);
1624 if (success) {
1625 // make sure the matching point completes the coincidence span
1626 success = false;
1627 do {
1628 if (oPeek->fOther == this) {
1629 success = true;
1630 break;
1631 }
1632 if (++oPeekIndex == oCount) {
1633 break;
1634 }
1635 oPeek = &other->fTs[oPeekIndex];
1636 } while (endPt == oPeek->fPt);
1637 }
1638 if (success) {
1639 do {
1640 if (!binary || test->fWindValue + oTest->fOppValue >= 0) {
1641 if (other->bumpCoincidentOther(*test, &oIndex, &oOutsidePts, endPt)) {
1642 break;
1643 }
1644 } else {
1645 if (!other->bumpCoincidentThis(*test, binary, &oIndex, &oOut sidePts)) {
1646 return false;
1647 }
1648 }
1649 oTest = &other->fTs[oIndex];
1650 oTestPt = &oTest->fPt;
1651 } while (endPt != *oTestPt);
1652 }
1653 }
1654 int outCount = outsidePts.count();
1655 if (!done() && outCount) {
1656 addCoinOutsides(outsidePts[0], endPt, other);
1657 }
1658 if (!other->done() && oOutsidePts.count()) {
1659 other->addCoinOutsides(oOutsidePts[0], endPt, this);
1660 }
1661 setCoincidentRange(startPt, endPt, other);
1662 other->setCoincidentRange(startPt, endPt, this);
1663 return true;
1664 }
1665
1666 // FIXME: this doesn't prevent the same span from being added twice
1667 // fix in caller, SkASSERT here?
1668 // FIXME: this may erroneously reject adds for cubic loops
1669 const SkOpSpan* SkOpSegment::addTPair(double t, SkOpSegment* other, double other T, bool borrowWind,
1670 const SkPoint& pt, const SkPoint& pt2) {
1671 int tCount = fTs.count();
1672 for (int tIndex = 0; tIndex < tCount; ++tIndex) {
1673 const SkOpSpan& span = fTs[tIndex];
1674 if (!approximately_negative(span.fT - t)) {
1675 break;
1676 }
1677 if (span.fOther == other) {
1678 bool tsMatch = approximately_equal(span.fT, t);
1679 bool otherTsMatch = approximately_equal(span.fOtherT, otherT);
1680 // FIXME: add cubic loop detecting logic here
1681 // if fLoop bit is set on span, that could be enough if addOtherT co pies the bit
1682 // or if a new bit is added ala fOtherLoop
1683 if (tsMatch || otherTsMatch) {
1684 #if DEBUG_ADD_T_PAIR
1685 SkDebugf("%s addTPair duplicate this=%d %1.9g other=%d %1.9g\n",
1686 __FUNCTION__, fID, t, other->fID, otherT);
1687 #endif
1688 return NULL;
1689 }
1690 }
1691 }
1692 int oCount = other->count();
1693 for (int oIndex = 0; oIndex < oCount; ++oIndex) {
1694 const SkOpSpan& oSpan = other->span(oIndex);
1695 if (!approximately_negative(oSpan.fT - otherT)) {
1696 break;
1697 }
1698 if (oSpan.fOther == this) {
1699 bool otherTsMatch = approximately_equal(oSpan.fT, otherT);
1700 bool tsMatch = approximately_equal(oSpan.fOtherT, t);
1701 if (otherTsMatch || tsMatch) {
1702 #if DEBUG_ADD_T_PAIR
1703 SkDebugf("%s addTPair other duplicate this=%d %1.9g other=%d %1. 9g\n",
1704 __FUNCTION__, fID, t, other->fID, otherT);
1705 #endif
1706 return NULL;
1707 }
1708 }
1709 }
1710 #if DEBUG_ADD_T_PAIR
1711 SkDebugf("%s addTPair this=%d %1.9g other=%d %1.9g\n",
1712 __FUNCTION__, fID, t, other->fID, otherT);
1713 #endif
1714 SkASSERT(other != this);
1715 int insertedAt = addT(other, pt, t);
1716 int otherInsertedAt = other->addT(this, pt2, otherT);
1717 this->addOtherT(insertedAt, otherT, otherInsertedAt);
1718 other->addOtherT(otherInsertedAt, t, insertedAt);
1719 this->matchWindingValue(insertedAt, t, borrowWind);
1720 other->matchWindingValue(otherInsertedAt, otherT, borrowWind);
1721 SkOpSpan& span = this->fTs[insertedAt];
1722 if (pt != pt2) {
1723 span.fNear = true;
1724 SkOpSpan& oSpan = other->fTs[otherInsertedAt];
1725 oSpan.fNear = true;
1726 }
1727 // if the newly inserted spans match a neighbor on one but not the other, ma ke them agree
1728 int lower = this->nextExactSpan(insertedAt, -1) + 1;
1729 int upper = this->nextExactSpan(insertedAt, 1) - 1;
1730 if (upper < 0) {
1731 upper = this->count() - 1;
1732 }
1733 int oLower = other->nextExactSpan(otherInsertedAt, -1) + 1;
1734 int oUpper = other->nextExactSpan(otherInsertedAt, 1) - 1;
1735 if (oUpper < 0) {
1736 oUpper = other->count() - 1;
1737 }
1738 if (lower == upper && oLower == oUpper) {
1739 return &span;
1740 }
1741 #if DEBUG_CONCIDENT
1742 SkDebugf("%s id=%d lower=%d upper=%d other=%d oLower=%d oUpper=%d\n", __FUNC TION__,
1743 debugID(), lower, upper, other->debugID(), oLower, oUpper);
1744 #endif
1745 // find the nearby spans in one range missing in the other
1746 this->alignRange(lower, upper, other, oLower, oUpper);
1747 other->alignRange(oLower, oUpper, this, lower, upper);
1748 return &span;
1749 }
1750
1751 const SkOpSpan* SkOpSegment::addTPair(double t, SkOpSegment* other, double other T, bool borrowWind,
1752 const SkPoint& pt) {
1753 return addTPair(t, other, otherT, borrowWind, pt, pt);
1754 }
1755
1756 bool SkOpSegment::betweenPoints(double midT, const SkPoint& pt1, const SkPoint& pt2) const {
1757 const SkPoint midPt = ptAtT(midT);
1758 SkPathOpsBounds bounds;
1759 bounds.set(pt1.fX, pt1.fY, pt2.fX, pt2.fY);
1760 bounds.sort();
1761 return bounds.almostContains(midPt);
1762 }
1763
1764 bool SkOpSegment::betweenTs(int lesser, double testT, int greater) const {
1765 if (lesser > greater) {
1766 SkTSwap<int>(lesser, greater);
1767 }
1768 return approximately_between(fTs[lesser].fT, testT, fTs[greater].fT);
1769 }
1770
1771 // in extreme cases (like the buffer overflow test) return false to abort
1772 // for now, if one t value represents two different points, then the values are too extreme
1773 // to generate meaningful results
1774 bool SkOpSegment::calcAngles() {
1775 int spanCount = fTs.count();
1776 if (spanCount <= 2) {
1777 return spanCount == 2;
1778 }
1779 int index = 1;
1780 const SkOpSpan* firstSpan = &fTs[index];
1781 int activePrior = checkSetAngle(0);
1782 const SkOpSpan* span = &fTs[0];
1783 if (firstSpan->fT == 0 || span->fTiny || span->fOtherT != 1 || span->fOther- >multipleEnds()) {
1784 index = findStartSpan(0); // curve start intersects
1785 if (fTs[index].fT == 0) {
1786 return false;
1787 }
1788 SkASSERT(index > 0);
1789 if (activePrior >= 0) {
1790 addStartSpan(index);
1791 }
1792 }
1793 bool addEnd;
1794 int endIndex = spanCount - 1;
1795 span = &fTs[endIndex - 1];
1796 if ((addEnd = span->fT == 1 || span->fTiny)) { // if curve end intersects
1797 endIndex = findEndSpan(endIndex);
1798 SkASSERT(endIndex > 0);
1799 } else {
1800 addEnd = fTs[endIndex].fOtherT != 0 || fTs[endIndex].fOther->multipleSta rts();
1801 }
1802 SkASSERT(endIndex >= index);
1803 int prior = 0;
1804 while (index < endIndex) {
1805 const SkOpSpan& fromSpan = fTs[index]; // for each intermediate interse ction
1806 const SkOpSpan* lastSpan;
1807 span = &fromSpan;
1808 int start = index;
1809 do {
1810 lastSpan = span;
1811 span = &fTs[++index];
1812 SkASSERT(index < spanCount);
1813 if (!precisely_negative(span->fT - lastSpan->fT) && !lastSpan->fTiny ) {
1814 break;
1815 }
1816 if (!SkDPoint::ApproximatelyEqual(lastSpan->fPt, span->fPt)) {
1817 return false;
1818 }
1819 } while (true);
1820 SkOpAngle* angle = NULL;
1821 SkOpAngle* priorAngle;
1822 if (activePrior >= 0) {
1823 int pActive = firstActive(prior);
1824 SkASSERT(pActive < start);
1825 priorAngle = &fAngles.push_back();
1826 priorAngle->set(this, start, pActive);
1827 }
1828 int active = checkSetAngle(start);
1829 if (active >= 0) {
1830 SkASSERT(active < index);
1831 angle = &fAngles.push_back();
1832 angle->set(this, active, index);
1833 }
1834 #if DEBUG_ANGLE
1835 debugCheckPointsEqualish(start, index);
1836 #endif
1837 prior = start;
1838 do {
1839 const SkOpSpan* startSpan = &fTs[start - 1];
1840 if (!startSpan->fSmall || isCanceled(start - 1) || startSpan->fFromA ngle
1841 || startSpan->fToAngle) {
1842 break;
1843 }
1844 --start;
1845 } while (start > 0);
1846 do {
1847 if (activePrior >= 0) {
1848 SkASSERT(fTs[start].fFromAngle == NULL);
1849 fTs[start].fFromAngle = priorAngle;
1850 }
1851 if (active >= 0) {
1852 SkASSERT(fTs[start].fToAngle == NULL);
1853 fTs[start].fToAngle = angle;
1854 }
1855 } while (++start < index);
1856 activePrior = active;
1857 }
1858 if (addEnd && activePrior >= 0) {
1859 addEndSpan(endIndex);
1860 }
1861 return true;
1862 }
1863
1864 int SkOpSegment::checkSetAngle(int tIndex) const {
1865 const SkOpSpan* span = &fTs[tIndex];
1866 while (span->fTiny /* || span->fSmall */) {
1867 span = &fTs[++tIndex];
1868 }
1869 return isCanceled(tIndex) ? -1 : tIndex;
1870 } 585 }
1871 586
1872 // at this point, the span is already ordered, or unorderable 587 // at this point, the span is already ordered, or unorderable
1873 int SkOpSegment::computeSum(int startIndex, int endIndex, SkOpAngle::IncludeType includeType) { 588 int SkOpSegment::computeSum(SkOpSpanBase* start, SkOpSpanBase* end,
589 SkOpAngle::IncludeType includeType) {
1874 SkASSERT(includeType != SkOpAngle::kUnaryXor); 590 SkASSERT(includeType != SkOpAngle::kUnaryXor);
1875 SkOpAngle* firstAngle = spanToAngle(endIndex, startIndex); 591 SkOpAngle* firstAngle = this->spanToAngle(end, start);
1876 if (NULL == firstAngle || NULL == firstAngle->next()) { 592 if (NULL == firstAngle || NULL == firstAngle->next()) {
1877 return SK_NaN32; 593 return SK_NaN32;
1878 } 594 }
1879 // if all angles have a computed winding, 595 // if all angles have a computed winding,
1880 // or if no adjacent angles are orderable, 596 // or if no adjacent angles are orderable,
1881 // or if adjacent orderable angles have no computed winding, 597 // or if adjacent orderable angles have no computed winding,
1882 // there's nothing to do 598 // there's nothing to do
1883 // if two orderable angles are adjacent, and both are next to orderable angl es, 599 // if two orderable angles are adjacent, and both are next to orderable angl es,
1884 // and one has winding computed, transfer to the other 600 // and one has winding computed, transfer to the other
1885 SkOpAngle* baseAngle = NULL; 601 SkOpAngle* baseAngle = NULL;
1886 bool tryReverse = false; 602 bool tryReverse = false;
1887 // look for counterclockwise transfers 603 // look for counterclockwise transfers
1888 SkOpAngle* angle = firstAngle->previous(); 604 SkOpAngle* angle = firstAngle->previous();
1889 SkOpAngle* next = angle->next(); 605 SkOpAngle* next = angle->next();
1890 firstAngle = next; 606 firstAngle = next;
1891 do { 607 do {
1892 SkOpAngle* prior = angle; 608 SkOpAngle* prior = angle;
1893 angle = next; 609 angle = next;
1894 next = angle->next(); 610 next = angle->next();
1895 SkASSERT(prior->next() == angle); 611 SkASSERT(prior->next() == angle);
1896 SkASSERT(angle->next() == next); 612 SkASSERT(angle->next() == next);
1897 if (prior->unorderable() || angle->unorderable() || next->unorderable()) { 613 if (prior->unorderable() || angle->unorderable() || next->unorderable()) {
1898 baseAngle = NULL; 614 baseAngle = NULL;
1899 continue; 615 continue;
1900 } 616 }
1901 int testWinding = angle->segment()->windSum(angle); 617 int testWinding = angle->starter()->windSum();
1902 if (SK_MinS32 != testWinding) { 618 if (SK_MinS32 != testWinding) {
1903 baseAngle = angle; 619 baseAngle = angle;
1904 tryReverse = true; 620 tryReverse = true;
1905 continue; 621 continue;
1906 } 622 }
1907 if (baseAngle) { 623 if (baseAngle) {
1908 ComputeOneSum(baseAngle, angle, includeType); 624 ComputeOneSum(baseAngle, angle, includeType);
1909 baseAngle = SK_MinS32 != angle->segment()->windSum(angle) ? angle : NULL; 625 baseAngle = SK_MinS32 != angle->starter()->windSum() ? angle : NULL;
1910 } 626 }
1911 } while (next != firstAngle); 627 } while (next != firstAngle);
1912 if (baseAngle && SK_MinS32 == firstAngle->segment()->windSum(firstAngle)) { 628 if (baseAngle && SK_MinS32 == firstAngle->starter()->windSum()) {
1913 firstAngle = baseAngle; 629 firstAngle = baseAngle;
1914 tryReverse = true; 630 tryReverse = true;
1915 } 631 }
1916 if (tryReverse) { 632 if (tryReverse) {
1917 baseAngle = NULL; 633 baseAngle = NULL;
1918 SkOpAngle* prior = firstAngle; 634 SkOpAngle* prior = firstAngle;
1919 do { 635 do {
1920 angle = prior; 636 angle = prior;
1921 prior = angle->previous(); 637 prior = angle->previous();
1922 SkASSERT(prior->next() == angle); 638 SkASSERT(prior->next() == angle);
1923 next = angle->next(); 639 next = angle->next();
1924 if (prior->unorderable() || angle->unorderable() || next->unorderabl e()) { 640 if (prior->unorderable() || angle->unorderable() || next->unorderabl e()) {
1925 baseAngle = NULL; 641 baseAngle = NULL;
1926 continue; 642 continue;
1927 } 643 }
1928 int testWinding = angle->segment()->windSum(angle); 644 int testWinding = angle->starter()->windSum();
1929 if (SK_MinS32 != testWinding) { 645 if (SK_MinS32 != testWinding) {
1930 baseAngle = angle; 646 baseAngle = angle;
1931 continue; 647 continue;
1932 } 648 }
1933 if (baseAngle) { 649 if (baseAngle) {
1934 ComputeOneSumReverse(baseAngle, angle, includeType); 650 ComputeOneSumReverse(baseAngle, angle, includeType);
1935 baseAngle = SK_MinS32 != angle->segment()->windSum(angle) ? angl e : NULL; 651 baseAngle = SK_MinS32 != angle->starter()->windSum() ? angle : N ULL;
1936 } 652 }
1937 } while (prior != firstAngle); 653 } while (prior != firstAngle);
1938 } 654 }
1939 int minIndex = SkMin32(startIndex, endIndex); 655 return start->starter(end)->windSum();
1940 return windSum(minIndex);
1941 } 656 }
1942 657
1943 void SkOpSegment::ComputeOneSum(const SkOpAngle* baseAngle, SkOpAngle* nextAngle , 658 SkOpSpan* SkOpSegment::crossedSpanY(const SkPoint& basePt, double mid, bool opp, bool current,
1944 SkOpAngle::IncludeType includeType) { 659 SkScalar* bestY, double* hitT, bool* hitSomething, bool* vertical) {
1945 const SkOpSegment* baseSegment = baseAngle->segment();
1946 int sumMiWinding = baseSegment->updateWindingReverse(baseAngle);
1947 int sumSuWinding;
1948 bool binary = includeType >= SkOpAngle::kBinarySingle;
1949 if (binary) {
1950 sumSuWinding = baseSegment->updateOppWindingReverse(baseAngle);
1951 if (baseSegment->operand()) {
1952 SkTSwap<int>(sumMiWinding, sumSuWinding);
1953 }
1954 }
1955 SkOpSegment* nextSegment = nextAngle->segment();
1956 int maxWinding, sumWinding;
1957 SkOpSpan* last;
1958 if (binary) {
1959 int oppMaxWinding, oppSumWinding;
1960 nextSegment->setUpWindings(nextAngle->start(), nextAngle->end(), &sumMiW inding,
1961 &sumSuWinding, &maxWinding, &sumWinding, &oppMaxWinding, &oppSum Winding);
1962 last = nextSegment->markAngle(maxWinding, sumWinding, oppMaxWinding, opp SumWinding,
1963 nextAngle);
1964 } else {
1965 nextSegment->setUpWindings(nextAngle->start(), nextAngle->end(), &sumMiW inding,
1966 &maxWinding, &sumWinding);
1967 last = nextSegment->markAngle(maxWinding, sumWinding, nextAngle);
1968 }
1969 nextAngle->setLastMarked(last);
1970 }
1971
1972 void SkOpSegment::ComputeOneSumReverse(const SkOpAngle* baseAngle, SkOpAngle* ne xtAngle,
1973 SkOpAngle::IncludeType includeType) {
1974 const SkOpSegment* baseSegment = baseAngle->segment();
1975 int sumMiWinding = baseSegment->updateWinding(baseAngle);
1976 int sumSuWinding;
1977 bool binary = includeType >= SkOpAngle::kBinarySingle;
1978 if (binary) {
1979 sumSuWinding = baseSegment->updateOppWinding(baseAngle);
1980 if (baseSegment->operand()) {
1981 SkTSwap<int>(sumMiWinding, sumSuWinding);
1982 }
1983 }
1984 SkOpSegment* nextSegment = nextAngle->segment();
1985 int maxWinding, sumWinding;
1986 SkOpSpan* last;
1987 if (binary) {
1988 int oppMaxWinding, oppSumWinding;
1989 nextSegment->setUpWindings(nextAngle->end(), nextAngle->start(), &sumMiW inding,
1990 &sumSuWinding, &maxWinding, &sumWinding, &oppMaxWinding, &oppSum Winding);
1991 last = nextSegment->markAngle(maxWinding, sumWinding, oppMaxWinding, opp SumWinding,
1992 nextAngle);
1993 } else {
1994 nextSegment->setUpWindings(nextAngle->end(), nextAngle->start(), &sumMiW inding,
1995 &maxWinding, &sumWinding);
1996 last = nextSegment->markAngle(maxWinding, sumWinding, nextAngle);
1997 }
1998 nextAngle->setLastMarked(last);
1999 }
2000
2001 bool SkOpSegment::containsPt(const SkPoint& pt, int index, int endIndex) const {
2002 int step = index < endIndex ? 1 : -1;
2003 do {
2004 const SkOpSpan& span = this->span(index);
2005 if (span.fPt == pt) {
2006 const SkOpSpan& endSpan = this->span(endIndex);
2007 return span.fT == endSpan.fT && pt != endSpan.fPt;
2008 }
2009 index += step;
2010 } while (index != endIndex);
2011 return false;
2012 }
2013
2014 bool SkOpSegment::containsT(double t, const SkOpSegment* other, double otherT) c onst {
2015 int count = this->count();
2016 for (int index = 0; index < count; ++index) {
2017 const SkOpSpan& span = fTs[index];
2018 if (t < span.fT) {
2019 return false;
2020 }
2021 if (t == span.fT) {
2022 if (other != span.fOther) {
2023 continue;
2024 }
2025 if (other->fVerb != SkPath::kCubic_Verb) {
2026 return true;
2027 }
2028 if (!other->fLoop) {
2029 return true;
2030 }
2031 double otherMidT = (otherT + span.fOtherT) / 2;
2032 SkPoint otherPt = other->ptAtT(otherMidT);
2033 return SkDPoint::ApproximatelyEqual(span.fPt, otherPt);
2034 }
2035 }
2036 return false;
2037 }
2038
2039 int SkOpSegment::crossedSpanY(const SkPoint& basePt, SkScalar* bestY, double* hi tT,
2040 bool* hitSomething, double mid, bool opp, bool cur rent) const {
2041 SkScalar bottom = fBounds.fBottom; 660 SkScalar bottom = fBounds.fBottom;
2042 int bestTIndex = -1; 661 *vertical = false;
2043 if (bottom <= *bestY) { 662 if (bottom <= *bestY) {
2044 return bestTIndex; 663 return NULL;
2045 } 664 }
2046 SkScalar top = fBounds.fTop; 665 SkScalar top = fBounds.fTop;
2047 if (top >= basePt.fY) { 666 if (top >= basePt.fY) {
2048 return bestTIndex; 667 return NULL;
2049 } 668 }
2050 if (fBounds.fLeft > basePt.fX) { 669 if (fBounds.fLeft > basePt.fX) {
2051 return bestTIndex; 670 return NULL;
2052 } 671 }
2053 if (fBounds.fRight < basePt.fX) { 672 if (fBounds.fRight < basePt.fX) {
2054 return bestTIndex; 673 return NULL;
2055 } 674 }
2056 if (fBounds.fLeft == fBounds.fRight) { 675 if (fBounds.fLeft == fBounds.fRight) {
2057 // if vertical, and directly above test point, wait for another one 676 // if vertical, and directly above test point, wait for another one
2058 return AlmostEqualUlps(basePt.fX, fBounds.fLeft) ? SK_MinS32 : bestTInde x; 677 *vertical = AlmostEqualUlps(basePt.fX, fBounds.fLeft);
678 return NULL;
2059 } 679 }
2060 // intersect ray starting at basePt with edge 680 // intersect ray starting at basePt with edge
2061 SkIntersections intersections; 681 SkIntersections intersections;
2062 // OPTIMIZE: use specialty function that intersects ray with curve, 682 // OPTIMIZE: use specialty function that intersects ray with curve,
2063 // returning t values only for curve (we don't care about t on ray) 683 // returning t values only for curve (we don't care about t on ray)
2064 intersections.allowNear(false); 684 intersections.allowNear(false);
2065 int pts = (intersections.*CurveVertical[SkPathOpsVerbToPoints(fVerb)]) 685 int pts = (intersections.*CurveVertical[SkPathOpsVerbToPoints(fVerb)])
2066 (fPts, top, bottom, basePt.fX, false); 686 (fPts, top, bottom, basePt.fX, false);
2067 if (pts == 0 || (current && pts == 1)) { 687 if (pts == 0 || (current && pts == 1)) {
2068 return bestTIndex; 688 return NULL;
2069 } 689 }
2070 if (current) { 690 if (current) {
2071 SkASSERT(pts > 1); 691 SkASSERT(pts > 1);
2072 int closestIdx = 0; 692 int closestIdx = 0;
2073 double closest = fabs(intersections[0][0] - mid); 693 double closest = fabs(intersections[0][0] - mid);
2074 for (int idx = 1; idx < pts; ++idx) { 694 for (int idx = 1; idx < pts; ++idx) {
2075 double test = fabs(intersections[0][idx] - mid); 695 double test = fabs(intersections[0][idx] - mid);
2076 if (closest > test) { 696 if (closest > test) {
2077 closestIdx = idx; 697 closestIdx = idx;
2078 closest = test; 698 closest = test;
2079 } 699 }
2080 } 700 }
2081 intersections.quickRemoveOne(closestIdx, --pts); 701 intersections.quickRemoveOne(closestIdx, --pts);
2082 } 702 }
2083 double bestT = -1; 703 double bestT = -1;
2084 for (int index = 0; index < pts; ++index) { 704 for (int index = 0; index < pts; ++index) {
2085 double foundT = intersections[0][index]; 705 double foundT = intersections[0][index];
2086 if (approximately_less_than_zero(foundT) 706 if (approximately_less_than_zero(foundT)
2087 || approximately_greater_than_one(foundT)) { 707 || approximately_greater_than_one(foundT)) {
2088 continue; 708 continue;
2089 } 709 }
2090 SkScalar testY = (*CurvePointAtT[SkPathOpsVerbToPoints(fVerb)])(fPts, fo undT).fY; 710 SkScalar testY = (*CurvePointAtT[SkPathOpsVerbToPoints(fVerb)])(fPts, fo undT).fY;
2091 if (approximately_negative(testY - *bestY) 711 if (approximately_negative(testY - *bestY)
2092 || approximately_negative(basePt.fY - testY)) { 712 || approximately_negative(basePt.fY - testY)) {
2093 continue; 713 continue;
2094 } 714 }
2095 if (pts > 1 && fVerb == SkPath::kLine_Verb) { 715 if (pts > 1 && fVerb == SkPath::kLine_Verb) {
2096 return SK_MinS32; // if the intersection is edge on, wait for anoth er one 716 *vertical = true;
717 return NULL; // if the intersection is edge on, wait for another on e
2097 } 718 }
2098 if (fVerb > SkPath::kLine_Verb) { 719 if (fVerb > SkPath::kLine_Verb) {
2099 SkScalar dx = (*CurveSlopeAtT[SkPathOpsVerbToPoints(fVerb)])(fPts, f oundT).fX; 720 SkScalar dx = (*CurveSlopeAtT[SkPathOpsVerbToPoints(fVerb)])(fPts, f oundT).fX;
2100 if (approximately_zero(dx)) { 721 if (approximately_zero(dx)) {
2101 return SK_MinS32; // hit vertical, wait for another one 722 *vertical = true;
723 return NULL; // hit vertical, wait for another one
2102 } 724 }
2103 } 725 }
2104 *bestY = testY; 726 *bestY = testY;
2105 bestT = foundT; 727 bestT = foundT;
2106 } 728 }
2107 if (bestT < 0) { 729 if (bestT < 0) {
2108 return bestTIndex; 730 return NULL;
2109 } 731 }
2110 SkASSERT(bestT >= 0); 732 SkASSERT(bestT >= 0);
2111 SkASSERT(bestT <= 1); 733 SkASSERT(bestT < 1);
2112 int start; 734 SkOpSpanBase* testTSpanBase = &this->fHead;
2113 int end = 0; 735 SkOpSpanBase* nextTSpan;
736 double endT = 0;
2114 do { 737 do {
2115 start = end; 738 nextTSpan = testTSpanBase->upCast()->next();
2116 end = nextSpan(start, 1); 739 endT = nextTSpan->t();
2117 } while (fTs[end].fT < bestT); 740 if (endT >= bestT) {
2118 // FIXME: see next candidate for a better pattern to find the next start/end pair 741 break;
2119 while (start + 1 < end && fTs[start].fDone) { 742 }
2120 ++start; 743 testTSpanBase = nextTSpan;
2121 } 744 } while (testTSpanBase);
2122 if (!isCanceled(start)) { 745 SkOpSpan* bestTSpan = NULL;
746 SkOpSpan* testTSpan = testTSpanBase->upCast();
747 if (!testTSpan->isCanceled()) {
2123 *hitT = bestT; 748 *hitT = bestT;
2124 bestTIndex = start; 749 bestTSpan = testTSpan;
2125 *hitSomething = true; 750 *hitSomething = true;
2126 } 751 }
2127 return bestTIndex; 752 return bestTSpan;
2128 } 753 }
2129 754
2130 bool SkOpSegment::decrementSpan(SkOpSpan* span) { 755 void SkOpSegment::detach(const SkOpSpan* span) {
2131 SkASSERT(span->fWindValue > 0); 756 if (span->done()) {
2132 if (--(span->fWindValue) == 0) { 757 --this->fDoneCount;
2133 if (!span->fOppValue && !span->fDone) { 758 }
2134 span->fDone = true; 759 --this->fCount;
2135 ++fDoneSpans; 760 }
2136 return true; 761
762 double SkOpSegment::distSq(double t, SkOpAngle* oppAngle) {
763 SkDPoint testPt = this->dPtAtT(t);
764 SkDLine testPerp = {{ testPt, testPt }};
765 SkDVector slope = this->dSlopeAtT(t);
766 testPerp[1].fX += slope.fY;
767 testPerp[1].fY -= slope.fX;
768 SkIntersections i;
769 SkOpSegment* oppSegment = oppAngle->segment();
770 int oppPtCount = SkPathOpsVerbToPoints(oppSegment->verb());
771 (*CurveIntersectRay[oppPtCount])(oppSegment->pts(), testPerp, &i);
772 double closestDistSq = SK_ScalarInfinity;
773 for (int index = 0; index < i.used(); ++index) {
774 if (!between(oppAngle->start()->t(), i[0][index], oppAngle->end()->t())) {
775 continue;
776 }
777 double testDistSq = testPt.distanceSquared(i.pt(index));
778 if (closestDistSq > testDistSq) {
779 closestDistSq = testDistSq;
2137 } 780 }
2138 } 781 }
2139 return false; 782 return closestDistSq;
2140 }
2141
2142 bool SkOpSegment::bumpSpan(SkOpSpan* span, int windDelta, int oppDelta) {
2143 SkASSERT(!span->fDone || span->fTiny || span->fSmall);
2144 span->fWindValue += windDelta;
2145 SkASSERT(span->fWindValue >= 0);
2146 span->fOppValue += oppDelta;
2147 SkASSERT(span->fOppValue >= 0);
2148 if (fXor) {
2149 span->fWindValue &= 1;
2150 }
2151 if (fOppXor) {
2152 span->fOppValue &= 1;
2153 }
2154 if (!span->fWindValue && !span->fOppValue) {
2155 if (!span->fDone) {
2156 span->fDone = true;
2157 ++fDoneSpans;
2158 }
2159 return true;
2160 }
2161 return false;
2162 }
2163
2164 const SkOpSpan& SkOpSegment::firstSpan(const SkOpSpan& thisSpan) const {
2165 const SkOpSpan* firstSpan = &thisSpan; // rewind to the start
2166 const SkOpSpan* beginSpan = fTs.begin();
2167 const SkPoint& testPt = thisSpan.fPt;
2168 while (firstSpan > beginSpan && firstSpan[-1].fPt == testPt) {
2169 --firstSpan;
2170 }
2171 return *firstSpan;
2172 }
2173
2174 const SkOpSpan& SkOpSegment::lastSpan(const SkOpSpan& thisSpan) const {
2175 const SkOpSpan* endSpan = fTs.end() - 1; // last can't be small
2176 const SkOpSpan* lastSpan = &thisSpan; // find the end
2177 const SkPoint& testPt = thisSpan.fPt;
2178 while (lastSpan < endSpan && lastSpan[1].fPt == testPt) {
2179 ++lastSpan;
2180 }
2181 return *lastSpan;
2182 }
2183
2184 // with a loop, the comparison is move involved
2185 // scan backwards and forwards to count all matching points
2186 // (verify that there are twp scans marked as loops)
2187 // compare that against 2 matching scans for loop plus other results
2188 bool SkOpSegment::calcLoopSpanCount(const SkOpSpan& thisSpan, int* smallCounts) {
2189 const SkOpSpan& firstSpan = this->firstSpan(thisSpan); // rewind to the star t
2190 const SkOpSpan& lastSpan = this->lastSpan(thisSpan); // find the end
2191 double firstLoopT = -1, lastLoopT = -1;
2192 const SkOpSpan* testSpan = &firstSpan - 1;
2193 while (++testSpan <= &lastSpan) {
2194 if (testSpan->fLoop) {
2195 firstLoopT = testSpan->fT;
2196 break;
2197 }
2198 }
2199 testSpan = &lastSpan + 1;
2200 while (--testSpan >= &firstSpan) {
2201 if (testSpan->fLoop) {
2202 lastLoopT = testSpan->fT;
2203 break;
2204 }
2205 }
2206 SkASSERT((firstLoopT == -1) == (lastLoopT == -1));
2207 if (firstLoopT == -1) {
2208 return false;
2209 }
2210 SkASSERT(firstLoopT < lastLoopT);
2211 testSpan = &firstSpan - 1;
2212 smallCounts[0] = smallCounts[1] = 0;
2213 while (++testSpan <= &lastSpan) {
2214 SkASSERT(approximately_equal(testSpan->fT, firstLoopT) +
2215 approximately_equal(testSpan->fT, lastLoopT) == 1);
2216 smallCounts[approximately_equal(testSpan->fT, lastLoopT)]++;
2217 }
2218 return true;
2219 }
2220
2221 double SkOpSegment::calcMissingTEnd(const SkOpSegment* ref, double loEnd, double min, double max,
2222 double hiEnd, const SkOpSegment* other, int thisStart) {
2223 if (max >= hiEnd) {
2224 return -1;
2225 }
2226 int end = findOtherT(hiEnd, ref);
2227 if (end < 0) {
2228 return -1;
2229 }
2230 double tHi = span(end).fT;
2231 double tLo, refLo;
2232 if (thisStart >= 0) {
2233 tLo = span(thisStart).fT;
2234 refLo = min;
2235 } else {
2236 int start1 = findOtherT(loEnd, ref);
2237 SkASSERT(start1 >= 0);
2238 tLo = span(start1).fT;
2239 refLo = loEnd;
2240 }
2241 double missingT = (max - refLo) / (hiEnd - refLo);
2242 missingT = tLo + missingT * (tHi - tLo);
2243 return missingT;
2244 }
2245
2246 double SkOpSegment::calcMissingTStart(const SkOpSegment* ref, double loEnd, doub le min, double max,
2247 double hiEnd, const SkOpSegment* other, int thisEnd) {
2248 if (min <= loEnd) {
2249 return -1;
2250 }
2251 int start = findOtherT(loEnd, ref);
2252 if (start < 0) {
2253 return -1;
2254 }
2255 double tLo = span(start).fT;
2256 double tHi, refHi;
2257 if (thisEnd >= 0) {
2258 tHi = span(thisEnd).fT;
2259 refHi = max;
2260 } else {
2261 int end1 = findOtherT(hiEnd, ref);
2262 if (end1 < 0) {
2263 return -1;
2264 }
2265 tHi = span(end1).fT;
2266 refHi = hiEnd;
2267 }
2268 double missingT = (min - loEnd) / (refHi - loEnd);
2269 missingT = tLo + missingT * (tHi - tLo);
2270 return missingT;
2271 }
2272
2273 // see if spans with two or more intersections have the same number on the other end
2274 void SkOpSegment::checkDuplicates() {
2275 debugValidate();
2276 SkSTArray<kMissingSpanCount, MissingSpan, true> missingSpans;
2277 int index;
2278 int endIndex = 0;
2279 bool endFound;
2280 do {
2281 index = endIndex;
2282 endIndex = nextExactSpan(index, 1);
2283 if ((endFound = endIndex < 0)) {
2284 endIndex = count();
2285 }
2286 int dupCount = endIndex - index;
2287 if (dupCount < 2) {
2288 continue;
2289 }
2290 do {
2291 const SkOpSpan* thisSpan = &fTs[index];
2292 if (thisSpan->fNear) {
2293 continue;
2294 }
2295 SkOpSegment* other = thisSpan->fOther;
2296 int oIndex = thisSpan->fOtherIndex;
2297 int oStart = other->nextExactSpan(oIndex, -1) + 1;
2298 int oEnd = other->nextExactSpan(oIndex, 1);
2299 if (oEnd < 0) {
2300 oEnd = other->count();
2301 }
2302 int oCount = oEnd - oStart;
2303 // force the other to match its t and this pt if not on an end point
2304 if (oCount != dupCount) {
2305 MissingSpan& missing = missingSpans.push_back();
2306 missing.fOther = NULL;
2307 SkDEBUGCODE(sk_bzero(&missing, sizeof(missing)));
2308 missing.fPt = thisSpan->fPt;
2309 const SkOpSpan& oSpan = other->span(oIndex);
2310 if (oCount > dupCount) {
2311 missing.fSegment = this;
2312 missing.fT = thisSpan->fT;
2313 other->checkLinks(&oSpan, &missingSpans);
2314 } else {
2315 missing.fSegment = other;
2316 missing.fT = oSpan.fT;
2317 checkLinks(thisSpan, &missingSpans);
2318 }
2319 if (!missingSpans.back().fOther) {
2320 missingSpans.pop_back();
2321 }
2322 }
2323 } while (++index < endIndex);
2324 } while (!endFound);
2325 int missingCount = missingSpans.count();
2326 if (missingCount == 0) {
2327 return;
2328 }
2329 SkSTArray<kMissingSpanCount, MissingSpan, true> missingCoincidence;
2330 for (index = 0; index < missingCount; ++index) {
2331 MissingSpan& missing = missingSpans[index];
2332 SkOpSegment* missingOther = missing.fOther;
2333 if (missing.fSegment == missing.fOther) {
2334 continue;
2335 }
2336 #if 0 // FIXME: this eliminates spurious data from skpwww_argus_presse_fr_41 bu t breaks
2337 // skpwww_fashionscandal_com_94 -- calcAngles complains, but I don't unde rstand why
2338 if (missing.fSegment->containsT(missing.fT, missing.fOther, missing.fOth erT)) {
2339 #if DEBUG_DUPLICATES
2340 SkDebugf("skip 1 id=%d t=%1.9g other=%d otherT=%1.9g\n", missing.fSe gment->fID,
2341 missing.fT, missing.fOther->fID, missing.fOtherT);
2342 #endif
2343 continue;
2344 }
2345 if (missing.fOther->containsT(missing.fOtherT, missing.fSegment, missing .fT)) {
2346 #if DEBUG_DUPLICATES
2347 SkDebugf("skip 2 id=%d t=%1.9g other=%d otherT=%1.9g\n", missing.fOt her->fID,
2348 missing.fOtherT, missing.fSegment->fID, missing.fT);
2349 #endif
2350 continue;
2351 }
2352 #endif
2353 // skip if adding would insert point into an existing coincindent span
2354 if (missing.fSegment->inCoincidentSpan(missing.fT, missingOther)
2355 && missingOther->inCoincidentSpan(missing.fOtherT, this)) {
2356 continue;
2357 }
2358 // skip if the created coincident spans are small
2359 if (missing.fSegment->coincidentSmall(missing.fPt, missing.fT, missingOt her)
2360 && missingOther->coincidentSmall(missing.fPt, missing.fOtherT, m issing.fSegment)) {
2361 continue;
2362 }
2363 const SkOpSpan* added = missing.fSegment->addTPair(missing.fT, missingOt her,
2364 missing.fOtherT, false, missing.fPt);
2365 if (added && added->fSmall) {
2366 missing.fSegment->checkSmallCoincidence(*added, &missingCoincidence) ;
2367 }
2368 }
2369 for (index = 0; index < missingCount; ++index) {
2370 MissingSpan& missing = missingSpans[index];
2371 missing.fSegment->fixOtherTIndex();
2372 missing.fOther->fixOtherTIndex();
2373 }
2374 for (index = 0; index < missingCoincidence.count(); ++index) {
2375 MissingSpan& missing = missingCoincidence[index];
2376 missing.fSegment->fixOtherTIndex();
2377 }
2378 debugValidate();
2379 }
2380
2381 // look to see if the curve end intersects an intermediary that intersects the o ther
2382 bool SkOpSegment::checkEnds() {
2383 debugValidate();
2384 SkSTArray<kMissingSpanCount, MissingSpan, true> missingSpans;
2385 int count = fTs.count();
2386 for (int index = 0; index < count; ++index) {
2387 const SkOpSpan& span = fTs[index];
2388 double otherT = span.fOtherT;
2389 if (otherT != 0 && otherT != 1) { // only check ends
2390 continue;
2391 }
2392 const SkOpSegment* other = span.fOther;
2393 // peek start/last describe the range of spans that match the other t of this span
2394 int peekStart = span.fOtherIndex;
2395 while (--peekStart >= 0 && other->fTs[peekStart].fT == otherT)
2396 ;
2397 int otherCount = other->fTs.count();
2398 int peekLast = span.fOtherIndex;
2399 while (++peekLast < otherCount && other->fTs[peekLast].fT == otherT)
2400 ;
2401 if (++peekStart == --peekLast) { // if there isn't a range, there's noth ing to do
2402 continue;
2403 }
2404 // t start/last describe the range of spans that match the t of this spa n
2405 double t = span.fT;
2406 double tBottom = -1;
2407 int tStart = -1;
2408 int tLast = count;
2409 bool lastSmall = false;
2410 double afterT = t;
2411 for (int inner = 0; inner < count; ++inner) {
2412 double innerT = fTs[inner].fT;
2413 if (innerT <= t && innerT > tBottom) {
2414 if (innerT < t || !lastSmall) {
2415 tStart = inner - 1;
2416 }
2417 tBottom = innerT;
2418 }
2419 if (innerT > afterT) {
2420 if (t == afterT && lastSmall) {
2421 afterT = innerT;
2422 } else {
2423 tLast = inner;
2424 break;
2425 }
2426 }
2427 lastSmall = innerT <= t ? fTs[inner].fSmall : false;
2428 }
2429 for (int peekIndex = peekStart; peekIndex <= peekLast; ++peekIndex) {
2430 if (peekIndex == span.fOtherIndex) { // skip the other span pointed to by this span
2431 continue;
2432 }
2433 const SkOpSpan& peekSpan = other->fTs[peekIndex];
2434 SkOpSegment* match = peekSpan.fOther;
2435 if (match->done()) {
2436 continue; // if the edge has already been eaten (likely coincid ence), ignore it
2437 }
2438 const double matchT = peekSpan.fOtherT;
2439 // see if any of the spans match the other spans
2440 for (int tIndex = tStart + 1; tIndex < tLast; ++tIndex) {
2441 const SkOpSpan& tSpan = fTs[tIndex];
2442 if (tSpan.fOther == match) {
2443 if (tSpan.fOtherT == matchT) {
2444 goto nextPeekIndex;
2445 }
2446 double midT = (tSpan.fOtherT + matchT) / 2;
2447 if (match->betweenPoints(midT, tSpan.fPt, peekSpan.fPt)) {
2448 goto nextPeekIndex;
2449 }
2450 }
2451 }
2452 if (missingSpans.count() > 0) {
2453 const MissingSpan& lastMissing = missingSpans.back();
2454 if (lastMissing.fT == t
2455 && lastMissing.fOther == match
2456 && lastMissing.fOtherT == matchT) {
2457 SkASSERT(SkDPoint::ApproximatelyEqual(lastMissing.fPt, peekS pan.fPt));
2458 continue;
2459 }
2460 }
2461 if (this == match) {
2462 return false; // extremely large paths can trigger this
2463 }
2464 #if DEBUG_CHECK_ALIGN
2465 SkDebugf("%s id=%d missing t=%1.9g other=%d otherT=%1.9g pt=(%1.9g,% 1.9g)\n",
2466 __FUNCTION__, fID, t, match->fID, matchT, peekSpan.fPt.fX, p eekSpan.fPt.fY);
2467 #endif
2468 // this segment is missing a entry that the other contains
2469 // remember so we can add the missing one and recompute the indices
2470 {
2471 MissingSpan& missing = missingSpans.push_back();
2472 SkDEBUGCODE(sk_bzero(&missing, sizeof(missing)));
2473 missing.fT = t;
2474 SkASSERT(this != match);
2475 missing.fOther = match;
2476 missing.fOtherT = matchT;
2477 missing.fPt = peekSpan.fPt;
2478 }
2479 break;
2480 nextPeekIndex:
2481 ;
2482 }
2483 }
2484 if (missingSpans.count() == 0) {
2485 debugValidate();
2486 return true;
2487 }
2488 debugValidate();
2489 int missingCount = missingSpans.count();
2490 for (int index = 0; index < missingCount; ++index) {
2491 MissingSpan& missing = missingSpans[index];
2492 if (this != missing.fOther) {
2493 addTPair(missing.fT, missing.fOther, missing.fOtherT, false, missing .fPt);
2494 }
2495 }
2496 fixOtherTIndex();
2497 // OPTIMIZATION: this may fix indices more than once. Build an array of uniq ue segments to
2498 // avoid this
2499 for (int index = 0; index < missingCount; ++index) {
2500 missingSpans[index].fOther->fixOtherTIndex();
2501 }
2502 debugValidate();
2503 return true;
2504 }
2505
2506 void SkOpSegment::checkLinks(const SkOpSpan* base,
2507 SkTArray<MissingSpan, true>* missingSpans) const {
2508 const SkOpSpan* first = fTs.begin();
2509 const SkOpSpan* last = fTs.end() - 1;
2510 SkASSERT(base >= first && last >= base);
2511 const SkOpSegment* other = base->fOther;
2512 const SkOpSpan* oFirst = other->fTs.begin();
2513 const SkOpSpan* oLast = other->fTs.end() - 1;
2514 const SkOpSpan* oSpan = &other->fTs[base->fOtherIndex];
2515 const SkOpSpan* test = base;
2516 const SkOpSpan* missing = NULL;
2517 while (test > first && (--test)->fPt == base->fPt) {
2518 if (this == test->fOther) {
2519 continue;
2520 }
2521 CheckOneLink(test, oSpan, oFirst, oLast, &missing, missingSpans);
2522 }
2523 test = base;
2524 while (test < last && (++test)->fPt == base->fPt) {
2525 SkASSERT(this != test->fOther || test->fLoop);
2526 CheckOneLink(test, oSpan, oFirst, oLast, &missing, missingSpans);
2527 }
2528 }
2529
2530 // see if spans with two or more intersections all agree on common t and point v alues
2531 void SkOpSegment::checkMultiples() {
2532 debugValidate();
2533 int index;
2534 int end = 0;
2535 while (fTs[++end].fT == 0)
2536 ;
2537 while (fTs[end].fT < 1) {
2538 int start = index = end;
2539 end = nextExactSpan(index, 1);
2540 if (end <= index) {
2541 return; // buffer overflow example triggers this
2542 }
2543 if (index + 1 == end) {
2544 continue;
2545 }
2546 // force the duplicates to agree on t and pt if not on the end
2547 SkOpSpan& span = fTs[index];
2548 double thisT = span.fT;
2549 const SkPoint& thisPt = span.fPt;
2550 span.fMultiple = true;
2551 bool aligned = false;
2552 while (++index < end) {
2553 aligned |= alignSpan(index, thisT, thisPt);
2554 }
2555 if (aligned) {
2556 alignSpanState(start, end);
2557 }
2558 fMultiples = true;
2559 }
2560 debugValidate();
2561 }
2562
2563 void SkOpSegment::CheckOneLink(const SkOpSpan* test, const SkOpSpan* oSpan,
2564 const SkOpSpan* oFirst, const SkOpSpan* oLast, const SkOpSpan** missingP tr,
2565 SkTArray<MissingSpan, true>* missingSpans) {
2566 SkASSERT(oSpan->fPt == test->fPt);
2567 const SkOpSpan* oTest = oSpan;
2568 while (oTest > oFirst && (--oTest)->fPt == test->fPt) {
2569 if (oTest->fOther == test->fOther && oTest->fOtherT == test->fOtherT) {
2570 return;
2571 }
2572 }
2573 oTest = oSpan;
2574 while (oTest < oLast && (++oTest)->fPt == test->fPt) {
2575 if (oTest->fOther == test->fOther && oTest->fOtherT == test->fOtherT) {
2576 return;
2577 }
2578 }
2579 if (*missingPtr) {
2580 missingSpans->push_back();
2581 }
2582 MissingSpan& lastMissing = missingSpans->back();
2583 if (*missingPtr) {
2584 lastMissing = missingSpans->end()[-2];
2585 }
2586 *missingPtr = test;
2587 lastMissing.fOther = test->fOther;
2588 lastMissing.fOtherT = test->fOtherT;
2589 }
2590
2591 bool SkOpSegment::checkSmall(int index) const {
2592 if (fTs[index].fSmall) {
2593 return true;
2594 }
2595 double tBase = fTs[index].fT;
2596 while (index > 0 && precisely_negative(tBase - fTs[--index].fT))
2597 ;
2598 return fTs[index].fSmall;
2599 }
2600
2601 // a pair of curves may turn into coincident lines -- small may be a hint that t hat happened
2602 // if a cubic contains a loop, the counts must be adjusted
2603 void SkOpSegment::checkSmall() {
2604 SkSTArray<kMissingSpanCount, MissingSpan, true> missingSpans;
2605 const SkOpSpan* beginSpan = fTs.begin();
2606 const SkOpSpan* thisSpan = beginSpan - 1;
2607 const SkOpSpan* endSpan = fTs.end() - 1; // last can't be small
2608 while (++thisSpan < endSpan) {
2609 if (!thisSpan->fSmall) {
2610 continue;
2611 }
2612 if (!thisSpan->fWindValue) {
2613 continue;
2614 }
2615 const SkOpSpan& firstSpan = this->firstSpan(*thisSpan);
2616 const SkOpSpan& lastSpan = this->lastSpan(*thisSpan);
2617 const SkOpSpan* nextSpan = &firstSpan + 1;
2618 ptrdiff_t smallCount = &lastSpan - &firstSpan + 1;
2619 SkASSERT(1 <= smallCount && smallCount < count());
2620 if (smallCount <= 1 && !nextSpan->fSmall) {
2621 SkASSERT(1 == smallCount);
2622 checkSmallCoincidence(firstSpan, NULL);
2623 continue;
2624 }
2625 // at this point, check for missing computed intersections
2626 const SkPoint& testPt = firstSpan.fPt;
2627 thisSpan = &firstSpan - 1;
2628 SkOpSegment* other = NULL;
2629 while (++thisSpan <= &lastSpan) {
2630 other = thisSpan->fOther;
2631 if (other != this) {
2632 break;
2633 }
2634 }
2635 SkASSERT(other != this);
2636 int oIndex = thisSpan->fOtherIndex;
2637 const SkOpSpan& oSpan = other->span(oIndex);
2638 const SkOpSpan& oFirstSpan = other->firstSpan(oSpan);
2639 const SkOpSpan& oLastSpan = other->lastSpan(oSpan);
2640 ptrdiff_t oCount = &oLastSpan - &oFirstSpan + 1;
2641 if (fLoop) {
2642 int smallCounts[2];
2643 SkASSERT(!other->fLoop); // FIXME: we need more complicated logic f or pair of loops
2644 if (calcLoopSpanCount(*thisSpan, smallCounts)) {
2645 if (smallCounts[0] && oCount != smallCounts[0]) {
2646 SkASSERT(0); // FIXME: need a working test case to properly code & debug
2647 }
2648 if (smallCounts[1] && oCount != smallCounts[1]) {
2649 SkASSERT(0); // FIXME: need a working test case to properly code & debug
2650 }
2651 goto nextSmallCheck;
2652 }
2653 }
2654 if (other->fLoop) {
2655 int otherCounts[2];
2656 if (other->calcLoopSpanCount(other->span(oIndex), otherCounts)) {
2657 if (otherCounts[0] && otherCounts[0] != smallCount) {
2658 SkASSERT(0); // FIXME: need a working test case to properly code & debug
2659 }
2660 if (otherCounts[1] && otherCounts[1] != smallCount) {
2661 SkASSERT(0); // FIXME: need a working test case to properly code & debug
2662 }
2663 goto nextSmallCheck;
2664 }
2665 }
2666 if (oCount != smallCount) { // check if number of pts in this match oth er
2667 MissingSpan& missing = missingSpans.push_back();
2668 missing.fOther = NULL;
2669 SkDEBUGCODE(sk_bzero(&missing, sizeof(missing)));
2670 missing.fPt = testPt;
2671 const SkOpSpan& oSpan = other->span(oIndex);
2672 if (oCount > smallCount) {
2673 missing.fSegment = this;
2674 missing.fT = thisSpan->fT;
2675 other->checkLinks(&oSpan, &missingSpans);
2676 } else {
2677 missing.fSegment = other;
2678 missing.fT = oSpan.fT;
2679 checkLinks(thisSpan, &missingSpans);
2680 }
2681 if (!missingSpans.back().fOther || missing.fSegment->done()) {
2682 missingSpans.pop_back();
2683 }
2684 }
2685 nextSmallCheck:
2686 thisSpan = &lastSpan;
2687 }
2688 int missingCount = missingSpans.count();
2689 for (int index = 0; index < missingCount; ++index) {
2690 MissingSpan& missing = missingSpans[index];
2691 SkOpSegment* missingOther = missing.fOther;
2692 // note that add t pair may edit span arrays, so prior pointers to spans are no longer valid
2693 if (!missing.fSegment->addTPair(missing.fT, missingOther, missing.fOther T, false,
2694 missing.fPt)) {
2695 continue;
2696 }
2697 int otherTIndex = missingOther->findT(missing.fOtherT, missing.fPt, miss ing.fSegment);
2698 const SkOpSpan& otherSpan = missingOther->span(otherTIndex);
2699 if (otherSpan.fSmall) {
2700 const SkOpSpan* nextSpan = &otherSpan;
2701 if (nextSpan->fPt == missing.fPt) {
2702 continue;
2703 }
2704 do {
2705 ++nextSpan;
2706 } while (nextSpan->fSmall);
2707 if (nextSpan->fT == 1) {
2708 continue;
2709 }
2710 SkAssertResult(missing.fSegment->addTCoincident(missing.fPt, nextSpa n->fPt,
2711 nextSpan->fT, missingOther));
2712 } else if (otherSpan.fT > 0) {
2713 const SkOpSpan* priorSpan = &otherSpan;
2714 do {
2715 --priorSpan;
2716 } while (priorSpan->fT == otherSpan.fT);
2717 if (priorSpan->fSmall) {
2718 missing.fSegment->addTCancel(missing.fPt, priorSpan->fPt, missin gOther);
2719 }
2720 }
2721 }
2722 // OPTIMIZATION: this may fix indices more than once. Build an array of uniq ue segments to
2723 // avoid this
2724 for (int index = 0; index < missingCount; ++index) {
2725 MissingSpan& missing = missingSpans[index];
2726 missing.fSegment->fixOtherTIndex();
2727 missing.fOther->fixOtherTIndex();
2728 }
2729 debugValidate();
2730 }
2731
2732 void SkOpSegment::checkSmallCoincidence(const SkOpSpan& span,
2733 SkTArray<MissingSpan, true>* checkMultiple) {
2734 SkASSERT(span.fSmall);
2735 if (0 && !span.fWindValue) {
2736 return;
2737 }
2738 SkASSERT(&span < fTs.end() - 1);
2739 const SkOpSpan* next = &span + 1;
2740 SkASSERT(!next->fSmall || checkMultiple);
2741 if (checkMultiple) {
2742 while (next->fSmall) {
2743 ++next;
2744 SkASSERT(next < fTs.end());
2745 }
2746 }
2747 SkOpSegment* other = span.fOther;
2748 while (other != next->fOther) {
2749 if (!checkMultiple) {
2750 return;
2751 }
2752 const SkOpSpan* test = next + 1;
2753 if (test == fTs.end()) {
2754 return;
2755 }
2756 if (test->fPt != next->fPt || !precisely_equal(test->fT, next->fT)) {
2757 return;
2758 }
2759 next = test;
2760 }
2761 SkASSERT(span.fT < next->fT);
2762 int oStartIndex = other->findExactT(span.fOtherT, this);
2763 int oEndIndex = other->findExactT(next->fOtherT, this);
2764 // FIXME: be overly conservative by limiting this to the caller that allows multiple smalls
2765 if (!checkMultiple || fVerb != SkPath::kLine_Verb || other->fVerb != SkPath: :kLine_Verb) {
2766 SkPoint mid = ptAtT((span.fT + next->fT) / 2);
2767 const SkOpSpan& oSpanStart = other->fTs[oStartIndex];
2768 const SkOpSpan& oSpanEnd = other->fTs[oEndIndex];
2769 SkPoint oMid = other->ptAtT((oSpanStart.fT + oSpanEnd.fT) / 2);
2770 if (!SkDPoint::ApproximatelyEqual(mid, oMid)) {
2771 return;
2772 }
2773 }
2774 // FIXME: again, be overly conservative to avoid breaking existing tests
2775 const SkOpSpan& oSpan = oStartIndex < oEndIndex ? other->fTs[oStartIndex]
2776 : other->fTs[oEndIndex];
2777 if (checkMultiple && !oSpan.fSmall) {
2778 return;
2779 }
2780 // SkASSERT(oSpan.fSmall);
2781 if (oStartIndex < oEndIndex) {
2782 SkAssertResult(addTCoincident(span.fPt, next->fPt, next->fT, other));
2783 } else {
2784 addTCancel(span.fPt, next->fPt, other);
2785 }
2786 if (!checkMultiple) {
2787 return;
2788 }
2789 // check to see if either segment is coincident with a third segment -- if i t is, and if
2790 // the opposite segment is not already coincident with the third, make it so
2791 // OPTIMIZE: to make this check easier, add coincident and cancel could set a coincident bit
2792 if (span.fWindValue != 1 || span.fOppValue != 0) {
2793 // start here;
2794 // iterate through the spans, looking for the third coincident case
2795 // if we find one, we need to return state to the caller so that the ind ices can be fixed
2796 // this also suggests that all of this function is fragile since it reli es on a valid index
2797 }
2798 // probably should make this a common function rather than copy/paste code
2799 if (oSpan.fWindValue != 1 || oSpan.fOppValue != 0) {
2800 const SkOpSpan* oTest = &oSpan;
2801 while (--oTest >= other->fTs.begin()) {
2802 if (oTest->fPt != oSpan.fPt || !precisely_equal(oTest->fT, oSpan.fT) ) {
2803 break;
2804 }
2805 SkOpSegment* testOther = oTest->fOther;
2806 SkASSERT(testOther != this);
2807 // look in both directions to see if there is a coincident span
2808 const SkOpSpan* tTest = testOther->fTs.begin();
2809 for (int testIndex = 0; testIndex < testOther->count(); ++testIndex) {
2810 if (tTest->fPt != span.fPt) {
2811 ++tTest;
2812 continue;
2813 }
2814 if (testOther->verb() != SkPath::kLine_Verb
2815 || other->verb() != SkPath::kLine_Verb) {
2816 SkPoint mid = ptAtT((span.fT + next->fT) / 2);
2817 SkPoint oMid = other->ptAtT((oTest->fOtherT + tTest->fT) / 2 );
2818 if (!SkDPoint::ApproximatelyEqual(mid, oMid)) {
2819 continue;
2820 }
2821 }
2822 #if DEBUG_CONCIDENT
2823 SkDebugf("%s coincident found=%d %1.9g %1.9g\n", __FUNCTION__, t estOther->fID,
2824 oTest->fOtherT, tTest->fT);
2825 #endif
2826 if (tTest->fT < oTest->fOtherT) {
2827 SkAssertResult(addTCoincident(span.fPt, next->fPt, next->fT, testOther));
2828 } else {
2829 addTCancel(span.fPt, next->fPt, testOther);
2830 }
2831 MissingSpan missing;
2832 missing.fSegment = testOther;
2833 checkMultiple->push_back(missing);
2834 break;
2835 }
2836 }
2837 oTest = &oSpan;
2838 while (++oTest < other->fTs.end()) {
2839 if (oTest->fPt != oSpan.fPt || !precisely_equal(oTest->fT, oSpan.fT) ) {
2840 break;
2841 }
2842
2843 }
2844 }
2845 }
2846
2847 // if pair of spans on either side of tiny have the same end point and mid point , mark
2848 // them as parallel
2849 void SkOpSegment::checkTiny() {
2850 SkSTArray<kMissingSpanCount, MissingSpan, true> missingSpans;
2851 SkOpSpan* thisSpan = fTs.begin() - 1;
2852 const SkOpSpan* endSpan = fTs.end() - 1; // last can't be tiny
2853 while (++thisSpan < endSpan) {
2854 if (!thisSpan->fTiny) {
2855 continue;
2856 }
2857 SkOpSpan* nextSpan = thisSpan + 1;
2858 double thisT = thisSpan->fT;
2859 double nextT = nextSpan->fT;
2860 if (thisT == nextT) {
2861 continue;
2862 }
2863 SkASSERT(thisT < nextT);
2864 SkASSERT(thisSpan->fPt == nextSpan->fPt);
2865 SkOpSegment* thisOther = thisSpan->fOther;
2866 SkOpSegment* nextOther = nextSpan->fOther;
2867 int oIndex = thisSpan->fOtherIndex;
2868 for (int oStep = -1; oStep <= 1; oStep += 2) {
2869 int oEnd = thisOther->nextExactSpan(oIndex, oStep);
2870 if (oEnd < 0) {
2871 continue;
2872 }
2873 const SkOpSpan& oSpan = thisOther->span(oEnd);
2874 int nIndex = nextSpan->fOtherIndex;
2875 for (int nStep = -1; nStep <= 1; nStep += 2) {
2876 int nEnd = nextOther->nextExactSpan(nIndex, nStep);
2877 if (nEnd < 0) {
2878 continue;
2879 }
2880 const SkOpSpan& nSpan = nextOther->span(nEnd);
2881 if (oSpan.fPt != nSpan.fPt) {
2882 continue;
2883 }
2884 double oMidT = (thisSpan->fOtherT + oSpan.fT) / 2;
2885 const SkPoint& oPt = thisOther->ptAtT(oMidT);
2886 double nMidT = (nextSpan->fOtherT + nSpan.fT) / 2;
2887 const SkPoint& nPt = nextOther->ptAtT(nMidT);
2888 if (!AlmostEqualUlps(oPt, nPt)) {
2889 continue;
2890 }
2891 #if DEBUG_CHECK_TINY
2892 SkDebugf("%s [%d] add coincidence [%d] [%d]\n", __FUNCTION__, fI D,
2893 thisOther->fID, nextOther->fID);
2894 #endif
2895 // this segment is missing a entry that the other contains
2896 // remember so we can add the missing one and recompute the indi ces
2897 MissingSpan& missing = missingSpans.push_back();
2898 SkDEBUGCODE(sk_bzero(&missing, sizeof(missing)));
2899 missing.fSegment = thisOther;
2900 missing.fT = thisSpan->fOtherT;
2901 SkASSERT(this != nextOther);
2902 missing.fOther = nextOther;
2903 missing.fOtherT = nextSpan->fOtherT;
2904 missing.fPt = thisSpan->fPt;
2905 }
2906 }
2907 }
2908 int missingCount = missingSpans.count();
2909 if (!missingCount) {
2910 return;
2911 }
2912 for (int index = 0; index < missingCount; ++index) {
2913 MissingSpan& missing = missingSpans[index];
2914 if (missing.fSegment != missing.fOther) {
2915 missing.fSegment->addTPair(missing.fT, missing.fOther, missing.fOthe rT, false,
2916 missing.fPt);
2917 }
2918 }
2919 // OPTIMIZE: consolidate to avoid multiple calls to fix index
2920 for (int index = 0; index < missingCount; ++index) {
2921 MissingSpan& missing = missingSpans[index];
2922 missing.fSegment->fixOtherTIndex();
2923 missing.fOther->fixOtherTIndex();
2924 }
2925 }
2926
2927 bool SkOpSegment::coincidentSmall(const SkPoint& pt, double t, const SkOpSegment * other) const {
2928 int count = this->count();
2929 for (int index = 0; index < count; ++index) {
2930 const SkOpSpan& span = this->span(index);
2931 if (span.fOther != other) {
2932 continue;
2933 }
2934 if (span.fPt == pt) {
2935 continue;
2936 }
2937 if (!AlmostEqualUlps(span.fPt, pt)) {
2938 continue;
2939 }
2940 if (fVerb != SkPath::kCubic_Verb) {
2941 return true;
2942 }
2943 double tInterval = t - span.fT;
2944 double tMid = t - tInterval / 2;
2945 SkDPoint midPt = dcubic_xy_at_t(fPts, tMid);
2946 return midPt.approximatelyEqual(xyAtT(t));
2947 }
2948 return false;
2949 }
2950
2951 bool SkOpSegment::findCoincidentMatch(const SkOpSpan* span, const SkOpSegment* o ther, int oStart,
2952 int oEnd, int step, SkPoint* startPt, SkPoint* endPt, double* endT) cons t {
2953 SkASSERT(span->fT == 0 || span->fT == 1);
2954 SkASSERT(span->fOtherT == 0 || span->fOtherT == 1);
2955 const SkOpSpan* otherSpan = &other->span(oEnd);
2956 double refT = otherSpan->fT;
2957 const SkPoint& refPt = otherSpan->fPt;
2958 const SkOpSpan* lastSpan = &other->span(step > 0 ? other->count() - 1 : 0);
2959 do {
2960 const SkOpSegment* match = span->fOther;
2961 if (match == otherSpan->fOther) {
2962 // find start of respective spans and see if both have winding
2963 int startIndex, endIndex;
2964 if (span->fOtherT == 1) {
2965 endIndex = span->fOtherIndex;
2966 startIndex = match->nextExactSpan(endIndex, -1);
2967 } else {
2968 startIndex = span->fOtherIndex;
2969 endIndex = match->nextExactSpan(startIndex, 1);
2970 }
2971 const SkOpSpan& startSpan = match->span(startIndex);
2972 if (startSpan.fWindValue != 0) {
2973 // draw ray from endSpan.fPt perpendicular to end tangent and me asure distance
2974 // to other segment.
2975 const SkOpSpan& endSpan = match->span(endIndex);
2976 SkDLine ray;
2977 SkVector dxdy;
2978 if (span->fOtherT == 1) {
2979 ray.fPts[0].set(startSpan.fPt);
2980 dxdy = match->dxdy(startIndex);
2981 } else {
2982 ray.fPts[0].set(endSpan.fPt);
2983 dxdy = match->dxdy(endIndex);
2984 }
2985 ray.fPts[1].fX = ray.fPts[0].fX + dxdy.fY;
2986 ray.fPts[1].fY = ray.fPts[0].fY - dxdy.fX;
2987 SkIntersections i;
2988 int roots = (i.*CurveRay[SkPathOpsVerbToPoints(other->verb())])( other->pts(), ray);
2989 for (int index = 0; index < roots; ++index) {
2990 if (ray.fPts[0].approximatelyEqual(i.pt(index))) {
2991 double matchMidT = (match->span(startIndex).fT
2992 + match->span(endIndex).fT) / 2;
2993 SkPoint matchMidPt = match->ptAtT(matchMidT);
2994 double otherMidT = (i[0][index] + other->span(oStart).fT ) / 2;
2995 SkPoint otherMidPt = other->ptAtT(otherMidT);
2996 if (SkDPoint::ApproximatelyEqual(matchMidPt, otherMidPt) ) {
2997 *startPt = startSpan.fPt;
2998 *endPt = endSpan.fPt;
2999 *endT = endSpan.fT;
3000 return true;
3001 }
3002 }
3003 }
3004 }
3005 return false;
3006 }
3007 if (otherSpan == lastSpan) {
3008 break;
3009 }
3010 otherSpan += step;
3011 } while (otherSpan->fT == refT || otherSpan->fPt == refPt);
3012 return false;
3013 }
3014
3015 int SkOpSegment::findEndSpan(int endIndex) const {
3016 const SkOpSpan* span = &fTs[--endIndex];
3017 const SkPoint& lastPt = span->fPt;
3018 double endT = span->fT;
3019 do {
3020 span = &fTs[--endIndex];
3021 } while (SkDPoint::ApproximatelyEqual(span->fPt, lastPt) && (span->fT == end T || span->fTiny));
3022 return endIndex + 1;
3023 } 783 }
3024 784
3025 /* 785 /*
3026 The M and S variable name parts stand for the operators. 786 The M and S variable name parts stand for the operators.
3027 Mi stands for Minuend (see wiki subtraction, analogous to difference) 787 Mi stands for Minuend (see wiki subtraction, analogous to difference)
3028 Su stands for Subtrahend 788 Su stands for Subtrahend
3029 The Opp variable name part designates that the value is for the Opposite operat or. 789 The Opp variable name part designates that the value is for the Opposite operat or.
3030 Opposite values result from combining coincident spans. 790 Opposite values result from combining coincident spans.
3031 */ 791 */
3032 SkOpSegment* SkOpSegment::findNextOp(SkTDArray<SkOpSpan*>* chase, int* nextStart , int* nextEnd, 792 SkOpSegment* SkOpSegment::findNextOp(SkTDArray<SkOpSpanBase*>* chase, SkOpSpanBa se** nextStart,
3033 bool* unsortable, SkPathOp op, const int xo rMiMask, 793 SkOpSpanBase** nextEnd, bool* unsortable, SkPathOp op, int xorMiMask, in t xorSuMask) {
3034 const int xorSuMask) { 794 SkOpSpanBase* start = *nextStart;
3035 const int startIndex = *nextStart; 795 SkOpSpanBase* end = *nextEnd;
3036 const int endIndex = *nextEnd; 796 SkASSERT(start != end);
3037 SkASSERT(startIndex != endIndex); 797 int step = start->step(end);
3038 SkDEBUGCODE(const int count = fTs.count()); 798 SkOpSegment* other = this->isSimple(nextStart, &step); // advances nextStar t
3039 SkASSERT(startIndex < endIndex ? startIndex < count - 1 : startIndex > 0); 799 if (other) {
3040 int step = SkSign32(endIndex - startIndex);
3041 *nextStart = startIndex;
3042 SkOpSegment* other = isSimple(nextStart, &step);
3043 if (other)
3044 {
3045 // mark the smaller of startIndex, endIndex done, and all adjacent 800 // mark the smaller of startIndex, endIndex done, and all adjacent
3046 // spans with the same T value (but not 'other' spans) 801 // spans with the same T value (but not 'other' spans)
3047 #if DEBUG_WINDING 802 #if DEBUG_WINDING
3048 SkDebugf("%s simple\n", __FUNCTION__); 803 SkDebugf("%s simple\n", __FUNCTION__);
3049 #endif 804 #endif
3050 int min = SkMin32(startIndex, endIndex); 805 SkOpSpan* startSpan = start->starter(end);
3051 if (fTs[min].fDone) { 806 if (startSpan->done()) {
3052 return NULL; 807 return NULL;
3053 } 808 }
3054 markDoneBinary(min); 809 markDone(startSpan);
3055 double startT = other->fTs[*nextStart].fT; 810 *nextEnd = step > 0 ? (*nextStart)->upCast()->next() : (*nextStart)->pre v();
3056 *nextEnd = *nextStart;
3057 do {
3058 *nextEnd += step;
3059 } while (precisely_zero(startT - other->fTs[*nextEnd].fT));
3060 SkASSERT(step < 0 ? *nextEnd >= 0 : *nextEnd < other->fTs.count());
3061 if (other->isTiny(SkMin32(*nextStart, *nextEnd))) {
3062 *unsortable = true;
3063 return NULL;
3064 }
3065 return other; 811 return other;
3066 } 812 }
3067 const int end = nextExactSpan(startIndex, step); 813 SkOpSpanBase* endNear = step > 0 ? (*nextStart)->upCast()->next() : (*nextSt art)->prev();
3068 SkASSERT(end >= 0); 814 SkASSERT(endNear == end); // is this ever not end?
3069 SkASSERT(startIndex - endIndex != 0); 815 SkASSERT(endNear);
3070 SkASSERT((startIndex - endIndex < 0) ^ (step < 0)); 816 SkASSERT(start != endNear);
817 SkASSERT((start->t() < endNear->t()) ^ (step < 0));
3071 // more than one viable candidate -- measure angles to find best 818 // more than one viable candidate -- measure angles to find best
3072 819 int calcWinding = computeSum(start, endNear, SkOpAngle::kBinaryOpp);
3073 int calcWinding = computeSum(startIndex, end, SkOpAngle::kBinaryOpp);
3074 bool sortable = calcWinding != SK_NaN32; 820 bool sortable = calcWinding != SK_NaN32;
3075 if (!sortable) { 821 if (!sortable) {
3076 *unsortable = true; 822 *unsortable = true;
3077 markDoneBinary(SkMin32(startIndex, endIndex)); 823 markDone(start->starter(end));
3078 return NULL; 824 return NULL;
3079 } 825 }
3080 SkOpAngle* angle = spanToAngle(end, startIndex); 826 SkOpAngle* angle = this->spanToAngle(end, start);
3081 if (angle->unorderable()) { 827 if (angle->unorderable()) {
3082 *unsortable = true; 828 *unsortable = true;
3083 markDoneBinary(SkMin32(startIndex, endIndex)); 829 markDone(start->starter(end));
3084 return NULL; 830 return NULL;
3085 } 831 }
3086 #if DEBUG_SORT 832 #if DEBUG_SORT
3087 SkDebugf("%s\n", __FUNCTION__); 833 SkDebugf("%s\n", __FUNCTION__);
3088 angle->debugLoop(); 834 angle->debugLoop();
3089 #endif 835 #endif
3090 int sumMiWinding = updateWinding(endIndex, startIndex); 836 int sumMiWinding = updateWinding(end, start);
3091 if (sumMiWinding == SK_MinS32) { 837 if (sumMiWinding == SK_MinS32) {
3092 *unsortable = true; 838 *unsortable = true;
3093 markDoneBinary(SkMin32(startIndex, endIndex)); 839 markDone(start->starter(end));
3094 return NULL; 840 return NULL;
3095 } 841 }
3096 int sumSuWinding = updateOppWinding(endIndex, startIndex); 842 int sumSuWinding = updateOppWinding(end, start);
3097 if (operand()) { 843 if (operand()) {
3098 SkTSwap<int>(sumMiWinding, sumSuWinding); 844 SkTSwap<int>(sumMiWinding, sumSuWinding);
3099 } 845 }
3100 SkOpAngle* nextAngle = angle->next(); 846 SkOpAngle* nextAngle = angle->next();
3101 const SkOpAngle* foundAngle = NULL; 847 const SkOpAngle* foundAngle = NULL;
3102 bool foundDone = false; 848 bool foundDone = false;
3103 // iterate through the angle, and compute everyone's winding 849 // iterate through the angle, and compute everyone's winding
3104 SkOpSegment* nextSegment; 850 SkOpSegment* nextSegment;
3105 int activeCount = 0; 851 int activeCount = 0;
3106 do { 852 do {
3107 nextSegment = nextAngle->segment(); 853 nextSegment = nextAngle->segment();
3108 bool activeAngle = nextSegment->activeOp(xorMiMask, xorSuMask, nextAngle ->start(), 854 bool activeAngle = nextSegment->activeOp(xorMiMask, xorSuMask, nextAngle ->start(),
3109 nextAngle->end(), op, &sumMiWinding, &sumSuWinding); 855 nextAngle->end(), op, &sumMiWinding, &sumSuWinding);
3110 if (activeAngle) { 856 if (activeAngle) {
3111 ++activeCount; 857 ++activeCount;
3112 if (!foundAngle || (foundDone && activeCount & 1)) { 858 if (!foundAngle || (foundDone && activeCount & 1)) {
3113 if (nextSegment->isTiny(nextAngle)) {
3114 *unsortable = true;
3115 markDoneBinary(SkMin32(startIndex, endIndex));
3116 return NULL;
3117 }
3118 foundAngle = nextAngle; 859 foundAngle = nextAngle;
3119 foundDone = nextSegment->done(nextAngle); 860 foundDone = nextSegment->done(nextAngle);
3120 } 861 }
3121 } 862 }
3122 if (nextSegment->done()) { 863 if (nextSegment->done()) {
3123 continue; 864 continue;
3124 } 865 }
3125 if (nextSegment->isTiny(nextAngle)) { 866 if (!activeAngle) {
3126 continue; 867 (void) nextSegment->markAndChaseDone(nextAngle->start(), nextAngle-> end());
3127 } 868 }
3128 if (!activeAngle) { 869 SkOpSpanBase* last = nextAngle->lastMarked();
3129 (void) nextSegment->markAndChaseDoneBinary(nextAngle->start(), nextA ngle->end());
3130 }
3131 SkOpSpan* last = nextAngle->lastMarked();
3132 if (last) { 870 if (last) {
3133 SkASSERT(!SkPathOpsDebug::ChaseContains(*chase, last)); 871 SkASSERT(!SkPathOpsDebug::ChaseContains(*chase, last));
3134 *chase->append() = last; 872 *chase->append() = last;
3135 #if DEBUG_WINDING 873 #if DEBUG_WINDING
3136 SkDebugf("%s chase.append id=%d windSum=%d small=%d\n", __FUNCTION__ , 874 SkDebugf("%s chase.append segment=%d span=%d", __FUNCTION__,
3137 last->fOther->fTs[last->fOtherIndex].fOther->debugID(), last ->fWindSum, 875 last->segment()->debugID(), last->debugID());
3138 last->fSmall); 876 if (!last->final()) {
877 SkDebugf(" windSum=%d", last->upCast()->windSum());
878 }
879 SkDebugf("\n");
3139 #endif 880 #endif
3140 } 881 }
3141 } while ((nextAngle = nextAngle->next()) != angle); 882 } while ((nextAngle = nextAngle->next()) != angle);
3142 #if DEBUG_ANGLE 883 start->segment()->markDone(start->starter(end));
3143 if (foundAngle) {
3144 foundAngle->debugSameAs(foundAngle);
3145 }
3146 #endif
3147
3148 markDoneBinary(SkMin32(startIndex, endIndex));
3149 if (!foundAngle) { 884 if (!foundAngle) {
3150 return NULL; 885 return NULL;
3151 } 886 }
3152 *nextStart = foundAngle->start(); 887 *nextStart = foundAngle->start();
3153 *nextEnd = foundAngle->end(); 888 *nextEnd = foundAngle->end();
3154 nextSegment = foundAngle->segment(); 889 nextSegment = foundAngle->segment();
3155 #if DEBUG_WINDING 890 #if DEBUG_WINDING
3156 SkDebugf("%s from:[%d] to:[%d] start=%d end=%d\n", 891 SkDebugf("%s from:[%d] to:[%d] start=%d end=%d\n",
3157 __FUNCTION__, debugID(), nextSegment->debugID(), *nextStart, *nextEn d); 892 __FUNCTION__, debugID(), nextSegment->debugID(), *nextStart, *nextEn d);
3158 #endif 893 #endif
3159 return nextSegment; 894 return nextSegment;
3160 } 895 }
3161 896
3162 SkOpSegment* SkOpSegment::findNextWinding(SkTDArray<SkOpSpan*>* chase, int* next Start, 897 SkOpSegment* SkOpSegment::findNextWinding(SkTDArray<SkOpSpanBase*>* chase,
3163 int* nextEnd, bool* unsortable) { 898 SkOpSpanBase** nextStart, SkOpSpanBase** nextEnd, bool* unsortable) {
3164 const int startIndex = *nextStart; 899 SkOpSpanBase* start = *nextStart;
3165 const int endIndex = *nextEnd; 900 SkOpSpanBase* end = *nextEnd;
3166 SkASSERT(startIndex != endIndex); 901 SkASSERT(start != end);
3167 SkDEBUGCODE(const int count = fTs.count()); 902 int step = start->step(end);
3168 SkASSERT(startIndex < endIndex ? startIndex < count - 1 : startIndex > 0); 903 SkOpSegment* other = this->isSimple(nextStart, &step); // advances nextStar t
3169 int step = SkSign32(endIndex - startIndex); 904 if (other) {
3170 *nextStart = startIndex;
3171 SkOpSegment* other = isSimple(nextStart, &step);
3172 if (other)
3173 {
3174 // mark the smaller of startIndex, endIndex done, and all adjacent 905 // mark the smaller of startIndex, endIndex done, and all adjacent
3175 // spans with the same T value (but not 'other' spans) 906 // spans with the same T value (but not 'other' spans)
3176 #if DEBUG_WINDING 907 #if DEBUG_WINDING
3177 SkDebugf("%s simple\n", __FUNCTION__); 908 SkDebugf("%s simple\n", __FUNCTION__);
3178 #endif 909 #endif
3179 int min = SkMin32(startIndex, endIndex); 910 SkOpSpan* startSpan = start->starter(end);
3180 if (fTs[min].fDone) { 911 if (startSpan->done()) {
3181 return NULL; 912 return NULL;
3182 } 913 }
3183 markDoneUnary(min); 914 markDone(startSpan);
3184 double startT = other->fTs[*nextStart].fT; 915 *nextEnd = step > 0 ? (*nextStart)->upCast()->next() : (*nextStart)->pre v();
3185 *nextEnd = *nextStart;
3186 do {
3187 *nextEnd += step;
3188 } while (precisely_zero(startT - other->fTs[*nextEnd].fT));
3189 SkASSERT(step < 0 ? *nextEnd >= 0 : *nextEnd < other->fTs.count());
3190 if (other->isTiny(SkMin32(*nextStart, *nextEnd))) {
3191 *unsortable = true;
3192 return NULL;
3193 }
3194 return other; 916 return other;
3195 } 917 }
3196 const int end = nextExactSpan(startIndex, step); 918 SkOpSpanBase* endNear = step > 0 ? (*nextStart)->upCast()->next() : (*nextSt art)->prev();
3197 SkASSERT(end >= 0); 919 SkASSERT(endNear == end); // is this ever not end?
3198 SkASSERT(startIndex - endIndex != 0); 920 SkASSERT(endNear);
3199 SkASSERT((startIndex - endIndex < 0) ^ (step < 0)); 921 SkASSERT(start != endNear);
922 SkASSERT((start->t() < endNear->t()) ^ (step < 0));
3200 // more than one viable candidate -- measure angles to find best 923 // more than one viable candidate -- measure angles to find best
3201 924 int calcWinding = computeSum(start, endNear, SkOpAngle::kUnaryWinding);
3202 int calcWinding = computeSum(startIndex, end, SkOpAngle::kUnaryWinding);
3203 bool sortable = calcWinding != SK_NaN32; 925 bool sortable = calcWinding != SK_NaN32;
3204 if (!sortable) { 926 if (!sortable) {
3205 *unsortable = true; 927 *unsortable = true;
3206 markDoneUnary(SkMin32(startIndex, endIndex)); 928 markDone(start->starter(end));
3207 return NULL; 929 return NULL;
3208 } 930 }
3209 SkOpAngle* angle = spanToAngle(end, startIndex); 931 SkOpAngle* angle = this->spanToAngle(end, start);
932 if (angle->unorderable()) {
933 *unsortable = true;
934 markDone(start->starter(end));
935 return NULL;
936 }
3210 #if DEBUG_SORT 937 #if DEBUG_SORT
3211 SkDebugf("%s\n", __FUNCTION__); 938 SkDebugf("%s\n", __FUNCTION__);
3212 angle->debugLoop(); 939 angle->debugLoop();
3213 #endif 940 #endif
3214 int sumWinding = updateWinding(endIndex, startIndex); 941 int sumWinding = updateWinding(end, start);
3215 SkOpAngle* nextAngle = angle->next(); 942 SkOpAngle* nextAngle = angle->next();
3216 const SkOpAngle* foundAngle = NULL; 943 const SkOpAngle* foundAngle = NULL;
3217 bool foundDone = false; 944 bool foundDone = false;
945 // iterate through the angle, and compute everyone's winding
3218 SkOpSegment* nextSegment; 946 SkOpSegment* nextSegment;
3219 int activeCount = 0; 947 int activeCount = 0;
3220 do { 948 do {
3221 nextSegment = nextAngle->segment(); 949 nextSegment = nextAngle->segment();
3222 bool activeAngle = nextSegment->activeWinding(nextAngle->start(), nextAn gle->end(), 950 bool activeAngle = nextSegment->activeWinding(nextAngle->start(), nextAn gle->end(),
3223 &sumWinding); 951 &sumWinding);
3224 if (activeAngle) { 952 if (activeAngle) {
3225 ++activeCount; 953 ++activeCount;
3226 if (!foundAngle || (foundDone && activeCount & 1)) { 954 if (!foundAngle || (foundDone && activeCount & 1)) {
3227 if (nextSegment->isTiny(nextAngle)) {
3228 *unsortable = true;
3229 markDoneUnary(SkMin32(startIndex, endIndex));
3230 return NULL;
3231 }
3232 foundAngle = nextAngle; 955 foundAngle = nextAngle;
3233 foundDone = nextSegment->done(nextAngle); 956 foundDone = nextSegment->done(nextAngle);
3234 } 957 }
3235 } 958 }
3236 if (nextSegment->done()) { 959 if (nextSegment->done()) {
3237 continue; 960 continue;
3238 } 961 }
3239 if (nextSegment->isTiny(nextAngle)) { 962 if (!activeAngle) {
3240 continue; 963 (void) nextSegment->markAndChaseDone(nextAngle->start(), nextAngle-> end());
3241 } 964 }
3242 if (!activeAngle) { 965 SkOpSpanBase* last = nextAngle->lastMarked();
3243 nextSegment->markAndChaseDoneUnary(nextAngle->start(), nextAngle->en d());
3244 }
3245 SkOpSpan* last = nextAngle->lastMarked();
3246 if (last) { 966 if (last) {
3247 SkASSERT(!SkPathOpsDebug::ChaseContains(*chase, last)); 967 SkASSERT(!SkPathOpsDebug::ChaseContains(*chase, last));
3248 *chase->append() = last; 968 *chase->append() = last;
3249 #if DEBUG_WINDING 969 #if DEBUG_WINDING
3250 SkDebugf("%s chase.append id=%d windSum=%d small=%d\n", __FUNCTION__ , 970 SkDebugf("%s chase.append segment=%d span=%d", __FUNCTION__,
3251 last->fOther->fTs[last->fOtherIndex].fOther->debugID(), last ->fWindSum, 971 last->segment()->debugID(), last->debugID());
3252 last->fSmall); 972 if (!last->final()) {
973 SkDebugf(" windSum=%d", last->upCast()->windSum());
974 }
975 SkDebugf("\n");
3253 #endif 976 #endif
3254 } 977 }
3255 } while ((nextAngle = nextAngle->next()) != angle); 978 } while ((nextAngle = nextAngle->next()) != angle);
3256 markDoneUnary(SkMin32(startIndex, endIndex)); 979 start->segment()->markDone(start->starter(end));
3257 if (!foundAngle) { 980 if (!foundAngle) {
3258 return NULL; 981 return NULL;
3259 } 982 }
3260 *nextStart = foundAngle->start(); 983 *nextStart = foundAngle->start();
3261 *nextEnd = foundAngle->end(); 984 *nextEnd = foundAngle->end();
3262 nextSegment = foundAngle->segment(); 985 nextSegment = foundAngle->segment();
3263 #if DEBUG_WINDING 986 #if DEBUG_WINDING
3264 SkDebugf("%s from:[%d] to:[%d] start=%d end=%d\n", 987 SkDebugf("%s from:[%d] to:[%d] start=%d end=%d\n",
3265 __FUNCTION__, debugID(), nextSegment->debugID(), *nextStart, *nextEn d); 988 __FUNCTION__, debugID(), nextSegment->debugID(), *nextStart, *nextEn d);
3266 #endif 989 #endif
3267 return nextSegment; 990 return nextSegment;
3268 } 991 }
3269 992
3270 SkOpSegment* SkOpSegment::findNextXor(int* nextStart, int* nextEnd, bool* unsort able) { 993 SkOpSegment* SkOpSegment::findNextXor(SkOpSpanBase** nextStart, SkOpSpanBase** n extEnd,
3271 const int startIndex = *nextStart; 994 bool* unsortable) {
3272 const int endIndex = *nextEnd; 995 SkOpSpanBase* start = *nextStart;
3273 SkASSERT(startIndex != endIndex); 996 SkOpSpanBase* end = *nextEnd;
3274 SkDEBUGCODE(int count = fTs.count()); 997 SkASSERT(start != end);
3275 SkASSERT(startIndex < endIndex ? startIndex < count - 1 : startIndex > 0); 998 int step = start->step(end);
3276 int step = SkSign32(endIndex - startIndex); 999 SkOpSegment* other = this->isSimple(nextStart, &step); // advances nextStar t
3277 // Detect cases where all the ends canceled out (e.g., 1000 if (other) {
3278 // there is no angle) and therefore there's only one valid connection 1001 // mark the smaller of startIndex, endIndex done, and all adjacent
3279 *nextStart = startIndex; 1002 // spans with the same T value (but not 'other' spans)
3280 SkOpSegment* other = isSimple(nextStart, &step);
3281 if (other)
3282 {
3283 #if DEBUG_WINDING 1003 #if DEBUG_WINDING
3284 SkDebugf("%s simple\n", __FUNCTION__); 1004 SkDebugf("%s simple\n", __FUNCTION__);
3285 #endif 1005 #endif
3286 int min = SkMin32(startIndex, endIndex); 1006 SkOpSpan* startSpan = start->starter(end);
3287 if (fTs[min].fDone) { 1007 if (startSpan->done()) {
3288 return NULL; 1008 return NULL;
3289 } 1009 }
3290 markDone(min, 1); 1010 markDone(startSpan);
3291 double startT = other->fTs[*nextStart].fT; 1011 *nextEnd = step > 0 ? (*nextStart)->upCast()->next() : (*nextStart)->pre v();
3292 // FIXME: I don't know why the logic here is difference from the winding case
3293 SkDEBUGCODE(bool firstLoop = true;)
3294 if ((approximately_less_than_zero(startT) && step < 0)
3295 || (approximately_greater_than_one(startT) && step > 0)) {
3296 step = -step;
3297 SkDEBUGCODE(firstLoop = false;)
3298 }
3299 do {
3300 *nextEnd = *nextStart;
3301 do {
3302 *nextEnd += step;
3303 } while (precisely_zero(startT - other->fTs[*nextEnd].fT));
3304 if (other->fTs[SkMin32(*nextStart, *nextEnd)].fWindValue) {
3305 break;
3306 }
3307 SkASSERT(firstLoop);
3308 SkDEBUGCODE(firstLoop = false;)
3309 step = -step;
3310 } while (true);
3311 SkASSERT(step < 0 ? *nextEnd >= 0 : *nextEnd < other->fTs.count());
3312 return other; 1012 return other;
3313 } 1013 }
3314 SkASSERT(startIndex - endIndex != 0); 1014 SkDEBUGCODE(SkOpSpanBase* endNear = step > 0 ? (*nextStart)->upCast()->next( ) \
3315 SkASSERT((startIndex - endIndex < 0) ^ (step < 0)); 1015 : (*nextStart)->prev());
3316 // parallel block above with presorted version 1016 SkASSERT(endNear == end); // is this ever not end?
3317 int end = nextExactSpan(startIndex, step); 1017 SkASSERT(endNear);
3318 SkASSERT(end >= 0); 1018 SkASSERT(start != endNear);
3319 SkOpAngle* angle = spanToAngle(end, startIndex); 1019 SkASSERT((start->t() < endNear->t()) ^ (step < 0));
3320 SkASSERT(angle); 1020 SkOpAngle* angle = this->spanToAngle(end, start);
1021 if (angle->unorderable()) {
1022 *unsortable = true;
1023 markDone(start->starter(end));
1024 return NULL;
1025 }
3321 #if DEBUG_SORT 1026 #if DEBUG_SORT
3322 SkDebugf("%s\n", __FUNCTION__); 1027 SkDebugf("%s\n", __FUNCTION__);
3323 angle->debugLoop(); 1028 angle->debugLoop();
3324 #endif 1029 #endif
3325 SkOpAngle* nextAngle = angle->next(); 1030 SkOpAngle* nextAngle = angle->next();
3326 const SkOpAngle* foundAngle = NULL; 1031 const SkOpAngle* foundAngle = NULL;
3327 bool foundDone = false; 1032 bool foundDone = false;
1033 // iterate through the angle, and compute everyone's winding
3328 SkOpSegment* nextSegment; 1034 SkOpSegment* nextSegment;
3329 int activeCount = 0; 1035 int activeCount = 0;
3330 do { 1036 do {
3331 nextSegment = nextAngle->segment(); 1037 nextSegment = nextAngle->segment();
3332 ++activeCount; 1038 ++activeCount;
3333 if (!foundAngle || (foundDone && activeCount & 1)) { 1039 if (!foundAngle || (foundDone && activeCount & 1)) {
3334 if (nextSegment->isTiny(nextAngle)) {
3335 *unsortable = true;
3336 return NULL;
3337 }
3338 foundAngle = nextAngle; 1040 foundAngle = nextAngle;
3339 if (!(foundDone = nextSegment->done(nextAngle))) { 1041 if (!(foundDone = nextSegment->done(nextAngle))) {
3340 break; 1042 break;
3341 } 1043 }
3342 } 1044 }
3343 nextAngle = nextAngle->next(); 1045 nextAngle = nextAngle->next();
3344 } while (nextAngle != angle); 1046 } while (nextAngle != angle);
3345 markDone(SkMin32(startIndex, endIndex), 1); 1047 start->segment()->markDone(start->starter(end));
3346 if (!foundAngle) { 1048 if (!foundAngle) {
3347 return NULL; 1049 return NULL;
3348 } 1050 }
3349 *nextStart = foundAngle->start(); 1051 *nextStart = foundAngle->start();
3350 *nextEnd = foundAngle->end(); 1052 *nextEnd = foundAngle->end();
3351 nextSegment = foundAngle->segment(); 1053 nextSegment = foundAngle->segment();
3352 #if DEBUG_WINDING 1054 #if DEBUG_WINDING
3353 SkDebugf("%s from:[%d] to:[%d] start=%d end=%d\n", 1055 SkDebugf("%s from:[%d] to:[%d] start=%d end=%d\n",
3354 __FUNCTION__, debugID(), nextSegment->debugID(), *nextStart, *nextEn d); 1056 __FUNCTION__, debugID(), nextSegment->debugID(), *nextStart, *nextEn d);
3355 #endif 1057 #endif
3356 return nextSegment; 1058 return nextSegment;
3357 } 1059 }
3358 1060
3359 int SkOpSegment::findStartSpan(int startIndex) const { 1061 SkOpSegment* SkOpSegment::findTop(bool firstPass, SkOpSpanBase** startPtr, SkOpS panBase** endPtr,
3360 int index = startIndex; 1062 bool* unsortable, SkChunkAlloc* allocator) {
3361 const SkOpSpan* span = &fTs[index];
3362 const SkPoint& firstPt = span->fPt;
3363 double firstT = span->fT;
3364 const SkOpSpan* prior;
3365 do {
3366 prior = span;
3367 span = &fTs[++index];
3368 } while (SkDPoint::ApproximatelyEqual(span->fPt, firstPt)
3369 && (span->fT == firstT || prior->fTiny));
3370 return index;
3371 }
3372
3373 int SkOpSegment::findExactT(double t, const SkOpSegment* match) const {
3374 int count = this->count();
3375 for (int index = 0; index < count; ++index) {
3376 const SkOpSpan& span = fTs[index];
3377 if (span.fT == t && span.fOther == match) {
3378 return index;
3379 }
3380 }
3381 SkASSERT(0);
3382 return -1;
3383 }
3384
3385
3386
3387 int SkOpSegment::findOtherT(double t, const SkOpSegment* match) const {
3388 int count = this->count();
3389 for (int index = 0; index < count; ++index) {
3390 const SkOpSpan& span = fTs[index];
3391 if (span.fOtherT == t && span.fOther == match) {
3392 return index;
3393 }
3394 }
3395 return -1;
3396 }
3397
3398 int SkOpSegment::findT(double t, const SkPoint& pt, const SkOpSegment* match) co nst {
3399 int count = this->count();
3400 // prefer exact matches over approximate matches
3401 for (int index = 0; index < count; ++index) {
3402 const SkOpSpan& span = fTs[index];
3403 if (span.fT == t && span.fOther == match) {
3404 return index;
3405 }
3406 }
3407 for (int index = 0; index < count; ++index) {
3408 const SkOpSpan& span = fTs[index];
3409 if (approximately_equal_orderable(span.fT, t) && span.fOther == match) {
3410 return index;
3411 }
3412 }
3413 // Usually, the pair of ts are an exact match. It's possible that the t valu es have
3414 // been adjusted to make multiple intersections align. In this rare case, lo ok for a
3415 // matching point / match pair instead.
3416 for (int index = 0; index < count; ++index) {
3417 const SkOpSpan& span = fTs[index];
3418 if (span.fPt == pt && span.fOther == match) {
3419 return index;
3420 }
3421 }
3422 SkASSERT(0);
3423 return -1;
3424 }
3425
3426 SkOpSegment* SkOpSegment::findTop(int* tIndexPtr, int* endIndexPtr, bool* unsort able,
3427 bool firstPass) {
3428 // iterate through T intersections and return topmost 1063 // iterate through T intersections and return topmost
3429 // topmost tangent from y-min to first pt is closer to horizontal 1064 // topmost tangent from y-min to first pt is closer to horizontal
3430 SkASSERT(!done()); 1065 SkASSERT(!done());
3431 int firstT = -1; 1066 SkOpSpanBase* firstT = NULL;
3432 /* SkPoint topPt = */ activeLeftTop(&firstT); 1067 (void) this->activeLeftTop(&firstT);
3433 if (firstT < 0) { 1068 if (!firstT) {
3434 *unsortable = !firstPass; 1069 *unsortable = !firstPass;
3435 firstT = 0; 1070 firstT = &fHead;
3436 while (fTs[firstT].fDone) { 1071 while (firstT->upCast()->done()) {
3437 SkASSERT(firstT < fTs.count()); 1072 firstT = firstT->upCast()->next();
3438 ++firstT;
3439 } 1073 }
3440 *tIndexPtr = firstT; 1074 *startPtr = firstT;
3441 *endIndexPtr = nextExactSpan(firstT, 1); 1075 *endPtr = firstT->upCast()->next();
3442 return this; 1076 return this;
3443 } 1077 }
3444 // sort the edges to find the leftmost 1078 // sort the edges to find the leftmost
3445 int step = 1; 1079 int step = 1;
3446 int end; 1080 SkOpSpanBase* end;
3447 if (span(firstT).fDone || (end = nextSpan(firstT, step)) == -1) { 1081 if (firstT->final() || firstT->upCast()->done()) {
3448 step = -1; 1082 step = -1;
3449 end = nextSpan(firstT, step); 1083 end = firstT->prev();
3450 SkASSERT(end != -1); 1084 SkASSERT(end);
1085 } else {
1086 end = firstT->upCast()->next();
3451 } 1087 }
3452 // if the topmost T is not on end, or is three-way or more, find left 1088 // if the topmost T is not on end, or is three-way or more, find left
3453 // look for left-ness from tLeft to firstT (matching y of other) 1089 // look for left-ness from tLeft to firstT (matching y of other)
3454 SkASSERT(firstT - end != 0); 1090 SkASSERT(firstT != end);
3455 SkOpAngle* markAngle = spanToAngle(firstT, end); 1091 SkOpAngle* markAngle = spanToAngle(firstT, end);
3456 if (!markAngle) { 1092 if (!markAngle) {
3457 markAngle = addSingletonAngles(step); 1093 markAngle = addSingletonAngles(step, allocator);
3458 } 1094 }
3459 markAngle->markStops(); 1095 markAngle->markStops();
3460 const SkOpAngle* baseAngle = markAngle->next() == markAngle && !isVertical() ? markAngle 1096 const SkOpAngle* baseAngle = markAngle->next() == markAngle && !isVertical() ? markAngle
3461 : markAngle->findFirst(); 1097 : markAngle->findFirst();
3462 if (!baseAngle) { 1098 if (!baseAngle) {
3463 return NULL; // nothing to do 1099 return NULL; // nothing to do
3464 } 1100 }
3465 SkScalar top = SK_ScalarMax; 1101 SkScalar top = SK_ScalarMax;
3466 const SkOpAngle* firstAngle = NULL; 1102 const SkOpAngle* firstAngle = NULL;
3467 const SkOpAngle* angle = baseAngle; 1103 const SkOpAngle* angle = baseAngle;
3468 do { 1104 do {
3469 if (!angle->unorderable()) { 1105 if (!angle->unorderable()) {
3470 SkOpSegment* next = angle->segment(); 1106 const SkOpSegment* next = angle->segment();
3471 SkPathOpsBounds bounds; 1107 SkPathOpsBounds bounds;
3472 next->subDivideBounds(angle->end(), angle->start(), &bounds); 1108 next->subDivideBounds(angle->end(), angle->start(), &bounds);
3473 bool nearSame = AlmostEqualUlps(top, bounds.top()); 1109 bool nearSame = AlmostEqualUlps(top, bounds.top());
3474 bool lowerSector = !firstAngle || angle->sectorEnd() < firstAngle->s ectorStart(); 1110 bool lowerSector = !firstAngle || angle->sectorEnd() < firstAngle->s ectorStart();
3475 bool lesserSector = top > bounds.fTop; 1111 bool lesserSector = top > bounds.fTop;
3476 if (lesserSector && (!nearSame || lowerSector)) { 1112 if (lesserSector && (!nearSame || lowerSector)) {
3477 top = bounds.fTop; 1113 top = bounds.fTop;
3478 firstAngle = angle; 1114 firstAngle = angle;
3479 } 1115 }
3480 } 1116 }
3481 angle = angle->next(); 1117 angle = angle->next();
3482 } while (angle != baseAngle); 1118 } while (angle != baseAngle);
3483 if (!firstAngle) { 1119 if (!firstAngle) {
3484 return NULL; // if all are unorderable, give up 1120 return NULL; // if all are unorderable, give up
3485 } 1121 }
3486 #if DEBUG_SORT 1122 #if DEBUG_SORT
3487 SkDebugf("%s\n", __FUNCTION__); 1123 SkDebugf("%s\n", __FUNCTION__);
3488 firstAngle->debugLoop(); 1124 firstAngle->debugLoop();
3489 #endif 1125 #endif
3490 // skip edges that have already been processed 1126 // skip edges that have already been processed
3491 angle = firstAngle; 1127 angle = firstAngle;
3492 SkOpSegment* leftSegment = NULL; 1128 SkOpSegment* leftSegment = NULL;
3493 bool looped = false; 1129 bool looped = false;
3494 do { 1130 do {
3495 *unsortable = angle->unorderable(); 1131 *unsortable = angle->unorderable();
3496 if (firstPass || !*unsortable) { 1132 if (firstPass || !*unsortable) {
3497 leftSegment = angle->segment(); 1133 leftSegment = angle->segment();
3498 *tIndexPtr = angle->end(); 1134 *startPtr = angle->end();
3499 *endIndexPtr = angle->start(); 1135 *endPtr = angle->start();
3500 if (!leftSegment->fTs[SkMin32(*tIndexPtr, *endIndexPtr)].fDone) { 1136 const SkOpSpan* firstSpan = (*startPtr)->starter(*endPtr);
1137 if (!firstSpan->done()) {
3501 break; 1138 break;
3502 } 1139 }
3503 } 1140 }
3504 angle = angle->next(); 1141 angle = angle->next();
3505 looped = true; 1142 looped = true;
3506 } while (angle != firstAngle); 1143 } while (angle != firstAngle);
3507 if (angle == firstAngle && looped) { 1144 if (angle == firstAngle && looped) {
3508 return NULL; 1145 return NULL;
3509 } 1146 }
3510 if (leftSegment->verb() >= SkPath::kQuad_Verb) { 1147 if (leftSegment->verb() >= SkPath::kQuad_Verb) {
3511 const int tIndex = *tIndexPtr; 1148 SkOpSpanBase* start = *startPtr;
3512 const int endIndex = *endIndexPtr; 1149 SkOpSpanBase* end = *endPtr;
3513 bool swap; 1150 bool swap;
3514 if (!leftSegment->clockwise(tIndex, endIndex, &swap)) { 1151 if (!leftSegment->clockwise(start, end, &swap)) {
3515 #if DEBUG_SWAP_TOP 1152 #if DEBUG_SWAP_TOP
3516 SkDebugf("%s swap=%d inflections=%d serpentine=%d controlledbyends=% d monotonic=%d\n", 1153 SkDebugf("%s swap=%d inflections=%d monotonic=%d\n",
3517 __FUNCTION__, 1154 __FUNCTION__,
3518 swap, leftSegment->debugInflections(tIndex, endIndex), 1155 swap, leftSegment->debugInflections(start, end),
3519 leftSegment->serpentine(tIndex, endIndex), 1156 leftSegment->monotonicInY(start, end));
3520 leftSegment->controlsContainedByEnds(tIndex, endIndex),
3521 leftSegment->monotonicInY(tIndex, endIndex));
3522 #endif 1157 #endif
3523 if (swap) { 1158 if (swap) {
3524 // FIXME: I doubt it makes sense to (necessarily) swap if the edge was not t he first 1159 // FIXME: I doubt it makes sense to (necessarily) swap if the edge was not t he first
3525 // sorted but merely the first not already processed (i.e., not done) 1160 // sorted but merely the first not already processed (i.e., not done)
3526 SkTSwap(*tIndexPtr, *endIndexPtr); 1161 SkTSwap(*startPtr, *endPtr);
3527 } 1162 }
3528 } 1163 }
3529 } 1164 }
3530 SkASSERT(!leftSegment->fTs[SkMin32(*tIndexPtr, *endIndexPtr)].fTiny);
3531 return leftSegment; 1165 return leftSegment;
3532 } 1166 }
3533 1167
3534 int SkOpSegment::firstActive(int tIndex) const { 1168 SkOpGlobalState* SkOpSegment::globalState() const {
3535 while (fTs[tIndex].fTiny) { 1169 return contour()->globalState();
3536 SkASSERT(!isCanceled(tIndex));
3537 ++tIndex;
3538 }
3539 return tIndex;
3540 } 1170 }
3541 1171
3542 // FIXME: not crazy about this 1172 void SkOpSegment::init(SkPoint pts[], SkOpContour* contour, SkPath::Verb verb) {
3543 // when the intersections are performed, the other index is into an 1173 fContour = contour;
3544 // incomplete array. As the array grows, the indices become incorrect 1174 fNext = NULL;
3545 // while the following fixes the indices up again, it isn't smart about 1175 fPts = pts;
3546 // skipping segments whose indices are already correct 1176 fVerb = verb;
3547 // assuming we leave the code that wrote the index in the first place 1177 fCount = 0;
3548 // FIXME: if called after remove, this needs to correct tiny 1178 fDoneCount = 0;
3549 void SkOpSegment::fixOtherTIndex() { 1179 fVisited = false;
3550 int iCount = fTs.count(); 1180 SkOpSpan* zeroSpan = &fHead;
3551 for (int i = 0; i < iCount; ++i) { 1181 zeroSpan->init(this, NULL, 0, fPts[0]);
3552 SkOpSpan& iSpan = fTs[i]; 1182 SkOpSpanBase* oneSpan = &fTail;
3553 double oT = iSpan.fOtherT; 1183 zeroSpan->setNext(oneSpan);
3554 SkOpSegment* other = iSpan.fOther; 1184 oneSpan->initBase(this, zeroSpan, 1, fPts[SkPathOpsVerbToPoints(fVerb)]);
3555 int oCount = other->fTs.count(); 1185 PATH_OPS_DEBUG_CODE(fID = globalState()->nextSegmentID());
3556 SkDEBUGCODE(iSpan.fOtherIndex = -1);
3557 for (int o = 0; o < oCount; ++o) {
3558 SkOpSpan& oSpan = other->fTs[o];
3559 if (oT == oSpan.fT && this == oSpan.fOther && oSpan.fOtherT == iSpan .fT) {
3560 iSpan.fOtherIndex = o;
3561 oSpan.fOtherIndex = i;
3562 break;
3563 }
3564 }
3565 SkASSERT(iSpan.fOtherIndex >= 0);
3566 }
3567 } 1186 }
3568 1187
3569 bool SkOpSegment::inCoincidentSpan(double t, const SkOpSegment* other) const { 1188 void SkOpSegment::initWinding(SkOpSpanBase* start, SkOpSpanBase* end,
3570 int foundEnds = 0; 1189 SkOpAngle::IncludeType angleIncludeType) {
3571 int count = this->count(); 1190 int local = SkOpSegment::SpanSign(start, end);
3572 for (int index = 0; index < count; ++index) {
3573 const SkOpSpan& span = this->span(index);
3574 if (span.fCoincident) {
3575 foundEnds |= (span.fOther == other) << ((t > span.fT) + (t >= span.f T));
3576 }
3577 }
3578 SkASSERT(foundEnds != 7);
3579 return foundEnds == 0x3 || foundEnds == 0x5 || foundEnds == 0x6; // two bit s set
3580 }
3581
3582 bool SkOpSegment::inconsistentAngle(int maxWinding, int sumWinding, int oppMaxWi nding,
3583 int oppSumWinding, const SkOpAngle* angle) const {
3584 SkASSERT(angle->segment() == this);
3585 if (UseInnerWinding(maxWinding, sumWinding)) {
3586 maxWinding = sumWinding;
3587 }
3588 if (oppMaxWinding != oppSumWinding && UseInnerWinding(oppMaxWinding, oppSumW inding)) {
3589 oppMaxWinding = oppSumWinding;
3590 }
3591 return inconsistentWinding(angle, maxWinding, oppMaxWinding);
3592 }
3593
3594 bool SkOpSegment::inconsistentWinding(const SkOpAngle* angle, int winding,
3595 int oppWinding) const {
3596 int index = angle->start();
3597 int endIndex = angle->end();
3598 int min = SkMin32(index, endIndex);
3599 int step = SkSign32(endIndex - index);
3600 if (inconsistentWinding(min, winding, oppWinding)) {
3601 return true;
3602 }
3603 const SkOpSegment* other = this;
3604 while ((other = other->nextChase(&index, &step, &min, NULL))) {
3605 if (other->fTs[min].fWindSum != SK_MinS32) {
3606 break;
3607 }
3608 if (fOperand == other->fOperand) {
3609 if (other->inconsistentWinding(min, winding, oppWinding)) {
3610 return true;
3611 }
3612 } else {
3613 if (other->inconsistentWinding(min, oppWinding, winding)) {
3614 return true;
3615 }
3616 }
3617 }
3618 return false;
3619 }
3620
3621 bool SkOpSegment::inconsistentWinding(int index, int winding, int oppWinding) co nst {
3622 SkASSERT(winding || oppWinding);
3623 double referenceT = this->span(index).fT;
3624 int lesser = index;
3625 while (--lesser >= 0 && precisely_negative(referenceT - fTs[lesser].fT)) {
3626 if (inconsistentWinding(__FUNCTION__, lesser, winding, oppWinding)) {
3627 return true;
3628 }
3629 }
3630 do {
3631 if (inconsistentWinding(__FUNCTION__, index, winding, oppWinding)) {
3632 return true;
3633 }
3634 } while (++index < fTs.count() && precisely_negative(fTs[index].fT - referenc eT));
3635 return false;
3636 }
3637
3638 bool SkOpSegment::inconsistentWinding(const char* funName, int tIndex, int windi ng,
3639 int oppWinding) const {
3640 const SkOpSpan& span = this->span(tIndex);
3641 if (span.fDone && !span.fSmall) {
3642 return false;
3643 }
3644 return (span.fWindSum != SK_MinS32 && span.fWindSum != winding)
3645 || (span.fOppSum != SK_MinS32 && span.fOppSum != oppWinding);
3646 }
3647
3648 void SkOpSegment::init(const SkPoint pts[], SkPath::Verb verb, bool operand, boo l evenOdd) {
3649 fDoneSpans = 0;
3650 fOperand = operand;
3651 fXor = evenOdd;
3652 fPts = pts;
3653 fVerb = verb;
3654 fLoop = fMultiples = fSmall = fTiny = false;
3655 }
3656
3657 void SkOpSegment::initWinding(int start, int end, SkOpAngle::IncludeType angleIn cludeType) {
3658 int local = spanSign(start, end);
3659 SkDEBUGCODE(bool success); 1191 SkDEBUGCODE(bool success);
3660 if (angleIncludeType == SkOpAngle::kBinarySingle) { 1192 if (angleIncludeType == SkOpAngle::kBinarySingle) {
3661 int oppLocal = oppSign(start, end); 1193 int oppLocal = SkOpSegment::OppSign(start, end);
3662 SkDEBUGCODE(success =) markAndChaseWinding(start, end, local, oppLocal, NULL); 1194 SkDEBUGCODE(success =) markAndChaseWinding(start, end, local, oppLocal, NULL);
3663 // OPTIMIZATION: the reverse mark and chase could skip the first marking 1195 // OPTIMIZATION: the reverse mark and chase could skip the first marking
3664 SkDEBUGCODE(success |=) markAndChaseWinding(end, start, local, oppLocal, NULL); 1196 SkDEBUGCODE(success |=) markAndChaseWinding(end, start, local, oppLocal, NULL);
3665 } else { 1197 } else {
3666 SkDEBUGCODE(success =) markAndChaseWinding(start, end, local, NULL); 1198 SkDEBUGCODE(success =) markAndChaseWinding(start, end, local, NULL);
3667 // OPTIMIZATION: the reverse mark and chase could skip the first marking 1199 // OPTIMIZATION: the reverse mark and chase could skip the first marking
3668 SkDEBUGCODE(success |=) markAndChaseWinding(end, start, local, NULL); 1200 SkDEBUGCODE(success |=) markAndChaseWinding(end, start, local, NULL);
3669 } 1201 }
3670 SkASSERT(success); 1202 SkASSERT(success);
3671 } 1203 }
3672 1204
3673 /* 1205 /*
3674 when we start with a vertical intersect, we try to use the dx to determine if th e edge is to 1206 when we start with a vertical intersect, we try to use the dx to determine if th e edge is to
3675 the left or the right of vertical. This determines if we need to add the span's 1207 the left or the right of vertical. This determines if we need to add the span's
3676 sign or not. However, this isn't enough. 1208 sign or not. However, this isn't enough.
3677 If the supplied sign (winding) is zero, then we didn't hit another vertical span , so dx is needed. 1209 If the supplied sign (winding) is zero, then we didn't hit another vertical span , so dx is needed.
3678 If there was a winding, then it may or may not need adjusting. If the span the w inding was borrowed 1210 If there was a winding, then it may or may not need adjusting. If the span the w inding was borrowed
3679 from has the same x direction as this span, the winding should change. If the dx is opposite, then 1211 from has the same x direction as this span, the winding should change. If the dx is opposite, then
3680 the same winding is shared by both. 1212 the same winding is shared by both.
3681 */ 1213 */
3682 bool SkOpSegment::initWinding(int start, int end, double tHit, int winding, SkSc alar hitDx, 1214 bool SkOpSegment::initWinding(SkOpSpanBase* start, SkOpSpanBase* end, double tHi t,
3683 int oppWind, SkScalar hitOppDx) { 1215 int winding, SkScalar hitDx, int oppWind, SkScalar hitOppDx) {
1216 SkASSERT(this == start->segment());
3684 SkASSERT(hitDx || !winding); 1217 SkASSERT(hitDx || !winding);
3685 SkScalar dx = (*CurveSlopeAtT[SkPathOpsVerbToPoints(fVerb)])(fPts, tHit).fX; 1218 SkScalar dx = (*CurveSlopeAtT[SkPathOpsVerbToPoints(fVerb)])(fPts, tHit).fX;
3686 SkASSERT(dx); 1219 // SkASSERT(dx);
3687 int windVal = windValue(SkMin32(start, end)); 1220 int windVal = start->starter(end)->windValue();
3688 #if DEBUG_WINDING_AT_T 1221 #if DEBUG_WINDING_AT_T
3689 SkDebugf("%s id=%d oldWinding=%d hitDx=%c dx=%c windVal=%d", __FUNCTION__, d ebugID(), winding, 1222 SkDebugf("%s id=%d oldWinding=%d hitDx=%c dx=%c windVal=%d", __FUNCTION__, d ebugID(), winding,
3690 hitDx ? hitDx > 0 ? '+' : '-' : '0', dx > 0 ? '+' : '-', windVal); 1223 hitDx ? hitDx > 0 ? '+' : '-' : '0', dx > 0 ? '+' : '-', windVal);
3691 #endif 1224 #endif
3692 int sideWind = winding + (dx < 0 ? windVal : -windVal); 1225 int sideWind = winding + (dx < 0 ? windVal : -windVal);
3693 if (abs(winding) < abs(sideWind)) { 1226 if (abs(winding) < abs(sideWind)) {
3694 winding = sideWind; 1227 winding = sideWind;
3695 } 1228 }
3696 SkDEBUGCODE(int oppLocal = oppSign(start, end)); 1229 SkDEBUGCODE(int oppLocal = SkOpSegment::OppSign(start, end));
3697 SkASSERT(hitOppDx || !oppWind || !oppLocal); 1230 SkASSERT(hitOppDx || !oppWind || !oppLocal);
3698 int oppWindVal = oppValue(SkMin32(start, end)); 1231 int oppWindVal = start->starter(end)->oppValue();
3699 if (!oppWind) { 1232 if (!oppWind) {
3700 oppWind = dx < 0 ? oppWindVal : -oppWindVal; 1233 oppWind = dx < 0 ? oppWindVal : -oppWindVal;
3701 } else if (hitOppDx * dx >= 0) { 1234 } else if (hitOppDx * dx >= 0) {
3702 int oppSideWind = oppWind + (dx < 0 ? oppWindVal : -oppWindVal); 1235 int oppSideWind = oppWind + (dx < 0 ? oppWindVal : -oppWindVal);
3703 if (abs(oppWind) < abs(oppSideWind)) { 1236 if (abs(oppWind) < abs(oppSideWind)) {
3704 oppWind = oppSideWind; 1237 oppWind = oppSideWind;
3705 } 1238 }
3706 } 1239 }
3707 #if DEBUG_WINDING_AT_T 1240 #if DEBUG_WINDING_AT_T
3708 SkDebugf(" winding=%d oppWind=%d\n", winding, oppWind); 1241 SkDebugf(" winding=%d oppWind=%d\n", winding, oppWind);
3709 #endif 1242 #endif
3710 // if this fails to mark (because the edges are too small) inform caller to try again 1243 // if this fails to mark (because the edges are too small) inform caller to try again
3711 bool success = markAndChaseWinding(start, end, winding, oppWind, NULL); 1244 bool success = markAndChaseWinding(start, end, winding, oppWind, NULL);
3712 // OPTIMIZATION: the reverse mark and chase could skip the first marking 1245 // OPTIMIZATION: the reverse mark and chase could skip the first marking
3713 success |= markAndChaseWinding(end, start, winding, oppWind, NULL); 1246 success |= markAndChaseWinding(end, start, winding, oppWind, NULL);
3714 return success; 1247 return success;
3715 } 1248 }
3716 1249
3717 bool SkOpSegment::inLoop(const SkOpAngle* baseAngle, int spanCount, int* indexPt r) const { 1250 bool SkOpSegment::isClose(double t, const SkOpSegment* opp) const {
3718 if (!baseAngle->inLoop()) { 1251 SkDPoint cPt = this->dPtAtT(t);
3719 return false; 1252 int pts = SkPathOpsVerbToPoints(this->verb());
3720 } 1253 SkDVector dxdy = (*CurveDSlopeAtT[pts])(this->pts(), t);
3721 int index = *indexPtr; 1254 SkDLine perp = {{ cPt, {cPt.fX + dxdy.fY, cPt.fY - dxdy.fX} }};
3722 SkOpAngle* from = fTs[index].fFromAngle; 1255 SkIntersections i;
3723 SkOpAngle* to = fTs[index].fToAngle; 1256 int oppPts = SkPathOpsVerbToPoints(opp->verb());
3724 while (++index < spanCount) { 1257 (*CurveIntersectRay[oppPts])(opp->pts(), perp, &i);
3725 SkOpAngle* nextFrom = fTs[index].fFromAngle; 1258 int used = i.used();
3726 SkOpAngle* nextTo = fTs[index].fToAngle; 1259 for (int index = 0; index < used; ++index) {
3727 if (from != nextFrom || to != nextTo) { 1260 if (cPt.roughlyEqual(i.pt(index))) {
3728 break; 1261 return true;
3729 } 1262 }
3730 } 1263 }
3731 *indexPtr = index;
3732 return true;
3733 }
3734
3735 // OPTIMIZE: successive calls could start were the last leaves off
3736 // or calls could specialize to walk forwards or backwards
3737 bool SkOpSegment::isMissing(double startT, const SkPoint& pt) const {
3738 int tCount = fTs.count();
3739 for (int index = 0; index < tCount; ++index) {
3740 const SkOpSpan& span = fTs[index];
3741 if (approximately_zero(startT - span.fT) && pt == span.fPt) {
3742 return false;
3743 }
3744 }
3745 return true;
3746 }
3747
3748
3749 SkOpSegment* SkOpSegment::isSimple(int* end, int* step) {
3750 return nextChase(end, step, NULL, NULL);
3751 }
3752
3753 bool SkOpSegment::isTiny(const SkOpAngle* angle) const {
3754 int start = angle->start();
3755 int end = angle->end();
3756 const SkOpSpan& mSpan = fTs[SkMin32(start, end)];
3757 return mSpan.fTiny;
3758 }
3759
3760 bool SkOpSegment::isTiny(int index) const {
3761 return fTs[index].fTiny;
3762 }
3763
3764 // look pair of active edges going away from coincident edge
3765 // one of them should be the continuation of other
3766 // if both are active, look to see if they both the connect to another coinciden t pair
3767 // if at least one is a line, then make the pair coincident
3768 // if neither is a line, test for coincidence
3769 bool SkOpSegment::joinCoincidence(SkOpSegment* other, double otherT, const SkPoi nt& otherPt,
3770 int step, bool cancel) {
3771 int otherTIndex = other->findT(otherT, otherPt, this);
3772 int next = other->nextExactSpan(otherTIndex, step);
3773 int otherMin = SkMin32(otherTIndex, next);
3774 int otherWind = other->span(otherMin).fWindValue;
3775 if (otherWind == 0) {
3776 return false;
3777 }
3778 if (next < 0) {
3779 return false; // can happen if t values were adjusted but coincident ts were not
3780 }
3781 int tIndex = 0;
3782 do {
3783 SkOpSpan* test = &fTs[tIndex];
3784 SkASSERT(test->fT == 0);
3785 if (test->fOther == other || test->fOtherT != 1) {
3786 continue;
3787 }
3788 SkPoint startPt, endPt;
3789 double endT;
3790 if (findCoincidentMatch(test, other, otherTIndex, next, step, &startPt, &endPt, &endT)) {
3791 SkOpSegment* match = test->fOther;
3792 if (cancel) {
3793 match->addTCancel(startPt, endPt, other);
3794 } else {
3795 if (!match->addTCoincident(startPt, endPt, endT, other)) {
3796 return false;
3797 }
3798 }
3799 return true;
3800 }
3801 } while (fTs[++tIndex].fT == 0);
3802 return false; 1264 return false;
3803 } 1265 }
3804 1266
3805 // this span is excluded by the winding rule -- chase the ends 1267 bool SkOpSegment::isXor() const {
3806 // as long as they are unambiguous to mark connections as done 1268 return fContour->isXor();
3807 // and give them the same winding value 1269 }
3808 1270
3809 SkOpSpan* SkOpSegment::markAndChaseDoneBinary(int index, int endIndex) { 1271 SkOpSpanBase* SkOpSegment::markAndChaseDone(SkOpSpanBase* start, SkOpSpanBase* e nd) {
3810 int step = SkSign32(endIndex - index); 1272 int step = start->step(end);
3811 int min = SkMin32(index, endIndex); 1273 SkOpSpan* minSpan = start->starter(end);
3812 markDoneBinary(min); 1274 markDone(minSpan);
3813 SkOpSpan* last = NULL; 1275 SkOpSpanBase* last = NULL;
3814 SkOpSegment* other = this; 1276 SkOpSegment* other = this;
3815 while ((other = other->nextChase(&index, &step, &min, &last))) { 1277 while ((other = other->nextChase(&start, &step, &minSpan, &last))) {
3816 if (other->done()) { 1278 if (other->done()) {
3817 SkASSERT(!last); 1279 SkASSERT(!last);
3818 break; 1280 break;
3819 } 1281 }
3820 other->markDoneBinary(min); 1282 other->markDone(minSpan);
3821 } 1283 }
3822 return last; 1284 return last;
3823 } 1285 }
3824 1286
3825 SkOpSpan* SkOpSegment::markAndChaseDoneUnary(int index, int endIndex) { 1287 bool SkOpSegment::markAndChaseWinding(SkOpSpanBase* start, SkOpSpanBase* end, in t winding,
3826 int step = SkSign32(endIndex - index); 1288 SkOpSpanBase** lastPtr) {
3827 int min = SkMin32(index, endIndex); 1289 SkOpSpan* spanStart = start->starter(end);
3828 markDoneUnary(min); 1290 int step = start->step(end);
3829 SkOpSpan* last = NULL; 1291 bool success = markWinding(spanStart, winding);
1292 SkOpSpanBase* last = NULL;
3830 SkOpSegment* other = this; 1293 SkOpSegment* other = this;
3831 while ((other = other->nextChase(&index, &step, &min, &last))) { 1294 while ((other = other->nextChase(&start, &step, &spanStart, &last))) {
3832 if (other->done()) { 1295 if (spanStart->windSum() != SK_MinS32) {
1296 SkASSERT(spanStart->windSum() == winding);
3833 SkASSERT(!last); 1297 SkASSERT(!last);
3834 break; 1298 break;
3835 } 1299 }
3836 other->markDoneUnary(min); 1300 (void) other->markWinding(spanStart, winding);
3837 }
3838 return last;
3839 }
3840
3841 bool SkOpSegment::markAndChaseWinding(const SkOpAngle* angle, int winding, SkOpS pan** lastPtr) {
3842 int index = angle->start();
3843 int endIndex = angle->end();
3844 return markAndChaseWinding(index, endIndex, winding, lastPtr);
3845 }
3846
3847 bool SkOpSegment::markAndChaseWinding(int index, int endIndex, int winding, SkOp Span** lastPtr) {
3848 int min = SkMin32(index, endIndex);
3849 int step = SkSign32(endIndex - index);
3850 bool success = markWinding(min, winding);
3851 SkOpSpan* last = NULL;
3852 SkOpSegment* other = this;
3853 while ((other = other->nextChase(&index, &step, &min, &last))) {
3854 if (other->fTs[min].fWindSum != SK_MinS32) {
3855 SkASSERT(other->fTs[min].fWindSum == winding || other->fTs[min].fLoo p);
3856 SkASSERT(!last);
3857 break;
3858 }
3859 (void) other->markWinding(min, winding);
3860 } 1301 }
3861 if (lastPtr) { 1302 if (lastPtr) {
3862 *lastPtr = last; 1303 *lastPtr = last;
3863 } 1304 }
3864 return success; 1305 return success;
3865 } 1306 }
3866 1307
3867 bool SkOpSegment::markAndChaseWinding(int index, int endIndex, int winding, int oppWinding, 1308 bool SkOpSegment::markAndChaseWinding(SkOpSpanBase* start, SkOpSpanBase* end,
3868 SkOpSpan** lastPtr) { 1309 int winding, int oppWinding, SkOpSpanBase** lastPtr) {
3869 int min = SkMin32(index, endIndex); 1310 SkOpSpan* spanStart = start->starter(end);
3870 int step = SkSign32(endIndex - index); 1311 int step = start->step(end);
3871 bool success = markWinding(min, winding, oppWinding); 1312 bool success = markWinding(spanStart, winding, oppWinding);
3872 SkOpSpan* last = NULL; 1313 SkOpSpanBase* last = NULL;
3873 SkOpSegment* other = this; 1314 SkOpSegment* other = this;
3874 while ((other = other->nextChase(&index, &step, &min, &last))) { 1315 while ((other = other->nextChase(&start, &step, &spanStart, &last))) {
3875 if (other->fTs[min].fWindSum != SK_MinS32) { 1316 if (spanStart->windSum() != SK_MinS32) {
3876 #ifdef SK_DEBUG 1317 if (this->operand() == other->operand()) {
3877 if (!other->fTs[min].fLoop) { 1318 SkASSERT(spanStart->windSum() == winding);
3878 if (fOperand == other->fOperand) { 1319 if (spanStart->oppSum() != oppWinding) {
3879 // FIXME: this is probably a bug -- rects4 asserts here 1320 this->globalState()->setWindingFailed();
3880 // SkASSERT(other->fTs[min].fWindSum == winding); 1321 return false;
3881 // FIXME: this is probably a bug -- rects3 asserts here
3882 // SkASSERT(other->fTs[min].fOppSum == oppWinding);
3883 } else {
3884 // FIXME: this is probably a bug -- issue414409b asserts here
3885 // SkASSERT(other->fTs[min].fWindSum == oppWinding);
3886 // FIXME: this is probably a bug -- skpwww_joomla_org_23 asserts here
3887 // SkASSERT(other->fTs[min].fOppSum == winding);
3888 } 1322 }
1323 } else {
1324 SkASSERT(spanStart->windSum() == oppWinding);
1325 SkASSERT(spanStart->oppSum() == winding);
3889 } 1326 }
3890 SkASSERT(!last); 1327 SkASSERT(!last);
3891 #endif
3892 break; 1328 break;
3893 } 1329 }
3894 if (fOperand == other->fOperand) { 1330 if (this->operand() == other->operand()) {
3895 (void) other->markWinding(min, winding, oppWinding); 1331 (void) other->markWinding(spanStart, winding, oppWinding);
3896 } else { 1332 } else {
3897 (void) other->markWinding(min, oppWinding, winding); 1333 (void) other->markWinding(spanStart, oppWinding, winding);
3898 } 1334 }
3899 } 1335 }
3900 if (lastPtr) { 1336 if (lastPtr) {
3901 *lastPtr = last; 1337 *lastPtr = last;
3902 } 1338 }
3903 return success; 1339 return success;
3904 } 1340 }
3905 1341
3906 bool SkOpSegment::markAndChaseWinding(const SkOpAngle* angle, int winding, int o ppWinding, 1342 SkOpSpanBase* SkOpSegment::markAngle(int maxWinding, int sumWinding, const SkOpA ngle* angle) {
3907 SkOpSpan** lastPtr) {
3908 int start = angle->start();
3909 int end = angle->end();
3910 return markAndChaseWinding(start, end, winding, oppWinding, lastPtr);
3911 }
3912
3913 SkOpSpan* SkOpSegment::markAngle(int maxWinding, int sumWinding, const SkOpAngle * angle) {
3914 SkASSERT(angle->segment() == this); 1343 SkASSERT(angle->segment() == this);
3915 if (UseInnerWinding(maxWinding, sumWinding)) { 1344 if (UseInnerWinding(maxWinding, sumWinding)) {
3916 maxWinding = sumWinding; 1345 maxWinding = sumWinding;
3917 } 1346 }
3918 SkOpSpan* last; 1347 SkOpSpanBase* last;
3919 SkAssertResult(markAndChaseWinding(angle, maxWinding, &last)); 1348 (void) markAndChaseWinding(angle->start(), angle->end(), maxWinding, &last);
3920 #if DEBUG_WINDING 1349 #if DEBUG_WINDING
3921 if (last) { 1350 if (last) {
3922 SkDebugf("%s last id=%d windSum=", __FUNCTION__, 1351 SkDebugf("%s last seg=%d span=%d", __FUNCTION__,
3923 last->fOther->fTs[last->fOtherIndex].fOther->debugID()); 1352 last->segment()->debugID(), last->debugID());
3924 SkPathOpsDebug::WindingPrintf(last->fWindSum); 1353 if (!last->final()) {
3925 SkDebugf(" small=%d\n", last->fSmall); 1354 SkDebugf(" windSum=");
1355 SkPathOpsDebug::WindingPrintf(last->upCast()->windSum());
1356 }
1357 SkDebugf("\n");
3926 } 1358 }
3927 #endif 1359 #endif
3928 return last; 1360 return last;
3929 } 1361 }
3930 1362
3931 SkOpSpan* SkOpSegment::markAngle(int maxWinding, int sumWinding, int oppMaxWindi ng, 1363 SkOpSpanBase* SkOpSegment::markAngle(int maxWinding, int sumWinding, int oppMaxW inding,
3932 int oppSumWinding, const SkOpAngle* angle) { 1364 int oppSumWinding, const SkOpAngle* angle) {
3933 SkASSERT(angle->segment() == this); 1365 SkASSERT(angle->segment() == this);
3934 if (UseInnerWinding(maxWinding, sumWinding)) { 1366 if (UseInnerWinding(maxWinding, sumWinding)) {
3935 maxWinding = sumWinding; 1367 maxWinding = sumWinding;
3936 } 1368 }
3937 if (oppMaxWinding != oppSumWinding && UseInnerWinding(oppMaxWinding, oppSumW inding)) { 1369 if (oppMaxWinding != oppSumWinding && UseInnerWinding(oppMaxWinding, oppSumW inding)) {
3938 oppMaxWinding = oppSumWinding; 1370 oppMaxWinding = oppSumWinding;
3939 } 1371 }
3940 SkOpSpan* last; 1372 SkOpSpanBase* last = NULL;
3941 // caller doesn't require that this marks anything 1373 // caller doesn't require that this marks anything
3942 (void) markAndChaseWinding(angle, maxWinding, oppMaxWinding, &last); 1374 (void) markAndChaseWinding(angle->start(), angle->end(), maxWinding, oppMaxW inding, &last);
3943 #if DEBUG_WINDING 1375 #if DEBUG_WINDING
3944 if (last) { 1376 if (last) {
3945 SkDebugf("%s last id=%d windSum=", __FUNCTION__, 1377 SkDebugf("%s last segment=%d span=%d", __FUNCTION__,
3946 last->fOther->fTs[last->fOtherIndex].fOther->debugID()); 1378 last->segment()->debugID(), last->debugID());
3947 SkPathOpsDebug::WindingPrintf(last->fWindSum); 1379 if (!last->final()) {
3948 SkDebugf(" small=%d\n", last->fSmall); 1380 SkDebugf(" windSum=");
1381 SkPathOpsDebug::WindingPrintf(last->upCast()->windSum());
1382 }
1383 SkDebugf(" \n");
3949 } 1384 }
3950 #endif 1385 #endif
3951 return last; 1386 return last;
3952 } 1387 }
3953 1388
3954 // FIXME: this should also mark spans with equal (x,y) 1389 void SkOpSegment::markDone(SkOpSpan* span) {
3955 // This may be called when the segment is already marked done. While this 1390 SkASSERT(this == span->segment());
3956 // wastes time, it shouldn't do any more than spin through the T spans. 1391 if (span->done()) {
3957 // OPTIMIZATION: abort on first done found (assuming that this code is 1392 return;
3958 // always called to mark segments done).
3959 void SkOpSegment::markDone(int index, int winding) {
3960 // SkASSERT(!done());
3961 SkASSERT(winding);
3962 double referenceT = fTs[index].fT;
3963 int lesser = index;
3964 while (--lesser >= 0 && precisely_negative(referenceT - fTs[lesser].fT)) {
3965 markOneDone(__FUNCTION__, lesser, winding);
3966 } 1393 }
3967 do { 1394 #if DEBUG_MARK_DONE
3968 markOneDone(__FUNCTION__, index, winding); 1395 debugShowNewWinding(__FUNCTION__, span, span->windSum(), span->oppSum());
3969 } while (++index < fTs.count() && precisely_negative(fTs[index].fT - referen ceT)); 1396 #endif
1397 span->setDone(true);
1398 ++fDoneCount;
3970 debugValidate(); 1399 debugValidate();
3971 } 1400 }
3972 1401
3973 void SkOpSegment::markDoneBinary(int index) { 1402 bool SkOpSegment::markWinding(SkOpSpan* span, int winding) {
3974 double referenceT = fTs[index].fT; 1403 SkASSERT(this == span->segment());
3975 int lesser = index; 1404 SkASSERT(winding);
3976 while (--lesser >= 0 && precisely_negative(referenceT - fTs[lesser].fT)) { 1405 if (span->done()) {
3977 markOneDoneBinary(__FUNCTION__, lesser);
3978 }
3979 do {
3980 markOneDoneBinary(__FUNCTION__, index);
3981 } while (++index < fTs.count() && precisely_negative(fTs[index].fT - referen ceT));
3982 debugValidate();
3983 }
3984
3985 void SkOpSegment::markDoneFinal(int index) {
3986 double referenceT = fTs[index].fT;
3987 int lesser = index;
3988 while (--lesser >= 0 && precisely_negative(referenceT - fTs[lesser].fT)) {
3989 markOneDoneFinal(__FUNCTION__, lesser);
3990 }
3991 do {
3992 markOneDoneFinal(__FUNCTION__, index);
3993 } while (++index < fTs.count() && precisely_negative(fTs[index].fT - referen ceT));
3994 debugValidate();
3995 }
3996
3997 void SkOpSegment::markDoneUnary(int index) {
3998 double referenceT = fTs[index].fT;
3999 int lesser = index;
4000 while (--lesser >= 0 && precisely_negative(referenceT - fTs[lesser].fT)) {
4001 markOneDoneUnary(__FUNCTION__, lesser);
4002 }
4003 do {
4004 markOneDoneUnary(__FUNCTION__, index);
4005 } while (++index < fTs.count() && precisely_negative(fTs[index].fT - referen ceT));
4006 debugValidate();
4007 }
4008
4009 void SkOpSegment::markOneDone(const char* funName, int tIndex, int winding) {
4010 SkOpSpan* span;
4011 (void) markOneWinding(funName, tIndex, winding, &span); // allowed to do no thing
4012 if (span->fDone) {
4013 return;
4014 }
4015 span->fDone = true;
4016 ++fDoneSpans;
4017 }
4018
4019 void SkOpSegment::markOneDoneFinal(const char* funName, int tIndex) {
4020 SkOpSpan* span = &fTs[tIndex];
4021 if (span->fDone) {
4022 return;
4023 }
4024 span->fDone = true;
4025 ++fDoneSpans;
4026 }
4027
4028 void SkOpSegment::markOneDoneBinary(const char* funName, int tIndex) {
4029 SkOpSpan* span = verifyOneWinding(funName, tIndex);
4030 if (!span) {
4031 return;
4032 }
4033 SkASSERT(!span->fDone);
4034 span->fDone = true;
4035 ++fDoneSpans;
4036 }
4037
4038 void SkOpSegment::markOneDoneUnary(const char* funName, int tIndex) {
4039 SkOpSpan* span = verifyOneWindingU(funName, tIndex);
4040 if (!span) {
4041 return;
4042 }
4043 if (span->fWindSum == SK_MinS32) {
4044 SkDebugf("%s uncomputed\n", __FUNCTION__);
4045 }
4046 SkASSERT(!span->fDone);
4047 span->fDone = true;
4048 ++fDoneSpans;
4049 }
4050
4051 bool SkOpSegment::markOneWinding(const char* funName, int tIndex, int winding, S kOpSpan** lastPtr) {
4052 SkOpSpan* span = &fTs[tIndex];
4053 if (lastPtr) {
4054 *lastPtr = span;
4055 }
4056 if (span->fDone && !span->fSmall) {
4057 return false; 1406 return false;
4058 } 1407 }
4059 #if DEBUG_MARK_DONE 1408 #if DEBUG_MARK_DONE
4060 debugShowNewWinding(funName, *span, winding); 1409 debugShowNewWinding(__FUNCTION__, span, winding);
4061 #endif 1410 #endif
4062 SkASSERT(span->fWindSum == SK_MinS32 || span->fWindSum == winding); 1411 span->setWindSum(winding);
4063 #if DEBUG_LIMIT_WIND_SUM 1412 debugValidate();
4064 SkASSERT(abs(winding) <= DEBUG_LIMIT_WIND_SUM);
4065 #endif
4066 span->fWindSum = winding;
4067 return true; 1413 return true;
4068 } 1414 }
4069 1415
4070 bool SkOpSegment::markOneWinding(const char* funName, int tIndex, int winding, 1416 bool SkOpSegment::markWinding(SkOpSpan* span, int winding, int oppWinding) {
4071 int oppWinding, SkOpSpan** lastPtr) { 1417 SkASSERT(this == span->segment());
4072 SkOpSpan* span = &fTs[tIndex]; 1418 SkASSERT(winding || oppWinding);
4073 if (span->fDone && !span->fSmall) { 1419 if (span->done()) {
4074 return false; 1420 return false;
4075 } 1421 }
4076 #if DEBUG_MARK_DONE 1422 #if DEBUG_MARK_DONE
4077 debugShowNewWinding(funName, *span, winding, oppWinding); 1423 debugShowNewWinding(__FUNCTION__, span, winding, oppWinding);
4078 #endif 1424 #endif
4079 SkASSERT(span->fWindSum == SK_MinS32 || span->fWindSum == winding); 1425 span->setWindSum(winding);
4080 #if DEBUG_LIMIT_WIND_SUM 1426 span->setOppSum(oppWinding);
4081 SkASSERT(abs(winding) <= DEBUG_LIMIT_WIND_SUM);
4082 #endif
4083 span->fWindSum = winding;
4084 SkASSERT(span->fOppSum == SK_MinS32 || span->fOppSum == oppWinding);
4085 #if DEBUG_LIMIT_WIND_SUM
4086 SkASSERT(abs(oppWinding) <= DEBUG_LIMIT_WIND_SUM);
4087 #endif
4088 span->fOppSum = oppWinding;
4089 debugValidate(); 1427 debugValidate();
4090 if (lastPtr) {
4091 *lastPtr = span;
4092 }
4093 return true; 1428 return true;
4094 } 1429 }
4095 1430
4096 // from http://stackoverflow.com/questions/1165647/how-to-determine-if-a-list-of -polygon-points-are-in-clockwise-order 1431 bool SkOpSegment::match(const SkOpPtT* base, const SkOpSegment* testParent, doub le testT,
4097 bool SkOpSegment::clockwise(int tStart, int tEnd, bool* swap) const { 1432 const SkPoint& testPt) const {
4098 SkASSERT(fVerb != SkPath::kLine_Verb); 1433 const SkOpSegment* baseParent = base->segment();
4099 SkPoint edge[4]; 1434 if (this == baseParent && this == testParent && precisely_equal(base->fT, te stT)) {
4100 subDivide(tStart, tEnd, edge); 1435 return true;
4101 int points = SkPathOpsVerbToPoints(fVerb);
4102 double sum = (edge[0].fX - edge[points].fX) * (edge[0].fY + edge[points].fY) ;
4103 bool sumSet = false;
4104 if (fVerb == SkPath::kCubic_Verb) {
4105 SkDCubic cubic;
4106 cubic.set(edge);
4107 double inflectionTs[2];
4108 int inflections = cubic.findInflections(inflectionTs);
4109 // FIXME: this fixes cubicOp114 and breaks cubicOp58d
4110 // the trouble is that cubics with inflections confuse whether the curve breaks towards
4111 // or away, which in turn is used to determine if it is on the far right or left.
4112 // Probably a totally different approach is in order. At one time I trie d to project a
4113 // horizontal ray to determine winding, but was confused by how to map t he vertically
4114 // oriented winding computation over.
4115 if (0 && inflections) {
4116 double tLo = this->span(tStart).fT;
4117 double tHi = this->span(tEnd).fT;
4118 double tLoStart = tLo;
4119 for (int index = 0; index < inflections; ++index) {
4120 if (between(tLo, inflectionTs[index], tHi)) {
4121 tLo = inflectionTs[index];
4122 }
4123 }
4124 if (tLo != tLoStart && tLo != tHi) {
4125 SkDPoint sub[2];
4126 sub[0] = cubic.ptAtT(tLo);
4127 sub[1].set(edge[3]);
4128 SkDPoint ctrl[2];
4129 SkDCubic::SubDivide(fPts, sub[0], sub[1], tLo, tHi, ctrl);
4130 edge[0] = sub[0].asSkPoint();
4131 edge[1] = ctrl[0].asSkPoint();
4132 edge[2] = ctrl[1].asSkPoint();
4133 sum = (edge[0].fX - edge[3].fX) * (edge[0].fY + edge[3].fY);
4134 }
4135 }
4136 SkScalar lesser = SkTMin<SkScalar>(edge[0].fY, edge[3].fY);
4137 if (edge[1].fY < lesser && edge[2].fY < lesser) {
4138 SkDLine tangent1 = {{ {edge[0].fX, edge[0].fY}, {edge[1].fX, edge[1] .fY} }};
4139 SkDLine tangent2 = {{ {edge[2].fX, edge[2].fY}, {edge[3].fX, edge[3] .fY} }};
4140 if (SkIntersections::Test(tangent1, tangent2)) {
4141 SkPoint topPt = cubic_top(fPts, fTs[tStart].fT, fTs[tEnd].fT);
4142 sum += (topPt.fX - edge[0].fX) * (topPt.fY + edge[0].fY);
4143 sum += (edge[3].fX - topPt.fX) * (edge[3].fY + topPt.fY);
4144 sumSet = true;
4145 }
4146 }
4147 } 1436 }
4148 if (!sumSet) { 1437 if (!SkDPoint::ApproximatelyEqual(testPt, base->fPt)) {
4149 for (int idx = 0; idx < points; ++idx){ 1438 return false;
4150 sum += (edge[idx + 1].fX - edge[idx].fX) * (edge[idx + 1].fY + edge[ idx].fY);
4151 }
4152 } 1439 }
4153 if (fVerb == SkPath::kCubic_Verb) { 1440 return !ptsDisjoint(base->fT, base->fPt, testT, testPt);
4154 SkDCubic cubic;
4155 cubic.set(edge);
4156 *swap = sum > 0 && !cubic.monotonicInY() && !cubic.serpentine();
4157 } else {
4158 SkDQuad quad;
4159 quad.set(edge);
4160 *swap = sum > 0 && !quad.monotonicInY();
4161 }
4162 return sum <= 0;
4163 } 1441 }
4164 1442
4165 bool SkOpSegment::monotonicInY(int tStart, int tEnd) const { 1443 static SkOpSegment* set_last(SkOpSpanBase** last, SkOpSpanBase* endSpan) {
4166 SkASSERT(fVerb != SkPath::kLine_Verb); 1444 if (last) {
4167 if (fVerb == SkPath::kQuad_Verb) { 1445 *last = endSpan;
4168 SkDQuad dst = SkDQuad::SubDivide(fPts, fTs[tStart].fT, fTs[tEnd].fT);
4169 return dst.monotonicInY();
4170 }
4171 SkASSERT(fVerb == SkPath::kCubic_Verb);
4172 SkDCubic dst = SkDCubic::SubDivide(fPts, fTs[tStart].fT, fTs[tEnd].fT);
4173 return dst.monotonicInY();
4174 }
4175
4176 bool SkOpSegment::serpentine(int tStart, int tEnd) const {
4177 if (fVerb != SkPath::kCubic_Verb) {
4178 return false;
4179 }
4180 SkDCubic dst = SkDCubic::SubDivide(fPts, fTs[tStart].fT, fTs[tEnd].fT);
4181 return dst.serpentine();
4182 }
4183
4184 SkOpSpan* SkOpSegment::verifyOneWinding(const char* funName, int tIndex) {
4185 SkOpSpan& span = fTs[tIndex];
4186 if (span.fDone) {
4187 return NULL;
4188 }
4189 #if DEBUG_MARK_DONE
4190 debugShowNewWinding(funName, span, span.fWindSum, span.fOppSum);
4191 #endif
4192 // If the prior angle in the sort is unorderable, the winding sum may not be com putable.
4193 // To enable the assert, the 'prior is unorderable' state could be
4194 // piped down to this test, but not sure it's worth it.
4195 // (Once the sort order is stored in the span, this test may be feasible.)
4196 // SkASSERT(span.fWindSum != SK_MinS32);
4197 // SkASSERT(span.fOppSum != SK_MinS32);
4198 return &span;
4199 }
4200
4201 SkOpSpan* SkOpSegment::verifyOneWindingU(const char* funName, int tIndex) {
4202 SkOpSpan& span = fTs[tIndex];
4203 if (span.fDone) {
4204 return NULL;
4205 }
4206 #if DEBUG_MARK_DONE
4207 debugShowNewWinding(funName, span, span.fWindSum);
4208 #endif
4209 // If the prior angle in the sort is unorderable, the winding sum may not be com putable.
4210 // To enable the assert, the 'prior is unorderable' state could be
4211 // piped down to this test, but not sure it's worth it.
4212 // (Once the sort order is stored in the span, this test may be feasible.)
4213 // SkASSERT(span.fWindSum != SK_MinS32);
4214 return &span;
4215 }
4216
4217 bool SkOpSegment::markWinding(int index, int winding) {
4218 // SkASSERT(!done());
4219 SkASSERT(winding);
4220 double referenceT = fTs[index].fT;
4221 int lesser = index;
4222 bool success = false;
4223 while (--lesser >= 0 && precisely_negative(referenceT - fTs[lesser].fT)) {
4224 success |= markOneWinding(__FUNCTION__, lesser, winding, NULL);
4225 }
4226 do {
4227 success |= markOneWinding(__FUNCTION__, index, winding, NULL);
4228 } while (++index < fTs.count() && precisely_negative(fTs[index].fT - referenc eT));
4229 debugValidate();
4230 return success;
4231 }
4232
4233 bool SkOpSegment::markWinding(int index, int winding, int oppWinding) {
4234 // SkASSERT(!done());
4235 SkASSERT(winding || oppWinding);
4236 double referenceT = fTs[index].fT;
4237 int lesser = index;
4238 bool success = false;
4239 while (--lesser >= 0 && precisely_negative(referenceT - fTs[lesser].fT)) {
4240 success |= markOneWinding(__FUNCTION__, lesser, winding, oppWinding, NUL L);
4241 }
4242 do {
4243 success |= markOneWinding(__FUNCTION__, index, winding, oppWinding, NULL );
4244 } while (++index < fTs.count() && precisely_negative(fTs[index].fT - referenc eT));
4245 debugValidate();
4246 return success;
4247 }
4248
4249 void SkOpSegment::matchWindingValue(int tIndex, double t, bool borrowWind) {
4250 int nextDoorWind = SK_MaxS32;
4251 int nextOppWind = SK_MaxS32;
4252 // prefer exact matches
4253 if (tIndex > 0) {
4254 const SkOpSpan& below = fTs[tIndex - 1];
4255 if (below.fT == t) {
4256 nextDoorWind = below.fWindValue;
4257 nextOppWind = below.fOppValue;
4258 }
4259 }
4260 if (nextDoorWind == SK_MaxS32 && tIndex + 1 < fTs.count()) {
4261 const SkOpSpan& above = fTs[tIndex + 1];
4262 if (above.fT == t) {
4263 nextDoorWind = above.fWindValue;
4264 nextOppWind = above.fOppValue;
4265 }
4266 }
4267 if (nextDoorWind == SK_MaxS32 && tIndex > 0) {
4268 const SkOpSpan& below = fTs[tIndex - 1];
4269 if (approximately_negative(t - below.fT)) {
4270 nextDoorWind = below.fWindValue;
4271 nextOppWind = below.fOppValue;
4272 }
4273 }
4274 if (nextDoorWind == SK_MaxS32 && tIndex + 1 < fTs.count()) {
4275 const SkOpSpan& above = fTs[tIndex + 1];
4276 if (approximately_negative(above.fT - t)) {
4277 nextDoorWind = above.fWindValue;
4278 nextOppWind = above.fOppValue;
4279 }
4280 }
4281 if (nextDoorWind == SK_MaxS32 && borrowWind && tIndex > 0 && t < 1) {
4282 const SkOpSpan& below = fTs[tIndex - 1];
4283 nextDoorWind = below.fWindValue;
4284 nextOppWind = below.fOppValue;
4285 }
4286 if (nextDoorWind != SK_MaxS32) {
4287 SkOpSpan& newSpan = fTs[tIndex];
4288 newSpan.fWindValue = nextDoorWind;
4289 newSpan.fOppValue = nextOppWind;
4290 if (!nextDoorWind && !nextOppWind && !newSpan.fDone) {
4291 newSpan.fDone = true;
4292 ++fDoneSpans;
4293 }
4294 }
4295 }
4296
4297 bool SkOpSegment::nextCandidate(int* start, int* end) const {
4298 while (fTs[*end].fDone) {
4299 if (fTs[*end].fT == 1) {
4300 return false;
4301 }
4302 ++(*end);
4303 }
4304 *start = *end;
4305 *end = nextExactSpan(*start, 1);
4306 return true;
4307 }
4308
4309 static SkOpSegment* set_last(SkOpSpan** last, const SkOpSpan* endSpan) {
4310 if (last && !endSpan->fSmall) {
4311 *last = const_cast<SkOpSpan*>(endSpan); // FIXME: get rid of cast
4312 } 1446 }
4313 return NULL; 1447 return NULL;
4314 } 1448 }
4315 1449
4316 SkOpSegment* SkOpSegment::nextChase(int* indexPtr, int* stepPtr, int* minPtr, 1450 bool SkOpSegment::monotonicInY(const SkOpSpanBase* start, const SkOpSpanBase* en d) const {
4317 SkOpSpan** last) const { 1451 SkASSERT(fVerb != SkPath::kLine_Verb);
4318 int origIndex = *indexPtr; 1452 if (fVerb == SkPath::kQuad_Verb) {
1453 SkDQuad dst = SkDQuad::SubDivide(fPts, start->t(), end->t());
1454 return dst.monotonicInY();
1455 }
1456 SkASSERT(fVerb == SkPath::kCubic_Verb);
1457 SkDCubic dst = SkDCubic::SubDivide(fPts, start->t(), end->t());
1458 return dst.monotonicInY();
1459 }
1460
1461 bool SkOpSegment::NextCandidate(SkOpSpanBase* span, SkOpSpanBase** start,
1462 SkOpSpanBase** end) {
1463 while (span->final() || span->upCast()->done()) {
1464 if (span->final()) {
1465 return false;
1466 }
1467 span = span->upCast()->next();
1468 }
1469 *start = span;
1470 *end = span->upCast()->next();
1471 return true;
1472 }
1473
1474 SkOpSegment* SkOpSegment::nextChase(SkOpSpanBase** startPtr, int* stepPtr, SkOpS pan** minPtr,
1475 SkOpSpanBase** last) const {
1476 SkOpSpanBase* origStart = *startPtr;
4319 int step = *stepPtr; 1477 int step = *stepPtr;
4320 int end = nextExactSpan(origIndex, step); 1478 SkOpSpanBase* endSpan = step > 0 ? origStart->upCast()->next() : origStart-> prev();
4321 SkASSERT(end >= 0); 1479 SkASSERT(endSpan);
4322 const SkOpSpan& endSpan = this->span(end); 1480 SkOpAngle* angle = step > 0 ? endSpan->fromAngle() : endSpan->upCast()->toAn gle();
4323 SkOpAngle* angle = step > 0 ? endSpan.fFromAngle : endSpan.fToAngle; 1481 SkOpSpanBase* foundSpan;
4324 int foundIndex; 1482 SkOpSpanBase* otherEnd;
4325 int otherEnd;
4326 SkOpSegment* other; 1483 SkOpSegment* other;
4327 if (angle == NULL) { 1484 if (angle == NULL) {
4328 if (endSpan.fT != 0 && endSpan.fT != 1) { 1485 if (endSpan->t() != 0 && endSpan->t() != 1) {
4329 return NULL; 1486 return NULL;
4330 } 1487 }
4331 other = endSpan.fOther; 1488 SkOpPtT* otherPtT = endSpan->ptT()->next();
4332 foundIndex = endSpan.fOtherIndex; 1489 other = otherPtT->segment();
4333 otherEnd = other->nextExactSpan(foundIndex, step); 1490 foundSpan = otherPtT->span();
1491 otherEnd = step > 0 ? foundSpan->upCast()->next() : foundSpan->prev();
4334 } else { 1492 } else {
4335 int loopCount = angle->loopCount(); 1493 int loopCount = angle->loopCount();
4336 if (loopCount > 2) { 1494 if (loopCount > 2) {
4337 return set_last(last, &endSpan); 1495 return set_last(last, endSpan);
4338 } 1496 }
4339 const SkOpAngle* next = angle->next(); 1497 const SkOpAngle* next = angle->next();
4340 if (NULL == next) { 1498 if (NULL == next) {
4341 return NULL; 1499 return NULL;
4342 } 1500 }
4343 if (angle->sign() != next->sign()) {
4344 #if DEBUG_WINDING 1501 #if DEBUG_WINDING
1502 if (angle->sign() != next->sign() && !angle->segment()->contour()->isXor ()
1503 && !next->segment()->contour()->isXor()) {
4345 SkDebugf("%s mismatched signs\n", __FUNCTION__); 1504 SkDebugf("%s mismatched signs\n", __FUNCTION__);
1505 }
4346 #endif 1506 #endif
4347 // return set_last(last, &endSpan);
4348 }
4349 other = next->segment(); 1507 other = next->segment();
4350 foundIndex = end = next->start(); 1508 foundSpan = endSpan = next->start();
4351 otherEnd = next->end(); 1509 otherEnd = next->end();
4352 } 1510 }
4353 int foundStep = foundIndex < otherEnd ? 1 : -1; 1511 int foundStep = foundSpan->step(otherEnd);
4354 if (*stepPtr != foundStep) { 1512 if (*stepPtr != foundStep) {
4355 return set_last(last, &endSpan); 1513 return set_last(last, endSpan);
4356 } 1514 }
4357 SkASSERT(*indexPtr >= 0); 1515 SkASSERT(*startPtr);
4358 if (otherEnd < 0) { 1516 if (!otherEnd) {
4359 return NULL; 1517 return NULL;
4360 } 1518 }
4361 // SkASSERT(otherEnd >= 0); 1519 // SkASSERT(otherEnd >= 0);
4362 #if 1 1520 SkOpSpan* origMin = step < 0 ? origStart->prev() : origStart->upCast();
4363 int origMin = origIndex + (step < 0 ? step : 0); 1521 SkOpSpan* foundMin = foundSpan->starter(otherEnd);
4364 const SkOpSpan& orig = this->span(origMin); 1522 if (foundMin->windValue() != origMin->windValue()
4365 #endif 1523 || foundMin->oppValue() != origMin->oppValue()) {
4366 int foundMin = SkMin32(foundIndex, otherEnd); 1524 return set_last(last, endSpan);
4367 #if 1
4368 const SkOpSpan& found = other->span(foundMin);
4369 if (found.fWindValue != orig.fWindValue || found.fOppValue != orig.fOppValue ) {
4370 return set_last(last, &endSpan);
4371 } 1525 }
4372 #endif 1526 *startPtr = foundSpan;
4373 *indexPtr = foundIndex;
4374 *stepPtr = foundStep; 1527 *stepPtr = foundStep;
4375 if (minPtr) { 1528 if (minPtr) {
4376 *minPtr = foundMin; 1529 *minPtr = foundMin;
4377 } 1530 }
4378 return other; 1531 return other;
4379 } 1532 }
4380 1533
4381 // This has callers for two different situations: one establishes the end 1534 static void clear_visited(SkOpSpan* span) {
4382 // of the current span, and one establishes the beginning of the next span 1535 // reset visited flag back to false
4383 // (thus the name). When this is looking for the end of the current span, 1536 do {
4384 // coincidence is found when the beginning Ts contain -step and the end 1537 SkOpPtT* ptT = span->ptT(), * stopPtT = ptT;
4385 // contains step. When it is looking for the beginning of the next, the 1538 while ((ptT = ptT->next()) != stopPtT) {
4386 // first Ts found can be ignored and the last Ts should contain -step. 1539 SkOpSegment* opp = ptT->segment();
4387 // OPTIMIZATION: probably should split into two functions 1540 opp->resetVisited();
4388 int SkOpSegment::nextSpan(int from, int step) const {
4389 const SkOpSpan& fromSpan = fTs[from];
4390 int count = fTs.count();
4391 int to = from;
4392 while (step > 0 ? ++to < count : --to >= 0) {
4393 const SkOpSpan& span = fTs[to];
4394 if (approximately_zero(span.fT - fromSpan.fT)) {
4395 continue;
4396 } 1541 }
4397 return to; 1542 } while ((span = span->next()->upCastable()));
1543 }
1544
1545 // look for pairs of undetected coincident curves
1546 // assumes that segments going in have visited flag clear
1547 // curve/curve intersection should now do a pretty good job of finding coinciden t runs so
1548 // this may be only be necessary for line/curve pairs -- so skip unless this is a line and the
1549 // the opp is not a line
1550 void SkOpSegment::missingCoincidence(SkOpCoincidence* coincidences, SkChunkAlloc * allocator) {
1551 if (this->verb() != SkPath::kLine_Verb) {
1552 return;
4398 } 1553 }
4399 return -1; 1554 SkOpSpan* prior = NULL;
4400 } 1555 SkOpSpan* span = &fHead;
4401 1556 do {
4402 // FIXME 1557 SkOpPtT* ptT = span->ptT(), * spanStopPtT = ptT;
4403 // this returns at any difference in T, vs. a preset minimum. It may be 1558 SkASSERT(ptT->span() == span);
4404 // that all callers to nextSpan should use this instead. 1559 while ((ptT = ptT->next()) != spanStopPtT) {
4405 int SkOpSegment::nextExactSpan(int from, int step) const { 1560 SkOpSegment* opp = ptT->span()->segment();
4406 int to = from; 1561 if (opp->setVisited()) {
4407 if (step < 0) { 1562 continue;
4408 const SkOpSpan& fromSpan = fTs[from]; 1563 }
4409 while (--to >= 0) { 1564 if (opp->verb() == SkPath::kLine_Verb) {
4410 const SkOpSpan& span = fTs[to]; 1565 continue;
4411 if (precisely_negative(fromSpan.fT - span.fT) || span.fTiny) { 1566 }
4412 continue; 1567 if (span->containsCoincidence(opp)) { // FIXME: this assumes that if the opposite
4413 } 1568 // segment is coincident then no more coincidence
4414 return to; 1569 // needs to be detected. This may not be true.
1570 continue;
1571 }
1572 if (span->containsCoinEnd(opp)) {
1573 continue;
1574 }
1575 // if already visited and visited again, check for coin
1576 if (span == &fHead) {
1577 continue;
1578 }
1579 SkOpPtT* priorPtT = NULL, * priorStopPtT;
1580 // find prior span containing opp segment
1581 SkOpSegment* priorOpp = NULL;
1582 prior = span;
1583 while (!priorOpp && (prior = prior->prev())) {
1584 priorStopPtT = priorPtT = prior->ptT();
1585 while ((priorPtT = priorPtT->next()) != priorStopPtT) {
1586 SkOpSegment* segment = priorPtT->span()->segment();
1587 if (segment == opp) {
1588 priorOpp = opp;
1589 break;
1590 }
1591 }
1592 }
1593 if (!priorOpp) {
1594 continue;
1595 }
1596 SkOpPtT* oppStart = prior->ptT();
1597 SkOpPtT* oppEnd = span->ptT();
1598 bool swapped = priorPtT->fT > ptT->fT;
1599 if (swapped) {
1600 SkTSwap(priorPtT, ptT);
1601 SkTSwap(oppStart, oppEnd);
1602 }
1603 bool flipped = oppStart->fT > oppEnd->fT;
1604 bool coincident;
1605 if (coincidences->contains(priorPtT, ptT, oppStart, oppEnd, flipped) ) {
1606 goto swapBack;
1607 }
1608 {
1609 // average t, find mid pt
1610 double midT = (prior->t() + span->t()) / 2;
1611 SkPoint midPt = this->ptAtT(midT);
1612 coincident = true;
1613 // if the mid pt is not near either end pt, project perpendicula r through opp seg
1614 if (!SkDPoint::ApproximatelyEqual(priorPtT->fPt, midPt)
1615 && !SkDPoint::ApproximatelyEqual(ptT->fPt, midPt)) {
1616 coincident = false;
1617 SkIntersections i;
1618 int ptCount = SkPathOpsVerbToPoints(this->verb());
1619 SkVector dxdy = (*CurveSlopeAtT[ptCount])(pts(), midT);
1620 SkDLine ray = {{{midPt.fX, midPt.fY},
1621 {midPt.fX + dxdy.fY, midPt.fY - dxdy.fX}}};
1622 int oppPtCount = SkPathOpsVerbToPoints(opp->verb());
1623 (*CurveIntersectRay[oppPtCount])(opp->pts(), ray, &i);
1624 // measure distance and see if it's small enough to denote c oincidence
1625 for (int index = 0; index < i.used(); ++index) {
1626 SkDPoint oppPt = i.pt(index);
1627 if (oppPt.approximatelyEqual(midPt)) {
1628 SkVector oppDxdy = (*CurveSlopeAtT[oppPtCount])(opp- >pts(),
1629 i[index][0]);
1630 oppDxdy.normalize();
1631 dxdy.normalize();
1632 SkScalar flatness = SkScalarAbs(dxdy.cross(oppDxdy) / FLT_EPSILON);
1633 coincident |= flatness < 5000; // FIXME: replace wi th tuned value
1634 }
1635 }
1636 }
1637 }
1638 if (coincident) {
1639 // mark coincidence
1640 coincidences->add(priorPtT, ptT, oppStart, oppEnd, allocator);
1641 clear_visited(&fHead);
1642 missingCoincidence(coincidences, allocator);
1643 return;
1644 }
1645 swapBack:
1646 if (swapped) {
1647 SkTSwap(priorPtT, ptT);
1648 }
4415 } 1649 }
4416 } else { 1650 } while ((span = span->next()->upCastable()));
4417 while (fTs[from].fTiny) { 1651 clear_visited(&fHead);
4418 from++; 1652 }
1653
1654 // Move nearby t values and pts so they all hang off the same span. Alignment ha ppens later.
1655 bool SkOpSegment::moveNearby() {
1656 debugValidate();
1657 SkOpSpanBase* spanS = &fHead;
1658 do {
1659 SkOpSpanBase* test = spanS->upCast()->next();
1660 SkOpSpanBase* next;
1661 if (spanS->contains(test)) {
1662 if (!test->final()) {
1663 test->upCast()->detach(spanS->ptT());
1664 continue;
1665 } else if (spanS != &fHead) {
1666 spanS->upCast()->detach(test->ptT());
1667 spanS = test;
1668 continue;
1669 }
4419 } 1670 }
4420 const SkOpSpan& fromSpan = fTs[from]; 1671 do { // iterate through all spans associated with start
4421 int count = fTs.count(); 1672 SkOpPtT* startBase = spanS->ptT();
4422 while (++to < count) { 1673 next = test->final() ? NULL : test->upCast()->next();
4423 const SkOpSpan& span = fTs[to]; 1674 do {
4424 if (precisely_negative(span.fT - fromSpan.fT)) { 1675 SkOpPtT* testBase = test->ptT();
4425 continue; 1676 do {
4426 } 1677 if (startBase == testBase) {
4427 return to; 1678 goto checkNextSpan;
4428 } 1679 }
1680 if (testBase->duplicate()) {
1681 continue;
1682 }
1683 if (this->match(startBase, testBase->segment(), testBase->fT , testBase->fPt)) {
1684 if (test == &this->fTail) {
1685 if (spanS == &fHead) {
1686 debugValidate();
1687 return true; // if this span has collapsed, rem ove it from parent
1688 }
1689 this->fTail.merge(spanS->upCast());
1690 debugValidate();
1691 return true;
1692 }
1693 spanS->merge(test->upCast());
1694 spanS->upCast()->setNext(next);
1695 goto checkNextSpan;
1696 }
1697 } while ((testBase = testBase->next()) != test->ptT());
1698 } while ((startBase = startBase->next()) != spanS->ptT());
1699 checkNextSpan:
1700 ;
1701 } while ((test = next));
1702 spanS = spanS->upCast()->next();
1703 } while (!spanS->final());
1704 debugValidate();
1705 return true;
1706 }
1707
1708 bool SkOpSegment::operand() const {
1709 return fContour->operand();
1710 }
1711
1712 bool SkOpSegment::oppXor() const {
1713 return fContour->oppXor();
1714 }
1715
1716 bool SkOpSegment::ptsDisjoint(double t1, const SkPoint& pt1, double t2, const Sk Point& pt2) const {
1717 if (fVerb == SkPath::kLine_Verb) {
1718 return false;
4429 } 1719 }
4430 return -1; 1720 // quads (and cubics) can loop back to nearly a line so that an opposite cur ve
4431 } 1721 // hits in two places with very different t values.
4432 1722 // OPTIMIZATION: curves could be preflighted so that, for example, something like
4433 void SkOpSegment::pinT(const SkPoint& pt, double* t) { 1723 // 'controls contained by ends' could avoid this check for common curves
4434 if (pt == fPts[0]) { 1724 // 'ends are extremes in x or y' is cheaper to compute and real-world common
4435 *t = 0; 1725 // on the other hand, the below check is relatively inexpensive
4436 } 1726 double midT = (t1 + t2) / 2;
4437 int count = SkPathOpsVerbToPoints(fVerb); 1727 SkPoint midPt = this->ptAtT(midT);
4438 if (pt == fPts[count]) { 1728 double seDistSq = SkTMax(pt1.distanceToSqd(pt2) * 2, FLT_EPSILON * 2);
4439 *t = 1; 1729 return midPt.distanceToSqd(pt1) > seDistSq || midPt.distanceToSqd(pt2) > seD istSq;
4440 } 1730 }
4441 } 1731
4442 1732 void SkOpSegment::setUpWindings(SkOpSpanBase* start, SkOpSpanBase* end, int* sum MiWinding,
4443 bool SkOpSegment::reversePoints(const SkPoint& p1, const SkPoint& p2) const { 1733 int* maxWinding, int* sumWinding) {
4444 SkASSERT(p1 != p2); 1734 int deltaSum = SpanSign(start, end);
4445 int spanCount = count(); 1735 *maxWinding = *sumMiWinding;
4446 int p1IndexMin = -1; 1736 *sumWinding = *sumMiWinding -= deltaSum;
4447 int p2IndexMax = spanCount; 1737 SkASSERT(!DEBUG_LIMIT_WIND_SUM || abs(*sumWinding) <= DEBUG_LIMIT_WIND_SUM);
4448 for (int index = 0; index < spanCount; ++index) { 1738 }
4449 const SkOpSpan& span = fTs[index]; 1739
4450 if (span.fPt == p1) { 1740 void SkOpSegment::setUpWindings(SkOpSpanBase* start, SkOpSpanBase* end, int* sum MiWinding,
4451 if (p1IndexMin < 0) { 1741 int* sumSuWinding, int* maxWinding, int* sumWinding, int* oppMaxWinding,
4452 p1IndexMin = index; 1742 int* oppSumWinding) {
4453 } 1743 int deltaSum = SpanSign(start, end);
4454 } else if (span.fPt == p2) { 1744 int oppDeltaSum = OppSign(start, end);
4455 p2IndexMax = index;
4456 }
4457 }
4458 return p1IndexMin > p2IndexMax;
4459 }
4460
4461 void SkOpSegment::setCoincidentRange(const SkPoint& startPt, const SkPoint& endP t,
4462 SkOpSegment* other) {
4463 int count = this->count();
4464 for (int index = 0; index < count; ++index) {
4465 SkOpSpan &span = fTs[index];
4466 if ((startPt == span.fPt || endPt == span.fPt) && other == span.fOther) {
4467 span.fCoincident = true;
4468 }
4469 }
4470 }
4471
4472 void SkOpSegment::setUpWindings(int index, int endIndex, int* sumMiWinding, int* sumSuWinding,
4473 int* maxWinding, int* sumWinding, int* oppMaxWinding, int* oppSumWinding ) {
4474 int deltaSum = spanSign(index, endIndex);
4475 int oppDeltaSum = oppSign(index, endIndex);
4476 if (operand()) { 1745 if (operand()) {
4477 *maxWinding = *sumSuWinding; 1746 *maxWinding = *sumSuWinding;
4478 *sumWinding = *sumSuWinding -= deltaSum; 1747 *sumWinding = *sumSuWinding -= deltaSum;
4479 *oppMaxWinding = *sumMiWinding; 1748 *oppMaxWinding = *sumMiWinding;
4480 *oppSumWinding = *sumMiWinding -= oppDeltaSum; 1749 *oppSumWinding = *sumMiWinding -= oppDeltaSum;
4481 } else { 1750 } else {
4482 *maxWinding = *sumMiWinding; 1751 *maxWinding = *sumMiWinding;
4483 *sumWinding = *sumMiWinding -= deltaSum; 1752 *sumWinding = *sumMiWinding -= deltaSum;
4484 *oppMaxWinding = *sumSuWinding; 1753 *oppMaxWinding = *sumSuWinding;
4485 *oppSumWinding = *sumSuWinding -= oppDeltaSum; 1754 *oppSumWinding = *sumSuWinding -= oppDeltaSum;
4486 } 1755 }
4487 #if DEBUG_LIMIT_WIND_SUM 1756 SkASSERT(!DEBUG_LIMIT_WIND_SUM || abs(*sumWinding) <= DEBUG_LIMIT_WIND_SUM);
4488 SkASSERT(abs(*sumWinding) <= DEBUG_LIMIT_WIND_SUM); 1757 SkASSERT(!DEBUG_LIMIT_WIND_SUM || abs(*oppSumWinding) <= DEBUG_LIMIT_WIND_SU M);
4489 SkASSERT(abs(*oppSumWinding) <= DEBUG_LIMIT_WIND_SUM);
4490 #endif
4491 }
4492
4493 void SkOpSegment::setUpWindings(int index, int endIndex, int* sumMiWinding,
4494 int* maxWinding, int* sumWinding) {
4495 int deltaSum = spanSign(index, endIndex);
4496 *maxWinding = *sumMiWinding;
4497 *sumWinding = *sumMiWinding -= deltaSum;
4498 #if DEBUG_LIMIT_WIND_SUM
4499 SkASSERT(abs(*sumWinding) <= DEBUG_LIMIT_WIND_SUM);
4500 #endif
4501 } 1758 }
4502 1759
4503 void SkOpSegment::sortAngles() { 1760 void SkOpSegment::sortAngles() {
4504 int spanCount = fTs.count(); 1761 SkOpSpanBase* span = &this->fHead;
4505 if (spanCount <= 2) {
4506 return;
4507 }
4508 int index = 0;
4509 do { 1762 do {
4510 SkOpAngle* fromAngle = fTs[index].fFromAngle; 1763 SkOpAngle* fromAngle = span->fromAngle();
4511 SkOpAngle* toAngle = fTs[index].fToAngle; 1764 SkOpAngle* toAngle = span->final() ? NULL : span->upCast()->toAngle();
4512 if (!fromAngle && !toAngle) { 1765 if (!fromAngle && !toAngle) {
4513 index += 1;
4514 continue; 1766 continue;
4515 } 1767 }
4516 SkOpAngle* baseAngle = NULL;
4517 if (fromAngle) {
4518 baseAngle = fromAngle;
4519 if (inLoop(baseAngle, spanCount, &index)) {
4520 continue;
4521 }
4522 }
4523 #if DEBUG_ANGLE 1768 #if DEBUG_ANGLE
4524 bool wroteAfterHeader = false; 1769 bool wroteAfterHeader = false;
4525 #endif 1770 #endif
4526 if (toAngle) { 1771 SkOpAngle* baseAngle = fromAngle;
4527 if (!baseAngle) { 1772 if (fromAngle && toAngle) {
4528 baseAngle = toAngle;
4529 if (inLoop(baseAngle, spanCount, &index)) {
4530 continue;
4531 }
4532 } else {
4533 SkDEBUGCODE(int newIndex = index);
4534 SkASSERT(!inLoop(baseAngle, spanCount, &newIndex) && newIndex == index);
4535 #if DEBUG_ANGLE 1773 #if DEBUG_ANGLE
4536 SkDebugf("%s [%d] tStart=%1.9g [%d]\n", __FUNCTION__, debugID(), fTs[index].fT, 1774 SkDebugf("%s [%d] tStart=%1.9g [%d]\n", __FUNCTION__, debugID(), spa n->t(),
4537 index); 1775 span->debugID());
4538 wroteAfterHeader = true; 1776 wroteAfterHeader = true;
4539 #endif 1777 #endif
4540 baseAngle->insert(toAngle); 1778 fromAngle->insert(toAngle);
1779 } else if (!fromAngle) {
1780 baseAngle = toAngle;
1781 }
1782 SkOpPtT* ptT = span->ptT(), * stopPtT = ptT;
1783 do {
1784 SkOpSpanBase* oSpan = ptT->span();
1785 if (oSpan == span) {
1786 continue;
4541 } 1787 }
4542 } 1788 SkOpAngle* oAngle = oSpan->fromAngle();
4543 SkOpAngle* nextFrom, * nextTo;
4544 int firstIndex = index;
4545 do {
4546 SkOpSpan& span = fTs[index];
4547 SkOpSegment* other = span.fOther;
4548 SkOpSpan& oSpan = other->fTs[span.fOtherIndex];
4549 SkOpAngle* oAngle = oSpan.fFromAngle;
4550 if (oAngle) { 1789 if (oAngle) {
4551 #if DEBUG_ANGLE 1790 #if DEBUG_ANGLE
4552 if (!wroteAfterHeader) { 1791 if (!wroteAfterHeader) {
4553 SkDebugf("%s [%d] tStart=%1.9g [%d]\n", __FUNCTION__, debugI D(), fTs[index].fT, 1792 SkDebugf("%s [%d] tStart=%1.9g [%d]\n", __FUNCTION__, debugI D(),
4554 index); 1793 span->t(), span->debugID());
4555 wroteAfterHeader = true; 1794 wroteAfterHeader = true;
4556 } 1795 }
4557 #endif 1796 #endif
4558 if (!oAngle->loopContains(*baseAngle)) { 1797 if (!oAngle->loopContains(baseAngle)) {
4559 baseAngle->insert(oAngle); 1798 baseAngle->insert(oAngle);
4560 } 1799 }
4561 } 1800 }
4562 oAngle = oSpan.fToAngle; 1801 if (!oSpan->final()) {
4563 if (oAngle) { 1802 oAngle = oSpan->upCast()->toAngle();
1803 if (oAngle) {
4564 #if DEBUG_ANGLE 1804 #if DEBUG_ANGLE
4565 if (!wroteAfterHeader) { 1805 if (!wroteAfterHeader) {
4566 SkDebugf("%s [%d] tStart=%1.9g [%d]\n", __FUNCTION__, debugI D(), fTs[index].fT, 1806 SkDebugf("%s [%d] tStart=%1.9g [%d]\n", __FUNCTION__, de bugID(),
4567 index); 1807 span->t(), span->debugID());
4568 wroteAfterHeader = true; 1808 wroteAfterHeader = true;
4569 } 1809 }
4570 #endif 1810 #endif
4571 if (!oAngle->loopContains(*baseAngle)) { 1811 if (!oAngle->loopContains(baseAngle)) {
4572 baseAngle->insert(oAngle); 1812 baseAngle->insert(oAngle);
1813 }
4573 } 1814 }
4574 } 1815 }
4575 if (++index == spanCount) { 1816 } while ((ptT = ptT->next()) != stopPtT);
4576 break; 1817 if (baseAngle->loopCount() == 1) {
1818 span->setFromAngle(NULL);
1819 if (toAngle) {
1820 span->upCast()->setToAngle(NULL);
4577 } 1821 }
4578 nextFrom = fTs[index].fFromAngle;
4579 nextTo = fTs[index].fToAngle;
4580 } while (fromAngle == nextFrom && toAngle == nextTo);
4581 if (baseAngle && baseAngle->loopCount() == 1) {
4582 index = firstIndex;
4583 do {
4584 SkOpSpan& span = fTs[index];
4585 span.fFromAngle = span.fToAngle = NULL;
4586 if (++index == spanCount) {
4587 break;
4588 }
4589 nextFrom = fTs[index].fFromAngle;
4590 nextTo = fTs[index].fToAngle;
4591 } while (fromAngle == nextFrom && toAngle == nextTo);
4592 baseAngle = NULL; 1822 baseAngle = NULL;
4593 } 1823 }
4594 #if DEBUG_SORT 1824 #if DEBUG_SORT
4595 SkASSERT(!baseAngle || baseAngle->loopCount() > 1); 1825 SkASSERT(!baseAngle || baseAngle->loopCount() > 1);
4596 #endif 1826 #endif
4597 } while (index < spanCount); 1827 } while (!span->final() && (span = span->upCast()->next()));
4598 } 1828 }
4599 1829
4600 // return true if midpoints were computed 1830 // return true if midpoints were computed
4601 bool SkOpSegment::subDivide(int start, int end, SkPoint edge[4]) const { 1831 bool SkOpSegment::subDivide(const SkOpSpanBase* start, const SkOpSpanBase* end,
1832 SkPoint edge[4]) const {
4602 SkASSERT(start != end); 1833 SkASSERT(start != end);
4603 edge[0] = fTs[start].fPt; 1834 const SkOpPtT& startPtT = *start->ptT();
1835 const SkOpPtT& endPtT = *end->ptT();
1836 edge[0] = startPtT.fPt;
4604 int points = SkPathOpsVerbToPoints(fVerb); 1837 int points = SkPathOpsVerbToPoints(fVerb);
4605 edge[points] = fTs[end].fPt; 1838 edge[points] = endPtT.fPt;
4606 if (fVerb == SkPath::kLine_Verb) { 1839 if (fVerb == SkPath::kLine_Verb) {
4607 return false; 1840 return false;
4608 } 1841 }
4609 double startT = fTs[start].fT; 1842 double startT = startPtT.fT;
4610 double endT = fTs[end].fT; 1843 double endT = endPtT.fT;
4611 if ((startT == 0 || endT == 0) && (startT == 1 || endT == 1)) { 1844 if ((startT == 0 || endT == 0) && (startT == 1 || endT == 1)) {
4612 // don't compute midpoints if we already have them 1845 // don't compute midpoints if we already have them
4613 if (fVerb == SkPath::kQuad_Verb) { 1846 if (fVerb == SkPath::kQuad_Verb) {
4614 edge[1] = fPts[1]; 1847 edge[1] = fPts[1];
4615 return false; 1848 return false;
4616 } 1849 }
4617 SkASSERT(fVerb == SkPath::kCubic_Verb); 1850 SkASSERT(fVerb == SkPath::kCubic_Verb);
4618 if (start < end) { 1851 if (start < end) {
4619 edge[1] = fPts[1]; 1852 edge[1] = fPts[1];
4620 edge[2] = fPts[2]; 1853 edge[2] = fPts[2];
4621 return false; 1854 return false;
4622 } 1855 }
4623 edge[1] = fPts[2]; 1856 edge[1] = fPts[2];
4624 edge[2] = fPts[1]; 1857 edge[2] = fPts[1];
4625 return false; 1858 return false;
4626 } 1859 }
4627 const SkDPoint sub[2] = {{ edge[0].fX, edge[0].fY}, {edge[points].fX, edge[p oints].fY }}; 1860 const SkDPoint sub[2] = {{ edge[0].fX, edge[0].fY}, {edge[points].fX, edge[p oints].fY }};
4628 if (fVerb == SkPath::kQuad_Verb) { 1861 if (fVerb == SkPath::kQuad_Verb) {
4629 edge[1] = SkDQuad::SubDivide(fPts, sub[0], sub[1], startT, endT).asSkPoi nt(); 1862 edge[1] = SkDQuad::SubDivide(fPts, sub[0], sub[1], startT, endT).asSkPoi nt();
4630 } else { 1863 } else {
4631 SkASSERT(fVerb == SkPath::kCubic_Verb); 1864 SkASSERT(fVerb == SkPath::kCubic_Verb);
4632 SkDPoint ctrl[2]; 1865 SkDPoint ctrl[2];
4633 SkDCubic::SubDivide(fPts, sub[0], sub[1], startT, endT, ctrl); 1866 SkDCubic::SubDivide(fPts, sub[0], sub[1], startT, endT, ctrl);
4634 edge[1] = ctrl[0].asSkPoint(); 1867 edge[1] = ctrl[0].asSkPoint();
4635 edge[2] = ctrl[1].asSkPoint(); 1868 edge[2] = ctrl[1].asSkPoint();
4636 } 1869 }
4637 return true; 1870 return true;
4638 } 1871 }
4639 1872
4640 // return true if midpoints were computed 1873 bool SkOpSegment::subDivide(const SkOpSpanBase* start, const SkOpSpanBase* end,
4641 bool SkOpSegment::subDivide(int start, int end, SkDCubic* result) const { 1874 SkDCubic* result) const {
4642 SkASSERT(start != end); 1875 SkASSERT(start != end);
4643 (*result)[0].set(fTs[start].fPt); 1876 const SkOpPtT& startPtT = *start->ptT();
1877 const SkOpPtT& endPtT = *end->ptT();
1878 (*result)[0].set(startPtT.fPt);
4644 int points = SkPathOpsVerbToPoints(fVerb); 1879 int points = SkPathOpsVerbToPoints(fVerb);
4645 (*result)[points].set(fTs[end].fPt); 1880 (*result)[points].set(endPtT.fPt);
4646 if (fVerb == SkPath::kLine_Verb) { 1881 if (fVerb == SkPath::kLine_Verb) {
4647 return false; 1882 return false;
4648 } 1883 }
4649 double startT = fTs[start].fT; 1884 double startT = startPtT.fT;
4650 double endT = fTs[end].fT; 1885 double endT = endPtT.fT;
4651 if ((startT == 0 || endT == 0) && (startT == 1 || endT == 1)) { 1886 if ((startT == 0 || endT == 0) && (startT == 1 || endT == 1)) {
4652 // don't compute midpoints if we already have them 1887 // don't compute midpoints if we already have them
4653 if (fVerb == SkPath::kQuad_Verb) { 1888 if (fVerb == SkPath::kQuad_Verb) {
4654 (*result)[1].set(fPts[1]); 1889 (*result)[1].set(fPts[1]);
4655 return false; 1890 return false;
4656 } 1891 }
4657 SkASSERT(fVerb == SkPath::kCubic_Verb); 1892 SkASSERT(fVerb == SkPath::kCubic_Verb);
4658 if (start < end) { 1893 if (startT == 0) {
4659 (*result)[1].set(fPts[1]); 1894 (*result)[1].set(fPts[1]);
4660 (*result)[2].set(fPts[2]); 1895 (*result)[2].set(fPts[2]);
4661 return false; 1896 return false;
4662 } 1897 }
4663 (*result)[1].set(fPts[2]); 1898 (*result)[1].set(fPts[2]);
4664 (*result)[2].set(fPts[1]); 1899 (*result)[2].set(fPts[1]);
4665 return false; 1900 return false;
4666 } 1901 }
4667 if (fVerb == SkPath::kQuad_Verb) { 1902 if (fVerb == SkPath::kQuad_Verb) {
4668 (*result)[1] = SkDQuad::SubDivide(fPts, (*result)[0], (*result)[2], star tT, endT); 1903 (*result)[1] = SkDQuad::SubDivide(fPts, (*result)[0], (*result)[2], star tT, endT);
4669 } else { 1904 } else {
4670 SkASSERT(fVerb == SkPath::kCubic_Verb); 1905 SkASSERT(fVerb == SkPath::kCubic_Verb);
4671 SkDCubic::SubDivide(fPts, (*result)[0], (*result)[3], startT, endT, &(*r esult)[1]); 1906 SkDCubic::SubDivide(fPts, (*result)[0], (*result)[3], startT, endT, &(*r esult)[1]);
4672 } 1907 }
4673 return true; 1908 return true;
4674 } 1909 }
4675 1910
4676 void SkOpSegment::subDivideBounds(int start, int end, SkPathOpsBounds* bounds) c onst { 1911 void SkOpSegment::subDivideBounds(const SkOpSpanBase* start, const SkOpSpanBase* end,
1912 SkPathOpsBounds* bounds) const {
4677 SkPoint edge[4]; 1913 SkPoint edge[4];
4678 subDivide(start, end, edge); 1914 subDivide(start, end, edge);
4679 (bounds->*SetCurveBounds[SkPathOpsVerbToPoints(fVerb)])(edge); 1915 (bounds->*SetCurveBounds[SkPathOpsVerbToPoints(fVerb)])(edge);
4680 } 1916 }
4681 1917
4682 void SkOpSegment::TrackOutsidePair(SkTArray<SkPoint, true>* outsidePts, const Sk Point& endPt, 1918 void SkOpSegment::undoneSpan(SkOpSpanBase** start, SkOpSpanBase** end) {
4683 const SkPoint& startPt) { 1919 SkOpSpan* span = this->head();
4684 int outCount = outsidePts->count(); 1920 do {
4685 if (outCount == 0 || endPt != (*outsidePts)[outCount - 2]) { 1921 if (!span->done()) {
4686 outsidePts->push_back(endPt); 1922 break;
4687 outsidePts->push_back(startPt); 1923 }
4688 } 1924 } while ((span = span->next()->upCastable()));
1925 SkASSERT(span);
1926 *start = span;
1927 *end = span->next();
4689 } 1928 }
4690 1929
4691 void SkOpSegment::TrackOutside(SkTArray<SkPoint, true>* outsidePts, const SkPoin t& startPt) { 1930 int SkOpSegment::updateOppWinding(const SkOpSpanBase* start, const SkOpSpanBase* end) const {
4692 int outCount = outsidePts->count(); 1931 const SkOpSpan* lesser = start->starter(end);
4693 if (outCount == 0 || startPt != (*outsidePts)[outCount - 1]) { 1932 int oppWinding = lesser->oppSum();
4694 outsidePts->push_back(startPt); 1933 int oppSpanWinding = SkOpSegment::OppSign(start, end);
4695 }
4696 }
4697
4698 void SkOpSegment::undoneSpan(int* start, int* end) {
4699 int tCount = fTs.count();
4700 int index;
4701 for (index = 0; index < tCount; ++index) {
4702 if (!fTs[index].fDone) {
4703 break;
4704 }
4705 }
4706 SkASSERT(index < tCount - 1);
4707 *start = index;
4708 double startT = fTs[index].fT;
4709 while (approximately_negative(fTs[++index].fT - startT))
4710 SkASSERT(index < tCount);
4711 SkASSERT(index < tCount);
4712 *end = index;
4713 }
4714
4715 int SkOpSegment::updateOppWinding(int index, int endIndex) const {
4716 int lesser = SkMin32(index, endIndex);
4717 int oppWinding = oppSum(lesser);
4718 int oppSpanWinding = oppSign(index, endIndex);
4719 if (oppSpanWinding && UseInnerWinding(oppWinding - oppSpanWinding, oppWindin g) 1934 if (oppSpanWinding && UseInnerWinding(oppWinding - oppSpanWinding, oppWindin g)
4720 && oppWinding != SK_MaxS32) { 1935 && oppWinding != SK_MaxS32) {
4721 oppWinding -= oppSpanWinding; 1936 oppWinding -= oppSpanWinding;
4722 } 1937 }
4723 return oppWinding; 1938 return oppWinding;
4724 } 1939 }
4725 1940
4726 int SkOpSegment::updateOppWinding(const SkOpAngle* angle) const { 1941 int SkOpSegment::updateOppWinding(const SkOpAngle* angle) const {
4727 int startIndex = angle->start(); 1942 const SkOpSpanBase* startSpan = angle->start();
4728 int endIndex = angle->end(); 1943 const SkOpSpanBase* endSpan = angle->end();
4729 return updateOppWinding(endIndex, startIndex); 1944 return updateOppWinding(endSpan, startSpan);
4730 } 1945 }
4731 1946
4732 int SkOpSegment::updateOppWindingReverse(const SkOpAngle* angle) const { 1947 int SkOpSegment::updateOppWindingReverse(const SkOpAngle* angle) const {
4733 int startIndex = angle->start(); 1948 const SkOpSpanBase* startSpan = angle->start();
4734 int endIndex = angle->end(); 1949 const SkOpSpanBase* endSpan = angle->end();
4735 return updateOppWinding(startIndex, endIndex); 1950 return updateOppWinding(startSpan, endSpan);
4736 } 1951 }
4737 1952
4738 int SkOpSegment::updateWinding(int index, int endIndex) const { 1953 int SkOpSegment::updateWinding(const SkOpSpanBase* start, const SkOpSpanBase* en d) const {
4739 int lesser = SkMin32(index, endIndex); 1954 const SkOpSpan* lesser = start->starter(end);
4740 int winding = windSum(lesser); 1955 int winding = lesser->windSum();
4741 if (winding == SK_MinS32) { 1956 if (winding == SK_MinS32) {
4742 return winding; 1957 return winding;
4743 } 1958 }
4744 int spanWinding = spanSign(index, endIndex); 1959 int spanWinding = SkOpSegment::SpanSign(start, end);
4745 if (winding && UseInnerWinding(winding - spanWinding, winding) 1960 if (winding && UseInnerWinding(winding - spanWinding, winding)
4746 && winding != SK_MaxS32) { 1961 && winding != SK_MaxS32) {
4747 winding -= spanWinding; 1962 winding -= spanWinding;
4748 } 1963 }
4749 return winding; 1964 return winding;
4750 } 1965 }
4751 1966
4752 int SkOpSegment::updateWinding(const SkOpAngle* angle) const { 1967 int SkOpSegment::updateWinding(const SkOpAngle* angle) const {
4753 int startIndex = angle->start(); 1968 const SkOpSpanBase* startSpan = angle->start();
4754 int endIndex = angle->end(); 1969 const SkOpSpanBase* endSpan = angle->end();
4755 return updateWinding(endIndex, startIndex); 1970 return updateWinding(endSpan, startSpan);
4756 }
4757
4758 int SkOpSegment::updateWindingReverse(int index, int endIndex) const {
4759 int lesser = SkMin32(index, endIndex);
4760 int winding = windSum(lesser);
4761 int spanWinding = spanSign(endIndex, index);
4762 if (winding && UseInnerWindingReverse(winding - spanWinding, winding)
4763 && winding != SK_MaxS32) {
4764 winding -= spanWinding;
4765 }
4766 return winding;
4767 } 1971 }
4768 1972
4769 int SkOpSegment::updateWindingReverse(const SkOpAngle* angle) const { 1973 int SkOpSegment::updateWindingReverse(const SkOpAngle* angle) const {
4770 int startIndex = angle->start(); 1974 const SkOpSpanBase* startSpan = angle->start();
4771 int endIndex = angle->end(); 1975 const SkOpSpanBase* endSpan = angle->end();
4772 return updateWindingReverse(endIndex, startIndex); 1976 return updateWinding(startSpan, endSpan);
4773 } 1977 }
4774 1978
4775 // OPTIMIZATION: does the following also work, and is it any faster? 1979 // OPTIMIZATION: does the following also work, and is it any faster?
4776 // return outerWinding * innerWinding > 0 1980 // return outerWinding * innerWinding > 0
4777 // || ((outerWinding + innerWinding < 0) ^ ((outerWinding - innerWinding) < 0))) 1981 // || ((outerWinding + innerWinding < 0) ^ ((outerWinding - innerWinding) < 0)))
4778 bool SkOpSegment::UseInnerWinding(int outerWinding, int innerWinding) { 1982 bool SkOpSegment::UseInnerWinding(int outerWinding, int innerWinding) {
4779 SkASSERT(outerWinding != SK_MaxS32); 1983 SkASSERT(outerWinding != SK_MaxS32);
4780 SkASSERT(innerWinding != SK_MaxS32); 1984 SkASSERT(innerWinding != SK_MaxS32);
4781 int absOut = abs(outerWinding); 1985 int absOut = abs(outerWinding);
4782 int absIn = abs(innerWinding); 1986 int absIn = abs(innerWinding);
4783 bool result = absOut == absIn ? outerWinding < 0 : absOut < absIn; 1987 bool result = absOut == absIn ? outerWinding < 0 : absOut < absIn;
4784 return result; 1988 return result;
4785 } 1989 }
4786 1990
4787 bool SkOpSegment::UseInnerWindingReverse(int outerWinding, int innerWinding) { 1991 int SkOpSegment::windingAtT(double tHit, const SkOpSpan* span, bool crossOpp,
4788 SkASSERT(outerWinding != SK_MaxS32); 1992 SkScalar* dx) const {
4789 SkASSERT(innerWinding != SK_MaxS32); 1993 if (approximately_zero(tHit - span->t())) { // if we hit the end of a span, disregard
4790 int absOut = abs(outerWinding);
4791 int absIn = abs(innerWinding);
4792 bool result = absOut == absIn ? true : absOut < absIn;
4793 return result;
4794 }
4795
4796 int SkOpSegment::windingAtT(double tHit, int tIndex, bool crossOpp, SkScalar* dx ) const {
4797 if (approximately_zero(tHit - t(tIndex))) { // if we hit the end of a span, disregard
4798 return SK_MinS32; 1994 return SK_MinS32;
4799 } 1995 }
4800 int winding = crossOpp ? oppSum(tIndex) : windSum(tIndex); 1996 int winding = crossOpp ? span->oppSum() : span->windSum();
4801 SkASSERT(winding != SK_MinS32); 1997 SkASSERT(winding != SK_MinS32);
4802 int windVal = crossOpp ? oppValue(tIndex) : windValue(tIndex); 1998 int windVal = crossOpp ? span->oppValue() : span->windValue();
4803 #if DEBUG_WINDING_AT_T 1999 #if DEBUG_WINDING_AT_T
4804 SkDebugf("%s id=%d opp=%d tHit=%1.9g t=%1.9g oldWinding=%d windValue=%d", __ FUNCTION__, 2000 SkDebugf("%s id=%d opp=%d tHit=%1.9g t=%1.9g oldWinding=%d windValue=%d", __ FUNCTION__,
4805 debugID(), crossOpp, tHit, t(tIndex), winding, windVal); 2001 debugID(), crossOpp, tHit, span->t(), winding, windVal);
4806 #endif 2002 #endif
4807 // see if a + change in T results in a +/- change in X (compute x'(T)) 2003 // see if a + change in T results in a +/- change in X (compute x'(T))
4808 *dx = (*CurveSlopeAtT[SkPathOpsVerbToPoints(fVerb)])(fPts, tHit).fX; 2004 *dx = (*CurveSlopeAtT[SkPathOpsVerbToPoints(fVerb)])(fPts, tHit).fX;
4809 if (fVerb > SkPath::kLine_Verb && approximately_zero(*dx)) { 2005 if (fVerb > SkPath::kLine_Verb && approximately_zero(*dx)) {
4810 *dx = fPts[2].fX - fPts[1].fX - *dx; 2006 *dx = fPts[2].fX - fPts[1].fX - *dx;
4811 } 2007 }
4812 if (*dx == 0) { 2008 if (*dx == 0) {
4813 #if DEBUG_WINDING_AT_T 2009 #if DEBUG_WINDING_AT_T
4814 SkDebugf(" dx=0 winding=SK_MinS32\n"); 2010 SkDebugf(" dx=0 winding=SK_MinS32\n");
4815 #endif 2011 #endif
4816 return SK_MinS32; 2012 return SK_MinS32;
4817 } 2013 }
4818 if (windVal < 0) { // reverse sign if opp contour traveled in reverse 2014 if (windVal < 0) { // reverse sign if opp contour traveled in reverse
4819 *dx = -*dx; 2015 *dx = -*dx;
4820 } 2016 }
4821 if (winding * *dx > 0) { // if same signs, result is negative 2017 if (winding * *dx > 0) { // if same signs, result is negative
4822 winding += *dx > 0 ? -windVal : windVal; 2018 winding += *dx > 0 ? -windVal : windVal;
4823 } 2019 }
4824 #if DEBUG_WINDING_AT_T 2020 #if DEBUG_WINDING_AT_T
4825 SkDebugf(" dx=%c winding=%d\n", *dx > 0 ? '+' : '-', winding); 2021 SkDebugf(" dx=%c winding=%d\n", *dx > 0 ? '+' : '-', winding);
4826 #endif 2022 #endif
4827 return winding; 2023 return winding;
4828 } 2024 }
4829 2025
4830 int SkOpSegment::windSum(const SkOpAngle* angle) const { 2026 int SkOpSegment::windSum(const SkOpAngle* angle) const {
4831 int start = angle->start(); 2027 const SkOpSpan* minSpan = angle->start()->starter(angle->end());
4832 int end = angle->end(); 2028 return minSpan->windSum();
4833 int index = SkMin32(start, end);
4834 return windSum(index);
4835 } 2029 }
4836
4837 void SkOpSegment::zeroSpan(SkOpSpan* span) {
4838 SkASSERT(span->fWindValue > 0 || span->fOppValue != 0);
4839 span->fWindValue = 0;
4840 span->fOppValue = 0;
4841 if (span->fTiny || span->fSmall) {
4842 return;
4843 }
4844 SkASSERT(!span->fDone);
4845 span->fDone = true;
4846 ++fDoneSpans;
4847 }
OLDNEW
« no previous file with comments | « src/pathops/SkOpSegment.h ('k') | src/pathops/SkOpSpan.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698