SkPathView for ownerless (can live on stack) SkPaths

Follow-on CLs will push higher up in SkDraw, so that everywhere today
we have to cons-up (with the associated mallocs) a temp SkPath we can
replace it with a stack-based SPath...
- drawRect
- drawOval
- drawRRect
- drawLine(s)
(similar to how this CL already handled quads and triangles)

Bug: skia:10566
Change-Id: I882b4f4c60e80235ca83c86c926e905b269a7afd
Reviewed-on: https://skia-review.googlesource.com/c/skia/+/307784
Reviewed-by: Mike Klein <mtklein@google.com>
Commit-Queue: Mike Reed <reed@google.com>
This commit is contained in:
Mike Reed 2020-08-07 16:00:06 -04:00 committed by Skia Commit-Bot
parent 24a7f071ae
commit 375721d7bb
23 changed files with 283 additions and 104 deletions

View File

@ -18,6 +18,7 @@
class SkAutoPathBoundsUpdate;
class SkData;
class SkRRect;
struct SkPathView;
class SkWStream;
// WIP -- define this locally, and fix call-sites to use SkPathBuilder (skbug.com/9000)
@ -1558,6 +1559,8 @@ public:
Verb autoClose(SkPoint pts[2]);
};
SkPathView view() const;
private:
/** \class SkPath::RangeIter
Iterates through a raw range of path verbs, points, and conics. All values are returned

View File

@ -9,6 +9,7 @@
#define SkPathRef_DEFINED
#include "include/core/SkMatrix.h"
#include "include/core/SkPathTypes.h"
#include "include/core/SkPoint.h"
#include "include/core/SkRRect.h"
#include "include/core/SkRect.h"
@ -23,6 +24,7 @@
#include <limits>
#include <tuple>
struct SkPathView;
class SkRBuffer;
class SkWBuffer;
@ -334,6 +336,8 @@ public:
bool isValid() const;
SkDEBUGCODE(void validate() const { SkASSERT(this->isValid()); } )
SkPathView view(SkPathFillType, SkPathConvexityType) const;
private:
enum SerializationOffsets {
kLegacyRRectOrOvalStartIdx_SerializationShift = 28, // requires 3 bits, ignored.

View File

@ -227,7 +227,7 @@ static SkPath clip(const SkPath& path, SkPoint p0, SkPoint p1) {
SkPoint fPrev = {0, 0};
} rec;
SkEdgeClipper::ClipPath(rotated, clip, false,
SkEdgeClipper::ClipPath(rotated.view(), clip, false,
[](SkEdgeClipper* clipper, bool newCtr, void* ctx) {
Rec* rec = (Rec*)ctx;

View File

@ -1336,11 +1336,12 @@ bool SkAAClip::setPath(const SkPath& path, const SkRegion* clip, bool doAA) {
Builder builder(ibounds);
BuilderBlitter blitter(&builder);
const SkPathView view = path.view();
if (doAA) {
SkScan::AntiFillPath(path, snugClip, &blitter, true);
SkScan::AntiFillPath(view, snugClip, &blitter, true);
} else {
SkScan::FillPath(path, snugClip, &blitter);
SkScan::FillPath(view, snugClip, &blitter);
}
blitter.finish();

View File

@ -836,7 +836,7 @@ void SkDraw::drawDevPath(const SkPath& devPath, const SkPaint& paint, bool drawC
}
}
void (*proc)(const SkPath&, const SkRasterClip&, SkBlitter*);
void (*proc)(const SkPathView&, const SkRasterClip&, SkBlitter*);
if (doFill) {
if (paint.isAntiAlias()) {
proc = SkScan::AntiFillPath;
@ -877,7 +877,7 @@ void SkDraw::drawDevPath(const SkPath& devPath, const SkPaint& paint, bool drawC
}
}
proc(devPath, *fRC, blitter);
proc(devPath.view(), *fRC, blitter);
}
void SkDraw::drawPath(const SkPath& origSrcPath, const SkPaint& origPaint,

View File

@ -22,7 +22,7 @@
#include "src/core/SkScan.h"
static void fill_rect(const SkMatrix& ctm, const SkRasterClip& rc,
const SkRect& r, SkBlitter* blitter, SkPath* scratchPath) {
const SkRect& r, SkBlitter* blitter) {
if (ctm.rectStaysRect()) {
SkRect dr;
ctm.mapRect(&dr, r);
@ -32,9 +32,10 @@ static void fill_rect(const SkMatrix& ctm, const SkRasterClip& rc,
r.toQuad(pts);
ctm.mapPoints(pts, pts, 4);
scratchPath->rewind();
scratchPath->addPoly(pts, 4, true);
SkScan::FillPath(*scratchPath, rc, blitter);
SkRect bounds;
bounds.setBounds(pts, 4);
SkScan::FillPath(SkPathView_quad(pts, bounds), rc, blitter);
}
}
@ -103,8 +104,6 @@ void SkDraw::drawAtlas(const SkImage* atlas, const SkRSXform xform[], const SkRe
if (auto blitter = SkCreateRasterPipelineBlitter(fDst, p, pipeline, isOpaque, &alloc,
fRC->clipShader())) {
SkPath scratchPath;
for (int i = 0; i < count; ++i) {
if (colors) {
SkColor4f c4 = SkColor4f::FromColor(colors[i]);
@ -118,7 +117,7 @@ void SkDraw::drawAtlas(const SkImage* atlas, const SkRSXform xform[], const SkRe
mx.postConcat(fMatrixProvider->localToDevice());
if (updator->update(mx, nullptr)) {
fill_rect(mx, *fRC, textures[i], blitter, &scratchPath);
fill_rect(mx, *fRC, textures[i], blitter);
}
}
}

View File

@ -14,6 +14,7 @@
#include "src/core/SkGeometry.h"
#include "src/core/SkLineClipper.h"
#include "src/core/SkPathPriv.h"
#include "src/core/SkPathView.h"
#include "src/core/SkSafeMath.h"
SkEdgeBuilder::Combine SkBasicEdgeBuilder::combineVertical(const SkEdge* edge, SkEdge* last) {
@ -218,8 +219,8 @@ char* SkAnalyticEdgeBuilder::allocEdges(size_t n, size_t* size) {
}
// TODO: maybe get rid of buildPoly() entirely?
int SkEdgeBuilder::buildPoly(const SkPath& path, const SkIRect* iclip, bool canCullToTheRight) {
size_t maxEdgeCount = path.countPoints();
int SkEdgeBuilder::buildPoly(const SkPathView& path, const SkIRect* iclip, bool canCullToTheRight) {
size_t maxEdgeCount = path.fPoints.size();
if (iclip) {
// clipping can turn 1 line into (up to) kMaxClippedLineSegments, since
// we turn portions that are clipped out on the left/right into vertical
@ -286,7 +287,7 @@ int SkEdgeBuilder::buildPoly(const SkPath& path, const SkIRect* iclip, bool canC
return SkToInt(edgePtr - (char**)fEdgeList);
}
int SkEdgeBuilder::build(const SkPath& path, const SkIRect* iclip, bool canCullToTheRight) {
int SkEdgeBuilder::build(const SkPathView& path, const SkIRect* iclip, bool canCullToTheRight) {
SkAutoConicToQuads quadder;
const SkScalar conicTol = SK_Scalar1 / 4;
bool is_finite = true;
@ -360,14 +361,14 @@ int SkEdgeBuilder::build(const SkPath& path, const SkIRect* iclip, bool canCullT
return is_finite ? fList.count() : 0;
}
int SkEdgeBuilder::buildEdges(const SkPath& path,
int SkEdgeBuilder::buildEdges(const SkPathView& path,
const SkIRect* shiftedClip) {
// If we're convex, then we need both edges, even if the right edge is past the clip.
const bool canCullToTheRight = !path.isConvex();
// We can use our buildPoly() optimization if all the segments are lines.
// (Edges are homogenous and stored contiguously in memory, no need for indirection.)
const int count = SkPath::kLine_SegmentMask == path.getSegmentMasks()
// (Edges are homogeneous and stored contiguously in memory, no need for indirection.)
const int count = SkPath::kLine_SegmentMask == path.fSegmentMask
? this->buildPoly(path, shiftedClip, canCullToTheRight)
: this->build (path, shiftedClip, canCullToTheRight);

View File

@ -13,12 +13,11 @@
#include "src/core/SkArenaAlloc.h"
#include "src/core/SkEdge.h"
class SkPath;
struct SkPathView;
class SkEdgeBuilder {
public:
int buildEdges(const SkPath& path,
const SkIRect* shiftedClip);
int buildEdges(const SkPathView&, const SkIRect* shiftedClip);
protected:
SkEdgeBuilder() = default;
@ -37,8 +36,8 @@ protected:
};
private:
int build (const SkPath& path, const SkIRect* clip, bool clipToTheRight);
int buildPoly(const SkPath& path, const SkIRect* clip, bool clipToTheRight);
int build (const SkPathView&, const SkIRect* clip, bool clipToTheRight);
int buildPoly(const SkPathView&, const SkIRect* clip, bool clipToTheRight);
virtual char* allocEdges(size_t n, size_t* sizeof_edge) = 0;
virtual SkRect recoverClip(const SkIRect&) const = 0;

View File

@ -560,14 +560,13 @@ void sk_assert_monotonic_x(const SkPoint pts[], int count) {
#include "src/core/SkPathPriv.h"
void SkEdgeClipper::ClipPath(const SkPath& path, const SkRect& clip, bool canCullToTheRight,
void SkEdgeClipper::ClipPath(const SkPathView& view, const SkRect& clip, bool canCullToTheRight,
void (*consume)(SkEdgeClipper*, bool newCtr, void* ctx), void* ctx) {
SkASSERT(path.isFinite());
SkASSERT(view.isFinite());
SkAutoConicToQuads quadder;
const SkScalar conicTol = SK_Scalar1 / 4;
SkPathEdgeIter iter(path);
SkPathEdgeIter iter(view);
SkEdgeClipper clipper(canCullToTheRight);
while (auto e = iter.next()) {

View File

@ -10,6 +10,7 @@
#define SkEdgeClipper_DEFINED
#include "include/core/SkPath.h"
#include "src/core/SkPathView.h"
/** This is basically an iterator. It is initialized with an edge and a clip,
and then next() is called until it returns kDone_Verb.
@ -30,7 +31,7 @@ public:
* Clips each segment from the path, and passes the result (in a clipper) to the
* consume proc.
*/
static void ClipPath(const SkPath& path, const SkRect& clip, bool canCullToTheRight,
static void ClipPath(const SkPathView& path, const SkRect& clip, bool canCullToTheRight,
void (*consume)(SkEdgeClipper*, bool newCtr, void* ctx), void* ctx);
private:

View File

@ -230,6 +230,12 @@ void SkPath::swap(SkPath& that) {
}
}
SkPathView SkPath::view() const {
return fPathRef->view(this->getFillType(), this->getConvexityType());
}
///////////////////////////////////////////////////////////////////////////////////////////////////
bool SkPath::isInterpolatable(const SkPath& compare) const {
// need the same structure (verbs, conicweights) and same point-count
return fPathRef->fPoints.count() == compare.fPathRef->fPoints.count() &&
@ -467,7 +473,7 @@ bool SkPath::isRect(SkRect* rect, bool* isClosed, SkPathDirection* direction) co
SkDEBUGCODE(this->validate();)
int currVerb = 0;
const SkPoint* pts = fPathRef->points();
return SkPathPriv::IsRectContour(*this, false, &currVerb, &pts, isClosed, direction, rect);
return SkPathPriv::IsRectContour(this->view(), false, &currVerb, &pts, isClosed, direction, rect);
}
bool SkPath::isOval(SkRect* bounds) const {
@ -3425,7 +3431,7 @@ SkPath SkPath::Polygon(const SkPoint pts[], int count, bool isClosed,
//////////////////////////////////////////////////////////////////////////////////////////////////
bool SkPathPriv::IsRectContour(const SkPath& path, bool allowPartial, int* currVerb,
bool SkPathPriv::IsRectContour(const SkPathView& path, bool allowPartial, int* currVerb,
const SkPoint** ptsPtr, bool* isClosed, SkPathDirection* direction,
SkRect* rect) {
int corners = 0;
@ -3442,9 +3448,9 @@ bool SkPathPriv::IsRectContour(const SkPath& path, bool allowPartial, int* currV
bool closedOrMoved = false;
bool autoClose = false;
bool insertClose = false;
int verbCnt = path.fPathRef->countVerbs();
int verbCnt = path.fVerbs.count();
while (*currVerb < verbCnt && (!allowPartial || !autoClose)) {
uint8_t verb = insertClose ? (uint8_t) SkPath::kClose_Verb : path.fPathRef->atVerb(*currVerb);
uint8_t verb = insertClose ? (uint8_t) SkPath::kClose_Verb : path.fVerbs[*currVerb];
switch (verb) {
case SkPath::kClose_Verb:
savePts = pts;
@ -3566,10 +3572,9 @@ bool SkPathPriv::IsRectContour(const SkPath& path, bool allowPartial, int* currV
}
bool SkPathPriv::IsNestedFillRects(const SkPath& path, SkRect rects[2], SkPathDirection dirs[2]) {
SkDEBUGCODE(path.validate();)
bool SkPathPriv::IsNestedFillRects(const SkPathView& path, SkRect rects[2], SkPathDirection dirs[2]) {
int currVerb = 0;
const SkPoint* pts = path.fPathRef->points();
const SkPoint* pts = path.fPoints.begin();
SkPathDirection testDirs[2];
SkRect testRects[2];
if (!IsRectContour(path, true, &currVerb, &pts, nullptr, &testDirs[0], &testRects[0])) {
@ -3706,7 +3711,7 @@ static void clip(const SkPath& path, const SkHalfPlane& plane, SkPath* clippedPa
SkPoint fPrev;
} rec = { clippedPath, {0, 0} };
SkEdgeClipper::ClipPath(rotated, clip, false,
SkEdgeClipper::ClipPath(rotated.view(), clip, false,
[](SkEdgeClipper* clipper, bool newCtr, void* ctx) {
Rec* rec = (Rec*)ctx;

View File

@ -10,6 +10,7 @@
#include "include/core/SkPath.h"
#include "include/private/SkIDChangeListener.h"
#include "src/core/SkPathView.h"
static_assert(0 == static_cast<int>(SkPathFillType::kWinding), "fill_type_mismatch");
static_assert(1 == static_cast<int>(SkPathFillType::kEvenOdd), "fill_type_mismatch");
@ -167,6 +168,13 @@ public:
: path.fPathRef->verbsEnd(),
path.fPathRef->points(), path.fPathRef->conicWeights()) {
}
Iterate(const SkPathView& path)
: Iterate(path.fVerbs.begin(),
// Don't allow iteration through non-finite points.
(!path.isFinite()) ? path.fVerbs.begin()
: path.fVerbs.end(),
path.fPoints.begin(), path.fWeights.begin()) {
}
Iterate(const uint8_t* verbsBegin, const uint8_t* verbsEnd, const SkPoint* points,
const SkScalar* weights)
: fVerbsBegin(verbsBegin), fVerbsEnd(verbsEnd), fPoints(points), fWeights(weights) {
@ -323,7 +331,7 @@ public:
return true;
}
static bool IsRectContour(const SkPath&, bool allowPartial, int* currVerb,
static bool IsRectContour(const SkPathView&, bool allowPartial, int* currVerb,
const SkPoint** ptsPtr, bool* isClosed, SkPathDirection* direction,
SkRect* rect);
@ -338,9 +346,14 @@ public:
@param dirs storage for SkPathDirection pair; may be nullptr
@return true if SkPath contains nested SkRect pair
*/
static bool IsNestedFillRects(const SkPath&, SkRect rect[2],
static bool IsNestedFillRects(const SkPathView&, SkRect rect[2],
SkPathDirection dirs[2] = nullptr);
static bool IsNestedFillRects(const SkPath& path, SkRect rect[2],
SkPathDirection dirs[2] = nullptr) {
return IsNestedFillRects(path.view(), rect, dirs);
}
static bool IsInverseFillType(SkPathFillType fill) {
return (static_cast<int>(fill) & 2) != 0;
}
@ -402,7 +415,8 @@ class SkPathEdgeIter {
};
public:
SkPathEdgeIter(const SkPath& path);
SkPathEdgeIter(const SkPath&);
SkPathEdgeIter(const SkPathView&);
SkScalar conicWeight() const {
SkASSERT(fIsConic);

View File

@ -13,6 +13,7 @@
#include "include/private/SkTo.h"
#include "src/core/SkBuffer.h"
#include "src/core/SkPathPriv.h"
#include "src/core/SkPathView.h"
#include "src/core/SkSafeMath.h"
//////////////////////////////////////////////////////////////////////////////
@ -683,6 +684,21 @@ bool SkPathRef::isValid() const {
//////////////////////////////////////////////////////////////////////////////////////////////////
#include "src/core/SkPathView.h"
SkPathView SkPathRef::view(SkPathFillType ft, SkPathConvexityType ct) const {
return {
{ fPoints.begin(), fPoints.size() },
{ fVerbs.begin(), fVerbs.size() },
{ fConicWeights.begin(), fConicWeights.size() },
ft,
ct,
this->getBounds(), // don't use fBounds, in case our bounds are dirty
fSegmentMask,
this->isFinite(),
};
}
SkPathEdgeIter::SkPathEdgeIter(const SkPath& path) {
fMoveToPtr = fPts = path.fPathRef->points();
fVerbs = path.fPathRef->verbsBegin();
@ -696,3 +712,55 @@ SkPathEdgeIter::SkPathEdgeIter(const SkPath& path) {
fNextIsNewContour = false;
SkDEBUGCODE(fIsConic = false;)
}
SkPathEdgeIter::SkPathEdgeIter(const SkPathView& path) {
fMoveToPtr = fPts = path.fPoints.cbegin();
fVerbs = path.fVerbs.cbegin();
fVerbsStop = path.fVerbs.cend();
fConicWeights = path.fWeights.cbegin();
if (fConicWeights) {
fConicWeights -= 1; // begin one behind
}
fNeedsCloseLine = false;
fNextIsNewContour = false;
SkDEBUGCODE(fIsConic = false;)
}
bool SkPathView::isRect(SkRect* rect) const {
SkPathDirection direction;
bool isClosed;
int currVerb = 0;
const SkPoint* pts = fPoints.begin();
return SkPathPriv::IsRectContour(*this, false, &currVerb, &pts, &isClosed, &direction, rect);
}
#ifdef SK_DEBUG
void SkPathView::validate() const {
bool finite = SkScalarsAreFinite((const SkScalar*)fPoints.begin(), fPoints.count() * 2)
&& SkScalarsAreFinite( fWeights.begin(), fWeights.count());
SkASSERT(fIsFinite == finite);
if (fIsFinite) {
SkRect bounds;
bounds.setBounds(fPoints.begin(), fPoints.count());
SkASSERT(bounds == fBounds);
} else {
SkASSERT(fBounds.isEmpty());
}
unsigned mask = 0;
for (auto v : fVerbs) {
switch (static_cast<SkPathVerb>(v)) {
default: break;
case SkPathVerb::kLine: mask |= kLine_SkPathSegmentMask; break;
case SkPathVerb::kQuad: mask |= kQuad_SkPathSegmentMask; break;
case SkPathVerb::kConic: mask |= kConic_SkPathSegmentMask; break;
case SkPathVerb::kCubic: mask |= kCubic_SkPathSegmentMask; break;
}
}
SkASSERT(mask == fSegmentMask);
}
#endif

78
src/core/SkPathView.h Normal file
View File

@ -0,0 +1,78 @@
/*
* Copyright 2020 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#pragma once
#include "include/core/SkPathTypes.h"
#include "include/core/SkPoint.h"
#include "include/core/SkRect.h"
#include "src/core/SkSpan.h"
struct SkPathView {
SkPathView(SkSpan<const SkPoint> points, SkSpan<const uint8_t> verbs, SkSpan<const float> weights,
SkPathFillType ft, SkPathConvexityType ct, const SkRect& bounds, unsigned segmentMask,
bool isFinite)
: fPoints(points)
, fVerbs(verbs)
, fWeights(weights)
, fBounds(bounds)
, fFillType(ft)
, fConvexity(ct)
, fSegmentMask(segmentMask)
, fIsFinite(isFinite)
{
this->validate();
}
bool isInverseFillType() const { return SkPathFillType_IsInverse(fFillType); }
bool isConvex() const { return fConvexity == SkPathConvexityType::kConvex; }
bool isEmpty() const { return fPoints.size() == 0; }
bool isRect(SkRect*) const;
bool isFinite() const { return fIsFinite; }
SkSpan<const SkPoint> fPoints;
SkSpan<const uint8_t> fVerbs;
SkSpan<const float> fWeights;
SkRect fBounds;
SkPathFillType fFillType;
SkPathConvexityType fConvexity;
uint8_t fSegmentMask;
bool fIsFinite;
#ifdef SK_DEBUG
void validate() const;
#else
void validate() const {}
#endif
};
static inline SkPathView SkPathView_triangle(const SkPoint pts[3], const SkRect& bounds) {
static constexpr uint8_t verbs[] = {
(uint8_t)SkPathVerb::kMove,
(uint8_t)SkPathVerb::kLine,
(uint8_t)SkPathVerb::kLine,
};
return SkPathView({pts, 3}, SkMakeSpan(verbs), {},
SkPathFillType::kWinding, SkPathConvexityType::kConvex,
bounds, kLine_SkPathSegmentMask, true);
}
static inline SkPathView SkPathView_quad(const SkPoint pts[4], const SkRect& bounds) {
static constexpr uint8_t verbs[] = {
(uint8_t)SkPathVerb::kMove,
(uint8_t)SkPathVerb::kLine,
(uint8_t)SkPathVerb::kLine,
(uint8_t)SkPathVerb::kLine,
};
return SkPathView({pts, 4}, SkMakeSpan(verbs), {},
SkPathFillType::kWinding, SkPathConvexityType::kConvex,
bounds, kLine_SkPathSegmentMask, true);
};

View File

@ -365,7 +365,7 @@ bool SkRegion::setPath(const SkPath& path, const SkRegion& clip) {
return this->setEmpty();
}
SkScan::FillPath(path, clip, &builder);
SkScan::FillPath(path.view(), clip, &builder);
builder.done();
int count = builder.computeRunCount();

View File

@ -11,12 +11,13 @@
#include "include/core/SkRect.h"
#include "include/private/SkFixed.h"
#include "src/core/SkPathView.h"
#include <atomic>
class SkRasterClip;
class SkRegion;
class SkBlitter;
class SkPath;
/** Defines a fixed-point rectangle, identical to the integer SkIRect, but its
coordinates are treated as SkFixed rather than int32_t.
@ -39,7 +40,7 @@ public:
typedef void (*HairRgnProc)(const SkPoint[], int count, const SkRegion*, SkBlitter*);
typedef void (*HairRCProc)(const SkPoint[], int count, const SkRasterClip&, SkBlitter*);
static void FillPath(const SkPath&, const SkIRect&, SkBlitter*);
static void FillPath(const SkPathView&, const SkIRect&, SkBlitter*);
///////////////////////////////////////////////////////////////////////////
// rasterclip
@ -49,8 +50,8 @@ public:
static void FillRect(const SkRect&, const SkRasterClip&, SkBlitter*);
static void AntiFillRect(const SkRect&, const SkRasterClip&, SkBlitter*);
static void AntiFillXRect(const SkXRect&, const SkRasterClip&, SkBlitter*);
static void FillPath(const SkPath&, const SkRasterClip&, SkBlitter*);
static void AntiFillPath(const SkPath&, const SkRasterClip&, SkBlitter*);
static void FillPath(const SkPathView&, const SkRasterClip&, SkBlitter*);
static void AntiFillPath(const SkPathView&, const SkRasterClip&, SkBlitter*);
static void FrameRect(const SkRect&, const SkPoint& strokeSize,
const SkRasterClip&, SkBlitter*);
static void AntiFrameRect(const SkRect&, const SkPoint& strokeSize,
@ -60,15 +61,15 @@ public:
static void AntiHairLine(const SkPoint[], int count, const SkRasterClip&, SkBlitter*);
static void HairRect(const SkRect&, const SkRasterClip&, SkBlitter*);
static void AntiHairRect(const SkRect&, const SkRasterClip&, SkBlitter*);
static void HairPath(const SkPath&, const SkRasterClip&, SkBlitter*);
static void AntiHairPath(const SkPath&, const SkRasterClip&, SkBlitter*);
static void HairSquarePath(const SkPath&, const SkRasterClip&, SkBlitter*);
static void AntiHairSquarePath(const SkPath&, const SkRasterClip&, SkBlitter*);
static void HairRoundPath(const SkPath&, const SkRasterClip&, SkBlitter*);
static void AntiHairRoundPath(const SkPath&, const SkRasterClip&, SkBlitter*);
static void HairPath(const SkPathView&, const SkRasterClip&, SkBlitter*);
static void AntiHairPath(const SkPathView&, const SkRasterClip&, SkBlitter*);
static void HairSquarePath(const SkPathView&, const SkRasterClip&, SkBlitter*);
static void AntiHairSquarePath(const SkPathView&, const SkRasterClip&, SkBlitter*);
static void HairRoundPath(const SkPathView&, const SkRasterClip&, SkBlitter*);
static void AntiHairRoundPath(const SkPathView&, const SkRasterClip&, SkBlitter*);
// Needed by do_fill_path in SkScanPriv.h
static void FillPath(const SkPath&, const SkRegion& clip, SkBlitter*);
static void FillPath(const SkPathView&, const SkRegion& clip, SkBlitter*);
private:
friend class SkAAClip;
@ -79,16 +80,16 @@ private:
static void FillRect(const SkRect&, const SkRegion* clip, SkBlitter*);
static void AntiFillRect(const SkRect&, const SkRegion* clip, SkBlitter*);
static void AntiFillXRect(const SkXRect&, const SkRegion*, SkBlitter*);
static void AntiFillPath(const SkPath&, const SkRegion& clip, SkBlitter*, bool forceRLE);
static void AntiFillPath(const SkPathView&, const SkRegion& clip, SkBlitter*, bool forceRLE);
static void FillTriangle(const SkPoint pts[], const SkRegion*, SkBlitter*);
static void AntiFrameRect(const SkRect&, const SkPoint& strokeSize,
const SkRegion*, SkBlitter*);
static void HairLineRgn(const SkPoint[], int count, const SkRegion*, SkBlitter*);
static void AntiHairLineRgn(const SkPoint[], int count, const SkRegion*, SkBlitter*);
static void AAAFillPath(const SkPath& path, SkBlitter* blitter, const SkIRect& pathIR,
static void AAAFillPath(const SkPathView& path, SkBlitter* blitter, const SkIRect& pathIR,
const SkIRect& clipBounds, bool forceRLE);
static void SAAFillPath(const SkPath& path, SkBlitter* blitter, const SkIRect& pathIR,
static void SAAFillPath(const SkPathView& path, SkBlitter* blitter, const SkIRect& pathIR,
const SkIRect& clipBounds, bool forceRLE);
};

View File

@ -33,7 +33,7 @@ private:
const SkIRect* fClipRect;
};
void sk_fill_path(const SkPath& path, const SkIRect& clipRect,
void sk_fill_path(const SkPathView&, const SkIRect& clipRect,
SkBlitter* blitter, int start_y, int stop_y, int shiftEdgesUp,
bool pathContainedInClip);
@ -81,7 +81,7 @@ static EdgeType* backward_insert_start(EdgeType* prev, SkFixed x) {
}
// Check if the path is a rect and fat enough after clipping; if so, blit it.
static inline bool TryBlitFatAntiRect(SkBlitter* blitter, const SkPath& path, const SkIRect& clip) {
static inline bool TryBlitFatAntiRect(SkBlitter* blitter, const SkPathView& path, const SkIRect& clip) {
SkRect rect;
if (!path.isRect(&rect)) {
return false; // not rect

View File

@ -25,7 +25,7 @@
#include <utility>
#if defined(SK_DISABLE_AAA)
void SkScan::AAAFillPath(const SkPath&, SkBlitter*, const SkIRect&, const SkIRect&, bool) {
void SkScan::AAAFillPath(const SkPathView&, SkBlitter*, const SkIRect&, const SkIRect&, bool) {
SkDEBUGFAIL("AAA Disabled");
return;
}
@ -1837,8 +1837,10 @@ static void aaa_walk_edges(SkAnalyticEdge* prevHead,
}
}
#include "src/core/SkPathView.h"
static SK_ALWAYS_INLINE void aaa_fill_path(
const SkPath& path,
const SkPathView& path,
const SkIRect& clipRect,
AdditiveBlitter* blitter,
int start_y,
@ -1914,7 +1916,7 @@ static SK_ALWAYS_INLINE void aaa_fill_path(
// If we're using mask, then we have to limit the bound within the path bounds.
// Otherwise, the edge drift may access an invalid address inside the mask.
SkIRect ir;
path.getBounds().roundOut(&ir);
path.fBounds.roundOut(&ir);
leftBound = std::max(leftBound, SkIntToFixed(ir.fLeft));
rightBound = std::min(rightBound, SkIntToFixed(ir.fRight));
}
@ -1930,11 +1932,11 @@ static SK_ALWAYS_INLINE void aaa_fill_path(
// We skip intersection computation if there are many points which probably already
// give us enough fractional scan lines.
bool skipIntersect = path.countPoints() > (stop_y - start_y) * 2;
bool skipIntersect = path.fPoints.count() > (stop_y - start_y) * 2;
aaa_walk_edges(&headEdge,
&tailEdge,
path.getFillType(),
path.fFillType,
blitter,
start_y,
stop_y,
@ -1947,7 +1949,7 @@ static SK_ALWAYS_INLINE void aaa_fill_path(
}
}
void SkScan::AAAFillPath(const SkPath& path,
void SkScan::AAAFillPath(const SkPathView& path,
SkBlitter* blitter,
const SkIRect& ir,
const SkIRect& clipBounds,

View File

@ -603,18 +603,18 @@ constexpr int kSampleSize = 8;
constexpr SkScalar kComplexityThreshold = 0.25;
#endif
static void compute_complexity(const SkPath& path, SkScalar& avgLength, SkScalar& complexity) {
int n = path.countPoints();
if (n < kSampleSize || path.getBounds().isEmpty()) {
static void compute_complexity(const SkPathView& path, SkScalar& avgLength, SkScalar& complexity) {
int n = path.fPoints.count();
if (n < kSampleSize || path.fBounds.isEmpty()) {
// set to invalid value to indicate that we failed to compute
avgLength = complexity = -1;
return;
}
SkScalar sumLength = 0;
SkPoint lastPoint = path.getPoint(0);
SkPoint lastPoint = path.fPoints[0];
for(int i = 1; i < kSampleSize; ++i) {
SkPoint point = path.getPoint(i);
SkPoint point = path.fPoints[i];
sumLength += SkPoint::Distance(lastPoint, point);
lastPoint = point;
}
@ -622,21 +622,21 @@ static void compute_complexity(const SkPath& path, SkScalar& avgLength, SkScalar
auto sqr = [](SkScalar x) { return x*x; };
SkScalar diagonalSqr = sqr(path.getBounds().width()) + sqr(path.getBounds().height());
SkScalar diagonalSqr = sqr(path.fBounds.width()) + sqr(path.fBounds.height());
// If the path consists of random line segments, the number of intersections should be
// proportional to this.
SkScalar intersections = sk_ieee_float_divide(sqr(n) * sqr(avgLength), diagonalSqr);
// The number of intersections per scanline should be proportional to this number.
complexity = sk_ieee_float_divide(intersections, path.getBounds().height());
complexity = sk_ieee_float_divide(intersections, path.fBounds.height());
if (sk_float_isnan(complexity)) { // it may be possible to have 0.0 / 0.0; inf is fine for us.
complexity = -1;
}
}
static bool ShouldUseAAA(const SkPath& path, SkScalar avgLength, SkScalar complexity) {
static bool ShouldUseAAA(const SkPathView& path, SkScalar avgLength, SkScalar complexity) {
#if defined(SK_DISABLE_AAA)
return false;
#else
@ -651,16 +651,16 @@ static bool ShouldUseAAA(const SkPath& path, SkScalar avgLength, SkScalar comple
}
#ifdef SK_SUPPORT_LEGACY_AAA_CHOICE
const SkRect& bounds = path.getBounds();
const SkRect& bounds = path.fBounds;
// When the path have so many points compared to the size of its
// bounds/resolution, it indicates that the path is not quite smooth in
// the current resolution: the expected number of turning points in
// every pixel row/column is significantly greater than zero. Hence
// Aanlytic AA is not likely to produce visible quality improvements,
// and Analytic AA might be slower than supersampling.
return path.countPoints() < std::max(bounds.width(), bounds.height()) / 2 - 10;
return path.fPoints.count() < std::max(bounds.width(), bounds.height()) / 2 - 10;
#else
if (path.countPoints() >= path.getBounds().height()) {
if (path.fPoints.count() >= path.fBounds.height()) {
// SAA is faster than AAA in this case even if there are no
// intersections because AAA will have too many scan lines. See
// skbug.com/8272
@ -672,7 +672,7 @@ static bool ShouldUseAAA(const SkPath& path, SkScalar avgLength, SkScalar comple
#endif
}
void SkScan::SAAFillPath(const SkPath& path, SkBlitter* blitter, const SkIRect& ir,
void SkScan::SAAFillPath(const SkPathView& path, SkBlitter* blitter, const SkIRect& ir,
const SkIRect& clipBounds, bool forceRLE) {
bool containedInClip = clipBounds.contains(ir);
bool isInverse = path.isInverseFillType();
@ -681,7 +681,7 @@ void SkScan::SAAFillPath(const SkPath& path, SkBlitter* blitter, const SkIRect&
// if we're an inverse filltype
if (!isInverse && MaskSuperBlitter::CanHandleRect(ir) && !forceRLE) {
MaskSuperBlitter superBlit(blitter, ir, clipBounds, isInverse);
SkASSERT(SkIntToScalar(ir.fTop) <= path.getBounds().fTop);
SkASSERT(SkIntToScalar(ir.fTop) <= path.fBounds.fTop);
sk_fill_path(path, clipBounds, &superBlit, ir.fTop, ir.fBottom, SHIFT, containedInClip);
} else {
SuperBlitter superBlit(blitter, ir, clipBounds, isInverse);
@ -712,14 +712,14 @@ static int rect_overflows_short_shift(SkIRect rect, int shift) {
overflows_short_shift(rect.fBottom, shift);
}
void SkScan::AntiFillPath(const SkPath& path, const SkRegion& origClip,
void SkScan::AntiFillPath(const SkPathView& path, const SkRegion& origClip,
SkBlitter* blitter, bool forceRLE) {
if (origClip.isEmpty()) {
return;
}
const bool isInverse = path.isInverseFillType();
SkIRect ir = safeRoundOut(path.getBounds());
SkIRect ir = safeRoundOut(path.fBounds);
if (ir.isEmpty()) {
if (isInverse) {
blitter->blitRegion(origClip);
@ -803,7 +803,7 @@ void SkScan::AntiFillPath(const SkPath& path, const SkRegion& origClip,
#include "src/core/SkRasterClip.h"
void SkScan::FillPath(const SkPath& path, const SkRasterClip& clip, SkBlitter* blitter) {
void SkScan::FillPath(const SkPathView& path, const SkRasterClip& clip, SkBlitter* blitter) {
if (clip.isEmpty() || !path.isFinite()) {
return;
}
@ -820,7 +820,7 @@ void SkScan::FillPath(const SkPath& path, const SkRasterClip& clip, SkBlitter* b
}
}
void SkScan::AntiFillPath(const SkPath& path, const SkRasterClip& clip, SkBlitter* blitter) {
void SkScan::AntiFillPath(const SkPathView& path, const SkRasterClip& clip, SkBlitter* blitter) {
if (clip.isEmpty() || !path.isFinite()) {
return;
}

View File

@ -499,7 +499,7 @@ void extend_pts(SkPath::Verb prevVerb, SkPath::Verb nextVerb, SkPoint* pts, int
}
template <SkPaint::Cap capStyle>
void hair_path(const SkPath& path, const SkRasterClip& rclip, SkBlitter* blitter,
void hair_path(const SkPathView& path, const SkRasterClip& rclip, SkBlitter* blitter,
SkScan::HairRgnProc lineproc) {
if (path.isEmpty()) {
return;
@ -513,7 +513,7 @@ void hair_path(const SkPath& path, const SkRasterClip& rclip, SkBlitter* blitter
{
const int capOut = SkPaint::kButt_Cap == capStyle ? 1 : 2;
const SkIRect ibounds = path.getBounds().roundOut().makeOutset(capOut, capOut);
const SkIRect ibounds = path.fBounds.roundOut().makeOutset(capOut, capOut);
if (rclip.quickReject(ibounds)) {
return;
}
@ -635,27 +635,27 @@ void hair_path(const SkPath& path, const SkRasterClip& rclip, SkBlitter* blitter
}
}
void SkScan::HairPath(const SkPath& path, const SkRasterClip& clip, SkBlitter* blitter) {
void SkScan::HairPath(const SkPathView& path, const SkRasterClip& clip, SkBlitter* blitter) {
hair_path<SkPaint::kButt_Cap>(path, clip, blitter, SkScan::HairLineRgn);
}
void SkScan::AntiHairPath(const SkPath& path, const SkRasterClip& clip, SkBlitter* blitter) {
void SkScan::AntiHairPath(const SkPathView& path, const SkRasterClip& clip, SkBlitter* blitter) {
hair_path<SkPaint::kButt_Cap>(path, clip, blitter, SkScan::AntiHairLineRgn);
}
void SkScan::HairSquarePath(const SkPath& path, const SkRasterClip& clip, SkBlitter* blitter) {
void SkScan::HairSquarePath(const SkPathView& path, const SkRasterClip& clip, SkBlitter* blitter) {
hair_path<SkPaint::kSquare_Cap>(path, clip, blitter, SkScan::HairLineRgn);
}
void SkScan::AntiHairSquarePath(const SkPath& path, const SkRasterClip& clip, SkBlitter* blitter) {
void SkScan::AntiHairSquarePath(const SkPathView& path, const SkRasterClip& clip, SkBlitter* blitter) {
hair_path<SkPaint::kSquare_Cap>(path, clip, blitter, SkScan::AntiHairLineRgn);
}
void SkScan::HairRoundPath(const SkPath& path, const SkRasterClip& clip, SkBlitter* blitter) {
void SkScan::HairRoundPath(const SkPathView& path, const SkRasterClip& clip, SkBlitter* blitter) {
hair_path<SkPaint::kRound_Cap>(path, clip, blitter, SkScan::HairLineRgn);
}
void SkScan::AntiHairRoundPath(const SkPath& path, const SkRasterClip& clip, SkBlitter* blitter) {
void SkScan::AntiHairRoundPath(const SkPathView& path, const SkRasterClip& clip, SkBlitter* blitter) {
hair_path<SkPaint::kRound_Cap>(path, clip, blitter, SkScan::AntiHairLineRgn);
}

View File

@ -391,8 +391,10 @@ static SkEdge* sort_edges(SkEdge* list[], int count, SkEdge** last) {
return list[0];
}
#include "src/core/SkPathView.h"
// clipRect has not been shifted up
void sk_fill_path(const SkPath& path, const SkIRect& clipRect, SkBlitter* blitter,
void sk_fill_path(const SkPathView& path, const SkIRect& clipRect, SkBlitter* blitter,
int start_y, int stop_y, int shiftEdgesUp, bool pathContainedInClip) {
SkASSERT(blitter);
@ -470,8 +472,7 @@ void sk_fill_path(const SkPath& path, const SkIRect& clipRect, SkBlitter* blitte
if (path.isConvex() && (nullptr == proc) && count >= 2) {
walk_simple_edges(&headEdge, blitter, start_y, stop_y);
} else {
walk_edges(&headEdge, path.getFillType(), blitter, start_y, stop_y, proc,
shiftedClip.right());
walk_edges(&headEdge, path.fFillType, blitter, start_y, stop_y, proc, shiftedClip.right());
}
}
@ -613,8 +614,7 @@ static SkIRect conservative_round_to_int(const SkRect& src) {
};
}
void SkScan::FillPath(const SkPath& path, const SkRegion& origClip,
SkBlitter* blitter) {
void SkScan::FillPath(const SkPathView& path, const SkRegion& origClip, SkBlitter* blitter) {
if (origClip.isEmpty()) {
return;
}
@ -632,7 +632,7 @@ void SkScan::FillPath(const SkPath& path, const SkRegion& origClip,
// don't reference "origClip" any more, just use clipPtr
SkRect bounds = path.getBounds();
SkRect bounds = path.fBounds;
bool irPreClipped = false;
if (!SkRectPriv::MakeLargeS32().contains(bounds)) {
if (!bounds.intersect(SkRectPriv::MakeLargeS32())) {
@ -670,7 +670,7 @@ void SkScan::FillPath(const SkPath& path, const SkRegion& origClip,
}
}
void SkScan::FillPath(const SkPath& path, const SkIRect& ir,
void SkScan::FillPath(const SkPathView& path, const SkIRect& ir,
SkBlitter* blitter) {
SkRegion rgn(ir);
FillPath(path, rgn, blitter);
@ -737,8 +737,7 @@ static void sk_fill_triangle(const SkPoint pts[], const SkIRect* clipRect,
walk_simple_edges(&headEdge, blitter, start_y, stop_y);
}
void SkScan::FillTriangle(const SkPoint pts[], const SkRasterClip& clip,
SkBlitter* blitter) {
void SkScan::FillTriangle(const SkPoint pts[], const SkRasterClip& clip, SkBlitter* blitter) {
if (clip.isEmpty()) {
return;
}
@ -750,9 +749,7 @@ void SkScan::FillTriangle(const SkPoint pts[], const SkRasterClip& clip,
// Use FixedMax/2 as the limit so we can subtract two edges and still store that in Fixed.
const SkScalar limit = SK_MaxS16 >> 1;
if (!SkRect::MakeLTRB(-limit, -limit, limit, limit).contains(r)) {
SkPath path;
path.addPoly(pts, 3, false);
FillPath(path, clip, blitter);
FillPath(SkPathView_triangle(pts, r), clip, blitter);
return;
}

View File

@ -42,7 +42,7 @@ DEF_TEST(FillPathInverse, reporter) {
SkIntToScalar(width), 0.0f)
.close()
.setFillType(SkPathFillType::kInverseWinding);
SkScan::FillPath(path, clip, &blitter);
SkScan::FillPath(path.view(), clip, &blitter);
REPORTER_ASSERT(reporter, blitter.m_blitCount == expected_lines);
}

View File

@ -22,6 +22,7 @@
#include "src/core/SkAutoMalloc.h"
#include "src/core/SkGeometry.h"
#include "src/core/SkPathPriv.h"
#include "src/core/SkPathView.h"
#include "src/core/SkReadBuffer.h"
#include "src/core/SkWriteBuffer.h"
#include "tests/Test.h"
@ -5554,12 +5555,13 @@ void survive(SkPath* path, const Xforms& x, bool isAxisAligned, skiatest::Report
// getConvexityTypeOrUnknown() instead of getConvexityType().
path->transform(x.fRM, &path2);
path->transform(x.fRM);
REPORTER_ASSERT(reporter, path2.getConvexityTypeOrUnknown() != SkPathConvexityType::kConvex);
REPORTER_ASSERT(reporter, path->getConvexityTypeOrUnknown() != SkPathConvexityType::kConvex);
if (isAxisAligned) {
REPORTER_ASSERT(reporter, !isa_proc(path2));
REPORTER_ASSERT(reporter, !isa_proc(*path));
}
REPORTER_ASSERT(reporter, path2.getConvexityTypeOrUnknown() != SkPathConvexityType::kConvex);
REPORTER_ASSERT(reporter, path->getConvexityTypeOrUnknown() != SkPathConvexityType::kConvex);
}
DEF_TEST(Path_survive_transform, r) {
@ -5626,13 +5628,18 @@ static void test_edger(skiatest::Reporter* r,
}
SkPathEdgeIter iter(path);
SkPathEdgeIter iter2(path.view());
for (auto v : expected) {
auto e = iter.next();
REPORTER_ASSERT(r, e);
REPORTER_ASSERT(r, SkPathEdgeIter::EdgeToVerb(e.fEdge) == v);
e = iter2.next();
REPORTER_ASSERT(r, e);
REPORTER_ASSERT(r, SkPathEdgeIter::EdgeToVerb(e.fEdge) == v);
}
auto e = iter.next();
REPORTER_ASSERT(r, !e);
REPORTER_ASSERT(r, !iter.next());
REPORTER_ASSERT(r, !iter2.next());
}
DEF_TEST(pathedger, r) {