Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(165)

Unified Diff: src/core/SkScan_AntiPath.cpp

Issue 1893433002: In SkDraw::drawRect, use SkPath for huge rects. Base URL: https://skia.googlesource.com/skia@fixed-assert
Patch Set: Created 4 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « src/core/SkScanPriv.h ('k') | src/core/SkScan_Hairline.cpp » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: src/core/SkScan_AntiPath.cpp
diff --git a/src/core/SkScan_AntiPath.cpp b/src/core/SkScan_AntiPath.cpp
index 3b0f3e189bed8fc8b5e36f36b2a7912e2267876c..cd7810638c18eb3e5ce41802c76e53c4fa587e81 100644
--- a/src/core/SkScan_AntiPath.cpp
+++ b/src/core/SkScan_AntiPath.cpp
@@ -583,12 +583,6 @@ void MaskSuperBlitter::blitH(int x, int y, int width) {
///////////////////////////////////////////////////////////////////////////////
-static bool fitsInsideLimit(const SkRect& r, SkScalar max) {
- const SkScalar min = -max;
- return r.fLeft > min && r.fTop > min &&
- r.fRight < max && r.fBottom < max;
-}
-
static int overflows_short_shift(int value, int shift) {
const int s = 16 + shift;
return (SkLeftShift(value, s) >> s) - value;
@@ -598,7 +592,7 @@ static int overflows_short_shift(int value, int shift) {
Would any of the coordinates of this rectangle not fit in a short,
when left-shifted by shift?
*/
-static int rect_overflows_short_shift(SkIRect rect, int shift) {
+static int rect_overflows_short_shift(SkIRect rect) {
SkASSERT(!overflows_short_shift(8191, SHIFT));
SkASSERT(overflows_short_shift(8192, SHIFT));
SkASSERT(!overflows_short_shift(32767, 0));
@@ -612,14 +606,21 @@ static int rect_overflows_short_shift(SkIRect rect, int shift) {
overflows_short_shift(rect.fBottom, SHIFT);
}
-static bool safeRoundOut(const SkRect& src, SkIRect* dst, int32_t maxInt) {
- const SkScalar maxScalar = SkIntToScalar(maxInt);
-
- if (fitsInsideLimit(src, maxScalar)) {
+// When src has huge coordinates, clips dst by clip and returns true.
+static bool roundOutClipped(const SkRect& src, const SkIRect& clip, SkIRect* dst) {
+ static const int32_t kLimit = (SK_MaxS32 >> SHIFT) - 1;
+ const SkIRect max = SkIRect::MakeLTRB(-kLimit, -kLimit, kLimit, kLimit);
+ if (max.contains(src)) {
src.roundOut(dst);
- return true;
+ return false;
+ }
+ SkRect clipped;
+ if (clipped.intersect(src, SkRect::Make(clip))) {
+ clipped.roundOut(dst);
+ } else {
+ dst->setEmpty();
}
- return false;
+ return true;
}
void SkScan::AntiFillPath(const SkPath& path, const SkRegion& origClip,
@@ -631,13 +632,7 @@ void SkScan::AntiFillPath(const SkPath& path, const SkRegion& origClip,
const bool isInverse = path.isInverseFillType();
SkIRect ir;
- if (!safeRoundOut(path.getBounds(), &ir, SK_MaxS32 >> SHIFT)) {
-#if 0
- const SkRect& r = path.getBounds();
- SkDebugf("--- bounds can't fit in SkIRect\n", r.fLeft, r.fTop, r.fRight, r.fBottom);
-#endif
- return;
- }
+ bool irIsClipped = roundOutClipped(path.getBounds(), origClip.getBounds(), &ir);
if (ir.isEmpty()) {
if (isInverse) {
blitter->blitRegion(origClip);
@@ -658,7 +653,7 @@ void SkScan::AntiFillPath(const SkPath& path, const SkRegion& origClip,
return;
}
}
- if (rect_overflows_short_shift(clippedIR, SHIFT)) {
+ if (rect_overflows_short_shift(clippedIR)) {
SkScan::FillPath(path, origClip, blitter);
return;
}
@@ -682,7 +677,7 @@ void SkScan::AntiFillPath(const SkPath& path, const SkRegion& origClip,
}
// for here down, use clipRgn, not origClip
- SkScanClipper clipper(blitter, clipRgn, ir);
+ SkScanClipper clipper(blitter, clipRgn, ir, /* skipRejectTest */ false, irIsClipped);
const SkIRect* clipRect = clipper.getClipRect();
if (clipper.getBlitter() == nullptr) { // clipped out
@@ -709,11 +704,9 @@ void SkScan::AntiFillPath(const SkPath& path, const SkRegion& origClip,
superClipRect = &superRect;
}
- SkASSERT(SkIntToScalar(ir.fTop) <= path.getBounds().fTop);
-
// MaskSuperBlitter can't handle drawing outside of ir, so we can't use it
// if we're an inverse filltype
- if (!isInverse && MaskSuperBlitter::CanHandleRect(ir) && !forceRLE) {
+ if (!isInverse && !irIsClipped && MaskSuperBlitter::CanHandleRect(ir) && !forceRLE) {
MaskSuperBlitter superBlit(blitter, ir, *clipRgn, isInverse);
SkASSERT(SkIntToScalar(ir.fTop) <= path.getBounds().fTop);
sk_fill_path(path, superClipRect, &superBlit, ir.fTop, ir.fBottom, SHIFT, *clipRgn);
« no previous file with comments | « src/core/SkScanPriv.h ('k') | src/core/SkScan_Hairline.cpp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698