470 lines
17 KiB
C++
470 lines
17 KiB
C++
|
/*
|
||
|
* 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
|
||
|
}
|