1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2023 Google LLC
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkFloatingPoint.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkSpan_impl.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkBezierCurves.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkQuads.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
13*c8dee2aaSAndroid Build Coastguard Worker
14*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm>
15*c8dee2aaSAndroid Build Coastguard Worker #include <cmath>
16*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
17*c8dee2aaSAndroid Build Coastguard Worker #include <initializer_list>
18*c8dee2aaSAndroid Build Coastguard Worker #include <limits>
19*c8dee2aaSAndroid Build Coastguard Worker #include <set>
20*c8dee2aaSAndroid Build Coastguard Worker #include <string>
21*c8dee2aaSAndroid Build Coastguard Worker
22*c8dee2aaSAndroid Build Coastguard Worker // Grouping the test inputs into DoublePoints makes the test cases easier to read.
23*c8dee2aaSAndroid Build Coastguard Worker struct DoublePoint {
24*c8dee2aaSAndroid Build Coastguard Worker double x;
25*c8dee2aaSAndroid Build Coastguard Worker double y;
26*c8dee2aaSAndroid Build Coastguard Worker };
27*c8dee2aaSAndroid Build Coastguard Worker
nearly_equal(double expected,double actual)28*c8dee2aaSAndroid Build Coastguard Worker static bool nearly_equal(double expected, double actual) {
29*c8dee2aaSAndroid Build Coastguard Worker if (sk_double_nearly_zero(expected)) {
30*c8dee2aaSAndroid Build Coastguard Worker return sk_double_nearly_zero(actual);
31*c8dee2aaSAndroid Build Coastguard Worker }
32*c8dee2aaSAndroid Build Coastguard Worker return sk_doubles_nearly_equal_ulps(expected, actual, 64);
33*c8dee2aaSAndroid Build Coastguard Worker }
34*c8dee2aaSAndroid Build Coastguard Worker
testCubicEvalAtT(skiatest::Reporter * reporter,const std::string & name,SkSpan<const DoublePoint> curveInputs,double t,const DoublePoint & expectedXY)35*c8dee2aaSAndroid Build Coastguard Worker static void testCubicEvalAtT(skiatest::Reporter* reporter, const std::string& name,
36*c8dee2aaSAndroid Build Coastguard Worker SkSpan<const DoublePoint> curveInputs, double t,
37*c8dee2aaSAndroid Build Coastguard Worker const DoublePoint& expectedXY) {
38*c8dee2aaSAndroid Build Coastguard Worker skiatest::ReporterContext subtest(reporter, name);
39*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, curveInputs.size() == 4,
40*c8dee2aaSAndroid Build Coastguard Worker "Invalid test case. Should have 4 input points.");
41*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, t >= 0.0 && t <= 1.0,
42*c8dee2aaSAndroid Build Coastguard Worker "Invalid test case. t %f should be in [0, 1]", t);
43*c8dee2aaSAndroid Build Coastguard Worker
44*c8dee2aaSAndroid Build Coastguard Worker auto [x, y] = SkBezierCubic::EvalAt(reinterpret_cast<const double*>(curveInputs.data()), t);
45*c8dee2aaSAndroid Build Coastguard Worker
46*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, nearly_equal(expectedXY.x, x),
47*c8dee2aaSAndroid Build Coastguard Worker "X wrong %1.16f != %1.16f", expectedXY.x, x);
48*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, nearly_equal(expectedXY.y, y),
49*c8dee2aaSAndroid Build Coastguard Worker "Y wrong %1.16f != %1.16f", expectedXY.y, y);
50*c8dee2aaSAndroid Build Coastguard Worker }
51*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(BezierCubicEvalAt,reporter)52*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(BezierCubicEvalAt, reporter) {
53*c8dee2aaSAndroid Build Coastguard Worker testCubicEvalAtT(reporter, "linear curve @0.1234",
54*c8dee2aaSAndroid Build Coastguard Worker {{ 0, 0 }, { 0, 0 }, { 10, 10 }, { 10, 10 }},
55*c8dee2aaSAndroid Build Coastguard Worker 0.1234,
56*c8dee2aaSAndroid Build Coastguard Worker { 0.4192451819200000, 0.4192451819200000 });
57*c8dee2aaSAndroid Build Coastguard Worker
58*c8dee2aaSAndroid Build Coastguard Worker testCubicEvalAtT(reporter, "linear curve @0.2345",
59*c8dee2aaSAndroid Build Coastguard Worker {{ 0, 0 }, { 5, 5 }, { 5, 5 }, { 10, 10 }},
60*c8dee2aaSAndroid Build Coastguard Worker 0.2345,
61*c8dee2aaSAndroid Build Coastguard Worker { 2.8215983862500000, 2.8215983862500000 });
62*c8dee2aaSAndroid Build Coastguard Worker
63*c8dee2aaSAndroid Build Coastguard Worker testCubicEvalAtT(reporter, "Arbitrary Cubic, t=0.0",
64*c8dee2aaSAndroid Build Coastguard Worker {{ -10, -20 }, { -7, 5 }, { 14, -2 }, { 3, 13 }},
65*c8dee2aaSAndroid Build Coastguard Worker 0.0,
66*c8dee2aaSAndroid Build Coastguard Worker { -10, -20 });
67*c8dee2aaSAndroid Build Coastguard Worker
68*c8dee2aaSAndroid Build Coastguard Worker testCubicEvalAtT(reporter, "Arbitrary Cubic, t=0.3456",
69*c8dee2aaSAndroid Build Coastguard Worker {{ -10, -20 }, { -7, 5 }, { 14, -2 }, { 3, 13 }},
70*c8dee2aaSAndroid Build Coastguard Worker 0.3456,
71*c8dee2aaSAndroid Build Coastguard Worker { -2.503786700800000, -3.31715344793600 });
72*c8dee2aaSAndroid Build Coastguard Worker
73*c8dee2aaSAndroid Build Coastguard Worker testCubicEvalAtT(reporter, "Arbitrary Cubic, t=0.5",
74*c8dee2aaSAndroid Build Coastguard Worker {{ -10, -20 }, { -7, 5 }, { 14, -2 }, { 3, 13 }},
75*c8dee2aaSAndroid Build Coastguard Worker 0.5,
76*c8dee2aaSAndroid Build Coastguard Worker { 1.75, 0.25 });
77*c8dee2aaSAndroid Build Coastguard Worker
78*c8dee2aaSAndroid Build Coastguard Worker testCubicEvalAtT(reporter, "Arbitrary Cubic, t=0.7891",
79*c8dee2aaSAndroid Build Coastguard Worker {{ -10, -20 }, { -7, 5 }, { 14, -2 }, { 3, 13 }},
80*c8dee2aaSAndroid Build Coastguard Worker 0.7891,
81*c8dee2aaSAndroid Build Coastguard Worker { 6.158763291450000, 5.938550084434000 });
82*c8dee2aaSAndroid Build Coastguard Worker
83*c8dee2aaSAndroid Build Coastguard Worker testCubicEvalAtT(reporter, "Arbitrary Cubic, t=1.0",
84*c8dee2aaSAndroid Build Coastguard Worker {{ -10, -20 }, { -7, 5 }, { 14, -2 }, { 3, 13 }},
85*c8dee2aaSAndroid Build Coastguard Worker 1.0,
86*c8dee2aaSAndroid Build Coastguard Worker { 3, 13 });
87*c8dee2aaSAndroid Build Coastguard Worker }
88*c8dee2aaSAndroid Build Coastguard Worker
testCubicConvertToPolynomial(skiatest::Reporter * reporter,const std::string & name,SkSpan<const DoublePoint> curveInputs,bool yValues,double expectedA,double expectedB,double expectedC,double expectedD)89*c8dee2aaSAndroid Build Coastguard Worker static void testCubicConvertToPolynomial(skiatest::Reporter* reporter, const std::string& name,
90*c8dee2aaSAndroid Build Coastguard Worker SkSpan<const DoublePoint> curveInputs, bool yValues,
91*c8dee2aaSAndroid Build Coastguard Worker double expectedA, double expectedB,
92*c8dee2aaSAndroid Build Coastguard Worker double expectedC, double expectedD) {
93*c8dee2aaSAndroid Build Coastguard Worker skiatest::ReporterContext subtest(reporter, name);
94*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, curveInputs.size() == 4,
95*c8dee2aaSAndroid Build Coastguard Worker "Invalid test case. Need 4 points (start, control, control, end)");
96*c8dee2aaSAndroid Build Coastguard Worker
97*c8dee2aaSAndroid Build Coastguard Worker skiatest::ReporterContext subsubtest(reporter, "SkBezierCurve Implementation");
98*c8dee2aaSAndroid Build Coastguard Worker const double* input = &curveInputs[0].x;
99*c8dee2aaSAndroid Build Coastguard Worker auto [A, B, C, D] = SkBezierCubic::ConvertToPolynomial(input, yValues);
100*c8dee2aaSAndroid Build Coastguard Worker
101*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, nearly_equal(expectedA, A), "%f != %f", expectedA, A);
102*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, nearly_equal(expectedB, B), "%f != %f", expectedB, B);
103*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, nearly_equal(expectedC, C), "%f != %f", expectedC, C);
104*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, nearly_equal(expectedD, D), "%f != %f", expectedD, D);
105*c8dee2aaSAndroid Build Coastguard Worker }
106*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(BezierCubicToPolynomials,reporter)107*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(BezierCubicToPolynomials, reporter) {
108*c8dee2aaSAndroid Build Coastguard Worker // See also tests/PathOpsDCubicTest.cpp->SkDCubicPolynomialCoefficients
109*c8dee2aaSAndroid Build Coastguard Worker testCubicConvertToPolynomial(reporter, "Arbitrary control points X direction",
110*c8dee2aaSAndroid Build Coastguard Worker {{1, 2}, {-3, 4}, {5, -6}, {7, 8}}, false, /*=yValues*/
111*c8dee2aaSAndroid Build Coastguard Worker -18, 36, -12, 1
112*c8dee2aaSAndroid Build Coastguard Worker );
113*c8dee2aaSAndroid Build Coastguard Worker testCubicConvertToPolynomial(reporter, "Arbitrary control points Y direction",
114*c8dee2aaSAndroid Build Coastguard Worker {{1, 2}, {-3, 4}, {5, -6}, {7, 8}}, true, /*=yValues*/
115*c8dee2aaSAndroid Build Coastguard Worker 36, -36, 6, 2
116*c8dee2aaSAndroid Build Coastguard Worker );
117*c8dee2aaSAndroid Build Coastguard Worker }
118*c8dee2aaSAndroid Build Coastguard Worker
119*c8dee2aaSAndroid Build Coastguard Worker // Since, Roots and EvalAt are separately unit tested, make sure that the parametric pramater t
120*c8dee2aaSAndroid Build Coastguard Worker // is correctly in range, and checked.
DEF_TEST(QuadRoots_CheckTRange,reporter)121*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(QuadRoots_CheckTRange, reporter) {
122*c8dee2aaSAndroid Build Coastguard Worker // Pick interesting numbers around 0 and 1.
123*c8dee2aaSAndroid Build Coastguard Worker const double interestingRoots[] =
124*c8dee2aaSAndroid Build Coastguard Worker {-1000, -10, -1, -0.1, -0.0001, 0, 0.0001, 0.1, 0.9, 0.9999, 1, 1.0001, 1.1, 10, 1000};
125*c8dee2aaSAndroid Build Coastguard Worker
126*c8dee2aaSAndroid Build Coastguard Worker // Interesting scales to make the quadratic.
127*c8dee2aaSAndroid Build Coastguard Worker const double interestingScales[] =
128*c8dee2aaSAndroid Build Coastguard Worker {-1000, -10, -1, -0.1, -0.0001, 0.0001, 0.1, 1, 10, 1000};
129*c8dee2aaSAndroid Build Coastguard Worker
130*c8dee2aaSAndroid Build Coastguard Worker auto outsideTRange = [](double r) {
131*c8dee2aaSAndroid Build Coastguard Worker return r < 0 || 1 < r;
132*c8dee2aaSAndroid Build Coastguard Worker };
133*c8dee2aaSAndroid Build Coastguard Worker
134*c8dee2aaSAndroid Build Coastguard Worker auto insideTRange = [&] (double r) {
135*c8dee2aaSAndroid Build Coastguard Worker return !outsideTRange(r);
136*c8dee2aaSAndroid Build Coastguard Worker };
137*c8dee2aaSAndroid Build Coastguard Worker
138*c8dee2aaSAndroid Build Coastguard Worker // The original test for AddValidTs (which quad intersect was based on) used 1 float ulp of
139*c8dee2aaSAndroid Build Coastguard Worker // leeway for comparison. Tighten this up to half a float ulp.
140*c8dee2aaSAndroid Build Coastguard Worker auto equalAsFloat = [] (double a, double b) {
141*c8dee2aaSAndroid Build Coastguard Worker // When converted to float, a double will be rounded up to half a float ulp for a double
142*c8dee2aaSAndroid Build Coastguard Worker // value between two float values.
143*c8dee2aaSAndroid Build Coastguard Worker return sk_double_to_float(a) == sk_double_to_float(b);
144*c8dee2aaSAndroid Build Coastguard Worker };
145*c8dee2aaSAndroid Build Coastguard Worker
146*c8dee2aaSAndroid Build Coastguard Worker for (double r1 : interestingRoots) {
147*c8dee2aaSAndroid Build Coastguard Worker for (double r0 : interestingRoots) {
148*c8dee2aaSAndroid Build Coastguard Worker for (double s : interestingScales) {
149*c8dee2aaSAndroid Build Coastguard Worker // Create a quadratic using the roots r0 and r1.
150*c8dee2aaSAndroid Build Coastguard Worker // s(x-r0)(x-r1) = s(x^2 - r0*x - r1*x + r0*r1)
151*c8dee2aaSAndroid Build Coastguard Worker const double A = s;
152*c8dee2aaSAndroid Build Coastguard Worker // Normally B = -(r0 + r1) but this needs the modified B' = (r0 + r1) / 2.
153*c8dee2aaSAndroid Build Coastguard Worker const double B = s * 0.5 * (r0 + r1);
154*c8dee2aaSAndroid Build Coastguard Worker const double C = s*r0*r1;
155*c8dee2aaSAndroid Build Coastguard Worker
156*c8dee2aaSAndroid Build Coastguard Worker float storage[2];
157*c8dee2aaSAndroid Build Coastguard Worker // The X coefficients are set to return t's generated by root intersection.
158*c8dee2aaSAndroid Build Coastguard Worker // The offset is set to 0, because an arbitrary offset is essentially encoded in C.
159*c8dee2aaSAndroid Build Coastguard Worker auto intersections = SkBezierQuad::Intersect(0, -0.5, 0, A, B, C, 0, storage);
160*c8dee2aaSAndroid Build Coastguard Worker
161*c8dee2aaSAndroid Build Coastguard Worker if (intersections.empty()) {
162*c8dee2aaSAndroid Build Coastguard Worker // Either imaginary or both roots are outside [0, 1].
163*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter,
164*c8dee2aaSAndroid Build Coastguard Worker SkQuads::Discriminant(A, B, C) < 0
165*c8dee2aaSAndroid Build Coastguard Worker || (outsideTRange(r0) && outsideTRange(r1)));
166*c8dee2aaSAndroid Build Coastguard Worker } else if (intersections.size() == 1) {
167*c8dee2aaSAndroid Build Coastguard Worker // One of the roots is outside [0, 1]
168*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, insideTRange(r0) || insideTRange(r1));
169*c8dee2aaSAndroid Build Coastguard Worker const double insideRoot = insideTRange(r0) ? r0 : r1;
170*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, equalAsFloat(insideRoot, intersections[0]));
171*c8dee2aaSAndroid Build Coastguard Worker } else {
172*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, intersections.size() == 2);
173*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, insideTRange(r0) && insideTRange(r1));
174*c8dee2aaSAndroid Build Coastguard Worker auto [smaller, bigger] = std::minmax(intersections[0], intersections[1]);
175*c8dee2aaSAndroid Build Coastguard Worker auto [smallerRoot, biggerRoot] = std::minmax(r0, r1);
176*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, equalAsFloat(smaller, smallerRoot));
177*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, equalAsFloat(bigger, biggerRoot));
178*c8dee2aaSAndroid Build Coastguard Worker }
179*c8dee2aaSAndroid Build Coastguard Worker }
180*c8dee2aaSAndroid Build Coastguard Worker }
181*c8dee2aaSAndroid Build Coastguard Worker }
182*c8dee2aaSAndroid Build Coastguard Worker
183*c8dee2aaSAndroid Build Coastguard Worker // Check when A == 0.
184*c8dee2aaSAndroid Build Coastguard Worker {
185*c8dee2aaSAndroid Build Coastguard Worker const double A = 0;
186*c8dee2aaSAndroid Build Coastguard Worker
187*c8dee2aaSAndroid Build Coastguard Worker // We need M = 4, so that will be a Kahan style B of -0.5 * M = -2.
188*c8dee2aaSAndroid Build Coastguard Worker const double B = -2;
189*c8dee2aaSAndroid Build Coastguard Worker const double C = -1;
190*c8dee2aaSAndroid Build Coastguard Worker float storage[2];
191*c8dee2aaSAndroid Build Coastguard Worker
192*c8dee2aaSAndroid Build Coastguard Worker // Assume the offset is already encoded in C.
193*c8dee2aaSAndroid Build Coastguard Worker auto intersections = SkBezierQuad::Intersect(0, -0.5, 0, A, B, C, 0, storage);
194*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, intersections.size() == 1);
195*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, intersections[0] == 0.25);
196*c8dee2aaSAndroid Build Coastguard Worker }
197*c8dee2aaSAndroid Build Coastguard Worker }
198*c8dee2aaSAndroid Build Coastguard Worker
199*c8dee2aaSAndroid Build Coastguard Worker // Since, Roots and EvalAt are separately unit tested, make sure that the parametric pramater t
200*c8dee2aaSAndroid Build Coastguard Worker // is correctly in range, and checked.
DEF_TEST(SkBezierCubic_CheckTRange,reporter)201*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkBezierCubic_CheckTRange, reporter) {
202*c8dee2aaSAndroid Build Coastguard Worker // Pick interesting numbers around 0 and 1.
203*c8dee2aaSAndroid Build Coastguard Worker const double interestingRoots[] =
204*c8dee2aaSAndroid Build Coastguard Worker {-10, -5, -2, -1, 0, 0.5, 1, 2, 5, 10};
205*c8dee2aaSAndroid Build Coastguard Worker
206*c8dee2aaSAndroid Build Coastguard Worker // Interesting scales to make the quadratic.
207*c8dee2aaSAndroid Build Coastguard Worker const double interestingScales[] =
208*c8dee2aaSAndroid Build Coastguard Worker {-1000, -10, -1, -0.1, -0.0001, 0.0001, 0.1, 1, 10, 1000};
209*c8dee2aaSAndroid Build Coastguard Worker
210*c8dee2aaSAndroid Build Coastguard Worker auto outsideTRange = [](double r) {
211*c8dee2aaSAndroid Build Coastguard Worker return r < 0 || 1 < r;
212*c8dee2aaSAndroid Build Coastguard Worker };
213*c8dee2aaSAndroid Build Coastguard Worker
214*c8dee2aaSAndroid Build Coastguard Worker auto insideTRange = [&] (double r) {
215*c8dee2aaSAndroid Build Coastguard Worker return !outsideTRange(r);
216*c8dee2aaSAndroid Build Coastguard Worker };
217*c8dee2aaSAndroid Build Coastguard Worker
218*c8dee2aaSAndroid Build Coastguard Worker auto specialEqual = [] (double actual, double test) {
219*c8dee2aaSAndroid Build Coastguard Worker // At least a floats worth of digits are correct.
220*c8dee2aaSAndroid Build Coastguard Worker const double errorFactor = std::numeric_limits<float>::epsilon();
221*c8dee2aaSAndroid Build Coastguard Worker return std::abs(test - actual) <= errorFactor * std::max(std::abs(test), std::abs(actual));
222*c8dee2aaSAndroid Build Coastguard Worker };
223*c8dee2aaSAndroid Build Coastguard Worker
224*c8dee2aaSAndroid Build Coastguard Worker for (double r2 : interestingRoots) {
225*c8dee2aaSAndroid Build Coastguard Worker for (double r1 : interestingRoots) {
226*c8dee2aaSAndroid Build Coastguard Worker for (double r0 : interestingRoots) {
227*c8dee2aaSAndroid Build Coastguard Worker for (double s : interestingScales) {
228*c8dee2aaSAndroid Build Coastguard Worker // Create a cubic using the roots r0, r1, and r2.
229*c8dee2aaSAndroid Build Coastguard Worker // s(x-r0)(x-r1)(x-r2) = s(x^3 - (r0+r1+r2)x^2 + (r0r1+r1r2+r0r2)x - r0r1r2)
230*c8dee2aaSAndroid Build Coastguard Worker const double A = s,
231*c8dee2aaSAndroid Build Coastguard Worker B = -s * (r0+r1+r2),
232*c8dee2aaSAndroid Build Coastguard Worker C = s * (r0*r1 + r1*r2 + r0*r2),
233*c8dee2aaSAndroid Build Coastguard Worker D = -s * r0 * r1 * r2;
234*c8dee2aaSAndroid Build Coastguard Worker
235*c8dee2aaSAndroid Build Coastguard Worker // Accumulate all the valid t's.
236*c8dee2aaSAndroid Build Coastguard Worker std::set<double> inRangeRoots;
237*c8dee2aaSAndroid Build Coastguard Worker for (auto r : {r0, r1, r2}) {
238*c8dee2aaSAndroid Build Coastguard Worker if (insideTRange(r)) {
239*c8dee2aaSAndroid Build Coastguard Worker inRangeRoots.insert(r);
240*c8dee2aaSAndroid Build Coastguard Worker }
241*c8dee2aaSAndroid Build Coastguard Worker }
242*c8dee2aaSAndroid Build Coastguard Worker
243*c8dee2aaSAndroid Build Coastguard Worker float storage[3];
244*c8dee2aaSAndroid Build Coastguard Worker // The X coefficients are set to return t's generated by root intersection.
245*c8dee2aaSAndroid Build Coastguard Worker // The offset is set to 0, because an arbitrary offset is essentially encoded
246*c8dee2aaSAndroid Build Coastguard Worker // in C.
247*c8dee2aaSAndroid Build Coastguard Worker auto intersections =
248*c8dee2aaSAndroid Build Coastguard Worker SkBezierCubic::Intersect(0, 0, 1, 0, A, B, C, D, 0, storage);
249*c8dee2aaSAndroid Build Coastguard Worker
250*c8dee2aaSAndroid Build Coastguard Worker size_t correct = 0;
251*c8dee2aaSAndroid Build Coastguard Worker for (auto candidate : intersections) {
252*c8dee2aaSAndroid Build Coastguard Worker for (auto answer : inRangeRoots) {
253*c8dee2aaSAndroid Build Coastguard Worker if (specialEqual(candidate, answer)) {
254*c8dee2aaSAndroid Build Coastguard Worker correct += 1;
255*c8dee2aaSAndroid Build Coastguard Worker break;
256*c8dee2aaSAndroid Build Coastguard Worker }
257*c8dee2aaSAndroid Build Coastguard Worker }
258*c8dee2aaSAndroid Build Coastguard Worker }
259*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, correct == intersections.size());
260*c8dee2aaSAndroid Build Coastguard Worker }
261*c8dee2aaSAndroid Build Coastguard Worker }
262*c8dee2aaSAndroid Build Coastguard Worker }
263*c8dee2aaSAndroid Build Coastguard Worker }
264*c8dee2aaSAndroid Build Coastguard Worker }
265*c8dee2aaSAndroid Build Coastguard Worker
266