|  | #include "CubicIntersection_TestData.h" | 
|  | #include "CubicUtilities.h" | 
|  | #include "Intersection_Tests.h" | 
|  | #include "QuadraticIntersection_TestData.h" | 
|  | #include "TestUtilities.h" | 
|  | #include "SkGeometry.h" | 
|  |  | 
|  | static void test(const Cubic* cubics, const char* name, int firstTest, size_t testCount) { | 
|  | SkTDArray<Quadratic> quads; | 
|  | for (size_t index = firstTest; index < testCount; ++index) { | 
|  | const Cubic& cubic = cubics[index]; | 
|  | double precision = calcPrecision(cubic); | 
|  | (void) cubic_to_quadratics(cubic, precision, quads); | 
|  | if (quads.count() != 1) { | 
|  | printf("%s [%d] cubic to quadratics failed count=%d\n", name, (int) index, | 
|  | quads.count()); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | static void test(const Quadratic* quadTests, const char* name, int firstTest, size_t testCount) { | 
|  | SkTDArray<Quadratic> quads; | 
|  | for (size_t index = firstTest; index < testCount; ++index) { | 
|  | const Quadratic& quad = quadTests[index]; | 
|  | Cubic cubic; | 
|  | quad_to_cubic(quad, cubic); | 
|  | double precision = calcPrecision(cubic); | 
|  | (void) cubic_to_quadratics(cubic, precision, quads); | 
|  | if (quads.count() != 1) { | 
|  | printf("%s [%d] cubic to quadratics failed count=%d\n", name, (int) index, | 
|  | quads.count()); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | static void testC(const Cubic* cubics, const char* name, int firstTest, size_t testCount) { | 
|  | SkTDArray<Quadratic> quads; | 
|  | // test if computed line end points are valid | 
|  | for (size_t index = firstTest; index < testCount; ++index) { | 
|  | const Cubic& cubic = cubics[index]; | 
|  | double precision = calcPrecision(cubic); | 
|  | int order = cubic_to_quadratics(cubic, precision, quads); | 
|  | SkASSERT(order != 4); | 
|  | if (order < 3) { | 
|  | continue; | 
|  | } | 
|  | if (!AlmostEqualUlps(cubic[0].x, quads[0][0].x) | 
|  | || !AlmostEqualUlps(cubic[0].y, quads[0][0].y)) { | 
|  | printf("[%d] unmatched start\n", (int) index); | 
|  | } | 
|  | int last = quads.count() - 1; | 
|  | if (!AlmostEqualUlps(cubic[3].x, quads[last][2].x) | 
|  | || !AlmostEqualUlps(cubic[3].y, quads[last][2].y)) { | 
|  | printf("[%d] unmatched end\n", (int) index); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | static void testC(const Cubic(* cubics)[2], const char* name, int firstTest, size_t testCount) { | 
|  | SkTDArray<Quadratic> quads; | 
|  | for (size_t index = firstTest; index < testCount; ++index) { | 
|  | for (int idx2 = 0; idx2 < 2; ++idx2) { | 
|  | const Cubic& cubic = cubics[index][idx2]; | 
|  | double precision = calcPrecision(cubic); | 
|  | int order = cubic_to_quadratics(cubic, precision, quads); | 
|  | SkASSERT(order != 4); | 
|  | if (order < 3) { | 
|  | continue; | 
|  | } | 
|  | if (!AlmostEqualUlps(cubic[0].x, quads[0][0].x) | 
|  | || !AlmostEqualUlps(cubic[0].y, quads[0][0].y)) { | 
|  | printf("[%d][%d] unmatched start\n", (int) index, idx2); | 
|  | } | 
|  | int last = quads.count() - 1; | 
|  | if (!AlmostEqualUlps(cubic[3].x, quads[last][2].x) | 
|  | || !AlmostEqualUlps(cubic[3].y, quads[last][2].y)) { | 
|  | printf("[%d][%d] unmatched end\n", (int) index, idx2); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void CubicToQuadratics_Test() { | 
|  | enum { | 
|  | RunAll, | 
|  | RunPointDegenerates, | 
|  | RunNotPointDegenerates, | 
|  | RunLines, | 
|  | RunNotLines, | 
|  | RunModEpsilonLines, | 
|  | RunLessEpsilonLines, | 
|  | RunNegEpsilonLines, | 
|  | RunQuadraticLines, | 
|  | RunQuadraticModLines, | 
|  | RunComputedLines, | 
|  | RunComputedTests, | 
|  | RunNone | 
|  | } run = RunAll; | 
|  | int firstTestIndex = 0; | 
|  | #if 0 | 
|  | run = RunComputedLines; | 
|  | firstTestIndex = 18; | 
|  | #endif | 
|  | int firstPointDegeneratesTest = run == RunAll ? 0 : run == RunPointDegenerates ? firstTestIndex : SK_MaxS32; | 
|  | int firstNotPointDegeneratesTest = run == RunAll ? 0 : run == RunNotPointDegenerates ? firstTestIndex : SK_MaxS32; | 
|  | int firstLinesTest = run == RunAll ? 0 : run == RunLines ? firstTestIndex : SK_MaxS32; | 
|  | int firstNotLinesTest = run == RunAll ? 0 : run == RunNotLines ? firstTestIndex : SK_MaxS32; | 
|  | int firstModEpsilonTest = run == RunAll ? 0 : run == RunModEpsilonLines ? firstTestIndex : SK_MaxS32; | 
|  | int firstLessEpsilonTest = run == RunAll ? 0 : run == RunLessEpsilonLines ? firstTestIndex : SK_MaxS32; | 
|  | int firstNegEpsilonTest = run == RunAll ? 0 : run == RunNegEpsilonLines ? firstTestIndex : SK_MaxS32; | 
|  | int firstQuadraticLineTest = run == RunAll ? 0 : run == RunQuadraticLines ? firstTestIndex : SK_MaxS32; | 
|  | int firstQuadraticModLineTest = run == RunAll ? 0 : run == RunQuadraticModLines ? firstTestIndex : SK_MaxS32; | 
|  | int firstComputedLinesTest = run == RunAll ? 0 : run == RunComputedLines ? firstTestIndex : SK_MaxS32; | 
|  | int firstComputedCubicsTest = run == RunAll ? 0 : run == RunComputedTests ? firstTestIndex : SK_MaxS32; | 
|  |  | 
|  | test(pointDegenerates, "pointDegenerates", firstPointDegeneratesTest, pointDegenerates_count); | 
|  | test(notPointDegenerates, "notPointDegenerates", firstNotPointDegeneratesTest, notPointDegenerates_count); | 
|  | test(lines, "lines", firstLinesTest, lines_count); | 
|  | test(notLines, "notLines", firstNotLinesTest, notLines_count); | 
|  | test(modEpsilonLines, "modEpsilonLines", firstModEpsilonTest, modEpsilonLines_count); | 
|  | test(lessEpsilonLines, "lessEpsilonLines", firstLessEpsilonTest, lessEpsilonLines_count); | 
|  | test(negEpsilonLines, "negEpsilonLines", firstNegEpsilonTest, negEpsilonLines_count); | 
|  | test(quadraticLines, "quadraticLines", firstQuadraticLineTest, quadraticLines_count); | 
|  | test(quadraticModEpsilonLines, "quadraticModEpsilonLines", firstQuadraticModLineTest, | 
|  | quadraticModEpsilonLines_count); | 
|  | testC(lines, "computed lines", firstComputedLinesTest, lines_count); | 
|  | testC(tests, "computed tests", firstComputedCubicsTest, tests_count); | 
|  | printf("%s end\n", __FUNCTION__); | 
|  | } | 
|  |  | 
|  | static Cubic locals[] = { | 
|  | {{0, 1}, {1.9274705288631189e-19, 1.0000000000000002}, {0.0017190297609673323, 0.99828097023903239}, | 
|  | {0.0053709083094631276, 0.99505672974365911}}, | 
|  |  | 
|  | {{14.5975863, 41.632436}, {16.3518929, 26.2639684}, {18.5165519, 7.68775139}, {8.03767257, 89.1628526}}, | 
|  | {{69.7292014, 38.6877352}, {24.7648688, 23.1501713}, {84.9283191, 90.2588441}, {80.392774, 61.3533852}}, | 
|  | {{ | 
|  | 60.776536520932126, | 
|  | 71.249307306133829 | 
|  | }, { | 
|  | 87.107894191103014, | 
|  | 22.377669868235323 | 
|  | }, { | 
|  | 1.4974754310666936, | 
|  | 68.069569937917208 | 
|  | }, { | 
|  | 45.261946574441133, | 
|  | 17.536076632112298 | 
|  | }}, | 
|  | }; | 
|  |  | 
|  | static size_t localsCount = sizeof(locals) / sizeof(locals[0]); | 
|  |  | 
|  | #define DEBUG_CRASH 0 | 
|  | #define TEST_AVERAGE_END_POINTS 0 // must take const off to test | 
|  | extern const bool AVERAGE_END_POINTS; | 
|  |  | 
|  | static void oneOff(size_t x) { | 
|  | const Cubic& cubic = locals[x]; | 
|  | const SkPoint skcubic[4] = {{(float) cubic[0].x, (float) cubic[0].y}, | 
|  | {(float) cubic[1].x, (float) cubic[1].y}, {(float) cubic[2].x, (float) cubic[2].y}, | 
|  | {(float) cubic[3].x, (float) cubic[3].y}}; | 
|  | SkScalar skinflect[2]; | 
|  | int skin = SkFindCubicInflections(skcubic, skinflect); | 
|  | SkDebugf("%s %d %1.9g\n", __FUNCTION__, skin, skinflect[0]); | 
|  | SkTDArray<Quadratic> quads; | 
|  | double precision = calcPrecision(cubic); | 
|  | (void) cubic_to_quadratics(cubic, precision, quads); | 
|  | SkDebugf("%s quads=%d\n", __FUNCTION__, quads.count()); | 
|  | } | 
|  |  | 
|  | void CubicsToQuadratics_OneOffTests() { | 
|  | for (size_t x = 0; x < localsCount; ++x) { | 
|  | oneOff(x); | 
|  | } | 
|  | } | 
|  |  | 
|  | void CubicsToQuadratics_OneOffTest() { | 
|  | oneOff(0); | 
|  | } | 
|  |  | 
|  | void CubicsToQuadratics_RandTest() { | 
|  | srand(0); | 
|  | const int arrayMax = 8; | 
|  | const int sampleMax = 10; | 
|  | const int tests = 1000000; // 10000000; | 
|  | int quadDist[arrayMax]; | 
|  | bzero(quadDist, sizeof(quadDist)); | 
|  | Cubic samples[arrayMax][sampleMax]; | 
|  | int sampleCount[arrayMax]; | 
|  | bzero(sampleCount, sizeof(sampleCount)); | 
|  | for (int x = 0; x < tests; ++x) { | 
|  | Cubic cubic; | 
|  | for (int i = 0; i < 4; ++i) { | 
|  | cubic[i].x = (double) rand() / RAND_MAX * 100; | 
|  | cubic[i].y = (double) rand() / RAND_MAX * 100; | 
|  | } | 
|  | #if DEBUG_CRASH | 
|  | char str[1024]; | 
|  | sprintf(str, "{{%1.9g, %1.9g}, {%1.9g, %1.9g}, {%1.9g, %1.9g}, {%1.9g, %1.9g}},\n", | 
|  | cubic[0].x, cubic[0].y,  cubic[1].x, cubic[1].y, cubic[2].x, cubic[2].y, | 
|  | cubic[3].x, cubic[3].y); | 
|  | #endif | 
|  | SkTDArray<Quadratic> quads; | 
|  | double precision = calcPrecision(cubic); | 
|  | (void) cubic_to_quadratics(cubic, precision, quads); | 
|  | int count = quads.count(); | 
|  | SkASSERT(count > 0); | 
|  | SkASSERT(--count < arrayMax); | 
|  | quadDist[count]++; | 
|  | int sCount = sampleCount[count]; | 
|  | if (sCount < sampleMax) { | 
|  | memcpy(samples[count][sCount], cubic, sizeof(Cubic)); | 
|  | sampleCount[count]++; | 
|  | } | 
|  | } | 
|  | for (int x = 0; x < arrayMax; ++x) { | 
|  | if (!quadDist[x]) { | 
|  | continue; | 
|  | } | 
|  | SkDebugf("%d %1.9g%%\n", x + 1, (double) quadDist[x] / tests * 100); | 
|  | } | 
|  | SkDebugf("\n"); | 
|  | for (int x = 0; x < arrayMax; ++x) { | 
|  | for (int y = 0; y < sampleCount[x]; ++y) { | 
|  | #if TEST_AVERAGE_END_POINTS | 
|  | for (int w = 0; w < 2; ++w) { | 
|  | AVERAGE_END_POINTS = w; | 
|  | #else | 
|  | int w = 0; | 
|  | #endif | 
|  | SkDebugf("<div id=\"cubic%dx%d%s\">\n", x + 1, y, w ? "x" : ""); | 
|  | const Cubic& cubic = samples[x][y]; | 
|  | SkDebugf("{{%1.9g, %1.9g}, {%1.9g, %1.9g}, {%1.9g, %1.9g}, {%1.9g, %1.9g}},\n", | 
|  | cubic[0].x, cubic[0].y,  cubic[1].x, cubic[1].y, cubic[2].x, cubic[2].y, | 
|  | cubic[3].x, cubic[3].y); | 
|  | SkTDArray<Quadratic> quads; | 
|  | double precision = calcPrecision(cubic); | 
|  | (void) cubic_to_quadratics(cubic, precision, quads); | 
|  | for (int z = 0; z < quads.count(); ++z) { | 
|  | const Quadratic& quad = quads[z]; | 
|  | SkDebugf("{{%1.9g, %1.9g}, {%1.9g, %1.9g}, {%1.9g, %1.9g}},\n", | 
|  | quad[0].x, quad[0].y,  quad[1].x, quad[1].y, quad[2].x, quad[2].y); | 
|  | } | 
|  | SkDebugf("</div>\n\n"); | 
|  | #if TEST_AVERAGE_END_POINTS | 
|  | } | 
|  | #endif | 
|  | } | 
|  | } | 
|  | SkDebugf("</div>\n\n"); | 
|  | SkDebugf("<script type=\"text/javascript\">\n\n"); | 
|  | SkDebugf("var testDivs = [\n"); | 
|  | for (int x = 0; x < arrayMax; ++x) { | 
|  | for (int y = 0; y < sampleCount[x]; ++y) { | 
|  | #if TEST_AVERAGE_END_POINTS | 
|  | for (int w = 0; w < 2; ++w) { | 
|  | #else | 
|  | int w = 0; | 
|  | #endif | 
|  | SkDebugf("    cubic%dx%d%s,\n", x + 1, y, w ? "x" : ""); | 
|  | #if TEST_AVERAGE_END_POINTS | 
|  | } | 
|  | #endif | 
|  | } | 
|  | } | 
|  | SkDebugf("\n\n\n"); | 
|  | SkDebugf("%s end\n", __FUNCTION__); | 
|  | } |