xref: /aosp_15_r20/external/skia/tests/CubicRootsTest.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
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/core/SkSpan.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkFloatingPoint.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkCubics.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkUtils.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/pathops/SkPathOpsCubic.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
15*c8dee2aaSAndroid Build Coastguard Worker 
16*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm>
17*c8dee2aaSAndroid Build Coastguard Worker #include <cmath>
18*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
19*c8dee2aaSAndroid Build Coastguard Worker #include <iterator>
20*c8dee2aaSAndroid Build Coastguard Worker #include <string>
21*c8dee2aaSAndroid Build Coastguard Worker 
testCubicRootsReal(skiatest::Reporter * reporter,std::string name,double A,double B,double C,double D,SkSpan<const double> expectedRoots,bool skipPathops=false,bool skipRootValidation=false)22*c8dee2aaSAndroid Build Coastguard Worker static void testCubicRootsReal(skiatest::Reporter* reporter, std::string name,
23*c8dee2aaSAndroid Build Coastguard Worker                                double A, double B, double C, double D,
24*c8dee2aaSAndroid Build Coastguard Worker                                SkSpan<const double> expectedRoots,
25*c8dee2aaSAndroid Build Coastguard Worker                                bool skipPathops = false,
26*c8dee2aaSAndroid Build Coastguard Worker                                bool skipRootValidation = false) {
27*c8dee2aaSAndroid Build Coastguard Worker     skiatest::ReporterContext subtest(reporter, name);
28*c8dee2aaSAndroid Build Coastguard Worker     // Validate test case
29*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, expectedRoots.size() <= 3,
30*c8dee2aaSAndroid Build Coastguard Worker                     "Invalid test case, up to 3 roots allowed");
31*c8dee2aaSAndroid Build Coastguard Worker 
32*c8dee2aaSAndroid Build Coastguard Worker     for (size_t i = 0; i < expectedRoots.size(); i++) {
33*c8dee2aaSAndroid Build Coastguard Worker         double x = expectedRoots[i];
34*c8dee2aaSAndroid Build Coastguard Worker         // A*x^3 + B*x^2 + C*x + D should equal 0 (unless floating point error causes issues)
35*c8dee2aaSAndroid Build Coastguard Worker         double y = A * x * x * x + B * x * x + C * x + D;
36*c8dee2aaSAndroid Build Coastguard Worker         if (!skipRootValidation) {
37*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, sk_double_nearly_zero(y),
38*c8dee2aaSAndroid Build Coastguard Worker                             "Invalid test case root %zu. %.16f != 0", i, y);
39*c8dee2aaSAndroid Build Coastguard Worker         }
40*c8dee2aaSAndroid Build Coastguard Worker 
41*c8dee2aaSAndroid Build Coastguard Worker         if (i > 0) {
42*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, expectedRoots[i-1] <= expectedRoots[i],
43*c8dee2aaSAndroid Build Coastguard Worker                     "Invalid test case root %zu. Roots should be sorted in ascending order", i);
44*c8dee2aaSAndroid Build Coastguard Worker         }
45*c8dee2aaSAndroid Build Coastguard Worker     }
46*c8dee2aaSAndroid Build Coastguard Worker 
47*c8dee2aaSAndroid Build Coastguard Worker     // The old pathops implementation sometimes gives incorrect solutions. We can opt
48*c8dee2aaSAndroid Build Coastguard Worker     // our tests out of checking that older implementation if that causes issues.
49*c8dee2aaSAndroid Build Coastguard Worker     if (!skipPathops) {
50*c8dee2aaSAndroid Build Coastguard Worker         skiatest::ReporterContext subsubtest(reporter, "Pathops Implementation");
51*c8dee2aaSAndroid Build Coastguard Worker         double roots[3] = {0, 0, 0};
52*c8dee2aaSAndroid Build Coastguard Worker         int rootCount = SkDCubic::RootsReal(A, B, C, D, roots);
53*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, expectedRoots.size() == size_t(rootCount),
54*c8dee2aaSAndroid Build Coastguard Worker                         "Wrong number of roots returned %zu != %d", expectedRoots.size(),
55*c8dee2aaSAndroid Build Coastguard Worker                         rootCount);
56*c8dee2aaSAndroid Build Coastguard Worker 
57*c8dee2aaSAndroid Build Coastguard Worker         // We don't care which order the roots are returned from the algorithm.
58*c8dee2aaSAndroid Build Coastguard Worker         // For determinism, we will sort them (and ensure the provided solutions are also sorted).
59*c8dee2aaSAndroid Build Coastguard Worker         std::sort(std::begin(roots), std::begin(roots) + rootCount);
60*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < rootCount; i++) {
61*c8dee2aaSAndroid Build Coastguard Worker             if (sk_double_nearly_zero(expectedRoots[i])) {
62*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(reporter, sk_double_nearly_zero(roots[i]),
63*c8dee2aaSAndroid Build Coastguard Worker                                 "0 != %.16f at index %d", roots[i], i);
64*c8dee2aaSAndroid Build Coastguard Worker             } else {
65*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(reporter,
66*c8dee2aaSAndroid Build Coastguard Worker                                 sk_doubles_nearly_equal_ulps(expectedRoots[i], roots[i], 64),
67*c8dee2aaSAndroid Build Coastguard Worker                                 "%.16f != %.16f at index %d", expectedRoots[i], roots[i], i);
68*c8dee2aaSAndroid Build Coastguard Worker             }
69*c8dee2aaSAndroid Build Coastguard Worker         }
70*c8dee2aaSAndroid Build Coastguard Worker     }
71*c8dee2aaSAndroid Build Coastguard Worker     {
72*c8dee2aaSAndroid Build Coastguard Worker         skiatest::ReporterContext subsubtest(reporter, "SkCubics Analytic Implementation");
73*c8dee2aaSAndroid Build Coastguard Worker         double roots[3] = {0, 0, 0};
74*c8dee2aaSAndroid Build Coastguard Worker         int rootCount = SkCubics::RootsReal(A, B, C, D, roots);
75*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, expectedRoots.size() == size_t(rootCount),
76*c8dee2aaSAndroid Build Coastguard Worker                         "Wrong number of roots returned %zu != %d", expectedRoots.size(),
77*c8dee2aaSAndroid Build Coastguard Worker                         rootCount);
78*c8dee2aaSAndroid Build Coastguard Worker 
79*c8dee2aaSAndroid Build Coastguard Worker         // We don't care which order the roots are returned from the algorithm.
80*c8dee2aaSAndroid Build Coastguard Worker         // For determinism, we will sort them (and ensure the provided solutions are also sorted).
81*c8dee2aaSAndroid Build Coastguard Worker         std::sort(std::begin(roots), std::begin(roots) + rootCount);
82*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < rootCount; i++) {
83*c8dee2aaSAndroid Build Coastguard Worker             if (sk_double_nearly_zero(expectedRoots[i])) {
84*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(reporter, sk_double_nearly_zero(roots[i]),
85*c8dee2aaSAndroid Build Coastguard Worker                                 "0 != %.16f at index %d", roots[i], i);
86*c8dee2aaSAndroid Build Coastguard Worker             } else {
87*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(reporter,
88*c8dee2aaSAndroid Build Coastguard Worker                                 sk_doubles_nearly_equal_ulps(expectedRoots[i], roots[i], 64),
89*c8dee2aaSAndroid Build Coastguard Worker                                 "%.16f != %.16f at index %d", expectedRoots[i], roots[i], i);
90*c8dee2aaSAndroid Build Coastguard Worker             }
91*c8dee2aaSAndroid Build Coastguard Worker         }
92*c8dee2aaSAndroid Build Coastguard Worker     }
93*c8dee2aaSAndroid Build Coastguard Worker }
94*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(CubicRootsReal_ActualCubics,reporter)95*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(CubicRootsReal_ActualCubics, reporter) {
96*c8dee2aaSAndroid Build Coastguard Worker     // All answers are given with 16 significant digits (max for a double) or as an integer
97*c8dee2aaSAndroid Build Coastguard Worker     // when the answer is exact.
98*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "one root 1x^3 + 2x^2 + 3x + 4",
99*c8dee2aaSAndroid Build Coastguard Worker                        1, 2, 3, 4,
100*c8dee2aaSAndroid Build Coastguard Worker                        {-1.650629191439388});
101*c8dee2aaSAndroid Build Coastguard Worker                       //-1.650629191439388218880801 from Wolfram Alpha
102*c8dee2aaSAndroid Build Coastguard Worker 
103*c8dee2aaSAndroid Build Coastguard Worker     // (3x-5)(6x-10)(x+4) = 18x^3 + 12x^2 - 190x + 200
104*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "touches y axis 18x^3 + 12x^2 - 190x + 200",
105*c8dee2aaSAndroid Build Coastguard Worker                        18, 12, -190, 200,
106*c8dee2aaSAndroid Build Coastguard Worker                        {-4.,
107*c8dee2aaSAndroid Build Coastguard Worker                          1.666666666666667, // 5/3
108*c8dee2aaSAndroid Build Coastguard Worker                        });
109*c8dee2aaSAndroid Build Coastguard Worker 
110*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "three roots 10x^3 - 20x^2 - 30x + 40",
111*c8dee2aaSAndroid Build Coastguard Worker                        10, -20, -30, 40,
112*c8dee2aaSAndroid Build Coastguard Worker                        {-1.561552812808830,
113*c8dee2aaSAndroid Build Coastguard Worker                       //-1.561552812808830274910705 from Wolfram Alpha
114*c8dee2aaSAndroid Build Coastguard Worker                          1.,
115*c8dee2aaSAndroid Build Coastguard Worker                          2.561552812808830,
116*c8dee2aaSAndroid Build Coastguard Worker                       // 2.561552812808830274910705 from Wolfram Alpha
117*c8dee2aaSAndroid Build Coastguard Worker                        });
118*c8dee2aaSAndroid Build Coastguard Worker 
119*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "three roots -10x^3 + 200x^2 + 300x - 400",
120*c8dee2aaSAndroid Build Coastguard Worker                        -10, 200, 300, -400,
121*c8dee2aaSAndroid Build Coastguard Worker                        {-2.179884793243323,
122*c8dee2aaSAndroid Build Coastguard Worker                       //-2.179884793243323422232630 from Wolfram Alpha
123*c8dee2aaSAndroid Build Coastguard Worker                          0.8607083693981839,
124*c8dee2aaSAndroid Build Coastguard Worker                       // 0.8607083693981838897123320 from Wolfram Alpha
125*c8dee2aaSAndroid Build Coastguard Worker                         21.31917642384514,
126*c8dee2aaSAndroid Build Coastguard Worker                       //21.31917642384513953252030 from Wolfram Alpha
127*c8dee2aaSAndroid Build Coastguard Worker                        });
128*c8dee2aaSAndroid Build Coastguard Worker 
129*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "one root -x^3 + 0x^2 + 5x - 7",
130*c8dee2aaSAndroid Build Coastguard Worker                        -1, 0, 5, -7,
131*c8dee2aaSAndroid Build Coastguard Worker                        {-2.747346540307211,
132*c8dee2aaSAndroid Build Coastguard Worker                       //-2.747346540307210849971436 from Wolfram Alpha
133*c8dee2aaSAndroid Build Coastguard Worker                        });
134*c8dee2aaSAndroid Build Coastguard Worker 
135*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "one root 2x^3 - 3x^2 + 0x + 3",
136*c8dee2aaSAndroid Build Coastguard Worker                        2, -3, 0, 3,
137*c8dee2aaSAndroid Build Coastguard Worker                        {-0.806443932358772,
138*c8dee2aaSAndroid Build Coastguard Worker                       //-0.8064439323587723772036250 from Wolfram Alpha
139*c8dee2aaSAndroid Build Coastguard Worker                        });
140*c8dee2aaSAndroid Build Coastguard Worker 
141*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "one root x^3 + 0x^2 + 0x - 9",
142*c8dee2aaSAndroid Build Coastguard Worker                        1, 0, 0, -9,
143*c8dee2aaSAndroid Build Coastguard Worker                        {2.080083823051904,
144*c8dee2aaSAndroid Build Coastguard Worker                       //2.0800838230519041145300568 from Wolfram Alpha
145*c8dee2aaSAndroid Build Coastguard Worker                        });
146*c8dee2aaSAndroid Build Coastguard Worker 
147*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "three roots 2x^3 - 3x^2 - 4x + 0",
148*c8dee2aaSAndroid Build Coastguard Worker                        2, -3, -4, 0,
149*c8dee2aaSAndroid Build Coastguard Worker                        {-0.8507810593582122,
150*c8dee2aaSAndroid Build Coastguard Worker                       //-0.8507810593582121716220544 from Wolfram Alpha
151*c8dee2aaSAndroid Build Coastguard Worker                         0.,
152*c8dee2aaSAndroid Build Coastguard Worker                         2.350781059358212
153*c8dee2aaSAndroid Build Coastguard Worker                       //2.350781059358212171622054 from Wolfram Alpha
154*c8dee2aaSAndroid Build Coastguard Worker                        });
155*c8dee2aaSAndroid Build Coastguard Worker 
156*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "R^2 and Q^3 are near zero",
157*c8dee2aaSAndroid Build Coastguard Worker                        -0.33790159225463867, -0.81997990608215332,
158*c8dee2aaSAndroid Build Coastguard Worker                        -0.66327774524688721, -0.17884063720703125,
159*c8dee2aaSAndroid Build Coastguard Worker                        {-0.7995944894729731});
160*c8dee2aaSAndroid Build Coastguard Worker 
161*c8dee2aaSAndroid Build Coastguard Worker     // The following three cases fallback to treating the cubic as a quadratic.
162*c8dee2aaSAndroid Build Coastguard Worker     // Otherwise, floating point error mangles the solutions near +- 1
163*c8dee2aaSAndroid Build Coastguard Worker     // This means we don't find all the roots, but usually we only care about roots
164*c8dee2aaSAndroid Build Coastguard Worker     // in the range [0, 1], so that is ok.
165*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "oss-fuzz:55625 Two roots near zero, one big root",
166*c8dee2aaSAndroid Build Coastguard Worker                        sk_bit_cast<double>(0xbf1a8de580000000), // -0.00010129655
167*c8dee2aaSAndroid Build Coastguard Worker                        sk_bit_cast<double>(0x4106c0c680000000), // 186392.8125
168*c8dee2aaSAndroid Build Coastguard Worker                        0.0,
169*c8dee2aaSAndroid Build Coastguard Worker                        sk_bit_cast<double>(0xc104c0ce80000000), // -170009.8125
170*c8dee2aaSAndroid Build Coastguard Worker                        { -0.9550418733785169, // Wolfram Alpha puts the root at X = 0.955042
171*c8dee2aaSAndroid Build Coastguard Worker                           0.9550418733785169, // (~2e7 error)
172*c8dee2aaSAndroid Build Coastguard Worker                          // 1.84007e9 is the other root, which we do not find.
173*c8dee2aaSAndroid Build Coastguard Worker                        },
174*c8dee2aaSAndroid Build Coastguard Worker                        true /* == skipPathops */, true /* == skipRootValidation */);
175*c8dee2aaSAndroid Build Coastguard Worker 
176*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "oss-fuzz:55625 Two roots near zero, one big root, near linear",
177*c8dee2aaSAndroid Build Coastguard Worker                        sk_bit_cast<double>(0x3c04040400000000), // -1.3563156-19
178*c8dee2aaSAndroid Build Coastguard Worker                        sk_bit_cast<double>(0x4106c0c680000000), // 186392.8125
179*c8dee2aaSAndroid Build Coastguard Worker                        0.0,
180*c8dee2aaSAndroid Build Coastguard Worker                        sk_bit_cast<double>(0xc104c0ce80000000), // -170009.8125
181*c8dee2aaSAndroid Build Coastguard Worker                        { -0.9550418733785169,
182*c8dee2aaSAndroid Build Coastguard Worker                           0.9550418733785169,
183*c8dee2aaSAndroid Build Coastguard Worker                          // 1.84007e9 is the other root, which we do not find.
184*c8dee2aaSAndroid Build Coastguard Worker                        },
185*c8dee2aaSAndroid Build Coastguard Worker                        true /* == skipPathops */);
186*c8dee2aaSAndroid Build Coastguard Worker 
187*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "oss-fuzz:55680 A nearly zero, C is zero",
188*c8dee2aaSAndroid Build Coastguard Worker                        sk_bit_cast<double>(0x3eb0000000000000), // 9.5367431640625000e-07
189*c8dee2aaSAndroid Build Coastguard Worker                        sk_bit_cast<double>(0x409278a560000000), // 1182.1614990234375
190*c8dee2aaSAndroid Build Coastguard Worker                        0.0,
191*c8dee2aaSAndroid Build Coastguard Worker                        sk_bit_cast<double>(0xc092706160000000), // -1180.0950927734375
192*c8dee2aaSAndroid Build Coastguard Worker                        { -0.9991256228290017,
193*c8dee2aaSAndroid Build Coastguard Worker                       // -0.9991256232316570469050229 according to Wolfram Alpha (~1e-09 error)
194*c8dee2aaSAndroid Build Coastguard Worker                           0.9991256228290017,
195*c8dee2aaSAndroid Build Coastguard Worker                        // 0.9991256224263463476403026 according to Wolfram Alpha (~1e-09 error)
196*c8dee2aaSAndroid Build Coastguard Worker                          // 1.239586176×10^9 is the other root, which we do not find.
197*c8dee2aaSAndroid Build Coastguard Worker                        },
198*c8dee2aaSAndroid Build Coastguard Worker                        true, true /* == skipRootValidation */);
199*c8dee2aaSAndroid Build Coastguard Worker }
200*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(CubicRootsReal_Quadratics,reporter)201*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(CubicRootsReal_Quadratics, reporter) {
202*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "two roots -2x^2 + 3x + 4",
203*c8dee2aaSAndroid Build Coastguard Worker                        0, -2, 3, 4,
204*c8dee2aaSAndroid Build Coastguard Worker                        {-0.8507810593582122,
205*c8dee2aaSAndroid Build Coastguard Worker                       //-0.8507810593582121716220544 from Wolfram Alpha
206*c8dee2aaSAndroid Build Coastguard Worker                          2.350781059358212,
207*c8dee2aaSAndroid Build Coastguard Worker                       // 2.350781059358212171622054 from Wolfram Alpha
208*c8dee2aaSAndroid Build Coastguard Worker                        });
209*c8dee2aaSAndroid Build Coastguard Worker 
210*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "touches y axis -x^2 + 3x + 4",
211*c8dee2aaSAndroid Build Coastguard Worker                        0, -2, 3, 4,
212*c8dee2aaSAndroid Build Coastguard Worker                        {-0.8507810593582122,
213*c8dee2aaSAndroid Build Coastguard Worker                       //-0.8507810593582121716220544 from Wolfram Alpha
214*c8dee2aaSAndroid Build Coastguard Worker                          2.350781059358212,
215*c8dee2aaSAndroid Build Coastguard Worker                       // 2.350781059358212171622054 from Wolfram Alpha
216*c8dee2aaSAndroid Build Coastguard Worker                        });
217*c8dee2aaSAndroid Build Coastguard Worker 
218*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "no roots x^2 + 2x + 7",
219*c8dee2aaSAndroid Build Coastguard Worker                        0, 1, 2, 7,
220*c8dee2aaSAndroid Build Coastguard Worker                        {});
221*c8dee2aaSAndroid Build Coastguard Worker 
222*c8dee2aaSAndroid Build Coastguard Worker     // similar to oss-fuzz:55680
223*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "two roots one small one big (and ignored)",
224*c8dee2aaSAndroid Build Coastguard Worker                        0, -0.01, 200000000000000, -120000000000000,
225*c8dee2aaSAndroid Build Coastguard Worker                        { 0.6 },
226*c8dee2aaSAndroid Build Coastguard Worker                         true /* == skipPathops */);
227*c8dee2aaSAndroid Build Coastguard Worker }
228*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(CubicRootsReal_Linear,reporter)229*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(CubicRootsReal_Linear, reporter) {
230*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "positive slope 3x + 4",
231*c8dee2aaSAndroid Build Coastguard Worker                        0, 0, 3, 4,
232*c8dee2aaSAndroid Build Coastguard Worker                        {-1.333333333333333});
233*c8dee2aaSAndroid Build Coastguard Worker 
234*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "negative slope -2x - 8",
235*c8dee2aaSAndroid Build Coastguard Worker                        0, 0, -2, -8,
236*c8dee2aaSAndroid Build Coastguard Worker                        {-4.});
237*c8dee2aaSAndroid Build Coastguard Worker }
238*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(CubicRootsReal_Constant,reporter)239*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(CubicRootsReal_Constant, reporter) {
240*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "No intersections y = 4",
241*c8dee2aaSAndroid Build Coastguard Worker                        0, 0, 0, 4,
242*c8dee2aaSAndroid Build Coastguard Worker                        {});
243*c8dee2aaSAndroid Build Coastguard Worker 
244*c8dee2aaSAndroid Build Coastguard Worker     testCubicRootsReal(reporter, "Infinite solutions y = 0",
245*c8dee2aaSAndroid Build Coastguard Worker                        0, 0, 0, 0,
246*c8dee2aaSAndroid Build Coastguard Worker                        {0.});
247*c8dee2aaSAndroid Build Coastguard Worker }
248*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(CubicRootsReal_NonFiniteNumbers,reporter)249*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(CubicRootsReal_NonFiniteNumbers, reporter) {
250*c8dee2aaSAndroid Build Coastguard Worker     // The Pathops implementation does not check for infinities nor nans in all cases.
251*c8dee2aaSAndroid Build Coastguard Worker     double roots[3] = {0, 0, 0};
252*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter,
253*c8dee2aaSAndroid Build Coastguard Worker         SkCubics::RootsReal(NAN, 1, 2, 3, roots) == 0,
254*c8dee2aaSAndroid Build Coastguard Worker         "Nan A"
255*c8dee2aaSAndroid Build Coastguard Worker     );
256*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter,
257*c8dee2aaSAndroid Build Coastguard Worker         SkCubics::RootsReal(1, NAN, 2, 3, roots) == 0,
258*c8dee2aaSAndroid Build Coastguard Worker         "Nan B"
259*c8dee2aaSAndroid Build Coastguard Worker     );
260*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter,
261*c8dee2aaSAndroid Build Coastguard Worker         SkCubics::RootsReal(1, 2, NAN, 3, roots) == 0,
262*c8dee2aaSAndroid Build Coastguard Worker         "Nan C"
263*c8dee2aaSAndroid Build Coastguard Worker     );
264*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter,
265*c8dee2aaSAndroid Build Coastguard Worker         SkCubics::RootsReal(1, 2, 3, NAN, roots) == 0,
266*c8dee2aaSAndroid Build Coastguard Worker         "Nan D"
267*c8dee2aaSAndroid Build Coastguard Worker     );
268*c8dee2aaSAndroid Build Coastguard Worker 
269*c8dee2aaSAndroid Build Coastguard Worker     {
270*c8dee2aaSAndroid Build Coastguard Worker         skiatest::ReporterContext subtest(reporter, "oss-fuzz:55419 C and D are large");
271*c8dee2aaSAndroid Build Coastguard Worker         int numRoots = SkCubics::RootsReal(
272*c8dee2aaSAndroid Build Coastguard Worker                                            -2, 0,
273*c8dee2aaSAndroid Build Coastguard Worker                                            sk_bit_cast<double>(0xd5422020202020ff), //-5.074559e+102
274*c8dee2aaSAndroid Build Coastguard Worker                                            sk_bit_cast<double>(0x600fff202020ff20), // 5.362551e+154
275*c8dee2aaSAndroid Build Coastguard Worker                                            roots);
276*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, numRoots == 0, "No finite roots expected, got %d", numRoots);
277*c8dee2aaSAndroid Build Coastguard Worker     }
278*c8dee2aaSAndroid Build Coastguard Worker     {
279*c8dee2aaSAndroid Build Coastguard Worker         skiatest::ReporterContext subtest(reporter, "oss-fuzz:55829 A is zero and B is NAN");
280*c8dee2aaSAndroid Build Coastguard Worker         int numRoots = SkCubics::RootsReal(
281*c8dee2aaSAndroid Build Coastguard Worker                                            0,
282*c8dee2aaSAndroid Build Coastguard Worker                                            sk_bit_cast<double>(0xffffffffffff2020), //-nan
283*c8dee2aaSAndroid Build Coastguard Worker                                            sk_bit_cast<double>(0x20202020202020ff), // 6.013470e-154
284*c8dee2aaSAndroid Build Coastguard Worker                                            sk_bit_cast<double>(0xff20202020202020), //-2.211661e+304
285*c8dee2aaSAndroid Build Coastguard Worker                                            roots);
286*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, numRoots == 0, "No finite roots expected, got %d", numRoots);
287*c8dee2aaSAndroid Build Coastguard Worker     }
288*c8dee2aaSAndroid Build Coastguard Worker }
289*c8dee2aaSAndroid Build Coastguard Worker 
testCubicValidT(skiatest::Reporter * reporter,std::string name,double A,double B,double C,double D,SkSpan<const double> expectedRoots)290*c8dee2aaSAndroid Build Coastguard Worker static void testCubicValidT(skiatest::Reporter* reporter, std::string name,
291*c8dee2aaSAndroid Build Coastguard Worker                             double A, double B, double C, double D,
292*c8dee2aaSAndroid Build Coastguard Worker                             SkSpan<const double> expectedRoots) {
293*c8dee2aaSAndroid Build Coastguard Worker     skiatest::ReporterContext subtest(reporter, name);
294*c8dee2aaSAndroid Build Coastguard Worker     // Validate test case
295*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, expectedRoots.size() <= 3,
296*c8dee2aaSAndroid Build Coastguard Worker                     "Invalid test case, up to 3 roots allowed");
297*c8dee2aaSAndroid Build Coastguard Worker 
298*c8dee2aaSAndroid Build Coastguard Worker     for (size_t i = 0; i < expectedRoots.size(); i++) {
299*c8dee2aaSAndroid Build Coastguard Worker         double x = expectedRoots[i];
300*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, x >= 0 && x <= 1,
301*c8dee2aaSAndroid Build Coastguard Worker                         "Invalid test case root %zu. Roots must be in [0, 1]", i);
302*c8dee2aaSAndroid Build Coastguard Worker 
303*c8dee2aaSAndroid Build Coastguard Worker         // A*x^3 + B*x^2 + C*x + D should equal 0
304*c8dee2aaSAndroid Build Coastguard Worker         double y = A * x * x * x + B * x * x + C * x + D;
305*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, sk_double_nearly_zero(y),
306*c8dee2aaSAndroid Build Coastguard Worker                     "Invalid test case root %zu. %.16f != 0", i, y);
307*c8dee2aaSAndroid Build Coastguard Worker 
308*c8dee2aaSAndroid Build Coastguard Worker         if (i > 0) {
309*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, expectedRoots[i-1] <= expectedRoots[i],
310*c8dee2aaSAndroid Build Coastguard Worker                     "Invalid test case root %zu. Roots should be sorted in ascending order", i);
311*c8dee2aaSAndroid Build Coastguard Worker         }
312*c8dee2aaSAndroid Build Coastguard Worker     }
313*c8dee2aaSAndroid Build Coastguard Worker 
314*c8dee2aaSAndroid Build Coastguard Worker     {
315*c8dee2aaSAndroid Build Coastguard Worker         skiatest::ReporterContext subsubtest(reporter, "Pathops Implementation");
316*c8dee2aaSAndroid Build Coastguard Worker         double roots[3] = {0, 0, 0};
317*c8dee2aaSAndroid Build Coastguard Worker         int rootCount = SkDCubic::RootsValidT(A, B, C, D, roots);
318*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, expectedRoots.size() == size_t(rootCount),
319*c8dee2aaSAndroid Build Coastguard Worker                         "Wrong number of roots returned %zu != %d",
320*c8dee2aaSAndroid Build Coastguard Worker                         expectedRoots.size(), rootCount);
321*c8dee2aaSAndroid Build Coastguard Worker 
322*c8dee2aaSAndroid Build Coastguard Worker         // We don't care which order the roots are returned from the algorithm.
323*c8dee2aaSAndroid Build Coastguard Worker         // For determinism, we will sort them (and ensure the provided solutions are also sorted).
324*c8dee2aaSAndroid Build Coastguard Worker         std::sort(std::begin(roots), std::begin(roots) + rootCount);
325*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < rootCount; i++) {
326*c8dee2aaSAndroid Build Coastguard Worker             if (sk_double_nearly_zero(expectedRoots[i])) {
327*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(reporter, sk_double_nearly_zero(roots[i]),
328*c8dee2aaSAndroid Build Coastguard Worker                                 "0 != %.16f at index %d", roots[i], i);
329*c8dee2aaSAndroid Build Coastguard Worker             } else {
330*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(reporter,
331*c8dee2aaSAndroid Build Coastguard Worker                                 sk_doubles_nearly_equal_ulps(expectedRoots[i], roots[i], 64),
332*c8dee2aaSAndroid Build Coastguard Worker                                 "%.16f != %.16f at index %d", expectedRoots[i], roots[i], i);
333*c8dee2aaSAndroid Build Coastguard Worker             }
334*c8dee2aaSAndroid Build Coastguard Worker         }
335*c8dee2aaSAndroid Build Coastguard Worker     }
336*c8dee2aaSAndroid Build Coastguard Worker     {
337*c8dee2aaSAndroid Build Coastguard Worker         skiatest::ReporterContext subsubtest(reporter, "SkCubics Analytic Implementation");
338*c8dee2aaSAndroid Build Coastguard Worker         double roots[3] = {0, 0, 0};
339*c8dee2aaSAndroid Build Coastguard Worker         int rootCount = SkCubics::RootsValidT(A, B, C, D, roots);
340*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, expectedRoots.size() == size_t(rootCount),
341*c8dee2aaSAndroid Build Coastguard Worker                         "Wrong number of roots returned %zu != %d",
342*c8dee2aaSAndroid Build Coastguard Worker                         expectedRoots.size(), rootCount);
343*c8dee2aaSAndroid Build Coastguard Worker 
344*c8dee2aaSAndroid Build Coastguard Worker         // We don't care which order the roots are returned from the algorithm.
345*c8dee2aaSAndroid Build Coastguard Worker         // For determinism, we will sort them (and ensure the provided solutions are also sorted).
346*c8dee2aaSAndroid Build Coastguard Worker         std::sort(std::begin(roots), std::begin(roots) + rootCount);
347*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < rootCount; i++) {
348*c8dee2aaSAndroid Build Coastguard Worker             if (sk_double_nearly_zero(expectedRoots[i])) {
349*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(reporter, sk_double_nearly_zero(roots[i]),
350*c8dee2aaSAndroid Build Coastguard Worker                                 "0 != %.16f at index %d", roots[i], i);
351*c8dee2aaSAndroid Build Coastguard Worker             } else {
352*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(reporter,
353*c8dee2aaSAndroid Build Coastguard Worker                                 sk_doubles_nearly_equal_ulps(expectedRoots[i], roots[i], 64),
354*c8dee2aaSAndroid Build Coastguard Worker                                 "%.16f != %.16f at index %d", expectedRoots[i], roots[i], i);
355*c8dee2aaSAndroid Build Coastguard Worker             }
356*c8dee2aaSAndroid Build Coastguard Worker         }
357*c8dee2aaSAndroid Build Coastguard Worker     }
358*c8dee2aaSAndroid Build Coastguard Worker     {
359*c8dee2aaSAndroid Build Coastguard Worker         skiatest::ReporterContext subsubtest(reporter, "SkCubics Binary Search Implementation");
360*c8dee2aaSAndroid Build Coastguard Worker         double roots[3] = {0, 0, 0};
361*c8dee2aaSAndroid Build Coastguard Worker         int rootCount = SkCubics::BinarySearchRootsValidT(A, B, C, D, roots);
362*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, expectedRoots.size() == size_t(rootCount),
363*c8dee2aaSAndroid Build Coastguard Worker                         "Wrong number of roots returned %zu != %d", expectedRoots.size(),
364*c8dee2aaSAndroid Build Coastguard Worker                         rootCount);
365*c8dee2aaSAndroid Build Coastguard Worker 
366*c8dee2aaSAndroid Build Coastguard Worker         // We don't care which order the roots are returned from the algorithm.
367*c8dee2aaSAndroid Build Coastguard Worker         // For determinism, we will sort them (and ensure the provided solutions are also sorted).
368*c8dee2aaSAndroid Build Coastguard Worker         std::sort(std::begin(roots), std::begin(roots) + rootCount);
369*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < rootCount; i++) {
370*c8dee2aaSAndroid Build Coastguard Worker             double delta = std::abs(roots[i] - expectedRoots[i]);
371*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter,
372*c8dee2aaSAndroid Build Coastguard Worker                             // Binary search is not absolutely accurate all the time, but
373*c8dee2aaSAndroid Build Coastguard Worker                             // it should be accurate enough reliably
374*c8dee2aaSAndroid Build Coastguard Worker                             delta < 0.000001,
375*c8dee2aaSAndroid Build Coastguard Worker                             "%.16f != %.16f at index %d", expectedRoots[i], roots[i], i);
376*c8dee2aaSAndroid Build Coastguard Worker         }
377*c8dee2aaSAndroid Build Coastguard Worker     }
378*c8dee2aaSAndroid Build Coastguard Worker }
379*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(CubicRootsValidT,reporter)380*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(CubicRootsValidT, reporter) {
381*c8dee2aaSAndroid Build Coastguard Worker     // All answers are given with 16 significant digits (max for a double) or as an integer
382*c8dee2aaSAndroid Build Coastguard Worker     // when the answer is exact.
383*c8dee2aaSAndroid Build Coastguard Worker     testCubicValidT(reporter, "three roots 24x^3 - 46x^2 + 29x - 6",
384*c8dee2aaSAndroid Build Coastguard Worker                     24, -46, 29, -6,
385*c8dee2aaSAndroid Build Coastguard Worker                     {0.5,
386*c8dee2aaSAndroid Build Coastguard Worker                      0.6666666666666667,
387*c8dee2aaSAndroid Build Coastguard Worker                      0.75});
388*c8dee2aaSAndroid Build Coastguard Worker 
389*c8dee2aaSAndroid Build Coastguard Worker     testCubicValidT(reporter, "three roots total, two in range 54x^3 - 117x^2 + 45x + 0",
390*c8dee2aaSAndroid Build Coastguard Worker                     54, -117, 45, 0,
391*c8dee2aaSAndroid Build Coastguard Worker                     {0.0,
392*c8dee2aaSAndroid Build Coastguard Worker                      0.5,
393*c8dee2aaSAndroid Build Coastguard Worker                      // 5/3 is the other root, but not in [0, 1]
394*c8dee2aaSAndroid Build Coastguard Worker                     });
395*c8dee2aaSAndroid Build Coastguard Worker 
396*c8dee2aaSAndroid Build Coastguard Worker     testCubicValidT(reporter, "one root = 1 10x^3 - 20x^2 - 30x + 40",
397*c8dee2aaSAndroid Build Coastguard Worker                     10, -20, -30, 40,
398*c8dee2aaSAndroid Build Coastguard Worker                     {1.0});
399*c8dee2aaSAndroid Build Coastguard Worker 
400*c8dee2aaSAndroid Build Coastguard Worker     testCubicValidT(reporter, "one root = 0 2x^3 - 3x^2 - 4x + 0",
401*c8dee2aaSAndroid Build Coastguard Worker                     2, -3, -4, 0,
402*c8dee2aaSAndroid Build Coastguard Worker                     {0.0});
403*c8dee2aaSAndroid Build Coastguard Worker 
404*c8dee2aaSAndroid Build Coastguard Worker     testCubicValidT(reporter, "three roots total, two in range -2x^3 - 3x^2 + 4x + 0",
405*c8dee2aaSAndroid Build Coastguard Worker                     -2, -3, 4, 0,
406*c8dee2aaSAndroid Build Coastguard Worker                     { 0.0,
407*c8dee2aaSAndroid Build Coastguard Worker                       0.8507810593582122,
408*c8dee2aaSAndroid Build Coastguard Worker                    // 0.8507810593582121716220544 from Wolfram Alpha
409*c8dee2aaSAndroid Build Coastguard Worker                     });
410*c8dee2aaSAndroid Build Coastguard Worker 
411*c8dee2aaSAndroid Build Coastguard Worker     // x(x-1) = x^2 - x
412*c8dee2aaSAndroid Build Coastguard Worker     testCubicValidT(reporter, "Two roots at exactly 0 and 1",
413*c8dee2aaSAndroid Build Coastguard Worker                     0, 1, -1, 0,
414*c8dee2aaSAndroid Build Coastguard Worker                     {0.0, 1.0});
415*c8dee2aaSAndroid Build Coastguard Worker 
416*c8dee2aaSAndroid Build Coastguard Worker     testCubicValidT(reporter, "Single point has one root",
417*c8dee2aaSAndroid Build Coastguard Worker                     0, 0, 0, 0,
418*c8dee2aaSAndroid Build Coastguard Worker                     {0.0});
419*c8dee2aaSAndroid Build Coastguard Worker }
420*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(CubicRootsValidT_ClampToZeroAndOne,reporter)421*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(CubicRootsValidT_ClampToZeroAndOne, reporter) {
422*c8dee2aaSAndroid Build Coastguard Worker     {
423*c8dee2aaSAndroid Build Coastguard Worker         // (x + 0.00001)(x - 1.00005), but the answers will be 0 and 1
424*c8dee2aaSAndroid Build Coastguard Worker         double A = 0.;
425*c8dee2aaSAndroid Build Coastguard Worker         double B = 1.;
426*c8dee2aaSAndroid Build Coastguard Worker         double C = -1.00004;
427*c8dee2aaSAndroid Build Coastguard Worker         double D = -0.0000100005;
428*c8dee2aaSAndroid Build Coastguard Worker         double roots[3] = {0, 0, 0};
429*c8dee2aaSAndroid Build Coastguard Worker         int rootCount = SkDCubic::RootsValidT(A, B, C, D, roots);
430*c8dee2aaSAndroid Build Coastguard Worker 
431*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, rootCount == 2);
432*c8dee2aaSAndroid Build Coastguard Worker         std::sort(std::begin(roots), std::begin(roots) + rootCount);
433*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, sk_double_nearly_zero(roots[0]), "%.16f != 0", roots[0]);
434*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, sk_doubles_nearly_equal_ulps(roots[1], 1), "%.16f != 1", roots[1]);
435*c8dee2aaSAndroid Build Coastguard Worker     }
436*c8dee2aaSAndroid Build Coastguard Worker     {
437*c8dee2aaSAndroid Build Coastguard Worker         // Three very small roots, all of them are nearly equal zero
438*c8dee2aaSAndroid Build Coastguard Worker         // (1 - 10000000000x)(1 - 20000000000x)(1 - 30000000000x)
439*c8dee2aaSAndroid Build Coastguard Worker         // -6000000000000000000000000000000 x^3 + 1100000000000000000000 x^2 - 60000000000 x + 1
440*c8dee2aaSAndroid Build Coastguard Worker         double A = -6.0e30;
441*c8dee2aaSAndroid Build Coastguard Worker         double B = 1.1e21;
442*c8dee2aaSAndroid Build Coastguard Worker         double C = -6.0e10;
443*c8dee2aaSAndroid Build Coastguard Worker         double D = 1;
444*c8dee2aaSAndroid Build Coastguard Worker         double roots[3] = {0, 0, 0};
445*c8dee2aaSAndroid Build Coastguard Worker         int rootCount = SkDCubic::RootsValidT(A, B, C, D, roots);
446*c8dee2aaSAndroid Build Coastguard Worker 
447*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, rootCount == 1);
448*c8dee2aaSAndroid Build Coastguard Worker         std::sort(std::begin(roots), std::begin(roots) + rootCount);
449*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, sk_double_nearly_zero(roots[0]), "%.16f != 0", roots[0]);
450*c8dee2aaSAndroid Build Coastguard Worker     }
451*c8dee2aaSAndroid Build Coastguard Worker }
452