blob: b7dc4b643ed321bb988f03da01e3e4be55b6e2e4 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +00002/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00003 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +00007 */
8
epoger@google.comec3ed6a2011-07-28 14:26:00 +00009
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000010#include "GrPathUtils.h"
bsalomon@google.com181e9bd2011-09-07 18:42:30 +000011
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000012#include "GrPoint.h"
13
bsalomon@google.com181e9bd2011-09-07 18:42:30 +000014GrScalar GrPathUtils::scaleToleranceToSrc(GrScalar devTol,
15 const GrMatrix& viewM) {
16 // In order to tesselate the path we get a bound on how much the matrix can
17 // stretch when mapping to screen coordinates.
18 GrScalar stretch = viewM.getMaxStretch();
19 GrScalar srcTol = devTol;
20
21 if (stretch < 0) {
22 // TODO: deal with perspective in some better way.
23 srcTol /= 5;
24 stretch = -stretch;
25 }
26 srcTol = GrScalarDiv(srcTol, stretch);
27 return srcTol;
28}
29
bsalomon@google.comb5b31682011-06-16 18:05:35 +000030static const int MAX_POINTS_PER_CURVE = 1 << 10;
bsalomon@google.com181e9bd2011-09-07 18:42:30 +000031static const GrScalar gMinCurveTol = GrFloatToScalar(0.0001f);
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000032
33uint32_t GrPathUtils::quadraticPointCount(const GrPoint points[],
tomhudson@google.comc10a8882011-06-28 15:19:32 +000034 GrScalar tol) {
35 if (tol < gMinCurveTol) {
tomhudson@google.comafec7ba2011-06-30 14:47:55 +000036 tol = gMinCurveTol;
tomhudson@google.comc10a8882011-06-28 15:19:32 +000037 }
38 GrAssert(tol > 0);
39
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000040 GrScalar d = points[1].distanceToLineSegmentBetween(points[0], points[2]);
tomhudson@google.comc10a8882011-06-28 15:19:32 +000041 if (d <= tol) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000042 return 1;
43 } else {
44 // Each time we subdivide, d should be cut in 4. So we need to
45 // subdivide x = log4(d/tol) times. x subdivisions creates 2^(x)
46 // points.
47 // 2^(log4(x)) = sqrt(x);
epoger@google.com2047f002011-05-17 17:36:59 +000048 int temp = SkScalarCeil(SkScalarSqrt(SkScalarDiv(d, tol)));
bsalomon@google.com61f3bde2011-06-17 20:06:49 +000049 int pow2 = GrNextPow2(temp);
50 // Because of NaNs & INFs we can wind up with a degenerate temp
51 // such that pow2 comes out negative. Also, our point generator
52 // will always output at least one pt.
53 if (pow2 < 1) {
54 pow2 = 1;
55 }
56 return GrMin(pow2, MAX_POINTS_PER_CURVE);
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000057 }
58}
59
60uint32_t GrPathUtils::generateQuadraticPoints(const GrPoint& p0,
tomhudson@google.comc10a8882011-06-28 15:19:32 +000061 const GrPoint& p1,
62 const GrPoint& p2,
63 GrScalar tolSqd,
64 GrPoint** points,
65 uint32_t pointsLeft) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000066 if (pointsLeft < 2 ||
67 (p1.distanceToLineSegmentBetweenSqd(p0, p2)) < tolSqd) {
68 (*points)[0] = p2;
69 *points += 1;
70 return 1;
71 }
72
73 GrPoint q[] = {
reed@google.com7744c202011-05-06 19:26:26 +000074 { GrScalarAve(p0.fX, p1.fX), GrScalarAve(p0.fY, p1.fY) },
75 { GrScalarAve(p1.fX, p2.fX), GrScalarAve(p1.fY, p2.fY) },
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000076 };
reed@google.com7744c202011-05-06 19:26:26 +000077 GrPoint r = { GrScalarAve(q[0].fX, q[1].fX), GrScalarAve(q[0].fY, q[1].fY) };
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000078
79 pointsLeft >>= 1;
80 uint32_t a = generateQuadraticPoints(p0, q[0], r, tolSqd, points, pointsLeft);
81 uint32_t b = generateQuadraticPoints(r, q[1], p2, tolSqd, points, pointsLeft);
82 return a + b;
83}
84
85uint32_t GrPathUtils::cubicPointCount(const GrPoint points[],
86 GrScalar tol) {
tomhudson@google.comc10a8882011-06-28 15:19:32 +000087 if (tol < gMinCurveTol) {
tomhudson@google.comafec7ba2011-06-30 14:47:55 +000088 tol = gMinCurveTol;
tomhudson@google.comc10a8882011-06-28 15:19:32 +000089 }
90 GrAssert(tol > 0);
91
92 GrScalar d = GrMax(
93 points[1].distanceToLineSegmentBetweenSqd(points[0], points[3]),
94 points[2].distanceToLineSegmentBetweenSqd(points[0], points[3]));
epoger@google.com2047f002011-05-17 17:36:59 +000095 d = SkScalarSqrt(d);
tomhudson@google.comc10a8882011-06-28 15:19:32 +000096 if (d <= tol) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000097 return 1;
98 } else {
epoger@google.com2047f002011-05-17 17:36:59 +000099 int temp = SkScalarCeil(SkScalarSqrt(SkScalarDiv(d, tol)));
bsalomon@google.com61f3bde2011-06-17 20:06:49 +0000100 int pow2 = GrNextPow2(temp);
101 // Because of NaNs & INFs we can wind up with a degenerate temp
102 // such that pow2 comes out negative. Also, our point generator
103 // will always output at least one pt.
104 if (pow2 < 1) {
105 pow2 = 1;
106 }
107 return GrMin(pow2, MAX_POINTS_PER_CURVE);
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000108 }
109}
110
111uint32_t GrPathUtils::generateCubicPoints(const GrPoint& p0,
tomhudson@google.comc10a8882011-06-28 15:19:32 +0000112 const GrPoint& p1,
113 const GrPoint& p2,
114 const GrPoint& p3,
115 GrScalar tolSqd,
116 GrPoint** points,
117 uint32_t pointsLeft) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000118 if (pointsLeft < 2 ||
119 (p1.distanceToLineSegmentBetweenSqd(p0, p3) < tolSqd &&
120 p2.distanceToLineSegmentBetweenSqd(p0, p3) < tolSqd)) {
121 (*points)[0] = p3;
122 *points += 1;
123 return 1;
124 }
125 GrPoint q[] = {
reed@google.com7744c202011-05-06 19:26:26 +0000126 { GrScalarAve(p0.fX, p1.fX), GrScalarAve(p0.fY, p1.fY) },
127 { GrScalarAve(p1.fX, p2.fX), GrScalarAve(p1.fY, p2.fY) },
128 { GrScalarAve(p2.fX, p3.fX), GrScalarAve(p2.fY, p3.fY) }
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000129 };
130 GrPoint r[] = {
reed@google.com7744c202011-05-06 19:26:26 +0000131 { GrScalarAve(q[0].fX, q[1].fX), GrScalarAve(q[0].fY, q[1].fY) },
132 { GrScalarAve(q[1].fX, q[2].fX), GrScalarAve(q[1].fY, q[2].fY) }
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000133 };
reed@google.com7744c202011-05-06 19:26:26 +0000134 GrPoint s = { GrScalarAve(r[0].fX, r[1].fX), GrScalarAve(r[0].fY, r[1].fY) };
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000135 pointsLeft >>= 1;
136 uint32_t a = generateCubicPoints(p0, q[0], r[0], s, tolSqd, points, pointsLeft);
137 uint32_t b = generateCubicPoints(s, r[1], q[2], p3, tolSqd, points, pointsLeft);
138 return a + b;
139}
140
reed@google.com07f3ee12011-05-16 17:21:57 +0000141int GrPathUtils::worstCasePointCount(const GrPath& path, int* subpaths,
142 GrScalar tol) {
tomhudson@google.comc10a8882011-06-28 15:19:32 +0000143 if (tol < gMinCurveTol) {
tomhudson@google.comafec7ba2011-06-30 14:47:55 +0000144 tol = gMinCurveTol;
tomhudson@google.comc10a8882011-06-28 15:19:32 +0000145 }
146 GrAssert(tol > 0);
147
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000148 int pointCount = 0;
149 *subpaths = 1;
150
151 bool first = true;
152
senorblanco@chromium.org129b8e32011-06-15 17:52:09 +0000153 SkPath::Iter iter(path, false);
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000154 GrPathCmd cmd;
155
156 GrPoint pts[4];
reed@google.com07f3ee12011-05-16 17:21:57 +0000157 while ((cmd = (GrPathCmd)iter.next(pts)) != kEnd_PathCmd) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000158
159 switch (cmd) {
160 case kLine_PathCmd:
161 pointCount += 1;
162 break;
163 case kQuadratic_PathCmd:
164 pointCount += quadraticPointCount(pts, tol);
165 break;
166 case kCubic_PathCmd:
167 pointCount += cubicPointCount(pts, tol);
168 break;
169 case kMove_PathCmd:
170 pointCount += 1;
171 if (!first) {
172 ++(*subpaths);
173 }
174 break;
175 default:
176 break;
177 }
178 first = false;
179 }
180 return pointCount;
181}