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:
parent
24a7f071ae
commit
375721d7bb
@ -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
|
||||
|
@ -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.
|
||||
|
@ -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;
|
||||
|
||||
|
@ -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();
|
||||
|
@ -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,
|
||||
|
@ -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);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -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);
|
||||
|
||||
|
@ -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;
|
||||
|
@ -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()) {
|
||||
|
@ -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:
|
||||
|
@ -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;
|
||||
|
||||
|
@ -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);
|
||||
|
@ -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
78
src/core/SkPathView.h
Normal 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);
|
||||
};
|
@ -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();
|
||||
|
@ -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);
|
||||
};
|
||||
|
||||
|
@ -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
|
||||
|
@ -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,
|
||||
|
@ -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;
|
||||
}
|
||||
|
@ -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);
|
||||
}
|
||||
|
||||
|
@ -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;
|
||||
}
|
||||
|
||||
|
@ -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);
|
||||
}
|
||||
|
@ -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) {
|
||||
|
Loading…
Reference in New Issue
Block a user