c836692f20
Change-Id: I4f0321dbf5c03adc7219ca2cfb6dbfbbaecc1e4f Reviewed-on: https://skia-review.googlesource.com/c/skia/+/553582 Commit-Queue: Herb Derby <herb@google.com> Reviewed-by: John Stiles <johnstiles@google.com>
446 lines
14 KiB
C++
446 lines
14 KiB
C++
/*
|
|
* Copyright 2011 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/SkBitmap.h"
|
|
#include "include/core/SkCanvas.h"
|
|
#include "include/core/SkColor.h"
|
|
#include "include/core/SkImageInfo.h"
|
|
#include "include/core/SkMatrix.h"
|
|
#include "include/core/SkPath.h"
|
|
#include "include/core/SkRRect.h"
|
|
#include "include/core/SkRect.h"
|
|
#include "include/core/SkRegion.h"
|
|
#include "include/core/SkScalar.h"
|
|
#include "include/core/SkTypes.h"
|
|
#include "include/private/SkMalloc.h"
|
|
#include "include/utils/SkRandom.h"
|
|
#include "src/core/SkAAClip.h"
|
|
#include "src/core/SkMask.h"
|
|
#include "src/core/SkRasterClip.h"
|
|
#include "tests/Test.h"
|
|
|
|
#include <string.h>
|
|
|
|
static bool operator==(const SkMask& a, const SkMask& b) {
|
|
if (a.fFormat != b.fFormat || a.fBounds != b.fBounds) {
|
|
return false;
|
|
}
|
|
if (!a.fImage && !b.fImage) {
|
|
return true;
|
|
}
|
|
if (!a.fImage || !b.fImage) {
|
|
return false;
|
|
}
|
|
|
|
size_t wbytes = a.fBounds.width();
|
|
switch (a.fFormat) {
|
|
case SkMask::kBW_Format:
|
|
wbytes = (wbytes + 7) >> 3;
|
|
break;
|
|
case SkMask::kA8_Format:
|
|
case SkMask::k3D_Format:
|
|
break;
|
|
case SkMask::kLCD16_Format:
|
|
wbytes <<= 1;
|
|
break;
|
|
case SkMask::kARGB32_Format:
|
|
wbytes <<= 2;
|
|
break;
|
|
default:
|
|
SkDEBUGFAIL("unknown mask format");
|
|
return false;
|
|
}
|
|
|
|
const int h = a.fBounds.height();
|
|
const char* aptr = (const char*)a.fImage;
|
|
const char* bptr = (const char*)b.fImage;
|
|
for (int y = 0; y < h; ++y) {
|
|
if (0 != memcmp(aptr, bptr, wbytes)) {
|
|
return false;
|
|
}
|
|
aptr += wbytes;
|
|
bptr += wbytes;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void copyToMask(const SkRegion& rgn, SkMask* mask) {
|
|
mask->fFormat = SkMask::kA8_Format;
|
|
|
|
if (rgn.isEmpty()) {
|
|
mask->fBounds.setEmpty();
|
|
mask->fRowBytes = 0;
|
|
mask->fImage = nullptr;
|
|
return;
|
|
}
|
|
|
|
mask->fBounds = rgn.getBounds();
|
|
mask->fRowBytes = mask->fBounds.width();
|
|
mask->fImage = SkMask::AllocImage(mask->computeImageSize());
|
|
sk_bzero(mask->fImage, mask->computeImageSize());
|
|
|
|
SkImageInfo info = SkImageInfo::Make(mask->fBounds.width(),
|
|
mask->fBounds.height(),
|
|
kAlpha_8_SkColorType,
|
|
kPremul_SkAlphaType);
|
|
SkBitmap bitmap;
|
|
bitmap.installPixels(info, mask->fImage, mask->fRowBytes);
|
|
|
|
// canvas expects its coordinate system to always be 0,0 in the top/left
|
|
// so we translate the rgn to match that before drawing into the mask.
|
|
//
|
|
SkRegion tmpRgn(rgn);
|
|
tmpRgn.translate(-rgn.getBounds().fLeft, -rgn.getBounds().fTop);
|
|
|
|
SkCanvas canvas(bitmap);
|
|
canvas.clipRegion(tmpRgn);
|
|
canvas.drawColor(SK_ColorBLACK);
|
|
}
|
|
|
|
static void copyToMask(const SkRasterClip& rc, SkMask* mask) {
|
|
if (rc.isBW()) {
|
|
copyToMask(rc.bwRgn(), mask);
|
|
} else {
|
|
rc.aaRgn().copyToMask(mask);
|
|
}
|
|
}
|
|
|
|
static bool operator==(const SkRasterClip& a, const SkRasterClip& b) {
|
|
if (a.isEmpty() && b.isEmpty()) {
|
|
return true;
|
|
} else if (a.isEmpty() != b.isEmpty() || a.isBW() != b.isBW() || a.isRect() != b.isRect()) {
|
|
return false;
|
|
}
|
|
|
|
SkMask mask0, mask1;
|
|
copyToMask(a, &mask0);
|
|
copyToMask(b, &mask1);
|
|
SkAutoMaskFreeImage free0(mask0.fImage);
|
|
SkAutoMaskFreeImage free1(mask1.fImage);
|
|
return mask0 == mask1;
|
|
}
|
|
|
|
static SkIRect rand_rect(SkRandom& rand, int n) {
|
|
int x = rand.nextS() % n;
|
|
int y = rand.nextS() % n;
|
|
int w = rand.nextU() % n;
|
|
int h = rand.nextU() % n;
|
|
return SkIRect::MakeXYWH(x, y, w, h);
|
|
}
|
|
|
|
static void make_rand_rgn(SkRegion* rgn, SkRandom& rand) {
|
|
int count = rand.nextU() % 20;
|
|
for (int i = 0; i < count; ++i) {
|
|
rgn->op(rand_rect(rand, 100), SkRegion::kXOR_Op);
|
|
}
|
|
}
|
|
|
|
static bool operator==(const SkRegion& rgn, const SkAAClip& aaclip) {
|
|
SkMask mask0, mask1;
|
|
|
|
copyToMask(rgn, &mask0);
|
|
aaclip.copyToMask(&mask1);
|
|
SkAutoMaskFreeImage free0(mask0.fImage);
|
|
SkAutoMaskFreeImage free1(mask1.fImage);
|
|
return mask0 == mask1;
|
|
}
|
|
|
|
static bool equalsAAClip(const SkRegion& rgn) {
|
|
SkAAClip aaclip;
|
|
aaclip.setRegion(rgn);
|
|
return rgn == aaclip;
|
|
}
|
|
|
|
static void setRgnToPath(SkRegion* rgn, const SkPath& path) {
|
|
SkIRect ir;
|
|
path.getBounds().round(&ir);
|
|
rgn->setPath(path, SkRegion(ir));
|
|
}
|
|
|
|
// aaclip.setRegion should create idential masks to the region
|
|
static void test_rgn(skiatest::Reporter* reporter) {
|
|
SkRandom rand;
|
|
for (int i = 0; i < 1000; i++) {
|
|
SkRegion rgn;
|
|
make_rand_rgn(&rgn, rand);
|
|
REPORTER_ASSERT(reporter, equalsAAClip(rgn));
|
|
}
|
|
|
|
{
|
|
SkRegion rgn;
|
|
SkPath path;
|
|
path.addCircle(0, 0, SkIntToScalar(30));
|
|
setRgnToPath(&rgn, path);
|
|
REPORTER_ASSERT(reporter, equalsAAClip(rgn));
|
|
|
|
path.reset();
|
|
path.moveTo(0, 0);
|
|
path.lineTo(SkIntToScalar(100), 0);
|
|
path.lineTo(SkIntToScalar(100 - 20), SkIntToScalar(20));
|
|
path.lineTo(SkIntToScalar(20), SkIntToScalar(20));
|
|
setRgnToPath(&rgn, path);
|
|
REPORTER_ASSERT(reporter, equalsAAClip(rgn));
|
|
}
|
|
}
|
|
|
|
static void imoveTo(SkPath& path, int x, int y) {
|
|
path.moveTo(SkIntToScalar(x), SkIntToScalar(y));
|
|
}
|
|
|
|
static void icubicTo(SkPath& path, int x0, int y0, int x1, int y1, int x2, int y2) {
|
|
path.cubicTo(SkIntToScalar(x0), SkIntToScalar(y0),
|
|
SkIntToScalar(x1), SkIntToScalar(y1),
|
|
SkIntToScalar(x2), SkIntToScalar(y2));
|
|
}
|
|
|
|
static void test_path_bounds(skiatest::Reporter* reporter) {
|
|
SkPath path;
|
|
SkAAClip clip;
|
|
const int height = 40;
|
|
const SkScalar sheight = SkIntToScalar(height);
|
|
|
|
path.addOval(SkRect::MakeWH(sheight, sheight));
|
|
REPORTER_ASSERT(reporter, sheight == path.getBounds().height());
|
|
clip.setPath(path, path.getBounds().roundOut(), true);
|
|
REPORTER_ASSERT(reporter, height == clip.getBounds().height());
|
|
|
|
// this is the trimmed height of this cubic (with aa). The critical thing
|
|
// for this test is that it is less than height, which represents just
|
|
// the bounds of the path's control-points.
|
|
//
|
|
// This used to fail until we tracked the MinY in the BuilderBlitter.
|
|
//
|
|
const int teardrop_height = 12;
|
|
path.reset();
|
|
imoveTo(path, 0, 20);
|
|
icubicTo(path, 40, 40, 40, 0, 0, 20);
|
|
REPORTER_ASSERT(reporter, sheight == path.getBounds().height());
|
|
clip.setPath(path, path.getBounds().roundOut(), true);
|
|
REPORTER_ASSERT(reporter, teardrop_height == clip.getBounds().height());
|
|
}
|
|
|
|
static void test_empty(skiatest::Reporter* reporter) {
|
|
SkAAClip clip;
|
|
|
|
REPORTER_ASSERT(reporter, clip.isEmpty());
|
|
REPORTER_ASSERT(reporter, clip.getBounds().isEmpty());
|
|
|
|
clip.translate(10, 10, &clip); // should have no effect on empty
|
|
REPORTER_ASSERT(reporter, clip.isEmpty());
|
|
REPORTER_ASSERT(reporter, clip.getBounds().isEmpty());
|
|
|
|
SkIRect r = { 10, 10, 40, 50 };
|
|
clip.setRect(r);
|
|
REPORTER_ASSERT(reporter, !clip.isEmpty());
|
|
REPORTER_ASSERT(reporter, !clip.getBounds().isEmpty());
|
|
REPORTER_ASSERT(reporter, clip.getBounds() == r);
|
|
|
|
clip.setEmpty();
|
|
REPORTER_ASSERT(reporter, clip.isEmpty());
|
|
REPORTER_ASSERT(reporter, clip.getBounds().isEmpty());
|
|
|
|
SkMask mask;
|
|
clip.copyToMask(&mask);
|
|
REPORTER_ASSERT(reporter, nullptr == mask.fImage);
|
|
REPORTER_ASSERT(reporter, mask.fBounds.isEmpty());
|
|
}
|
|
|
|
static void rand_irect(SkIRect* r, int N, SkRandom& rand) {
|
|
r->setXYWH(0, 0, rand.nextU() % N, rand.nextU() % N);
|
|
int dx = rand.nextU() % (2*N);
|
|
int dy = rand.nextU() % (2*N);
|
|
// use int dx,dy to make the subtract be signed
|
|
r->offset(N - dx, N - dy);
|
|
}
|
|
|
|
static void test_irect(skiatest::Reporter* reporter) {
|
|
SkRandom rand;
|
|
|
|
for (int i = 0; i < 10000; i++) {
|
|
SkAAClip clip0, clip1;
|
|
SkRegion rgn0, rgn1;
|
|
SkIRect r0, r1;
|
|
|
|
rand_irect(&r0, 10, rand);
|
|
rand_irect(&r1, 10, rand);
|
|
clip0.setRect(r0);
|
|
clip1.setRect(r1);
|
|
rgn0.setRect(r0);
|
|
rgn1.setRect(r1);
|
|
for (SkClipOp op : {SkClipOp::kDifference, SkClipOp::kIntersect}) {
|
|
SkAAClip clip2 = clip0; // leave clip0 unchanged for future iterations
|
|
SkRegion rgn2;
|
|
bool nonEmptyAA = clip2.op(clip1, op);
|
|
bool nonEmptyBW = rgn2.op(rgn0, rgn1, (SkRegion::Op) op);
|
|
if (nonEmptyAA != nonEmptyBW || clip2.getBounds() != rgn2.getBounds()) {
|
|
ERRORF(reporter, "%s %s "
|
|
"[%d %d %d %d] %s [%d %d %d %d] = BW:[%d %d %d %d] AA:[%d %d %d %d]\n",
|
|
nonEmptyAA == nonEmptyBW ? "true" : "false",
|
|
clip2.getBounds() == rgn2.getBounds() ? "true" : "false",
|
|
r0.fLeft, r0.fTop, r0.right(), r0.bottom(),
|
|
op == SkClipOp::kDifference ? "DIFF" : "INTERSECT",
|
|
r1.fLeft, r1.fTop, r1.right(), r1.bottom(),
|
|
rgn2.getBounds().fLeft, rgn2.getBounds().fTop,
|
|
rgn2.getBounds().right(), rgn2.getBounds().bottom(),
|
|
clip2.getBounds().fLeft, clip2.getBounds().fTop,
|
|
clip2.getBounds().right(), clip2.getBounds().bottom());
|
|
}
|
|
|
|
SkMask maskBW, maskAA;
|
|
copyToMask(rgn2, &maskBW);
|
|
clip2.copyToMask(&maskAA);
|
|
SkAutoMaskFreeImage freeBW(maskBW.fImage);
|
|
SkAutoMaskFreeImage freeAA(maskAA.fImage);
|
|
REPORTER_ASSERT(reporter, maskBW == maskAA);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void test_path_with_hole(skiatest::Reporter* reporter) {
|
|
static const uint8_t gExpectedImage[] = {
|
|
0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF,
|
|
0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00,
|
|
0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF,
|
|
};
|
|
SkMask expected;
|
|
expected.fBounds.setWH(4, 6);
|
|
expected.fRowBytes = 4;
|
|
expected.fFormat = SkMask::kA8_Format;
|
|
expected.fImage = (uint8_t*)gExpectedImage;
|
|
|
|
SkPath path;
|
|
path.addRect(SkRect::MakeXYWH(0, 0,
|
|
SkIntToScalar(4), SkIntToScalar(2)));
|
|
path.addRect(SkRect::MakeXYWH(0, SkIntToScalar(4),
|
|
SkIntToScalar(4), SkIntToScalar(2)));
|
|
|
|
for (int i = 0; i < 2; ++i) {
|
|
SkAAClip clip;
|
|
clip.setPath(path, path.getBounds().roundOut(), 1 == i);
|
|
|
|
SkMask mask;
|
|
clip.copyToMask(&mask);
|
|
SkAutoMaskFreeImage freeM(mask.fImage);
|
|
|
|
REPORTER_ASSERT(reporter, expected == mask);
|
|
}
|
|
}
|
|
|
|
static void test_really_a_rect(skiatest::Reporter* reporter) {
|
|
SkRRect rrect;
|
|
rrect.setRectXY(SkRect::MakeWH(100, 100), 5, 5);
|
|
|
|
SkPath path;
|
|
path.addRRect(rrect);
|
|
|
|
SkAAClip clip;
|
|
clip.setPath(path, path.getBounds().roundOut(), true);
|
|
|
|
REPORTER_ASSERT(reporter, clip.getBounds() == SkIRect::MakeWH(100, 100));
|
|
REPORTER_ASSERT(reporter, !clip.isRect());
|
|
|
|
// This rect should intersect the clip, but slice-out all of the "soft" parts,
|
|
// leaving just a rect.
|
|
const SkIRect ir = SkIRect::MakeLTRB(10, -10, 50, 90);
|
|
|
|
clip.op(ir, SkClipOp::kIntersect);
|
|
|
|
REPORTER_ASSERT(reporter, clip.getBounds() == SkIRect::MakeLTRB(10, 0, 50, 90));
|
|
// the clip recognized that that it is just a rect!
|
|
REPORTER_ASSERT(reporter, clip.isRect());
|
|
}
|
|
|
|
static void did_dx_affect(skiatest::Reporter* reporter, const SkScalar dx[],
|
|
size_t count, bool changed) {
|
|
SkIRect ir = { 0, 0, 10, 10 };
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
SkRect r;
|
|
r.set(ir);
|
|
|
|
SkRasterClip rc0(ir);
|
|
SkRasterClip rc1(ir);
|
|
SkRasterClip rc2(ir);
|
|
|
|
rc0.op(r, SkMatrix::I(), SkClipOp::kIntersect, false);
|
|
r.offset(dx[i], 0);
|
|
rc1.op(r, SkMatrix::I(), SkClipOp::kIntersect, true);
|
|
r.offset(-2*dx[i], 0);
|
|
rc2.op(r, SkMatrix::I(), SkClipOp::kIntersect, true);
|
|
|
|
REPORTER_ASSERT(reporter, changed != (rc0 == rc1));
|
|
REPORTER_ASSERT(reporter, changed != (rc0 == rc2));
|
|
}
|
|
}
|
|
|
|
static void test_nearly_integral(skiatest::Reporter* reporter) {
|
|
// All of these should generate equivalent rasterclips
|
|
|
|
static const SkScalar gSafeX[] = {
|
|
0, SK_Scalar1/1000, SK_Scalar1/100, SK_Scalar1/10,
|
|
};
|
|
did_dx_affect(reporter, gSafeX, std::size(gSafeX), false);
|
|
|
|
static const SkScalar gUnsafeX[] = {
|
|
SK_Scalar1/4, SK_Scalar1/3,
|
|
};
|
|
did_dx_affect(reporter, gUnsafeX, std::size(gUnsafeX), true);
|
|
}
|
|
|
|
static void test_regressions() {
|
|
// these should not assert in the debug build
|
|
// bug was introduced in rev. 3209
|
|
{
|
|
SkAAClip clip;
|
|
SkRect r;
|
|
r.fLeft = 129.892181f;
|
|
r.fTop = 10.3999996f;
|
|
r.fRight = 130.892181f;
|
|
r.fBottom = 20.3999996f;
|
|
clip.setPath(SkPath::Rect(r), r.roundOut(), true);
|
|
}
|
|
}
|
|
|
|
// Building aaclip meant aa-scan-convert a path into a huge clip.
|
|
// the old algorithm sized the supersampler to the size of the clip, which overflowed
|
|
// its internal 16bit coordinates. The fix was to intersect the clip+path_bounds before
|
|
// sizing the supersampler.
|
|
//
|
|
// Before the fix, the following code would assert in debug builds.
|
|
//
|
|
static void test_crbug_422693(skiatest::Reporter* reporter) {
|
|
SkRasterClip rc(SkIRect::MakeLTRB(-25000, -25000, 25000, 25000));
|
|
SkPath path;
|
|
path.addCircle(50, 50, 50);
|
|
rc.op(path, SkMatrix::I(), SkClipOp::kIntersect, true);
|
|
}
|
|
|
|
static void test_huge(skiatest::Reporter* reporter) {
|
|
SkAAClip clip;
|
|
int big = 0x70000000;
|
|
SkIRect r = { -big, -big, big, big };
|
|
SkASSERT(r.width() < 0 && r.height() < 0);
|
|
|
|
clip.setRect(r);
|
|
}
|
|
|
|
DEF_TEST(AAClip, reporter) {
|
|
test_empty(reporter);
|
|
test_path_bounds(reporter);
|
|
test_irect(reporter);
|
|
test_rgn(reporter);
|
|
test_path_with_hole(reporter);
|
|
test_regressions();
|
|
test_nearly_integral(reporter);
|
|
test_really_a_rect(reporter);
|
|
test_crbug_422693(reporter);
|
|
test_huge(reporter);
|
|
}
|