| /* |
| * Copyright 2014 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "include/core/SkPaint.h" |
| #include "include/core/SkPath.h" |
| #include "include/core/SkTime.h" |
| #include "include/utils/SkRandom.h" |
| #include "src/core/SkPointPriv.h" |
| #include "src/core/SkStrokerPriv.h" |
| #include "src/pathops/SkPathOpsCubic.h" |
| #include "tests/PathOpsCubicIntersectionTestData.h" |
| #include "tests/PathOpsQuadIntersectionTestData.h" |
| #include "tests/Test.h" |
| #include "tools/flags/CommandLineFlags.h" |
| |
| static DEFINE_bool(timeout, true, "run until alloted time expires"); |
| |
| #define MS_TEST_DURATION 10 |
| |
| const SkScalar widths[] = {-FLT_MAX, -1, -0.1f, -FLT_EPSILON, 0, FLT_EPSILON, |
| 0.0000001f, 0.000001f, 0.00001f, 0.0001f, 0.001f, 0.01f, |
| 0.1f, 0.2f, 0.3f, 0.4f, 0.5f, 1, 1.1f, 2, 10, 10e2f, 10e3f, 10e4f, 10e5f, 10e6f, 10e7f, |
| 10e8f, 10e9f, 10e10f, 10e20f, FLT_MAX }; |
| size_t widths_count = SK_ARRAY_COUNT(widths); |
| |
| static void pathTest(const SkPath& path) { |
| SkPaint p; |
| SkPath fill; |
| p.setStyle(SkPaint::kStroke_Style); |
| for (size_t index = 0; index < widths_count; ++index) { |
| p.setStrokeWidth(widths[index]); |
| p.getFillPath(path, &fill); |
| } |
| } |
| |
| static void cubicTest(const SkPoint c[4]) { |
| SkPath path; |
| path.moveTo(c[0].fX, c[0].fY); |
| path.cubicTo(c[1].fX, c[1].fY, c[2].fX, c[2].fY, c[3].fX, c[3].fY); |
| pathTest(path); |
| } |
| |
| static void quadTest(const SkPoint c[3]) { |
| SkPath path; |
| path.moveTo(c[0].fX, c[0].fY); |
| path.quadTo(c[1].fX, c[1].fY, c[2].fX, c[2].fY); |
| pathTest(path); |
| } |
| |
| static void cubicSetTest(const CubicPts* dCubic, size_t count) { |
| skiatest::Timer timer; |
| for (size_t index = 0; index < count; ++index) { |
| const CubicPts& dPts = dCubic[index]; |
| SkDCubic d; |
| d.debugSet(dPts.fPts); |
| SkPoint c[4] = { {(float) d[0].fX, (float) d[0].fY}, {(float) d[1].fX, (float) d[1].fY}, |
| {(float) d[2].fX, (float) d[2].fY}, {(float) d[3].fX, (float) d[3].fY} }; |
| cubicTest(c); |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| } |
| |
| static void cubicPairSetTest(const CubicPts dCubic[][2], size_t count) { |
| skiatest::Timer timer; |
| for (size_t index = 0; index < count; ++index) { |
| for (int pair = 0; pair < 2; ++pair) { |
| const CubicPts& dPts = dCubic[index][pair]; |
| SkDCubic d; |
| d.debugSet(dPts.fPts); |
| SkPoint c[4] = { {(float) d[0].fX, (float) d[0].fY}, {(float) d[1].fX, (float) d[1].fY}, |
| {(float) d[2].fX, (float) d[2].fY}, {(float) d[3].fX, (float) d[3].fY} }; |
| cubicTest(c); |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| } |
| } |
| |
| static void quadSetTest(const QuadPts* dQuad, size_t count) { |
| skiatest::Timer timer; |
| for (size_t index = 0; index < count; ++index) { |
| const QuadPts& dPts = dQuad[index]; |
| SkDQuad d; |
| d.debugSet(dPts.fPts); |
| SkPoint c[3] = { {(float) d[0].fX, (float) d[0].fY}, {(float) d[1].fX, (float) d[1].fY}, |
| {(float) d[2].fX, (float) d[2].fY} }; |
| quadTest(c); |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| } |
| |
| static void quadPairSetTest(const QuadPts dQuad[][2], size_t count) { |
| skiatest::Timer timer; |
| for (size_t index = 0; index < count; ++index) { |
| for (int pair = 0; pair < 2; ++pair) { |
| const QuadPts& dPts = dQuad[index][pair]; |
| SkDQuad d; |
| d.debugSet(dPts.fPts); |
| SkPoint c[3] = { {(float) d[0].fX, (float) d[0].fY}, {(float) d[1].fX, (float) d[1].fY}, |
| {(float) d[2].fX, (float) d[2].fY} }; |
| quadTest(c); |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| } |
| } |
| |
| DEF_TEST(QuadStrokerSet, reporter) { |
| quadSetTest(quadraticLines, quadraticLines_count); |
| quadSetTest(quadraticPoints, quadraticPoints_count); |
| quadSetTest(quadraticModEpsilonLines, quadraticModEpsilonLines_count); |
| quadPairSetTest(quadraticTests, quadraticTests_count); |
| } |
| |
| DEF_TEST(CubicStrokerSet, reporter) { |
| cubicSetTest(pointDegenerates, pointDegenerates_count); |
| cubicSetTest(notPointDegenerates, notPointDegenerates_count); |
| cubicSetTest(lines, lines_count); |
| cubicSetTest(notLines, notLines_count); |
| cubicSetTest(modEpsilonLines, modEpsilonLines_count); |
| cubicSetTest(lessEpsilonLines, lessEpsilonLines_count); |
| cubicSetTest(negEpsilonLines, negEpsilonLines_count); |
| cubicPairSetTest(tests, tests_count); |
| } |
| |
| static SkScalar unbounded(SkRandom& r) { |
| uint32_t val = r.nextU(); |
| return SkBits2Float(val); |
| } |
| |
| static SkScalar unboundedPos(SkRandom& r) { |
| uint32_t val = r.nextU() & 0x7fffffff; |
| return SkBits2Float(val); |
| } |
| |
| DEF_TEST(QuadStrokerUnbounded, reporter) { |
| SkRandom r; |
| SkPaint p; |
| p.setStyle(SkPaint::kStroke_Style); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| int best = 0; |
| sk_bzero(gMaxRecursion, sizeof(gMaxRecursion[0]) * 3); |
| #endif |
| skiatest::Timer timer; |
| for (int i = 0; i < 1000000; ++i) { |
| SkPath path, fill; |
| path.moveTo(unbounded(r), unbounded(r)); |
| path.quadTo(unbounded(r), unbounded(r), unbounded(r), unbounded(r)); |
| p.setStrokeWidth(unboundedPos(r)); |
| p.getFillPath(path, &fill); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (best < gMaxRecursion[2]) { |
| if (reporter->verbose()) { |
| SkDebugf("\n%s quad=%d width=%1.9g\n", __FUNCTION__, gMaxRecursion[2], |
| p.getStrokeWidth()); |
| path.dumpHex(); |
| SkDebugf("fill:\n"); |
| fill.dumpHex(); |
| } |
| best = gMaxRecursion[2]; |
| } |
| #endif |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (reporter->verbose()) { |
| SkDebugf("\n%s max quad=%d\n", __FUNCTION__, best); |
| } |
| #endif |
| } |
| |
| DEF_TEST(CubicStrokerUnbounded, reporter) { |
| SkRandom r; |
| SkPaint p; |
| p.setStyle(SkPaint::kStroke_Style); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| int bestTan = 0; |
| int bestCubic = 0; |
| sk_bzero(gMaxRecursion, sizeof(gMaxRecursion[0]) * 3); |
| #endif |
| skiatest::Timer timer; |
| for (int i = 0; i < 1000000; ++i) { |
| SkPath path, fill; |
| path.moveTo(unbounded(r), unbounded(r)); |
| path.cubicTo(unbounded(r), unbounded(r), unbounded(r), unbounded(r), |
| unbounded(r), unbounded(r)); |
| p.setStrokeWidth(unboundedPos(r)); |
| p.getFillPath(path, &fill); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (bestTan < gMaxRecursion[0] || bestCubic < gMaxRecursion[1]) { |
| if (reporter->verbose()) { |
| SkDebugf("\n%s tan=%d cubic=%d width=%1.9g\n", __FUNCTION__, gMaxRecursion[0], |
| gMaxRecursion[1], p.getStrokeWidth()); |
| path.dumpHex(); |
| SkDebugf("fill:\n"); |
| fill.dumpHex(); |
| } |
| bestTan = std::max(bestTan, gMaxRecursion[0]); |
| bestCubic = std::max(bestCubic, gMaxRecursion[1]); |
| } |
| #endif |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (reporter->verbose()) { |
| SkDebugf("\n%s max tan=%d cubic=%d\n", __FUNCTION__, bestTan, bestCubic); |
| } |
| #endif |
| } |
| |
| DEF_TEST(QuadStrokerConstrained, reporter) { |
| SkRandom r; |
| SkPaint p; |
| p.setStyle(SkPaint::kStroke_Style); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| int best = 0; |
| sk_bzero(gMaxRecursion, sizeof(gMaxRecursion[0]) * 3); |
| #endif |
| skiatest::Timer timer; |
| for (int i = 0; i < 1000000; ++i) { |
| SkPath path, fill; |
| SkPoint quad[3]; |
| quad[0].fX = r.nextRangeF(0, 500); |
| quad[0].fY = r.nextRangeF(0, 500); |
| const SkScalar halfSquared = 0.5f * 0.5f; |
| do { |
| quad[1].fX = r.nextRangeF(0, 500); |
| quad[1].fY = r.nextRangeF(0, 500); |
| } while (SkPointPriv::DistanceToSqd(quad[0], quad[1]) < halfSquared); |
| do { |
| quad[2].fX = r.nextRangeF(0, 500); |
| quad[2].fY = r.nextRangeF(0, 500); |
| } while (SkPointPriv::DistanceToSqd(quad[0], quad[2]) < halfSquared |
| || SkPointPriv::DistanceToSqd(quad[1], quad[2]) < halfSquared); |
| path.moveTo(quad[0].fX, quad[0].fY); |
| path.quadTo(quad[1].fX, quad[1].fY, quad[2].fX, quad[2].fY); |
| p.setStrokeWidth(r.nextRangeF(0, 500)); |
| p.getFillPath(path, &fill); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (best < gMaxRecursion[2]) { |
| if (reporter->verbose()) { |
| SkDebugf("\n%s quad=%d width=%1.9g\n", __FUNCTION__, gMaxRecursion[2], |
| p.getStrokeWidth()); |
| path.dumpHex(); |
| SkDebugf("fill:\n"); |
| fill.dumpHex(); |
| } |
| best = gMaxRecursion[2]; |
| } |
| #endif |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (reporter->verbose()) { |
| SkDebugf("\n%s max quad=%d\n", __FUNCTION__, best); |
| } |
| #endif |
| } |
| |
| DEF_TEST(CubicStrokerConstrained, reporter) { |
| SkRandom r; |
| SkPaint p; |
| p.setStyle(SkPaint::kStroke_Style); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| int bestTan = 0; |
| int bestCubic = 0; |
| sk_bzero(gMaxRecursion, sizeof(gMaxRecursion[0]) * 3); |
| #endif |
| skiatest::Timer timer; |
| for (int i = 0; i < 1000000; ++i) { |
| SkPath path, fill; |
| SkPoint cubic[4]; |
| cubic[0].fX = r.nextRangeF(0, 500); |
| cubic[0].fY = r.nextRangeF(0, 500); |
| const SkScalar halfSquared = 0.5f * 0.5f; |
| do { |
| cubic[1].fX = r.nextRangeF(0, 500); |
| cubic[1].fY = r.nextRangeF(0, 500); |
| } while (SkPointPriv::DistanceToSqd(cubic[0], cubic[1]) < halfSquared); |
| do { |
| cubic[2].fX = r.nextRangeF(0, 500); |
| cubic[2].fY = r.nextRangeF(0, 500); |
| } while ( SkPointPriv::DistanceToSqd(cubic[0], cubic[2]) < halfSquared |
| || SkPointPriv::DistanceToSqd(cubic[1], cubic[2]) < halfSquared); |
| do { |
| cubic[3].fX = r.nextRangeF(0, 500); |
| cubic[3].fY = r.nextRangeF(0, 500); |
| } while ( SkPointPriv::DistanceToSqd(cubic[0], cubic[3]) < halfSquared |
| || SkPointPriv::DistanceToSqd(cubic[1], cubic[3]) < halfSquared |
| || SkPointPriv::DistanceToSqd(cubic[2], cubic[3]) < halfSquared); |
| path.moveTo(cubic[0].fX, cubic[0].fY); |
| path.cubicTo(cubic[1].fX, cubic[1].fY, cubic[2].fX, cubic[2].fY, cubic[3].fX, cubic[3].fY); |
| p.setStrokeWidth(r.nextRangeF(0, 500)); |
| p.getFillPath(path, &fill); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (bestTan < gMaxRecursion[0] || bestCubic < gMaxRecursion[1]) { |
| if (reporter->verbose()) { |
| SkDebugf("\n%s tan=%d cubic=%d width=%1.9g\n", __FUNCTION__, gMaxRecursion[0], |
| gMaxRecursion[1], p.getStrokeWidth()); |
| path.dumpHex(); |
| SkDebugf("fill:\n"); |
| fill.dumpHex(); |
| } |
| bestTan = std::max(bestTan, gMaxRecursion[0]); |
| bestCubic = std::max(bestCubic, gMaxRecursion[1]); |
| } |
| #endif |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (reporter->verbose()) { |
| SkDebugf("\n%s max tan=%d cubic=%d\n", __FUNCTION__, bestTan, bestCubic); |
| } |
| #endif |
| } |
| |
| DEF_TEST(QuadStrokerRange, reporter) { |
| SkRandom r; |
| SkPaint p; |
| p.setStyle(SkPaint::kStroke_Style); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| int best = 0; |
| sk_bzero(gMaxRecursion, sizeof(gMaxRecursion[0]) * 3); |
| #endif |
| skiatest::Timer timer; |
| for (int i = 0; i < 1000000; ++i) { |
| SkPath path, fill; |
| SkPoint quad[3]; |
| quad[0].fX = r.nextRangeF(0, 500); |
| quad[0].fY = r.nextRangeF(0, 500); |
| quad[1].fX = r.nextRangeF(0, 500); |
| quad[1].fY = r.nextRangeF(0, 500); |
| quad[2].fX = r.nextRangeF(0, 500); |
| quad[2].fY = r.nextRangeF(0, 500); |
| path.moveTo(quad[0].fX, quad[0].fY); |
| path.quadTo(quad[1].fX, quad[1].fY, quad[2].fX, quad[2].fY); |
| p.setStrokeWidth(r.nextRangeF(0, 500)); |
| p.getFillPath(path, &fill); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (best < gMaxRecursion[2]) { |
| if (reporter->verbose()) { |
| SkDebugf("\n%s quad=%d width=%1.9g\n", __FUNCTION__, gMaxRecursion[2], |
| p.getStrokeWidth()); |
| path.dumpHex(); |
| SkDebugf("fill:\n"); |
| fill.dumpHex(); |
| } |
| best = gMaxRecursion[2]; |
| } |
| #endif |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (reporter->verbose()) { |
| SkDebugf("\n%s max quad=%d\n", __FUNCTION__, best); |
| } |
| #endif |
| } |
| |
| DEF_TEST(CubicStrokerRange, reporter) { |
| SkRandom r; |
| SkPaint p; |
| p.setStyle(SkPaint::kStroke_Style); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| int best[2] = { 0 }; |
| sk_bzero(gMaxRecursion, sizeof(gMaxRecursion[0]) * 3); |
| #endif |
| skiatest::Timer timer; |
| for (int i = 0; i < 1000000; ++i) { |
| SkPath path, fill; |
| path.moveTo(r.nextRangeF(0, 500), r.nextRangeF(0, 500)); |
| path.cubicTo(r.nextRangeF(0, 500), r.nextRangeF(0, 500), r.nextRangeF(0, 500), |
| r.nextRangeF(0, 500), r.nextRangeF(0, 500), r.nextRangeF(0, 500)); |
| p.setStrokeWidth(r.nextRangeF(0, 100)); |
| p.getFillPath(path, &fill); |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (best[0] < gMaxRecursion[0] || best[1] < gMaxRecursion[1]) { |
| if (reporter->verbose()) { |
| SkDebugf("\n%s tan=%d cubic=%d width=%1.9g\n", __FUNCTION__, gMaxRecursion[0], |
| gMaxRecursion[1], p.getStrokeWidth()); |
| path.dumpHex(); |
| SkDebugf("fill:\n"); |
| fill.dumpHex(); |
| } |
| best[0] = std::max(best[0], gMaxRecursion[0]); |
| best[1] = std::max(best[1], gMaxRecursion[1]); |
| } |
| #endif |
| if (FLAGS_timeout && timer.elapsedMs() > MS_TEST_DURATION) { |
| return; |
| } |
| } |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (reporter->verbose()) { |
| SkDebugf("\n%s max tan=%d cubic=%d\n", __FUNCTION__, best[0], best[1]); |
| } |
| #endif |
| } |
| |
| |
| DEF_TEST(QuadStrokerOneOff, reporter) { |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| sk_bzero(gMaxRecursion, sizeof(gMaxRecursion[0]) * 3); |
| #endif |
| SkPaint p; |
| p.setStyle(SkPaint::kStroke_Style); |
| p.setStrokeWidth(SkDoubleToScalar(164.683548)); |
| |
| SkPath path, fill; |
| path.moveTo(SkBits2Float(0x43c99223), SkBits2Float(0x42b7417e)); |
| path.quadTo(SkBits2Float(0x4285d839), SkBits2Float(0x43ed6645), SkBits2Float(0x43c941c8), SkBits2Float(0x42b3ace3)); |
| p.getFillPath(path, &fill); |
| if (reporter->verbose()) { |
| SkDebugf("\n%s path\n", __FUNCTION__); |
| path.dump(); |
| SkDebugf("fill:\n"); |
| fill.dump(); |
| } |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (reporter->verbose()) { |
| SkDebugf("max quad=%d\n", gMaxRecursion[2]); |
| } |
| #endif |
| } |
| |
| DEF_TEST(CubicStrokerOneOff, reporter) { |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| sk_bzero(gMaxRecursion, sizeof(gMaxRecursion[0]) * 3); |
| #endif |
| SkPaint p; |
| p.setStyle(SkPaint::kStroke_Style); |
| p.setStrokeWidth(SkDoubleToScalar(42.835968)); |
| |
| SkPath path, fill; |
| path.moveTo(SkBits2Float(0x433f5370), SkBits2Float(0x43d1f4b3)); |
| path.cubicTo(SkBits2Float(0x4331cb76), SkBits2Float(0x43ea3340), SkBits2Float(0x4388f498), SkBits2Float(0x42f7f08d), SkBits2Float(0x43f1cd32), SkBits2Float(0x42802ec1)); |
| p.getFillPath(path, &fill); |
| if (reporter->verbose()) { |
| SkDebugf("\n%s path\n", __FUNCTION__); |
| path.dump(); |
| SkDebugf("fill:\n"); |
| fill.dump(); |
| } |
| #if defined(SK_DEBUG) && QUAD_STROKE_APPROX_EXTENDED_DEBUGGING |
| if (reporter->verbose()) { |
| SkDebugf("max tan=%d cubic=%d\n", gMaxRecursion[0], gMaxRecursion[1]); |
| } |
| #endif |
| } |