blob: 429b2945d3cc121e58c7965365367b3f3fe303f9 [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"
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000011#include "GrPoint.h"
12
bsalomon@google.comb5b31682011-06-16 18:05:35 +000013static const int MAX_POINTS_PER_CURVE = 1 << 10;
tomhudson@google.comc10a8882011-06-28 15:19:32 +000014const GrScalar GrPathUtils::gMinCurveTol (GrFloatToScalar(0.0001f));
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000015
16uint32_t GrPathUtils::quadraticPointCount(const GrPoint points[],
tomhudson@google.comc10a8882011-06-28 15:19:32 +000017 GrScalar tol) {
18 if (tol < gMinCurveTol) {
tomhudson@google.comafec7ba2011-06-30 14:47:55 +000019 tol = gMinCurveTol;
tomhudson@google.comc10a8882011-06-28 15:19:32 +000020 }
21 GrAssert(tol > 0);
22
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000023 GrScalar d = points[1].distanceToLineSegmentBetween(points[0], points[2]);
tomhudson@google.comc10a8882011-06-28 15:19:32 +000024 if (d <= tol) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000025 return 1;
26 } else {
27 // Each time we subdivide, d should be cut in 4. So we need to
28 // subdivide x = log4(d/tol) times. x subdivisions creates 2^(x)
29 // points.
30 // 2^(log4(x)) = sqrt(x);
epoger@google.com2047f002011-05-17 17:36:59 +000031 int temp = SkScalarCeil(SkScalarSqrt(SkScalarDiv(d, tol)));
bsalomon@google.com61f3bde2011-06-17 20:06:49 +000032 int pow2 = GrNextPow2(temp);
33 // Because of NaNs & INFs we can wind up with a degenerate temp
34 // such that pow2 comes out negative. Also, our point generator
35 // will always output at least one pt.
36 if (pow2 < 1) {
37 pow2 = 1;
38 }
39 return GrMin(pow2, MAX_POINTS_PER_CURVE);
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000040 }
41}
42
43uint32_t GrPathUtils::generateQuadraticPoints(const GrPoint& p0,
tomhudson@google.comc10a8882011-06-28 15:19:32 +000044 const GrPoint& p1,
45 const GrPoint& p2,
46 GrScalar tolSqd,
47 GrPoint** points,
48 uint32_t pointsLeft) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000049 if (pointsLeft < 2 ||
50 (p1.distanceToLineSegmentBetweenSqd(p0, p2)) < tolSqd) {
51 (*points)[0] = p2;
52 *points += 1;
53 return 1;
54 }
55
56 GrPoint q[] = {
reed@google.com7744c202011-05-06 19:26:26 +000057 { GrScalarAve(p0.fX, p1.fX), GrScalarAve(p0.fY, p1.fY) },
58 { GrScalarAve(p1.fX, p2.fX), GrScalarAve(p1.fY, p2.fY) },
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000059 };
reed@google.com7744c202011-05-06 19:26:26 +000060 GrPoint r = { GrScalarAve(q[0].fX, q[1].fX), GrScalarAve(q[0].fY, q[1].fY) };
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000061
62 pointsLeft >>= 1;
63 uint32_t a = generateQuadraticPoints(p0, q[0], r, tolSqd, points, pointsLeft);
64 uint32_t b = generateQuadraticPoints(r, q[1], p2, tolSqd, points, pointsLeft);
65 return a + b;
66}
67
68uint32_t GrPathUtils::cubicPointCount(const GrPoint points[],
69 GrScalar tol) {
tomhudson@google.comc10a8882011-06-28 15:19:32 +000070 if (tol < gMinCurveTol) {
tomhudson@google.comafec7ba2011-06-30 14:47:55 +000071 tol = gMinCurveTol;
tomhudson@google.comc10a8882011-06-28 15:19:32 +000072 }
73 GrAssert(tol > 0);
74
75 GrScalar d = GrMax(
76 points[1].distanceToLineSegmentBetweenSqd(points[0], points[3]),
77 points[2].distanceToLineSegmentBetweenSqd(points[0], points[3]));
epoger@google.com2047f002011-05-17 17:36:59 +000078 d = SkScalarSqrt(d);
tomhudson@google.comc10a8882011-06-28 15:19:32 +000079 if (d <= tol) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000080 return 1;
81 } else {
epoger@google.com2047f002011-05-17 17:36:59 +000082 int temp = SkScalarCeil(SkScalarSqrt(SkScalarDiv(d, tol)));
bsalomon@google.com61f3bde2011-06-17 20:06:49 +000083 int pow2 = GrNextPow2(temp);
84 // Because of NaNs & INFs we can wind up with a degenerate temp
85 // such that pow2 comes out negative. Also, our point generator
86 // will always output at least one pt.
87 if (pow2 < 1) {
88 pow2 = 1;
89 }
90 return GrMin(pow2, MAX_POINTS_PER_CURVE);
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +000091 }
92}
93
94uint32_t GrPathUtils::generateCubicPoints(const GrPoint& p0,
tomhudson@google.comc10a8882011-06-28 15:19:32 +000095 const GrPoint& p1,
96 const GrPoint& p2,
97 const GrPoint& p3,
98 GrScalar tolSqd,
99 GrPoint** points,
100 uint32_t pointsLeft) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000101 if (pointsLeft < 2 ||
102 (p1.distanceToLineSegmentBetweenSqd(p0, p3) < tolSqd &&
103 p2.distanceToLineSegmentBetweenSqd(p0, p3) < tolSqd)) {
104 (*points)[0] = p3;
105 *points += 1;
106 return 1;
107 }
108 GrPoint q[] = {
reed@google.com7744c202011-05-06 19:26:26 +0000109 { GrScalarAve(p0.fX, p1.fX), GrScalarAve(p0.fY, p1.fY) },
110 { GrScalarAve(p1.fX, p2.fX), GrScalarAve(p1.fY, p2.fY) },
111 { GrScalarAve(p2.fX, p3.fX), GrScalarAve(p2.fY, p3.fY) }
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000112 };
113 GrPoint r[] = {
reed@google.com7744c202011-05-06 19:26:26 +0000114 { GrScalarAve(q[0].fX, q[1].fX), GrScalarAve(q[0].fY, q[1].fY) },
115 { GrScalarAve(q[1].fX, q[2].fX), GrScalarAve(q[1].fY, q[2].fY) }
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000116 };
reed@google.com7744c202011-05-06 19:26:26 +0000117 GrPoint s = { GrScalarAve(r[0].fX, r[1].fX), GrScalarAve(r[0].fY, r[1].fY) };
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000118 pointsLeft >>= 1;
119 uint32_t a = generateCubicPoints(p0, q[0], r[0], s, tolSqd, points, pointsLeft);
120 uint32_t b = generateCubicPoints(s, r[1], q[2], p3, tolSqd, points, pointsLeft);
121 return a + b;
122}
123
reed@google.com07f3ee12011-05-16 17:21:57 +0000124int GrPathUtils::worstCasePointCount(const GrPath& path, int* subpaths,
125 GrScalar tol) {
tomhudson@google.comc10a8882011-06-28 15:19:32 +0000126 if (tol < gMinCurveTol) {
tomhudson@google.comafec7ba2011-06-30 14:47:55 +0000127 tol = gMinCurveTol;
tomhudson@google.comc10a8882011-06-28 15:19:32 +0000128 }
129 GrAssert(tol > 0);
130
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000131 int pointCount = 0;
132 *subpaths = 1;
133
134 bool first = true;
135
senorblanco@chromium.org129b8e32011-06-15 17:52:09 +0000136 SkPath::Iter iter(path, false);
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000137 GrPathCmd cmd;
138
139 GrPoint pts[4];
reed@google.com07f3ee12011-05-16 17:21:57 +0000140 while ((cmd = (GrPathCmd)iter.next(pts)) != kEnd_PathCmd) {
senorblanco@chromium.org9d18b782011-03-28 20:47:09 +0000141
142 switch (cmd) {
143 case kLine_PathCmd:
144 pointCount += 1;
145 break;
146 case kQuadratic_PathCmd:
147 pointCount += quadraticPointCount(pts, tol);
148 break;
149 case kCubic_PathCmd:
150 pointCount += cubicPointCount(pts, tol);
151 break;
152 case kMove_PathCmd:
153 pointCount += 1;
154 if (!first) {
155 ++(*subpaths);
156 }
157 break;
158 default:
159 break;
160 }
161 first = false;
162 }
163 return pointCount;
164}