Revert of Hairline batch (patchset #17 id:360001 of https://codereview.chromium.org/876673002/)

Reason for revert:
still a performance regression

Original issue's description:
> Hairline batch
>
> BUG=skia:
>
> Committed: https://skia.googlesource.com/skia/+/6eff8701f027016fbb3147412ec2292dcec2b7f5
>
> Committed: https://skia.googlesource.com/skia/+/658d55cd6121c67488aaf5d0832c9712737f26a5

TBR=bsalomon@google.com,joshualitt@chromium.org
NOPRESUBMIT=true
NOTREECHECKS=true
NOTRY=true
BUG=skia:

Review URL: https://codereview.chromium.org/882883003
This commit is contained in:
joshualitt 2015-02-03 08:40:22 -08:00 committed by Commit bot
parent 7e80c889de
commit 46c77f76be
7 changed files with 322 additions and 503 deletions

View File

@ -7,9 +7,6 @@
#include "GrAAHairLinePathRenderer.h" #include "GrAAHairLinePathRenderer.h"
#include "GrBatch.h"
#include "GrBatchTarget.h"
#include "GrBufferAllocPool.h"
#include "GrContext.h" #include "GrContext.h"
#include "GrDefaultGeoProcFactory.h" #include "GrDefaultGeoProcFactory.h"
#include "GrDrawTargetCaps.h" #include "GrDrawTargetCaps.h"
@ -256,14 +253,14 @@ int num_quad_subdivs(const SkPoint p[3]) {
* subdivide large quads to reduce over-fill. This subdivision has to be * subdivide large quads to reduce over-fill. This subdivision has to be
* performed before applying the perspective matrix. * performed before applying the perspective matrix.
*/ */
int gather_lines_and_quads(const SkPath& path, int generate_lines_and_quads(const SkPath& path,
const SkMatrix& m, const SkMatrix& m,
const SkIRect& devClipBounds, const SkIRect& devClipBounds,
GrAAHairLinePathRenderer::PtArray* lines, GrAAHairLinePathRenderer::PtArray* lines,
GrAAHairLinePathRenderer::PtArray* quads, GrAAHairLinePathRenderer::PtArray* quads,
GrAAHairLinePathRenderer::PtArray* conics, GrAAHairLinePathRenderer::PtArray* conics,
GrAAHairLinePathRenderer::IntArray* quadSubdivCnts, GrAAHairLinePathRenderer::IntArray* quadSubdivCnts,
GrAAHairLinePathRenderer::FloatArray* conicWeights) { GrAAHairLinePathRenderer::FloatArray* conicWeights) {
SkPath::Iter iter(path, false); SkPath::Iter iter(path, false);
int totalQuadCount = 0; int totalQuadCount = 0;
@ -472,7 +469,8 @@ void set_uv_quad(const SkPoint qpts[3], BezierVertex verts[kQuadNumVertices]) {
} }
void bloat_quad(const SkPoint qpts[3], const SkMatrix* toDevice, void bloat_quad(const SkPoint qpts[3], const SkMatrix* toDevice,
const SkMatrix* toSrc, BezierVertex verts[kQuadNumVertices]) { const SkMatrix* toSrc, BezierVertex verts[kQuadNumVertices],
SkRect* devBounds) {
SkASSERT(!toDevice == !toSrc); SkASSERT(!toDevice == !toSrc);
// original quad is specified by tri a,b,c // original quad is specified by tri a,b,c
SkPoint a = qpts[0]; SkPoint a = qpts[0];
@ -537,6 +535,7 @@ void bloat_quad(const SkPoint qpts[3], const SkMatrix* toDevice,
c1.fPos -= cbN; c1.fPos -= cbN;
intersect_lines(a0.fPos, abN, c0.fPos, cbN, &b0.fPos); intersect_lines(a0.fPos, abN, c0.fPos, cbN, &b0.fPos);
devBounds->growToInclude(&verts[0].fPos, sizeof(BezierVertex), kQuadNumVertices);
if (toSrc) { if (toSrc) {
toSrc->mapPointsWithStride(&verts[0].fPos, sizeof(BezierVertex), kQuadNumVertices); toSrc->mapPointsWithStride(&verts[0].fPos, sizeof(BezierVertex), kQuadNumVertices);
@ -568,8 +567,9 @@ void add_conics(const SkPoint p[3],
const SkScalar weight, const SkScalar weight,
const SkMatrix* toDevice, const SkMatrix* toDevice,
const SkMatrix* toSrc, const SkMatrix* toSrc,
BezierVertex** vert) { BezierVertex** vert,
bloat_quad(p, toDevice, toSrc, *vert); SkRect* devBounds) {
bloat_quad(p, toDevice, toSrc, *vert, devBounds);
set_conic_coeffs(p, *vert, weight); set_conic_coeffs(p, *vert, weight);
*vert += kQuadNumVertices; *vert += kQuadNumVertices;
} }
@ -578,15 +578,16 @@ void add_quads(const SkPoint p[3],
int subdiv, int subdiv,
const SkMatrix* toDevice, const SkMatrix* toDevice,
const SkMatrix* toSrc, const SkMatrix* toSrc,
BezierVertex** vert) { BezierVertex** vert,
SkRect* devBounds) {
SkASSERT(subdiv >= 0); SkASSERT(subdiv >= 0);
if (subdiv) { if (subdiv) {
SkPoint newP[5]; SkPoint newP[5];
SkChopQuadAtHalf(p, newP); SkChopQuadAtHalf(p, newP);
add_quads(newP + 0, subdiv-1, toDevice, toSrc, vert); add_quads(newP + 0, subdiv-1, toDevice, toSrc, vert, devBounds);
add_quads(newP + 2, subdiv-1, toDevice, toSrc, vert); add_quads(newP + 2, subdiv-1, toDevice, toSrc, vert, devBounds);
} else { } else {
bloat_quad(p, toDevice, toSrc, *vert); bloat_quad(p, toDevice, toSrc, *vert, devBounds);
set_uv_quad(p, *vert); set_uv_quad(p, *vert);
*vert += kQuadNumVertices; *vert += kQuadNumVertices;
} }
@ -641,6 +642,106 @@ void add_line(const SkPoint p[2],
/////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////
bool GrAAHairLinePathRenderer::createLineGeom(GrDrawTarget* target,
GrPipelineBuilder* pipelineBuilder,
const SkMatrix& viewMatrix,
uint8_t coverage,
size_t vertexStride,
GrDrawTarget::AutoReleaseGeometry* arg,
SkRect* devBounds,
const SkPath& path,
const PtArray& lines,
int lineCnt) {
int vertCnt = kLineSegNumVertices * lineCnt;
SkASSERT(vertexStride == sizeof(LineVertex));
if (!arg->set(target, vertCnt, vertexStride, 0)) {
return false;
}
LineVertex* verts = reinterpret_cast<LineVertex*>(arg->vertices());
const SkMatrix* toSrc = NULL;
SkMatrix ivm;
if (viewMatrix.hasPerspective()) {
if (viewMatrix.invert(&ivm)) {
toSrc = &ivm;
}
}
devBounds->set(lines.begin(), lines.count());
for (int i = 0; i < lineCnt; ++i) {
add_line(&lines[2*i], toSrc, coverage, &verts);
}
// All the verts computed by add_line are within sqrt(1^2 + 0.5^2) of the end points.
static const SkScalar kSqrtOfOneAndAQuarter = 1.118f;
// Add a little extra to account for vector normalization precision.
static const SkScalar kOutset = kSqrtOfOneAndAQuarter + SK_Scalar1 / 20;
devBounds->outset(kOutset, kOutset);
return true;
}
bool GrAAHairLinePathRenderer::createBezierGeom(GrDrawTarget* target,
GrPipelineBuilder* pipelineBuilder,
const SkMatrix& viewMatrix,
GrDrawTarget::AutoReleaseGeometry* arg,
SkRect* devBounds,
const SkPath& path,
const PtArray& quads,
int quadCnt,
const PtArray& conics,
int conicCnt,
const IntArray& qSubdivs,
const FloatArray& cWeights,
size_t vertexStride) {
int vertCnt = kQuadNumVertices * quadCnt + kQuadNumVertices * conicCnt;
if (!arg->set(target, vertCnt, vertexStride, 0)) {
return false;
}
BezierVertex* verts = reinterpret_cast<BezierVertex*>(arg->vertices());
const SkMatrix* toDevice = NULL;
const SkMatrix* toSrc = NULL;
SkMatrix ivm;
if (viewMatrix.hasPerspective()) {
if (viewMatrix.invert(&ivm)) {
toDevice = &viewMatrix;
toSrc = &ivm;
}
}
// Seed the dev bounds with some pts known to be inside. Each quad and conic grows the bounding
// box to include its vertices.
SkPoint seedPts[2];
if (quadCnt) {
seedPts[0] = quads[0];
seedPts[1] = quads[2];
} else if (conicCnt) {
seedPts[0] = conics[0];
seedPts[1] = conics[2];
}
if (toDevice) {
toDevice->mapPoints(seedPts, 2);
}
devBounds->set(seedPts[0], seedPts[1]);
int unsubdivQuadCnt = quads.count() / 3;
for (int i = 0; i < unsubdivQuadCnt; ++i) {
SkASSERT(qSubdivs[i] >= 0);
add_quads(&quads[3*i], qSubdivs[i], toDevice, toSrc, &verts, devBounds);
}
// Start Conics
for (int i = 0; i < conicCnt; ++i) {
add_conics(&conics[3*i], cWeights[i], toDevice, toSrc, &verts, devBounds);
}
return true;
}
bool GrAAHairLinePathRenderer::canDrawPath(const GrDrawTarget* target, bool GrAAHairLinePathRenderer::canDrawPath(const GrDrawTarget* target,
const GrPipelineBuilder* pipelineBuilder, const GrPipelineBuilder* pipelineBuilder,
const SkMatrix& viewMatrix, const SkMatrix& viewMatrix,
@ -699,444 +800,13 @@ bool check_bounds(const SkMatrix& viewMatrix, const SkRect& devBounds, void* ver
return true; return true;
} }
class AAHairlineBatch : public GrBatch {
public:
struct Geometry {
GrColor fColor;
uint8_t fCoverage;
SkMatrix fViewMatrix;
SkPath fPath;
SkDEBUGCODE(SkRect fDevBounds;)
SkIRect fDevClipBounds;
};
// TODO Batch itself should not hold on to index buffers. Instead, these should live in the
// cache.
static GrBatch* Create(const Geometry& geometry, const GrIndexBuffer* linesIndexBuffer,
const GrIndexBuffer* quadsIndexBuffer) {
return SkNEW_ARGS(AAHairlineBatch, (geometry, linesIndexBuffer, quadsIndexBuffer));
}
const char* name() const SK_OVERRIDE { return "AAHairlineBatch"; }
void getInvariantOutputColor(GrInitInvariantOutput* out) const SK_OVERRIDE {
// When this is called on a batch, there is only one geometry bundle
out->setKnownFourComponents(fGeoData[0].fColor);
}
void getInvariantOutputCoverage(GrInitInvariantOutput* out) const SK_OVERRIDE {
out->setUnknownSingleComponent();
}
void initBatchOpt(const GrBatchOpt& batchOpt) {
fBatchOpt = batchOpt;
}
void initBatchTracker(const GrPipelineInfo& init) SK_OVERRIDE {
// Handle any color overrides
if (init.fColorIgnored) {
fGeoData[0].fColor = GrColor_ILLEGAL;
} else if (GrColor_ILLEGAL != init.fOverrideColor) {
fGeoData[0].fColor = init.fOverrideColor;
}
// setup batch properties
fBatch.fColorIgnored = init.fColorIgnored;
fBatch.fColor = fGeoData[0].fColor;
fBatch.fUsesLocalCoords = init.fUsesLocalCoords;
fBatch.fCoverageIgnored = init.fCoverageIgnored;
fBatch.fCoverage = fGeoData[0].fCoverage;
SkDEBUGCODE(fBatch.fDevBounds = fGeoData[0].fDevBounds;)
}
void generateGeometry(GrBatchTarget* batchTarget, const GrPipeline* pipeline) SK_OVERRIDE {
int instanceCount = fGeoData.count();
for (int i = 0; i < instanceCount; i++) {
const Geometry& args = fGeoData[i];
// createGeom transforms the geometry to device space when the matrix does not have
// perspective.
SkMatrix vm = args.fViewMatrix;
SkMatrix invert = SkMatrix::I();
if (!args.fViewMatrix.hasPerspective()) {
vm = SkMatrix::I();
if (!args.fViewMatrix.invert(&invert)) {
return;
}
}
int lineCount;
int quadCount;
int conicCount;
PREALLOC_PTARRAY(128) lines;
PREALLOC_PTARRAY(128) quads;
PREALLOC_PTARRAY(128) conics;
IntArray qSubdivs;
FloatArray cWeights;
quadCount = gather_lines_and_quads(args.fPath, args.fViewMatrix, args.fDevClipBounds,
&lines, &quads, &conics, &qSubdivs, &cWeights);
lineCount = lines.count() / 2;
conicCount = conics.count() / 3;
// do lines first
if (lineCount) {
this->generateLines(batchTarget, pipeline, args, vm, invert, lines, lineCount);
}
if (quadCount || conicCount) {
this->generateQuadsAndConics(batchTarget,
pipeline,
args,
quads,
quadCount,
conics,
conicCount,
qSubdivs,
cWeights,
vm,
invert);
}
}
}
SkSTArray<1, Geometry, true>* geoData() { return &fGeoData; }
private:
typedef SkTArray<SkPoint, true> PtArray;
typedef SkTArray<int, true> IntArray;
typedef SkTArray<float, true> FloatArray;
AAHairlineBatch(const Geometry& geometry, const GrIndexBuffer* linesIndexBuffer,
const GrIndexBuffer* quadsIndexBuffer)
: fLinesIndexBuffer(linesIndexBuffer)
, fQuadsIndexBuffer(quadsIndexBuffer) {
SkASSERT(linesIndexBuffer && quadsIndexBuffer);
this->initClassID<AAHairlineBatch>();
fGeoData.push_back(geometry);
}
bool onCombineIfPossible(GrBatch* t) SK_OVERRIDE {
AAHairlineBatch* that = t->cast<AAHairlineBatch>();
// We go to identity if we don't have perspective
if (this->viewMatrix().hasPerspective() &&
!this->viewMatrix().cheapEqualTo(that->viewMatrix())) {
return false;
}
// TODO we can actually batch hairlines if they are the same color in a kind of bulk method
// but we haven't implemented this yet
// TODO investigate going to vertex color and coverage?
if (this->coverage() != that->coverage()) {
return false;
}
if (this->color() != that->color()) {
return false;
}
SkASSERT(this->usesLocalCoords() == that->usesLocalCoords());
if (this->usesLocalCoords() && !this->viewMatrix().cheapEqualTo(that->viewMatrix())) {
return false;
}
fGeoData.push_back_n(that->geoData()->count(), that->geoData()->begin());
return true;
}
GrColor color() const { return fBatch.fColor; }
uint8_t coverage() const { return fBatch.fCoverage; }
bool usesLocalCoords() const { return fBatch.fUsesLocalCoords; }
const SkMatrix& viewMatrix() const { return fGeoData[0].fViewMatrix; }
void generateLines(GrBatchTarget* batchTarget,
const GrPipeline* pipeline,
const Geometry& args,
const SkMatrix& viewMatrix,
const SkMatrix& invert,
const PtArray& lines,
int lineCnt) {
uint32_t gpFlags = GrDefaultGeoProcFactory::kPosition_GPType |
GrDefaultGeoProcFactory::kCoverage_GPType;
SkAutoTUnref<const GrGeometryProcessor> gp(GrDefaultGeoProcFactory::Create(gpFlags,
args.fColor,
viewMatrix,
invert,
false,
args.fCoverage));
batchTarget->initDraw(gp, pipeline);
// TODO remove this when batch is everywhere
GrPipelineInfo init;
init.fColorIgnored = fBatch.fColorIgnored;
init.fOverrideColor = GrColor_ILLEGAL;
init.fCoverageIgnored = fBatch.fCoverageIgnored;
init.fUsesLocalCoords = this->usesLocalCoords();
gp->initBatchTracker(batchTarget->currentBatchTracker(), init);
const GrVertexBuffer* vertexBuffer;
int firstVertex;
size_t vertexStride = gp->getVertexStride();
int vertexCount = kLineSegNumVertices * lineCnt;
void *vertices = batchTarget->vertexPool()->makeSpace(vertexStride,
vertexCount,
&vertexBuffer,
&firstVertex);
SkASSERT(gp->getVertexStride() == sizeof(LineVertex));
// generate lines
const SkMatrix* toSrc = NULL;
if (args.fViewMatrix.hasPerspective()) {
SkMatrix perspectiveInvert;
if (!args.fViewMatrix.invert(&perspectiveInvert)) {
return;
}
toSrc = &perspectiveInvert;
}
LineVertex* verts = reinterpret_cast<LineVertex*>(vertices);
for (int i = 0; i < lineCnt; ++i) {
add_line(&lines[2*i], toSrc, args.fCoverage, &verts);
}
// Check devBounds
SkASSERT(check_bounds<LineVertex>(viewMatrix.hasPerspective() ? viewMatrix : SkMatrix::I(),
args.fDevBounds,
vertices,
kLineSegNumVertices * lineCnt));
{
GrDrawTarget::DrawInfo info;
info.setVertexBuffer(vertexBuffer);
info.setIndexBuffer(fLinesIndexBuffer);
info.setPrimitiveType(kTriangles_GrPrimitiveType);
info.setStartIndex(0);
int lines = 0;
while (lines < lineCnt) {
int n = SkTMin(lineCnt - lines, kLineSegsNumInIdxBuffer);
info.setStartVertex(kLineSegNumVertices*lines + firstVertex);
info.setVertexCount(kLineSegNumVertices*n);
info.setIndexCount(kIdxsPerLineSeg*n);
batchTarget->draw(info);
lines += n;
}
}
}
void generateQuadsAndConics(GrBatchTarget* batchTarget,
const GrPipeline* pipeline,
const Geometry& args,
const PREALLOC_PTARRAY(128)& quads,
int quadCount,
const PREALLOC_PTARRAY(128)& conics,
int conicCount,
const IntArray& qSubdivs,
const FloatArray& cWeights,
const SkMatrix& vm,
const SkMatrix& invert) {
const GrVertexBuffer* vertexBuffer;
int firstVertex;
size_t vertexStride = sizeof(BezierVertex);
int vertexCount = kQuadNumVertices * quadCount + kQuadNumVertices * conicCount;
void *vertices = batchTarget->vertexPool()->makeSpace(vertexStride,
vertexCount,
&vertexBuffer,
&firstVertex);
if (!this->createBezierGeom(vertices,
args.fViewMatrix,
args.fPath,
quads,
quadCount,
conics,
conicCount,
qSubdivs,
cWeights,
vertexStride)) {
SkDebugf("Couldn't create bezier geometry\n");
return;
}
// Check devBounds
SkASSERT(check_bounds<BezierVertex>(vm,
args.fDevBounds,
vertices,
kQuadNumVertices * quadCount +
kQuadNumVertices * conicCount));
if (quadCount > 0) {
SkAutoTUnref<GrGeometryProcessor> hairQuadProcessor(
GrQuadEffect::Create(args.fColor,
vm,
kHairlineAA_GrProcessorEdgeType,
batchTarget->caps(),
invert,
args.fCoverage));
batchTarget->initDraw(hairQuadProcessor, pipeline);
// TODO remove this when batch is everywhere
GrPipelineInfo init;
init.fColorIgnored = fBatch.fColorIgnored;
init.fOverrideColor = GrColor_ILLEGAL;
init.fCoverageIgnored = fBatch.fCoverageIgnored;
init.fUsesLocalCoords = this->usesLocalCoords();
hairQuadProcessor->initBatchTracker(batchTarget->currentBatchTracker(), init);
this->drawBeziers(batchTarget,
hairQuadProcessor,
pipeline,
vertexBuffer,
firstVertex,
quadCount);
}
if (conicCount > 0) {
SkAutoTUnref<GrGeometryProcessor> hairConicProcessor(
GrConicEffect::Create(args.fColor,
vm,
kHairlineAA_GrProcessorEdgeType,
batchTarget->caps(),
invert,
args.fCoverage));
batchTarget->initDraw(hairConicProcessor, pipeline);
// TODO remove this when batch is everywhere
GrPipelineInfo init;
init.fColorIgnored = fBatch.fColorIgnored;
init.fOverrideColor = GrColor_ILLEGAL;
init.fCoverageIgnored = fBatch.fCoverageIgnored;
init.fUsesLocalCoords = this->usesLocalCoords();
hairConicProcessor->initBatchTracker(batchTarget->currentBatchTracker(), init);
this->drawConics(batchTarget,
hairConicProcessor,
pipeline,
vertexBuffer,
firstVertex,
conicCount,
quadCount);
}
}
bool createBezierGeom(void* vertices,
const SkMatrix& viewMatrix,
const SkPath& path,
const PtArray& quads,
int quadCnt,
const PtArray& conics,
int conicCnt,
const IntArray& qSubdivs,
const FloatArray& cWeights,
size_t vertexStride) {
BezierVertex* verts = reinterpret_cast<BezierVertex*>(vertices);
const SkMatrix* toDevice = NULL;
const SkMatrix* toSrc = NULL;
SkMatrix ivm;
if (viewMatrix.hasPerspective()) {
if (viewMatrix.invert(&ivm)) {
toDevice = &viewMatrix;
toSrc = &ivm;
}
}
int unsubdivQuadCnt = quads.count() / 3;
for (int i = 0; i < unsubdivQuadCnt; ++i) {
SkASSERT(qSubdivs[i] >= 0);
add_quads(&quads[3*i], qSubdivs[i], toDevice, toSrc, &verts);
}
// Start Conics
for (int i = 0; i < conicCnt; ++i) {
add_conics(&conics[3*i], cWeights[i], toDevice, toSrc, &verts);
}
return true;
}
void drawBeziers(GrBatchTarget* batchTarget,
const GrGeometryProcessor* hairQuadProcessor,
const GrPipeline* pipeline,
const GrVertexBuffer* vertexBuffer,
int firstVertex,
int quadCount) {
GrDrawTarget::DrawInfo info;
info.setVertexBuffer(vertexBuffer);
info.setIndexBuffer(fQuadsIndexBuffer);
info.setPrimitiveType(kTriangles_GrPrimitiveType);
info.setStartIndex(0);
int quads = 0;
while (quads < quadCount) {
int n = SkTMin(quadCount - quads, kQuadsNumInIdxBuffer);
info.setStartVertex(kQuadNumVertices*quads + firstVertex);
info.setVertexCount(kQuadNumVertices*n);
info.setIndexCount(kIdxsPerQuad*n);
batchTarget->draw(info);
quads += n;
}
}
void drawConics(GrBatchTarget* batchTarget,
const GrGeometryProcessor* hairConicProcessor,
const GrPipeline* pipeline,
const GrVertexBuffer* vertexBuffer,
int firstVertex,
int conicCount,
int quadCount) {
GrDrawTarget::DrawInfo info;
info.setVertexBuffer(vertexBuffer);
info.setIndexBuffer(fQuadsIndexBuffer);
info.setPrimitiveType(kTriangles_GrPrimitiveType);
info.setStartIndex(0);
int conics = 0;
while (conics < conicCount) {
int n = SkTMin(conicCount - conics, kQuadsNumInIdxBuffer);
info.setStartVertex(kQuadNumVertices*(quadCount + conics) + firstVertex);
info.setVertexCount(kQuadNumVertices*n);
info.setIndexCount(kIdxsPerQuad*n);
batchTarget->draw(info);
conics += n;
}
}
struct BatchTracker {
GrColor fColor;
uint8_t fCoverage;
SkRect fDevBounds;
bool fUsesLocalCoords;
bool fColorIgnored;
bool fCoverageIgnored;
};
GrBatchOpt fBatchOpt;
BatchTracker fBatch;
SkSTArray<1, Geometry, true> fGeoData;
const GrIndexBuffer* fLinesIndexBuffer;
const GrIndexBuffer* fQuadsIndexBuffer;
};
bool GrAAHairLinePathRenderer::onDrawPath(GrDrawTarget* target, bool GrAAHairLinePathRenderer::onDrawPath(GrDrawTarget* target,
GrPipelineBuilder* pipelineBuilder, GrPipelineBuilder* pipelineBuilder,
GrColor color, GrColor color,
const SkMatrix& viewMatrix, const SkMatrix& viewMatrix,
const SkPath& path, const SkPath& path,
const SkStrokeRec& stroke, const SkStrokeRec& stroke,
bool) { bool antiAlias) {
SkScalar hairlineCoverage; SkScalar hairlineCoverage;
uint8_t newCoverage = 0xff; uint8_t newCoverage = 0xff;
if (IsStrokeHairlineOrEquivalent(stroke, viewMatrix, &hairlineCoverage)) { if (IsStrokeHairlineOrEquivalent(stroke, viewMatrix, &hairlineCoverage)) {
@ -1146,22 +816,163 @@ bool GrAAHairLinePathRenderer::onDrawPath(GrDrawTarget* target,
SkIRect devClipBounds; SkIRect devClipBounds;
target->getClip()->getConservativeBounds(pipelineBuilder->getRenderTarget(), &devClipBounds); target->getClip()->getConservativeBounds(pipelineBuilder->getRenderTarget(), &devClipBounds);
// This outset was determined experimentally by running skps and gms. It probably could be a int lineCnt;
// bit tighter int quadCnt;
SkRect devRect = path.getBounds(); int conicCnt;
devRect.outset(7, 7); PREALLOC_PTARRAY(128) lines;
viewMatrix.mapRect(&devRect); PREALLOC_PTARRAY(128) quads;
PREALLOC_PTARRAY(128) conics;
IntArray qSubdivs;
FloatArray cWeights;
quadCnt = generate_lines_and_quads(path, viewMatrix, devClipBounds,
&lines, &quads, &conics, &qSubdivs, &cWeights);
lineCnt = lines.count() / 2;
conicCnt = conics.count() / 3;
AAHairlineBatch::Geometry geometry; // createGeom transforms the geometry to device space when the matrix does not have
geometry.fColor = color; // perspective.
geometry.fCoverage = newCoverage; SkMatrix vm = viewMatrix;
geometry.fViewMatrix = viewMatrix; SkMatrix invert = SkMatrix::I();
geometry.fPath = path; if (!viewMatrix.hasPerspective()) {
SkDEBUGCODE(geometry.fDevBounds = devRect;) vm = SkMatrix::I();
geometry.fDevClipBounds = devClipBounds; if (!viewMatrix.invert(&invert)) {
return false;
}
}
GrBatch* batch = AAHairlineBatch::Create(geometry, fLinesIndexBuffer, fQuadsIndexBuffer); // do lines first
target->drawBatch(pipelineBuilder, batch, &devRect); if (lineCnt) {
GrDrawTarget::AutoReleaseGeometry arg;
SkRect devBounds;
GrPipelineBuilder::AutoRestoreEffects are(pipelineBuilder);
uint32_t gpFlags = GrDefaultGeoProcFactory::kPosition_GPType |
GrDefaultGeoProcFactory::kCoverage_GPType;
SkAutoTUnref<const GrGeometryProcessor> gp(GrDefaultGeoProcFactory::Create(gpFlags,
color,
vm,
invert,
false,
newCoverage));
if (!this->createLineGeom(target,
pipelineBuilder,
viewMatrix,
newCoverage,
gp->getVertexStride(),
&arg,
&devBounds,
path,
lines,
lineCnt)) {
return false;
}
// Check devBounds
SkASSERT(check_bounds<LineVertex>(viewMatrix.hasPerspective() ? viewMatrix : SkMatrix::I(),
devBounds,
arg.vertices(),
kLineSegNumVertices * lineCnt));
{
target->setIndexSourceToBuffer(fLinesIndexBuffer);
int lines = 0;
while (lines < lineCnt) {
int n = SkTMin(lineCnt - lines, kLineSegsNumInIdxBuffer);
target->drawIndexed(pipelineBuilder,
gp,
kTriangles_GrPrimitiveType,
kLineSegNumVertices*lines, // startV
0, // startI
kLineSegNumVertices*n, // vCount
kIdxsPerLineSeg*n, // iCount
&devBounds);
lines += n;
}
}
}
// then quadratics/conics
if (quadCnt || conicCnt) {
GrDrawTarget::AutoReleaseGeometry arg;
SkRect devBounds;
if (!this->createBezierGeom(target,
pipelineBuilder,
viewMatrix,
&arg,
&devBounds,
path,
quads,
quadCnt,
conics,
conicCnt,
qSubdivs,
cWeights,
sizeof(BezierVertex))) {
return false;
}
// Check devBounds
SkASSERT(check_bounds<BezierVertex>(viewMatrix.hasPerspective() ? viewMatrix :
SkMatrix::I(),
devBounds,
arg.vertices(),
kQuadNumVertices * quadCnt +
kQuadNumVertices * conicCnt));
if (quadCnt > 0) {
SkAutoTUnref<GrGeometryProcessor> hairQuadProcessor(
GrQuadEffect::Create(color,
vm,
kHairlineAA_GrProcessorEdgeType,
*target->caps(),
invert,
newCoverage));
SkASSERT(hairQuadProcessor);
GrPipelineBuilder::AutoRestoreEffects are(pipelineBuilder);
target->setIndexSourceToBuffer(fQuadsIndexBuffer);
int quads = 0;
while (quads < quadCnt) {
int n = SkTMin(quadCnt - quads, kQuadsNumInIdxBuffer);
target->drawIndexed(pipelineBuilder,
hairQuadProcessor,
kTriangles_GrPrimitiveType,
kQuadNumVertices*quads, // startV
0, // startI
kQuadNumVertices*n, // vCount
kIdxsPerQuad*n, // iCount
&devBounds);
quads += n;
}
}
if (conicCnt > 0) {
SkAutoTUnref<GrGeometryProcessor> hairConicProcessor(
GrConicEffect::Create(color, vm, kHairlineAA_GrProcessorEdgeType,
*target->caps(), invert, newCoverage));
SkASSERT(hairConicProcessor);
GrPipelineBuilder::AutoRestoreEffects are(pipelineBuilder);
target->setIndexSourceToBuffer(fQuadsIndexBuffer);
int conics = 0;
while (conics < conicCnt) {
int n = SkTMin(conicCnt - conics, kQuadsNumInIdxBuffer);
target->drawIndexed(pipelineBuilder,
hairConicProcessor,
kTriangles_GrPrimitiveType,
kQuadNumVertices*(quadCnt + conics), // startV
0, // startI
kQuadNumVertices*n, // vCount
kIdxsPerQuad*n, // iCount
&devBounds);
conics += n;
}
}
}
target->resetIndexSource();
return true; return true;
} }

View File

@ -42,6 +42,31 @@ private:
const GrIndexBuffer* fLinesIndexBuffer, const GrIndexBuffer* fLinesIndexBuffer,
const GrIndexBuffer* fQuadsIndexBuffer); const GrIndexBuffer* fQuadsIndexBuffer);
bool createLineGeom(GrDrawTarget* target,
GrPipelineBuilder*,
const SkMatrix& viewMatrix,
uint8_t coverage,
size_t vertexStride,
GrDrawTarget::AutoReleaseGeometry* arg,
SkRect* devBounds,
const SkPath& path,
const PtArray& lines,
int lineCnt);
bool createBezierGeom(GrDrawTarget* target,
GrPipelineBuilder*,
const SkMatrix& viewMatrix,
GrDrawTarget::AutoReleaseGeometry* arg,
SkRect* devBounds,
const SkPath& path,
const PtArray& quads,
int quadCnt,
const PtArray& conics,
int conicCnt,
const IntArray& qSubdivs,
const FloatArray& cWeights,
size_t vertexStride);
const GrIndexBuffer* fLinesIndexBuffer; const GrIndexBuffer* fLinesIndexBuffer;
const GrIndexBuffer* fQuadsIndexBuffer; const GrIndexBuffer* fQuadsIndexBuffer;

View File

@ -208,7 +208,10 @@ private:
return false; return false;
} }
SkASSERT(this->usesLocalCoords() == that->usesLocalCoords()); if (this->usesLocalCoords() != that->usesLocalCoords()) {
return false;
}
// We apply the viewmatrix to the rect points on the cpu. However, if the pipeline uses // We apply the viewmatrix to the rect points on the cpu. However, if the pipeline uses
// local coords then we won't be able to batch. We could actually upload the viewmatrix // local coords then we won't be able to batch. We could actually upload the viewmatrix
// using vertex attributes in these cases, but haven't investigated that // using vertex attributes in these cases, but haven't investigated that

View File

@ -46,7 +46,7 @@ struct GrBatchOpt {
class GrBatch : public SkRefCnt { class GrBatch : public SkRefCnt {
public: public:
SK_DECLARE_INST_COUNT(GrBatch) SK_DECLARE_INST_COUNT(GrBatch)
GrBatch() : fNumberOfDraws(0) { SkDEBUGCODE(fUsed = false;) } GrBatch() { SkDEBUGCODE(fUsed = false;) }
virtual ~GrBatch() {} virtual ~GrBatch() {}
virtual const char* name() const = 0; virtual const char* name() const = 0;
@ -75,10 +75,6 @@ public:
virtual void generateGeometry(GrBatchTarget*, const GrPipeline*) = 0; virtual void generateGeometry(GrBatchTarget*, const GrPipeline*) = 0;
// TODO this goes away when batches are everywhere
void setNumberOfDraws(int numberOfDraws) { fNumberOfDraws = numberOfDraws; }
int numberOfDraws() const { return fNumberOfDraws; }
void* operator new(size_t size); void* operator new(size_t size);
void operator delete(void* target); void operator delete(void* target);
@ -130,8 +126,6 @@ private:
SkDEBUGCODE(bool fUsed;) SkDEBUGCODE(bool fUsed;)
int fNumberOfDraws;
typedef SkRefCnt INHERITED; typedef SkRefCnt INHERITED;
}; };

View File

@ -32,20 +32,17 @@ void GrBatchTarget::flush() {
fFlushBuffer.reset(); fFlushBuffer.reset();
}*/ }*/
void GrBatchTarget::flushNext(int n) { void GrBatchTarget::flushNext() {
for (; n > 0; n--) { fIter.next();
SkDEBUGCODE(bool verify =) fIter.next(); GrProgramDesc desc;
SkASSERT(verify); BufferedFlush* bf = fIter.get();
GrProgramDesc desc; const GrPipeline* pipeline = bf->fPipeline;
BufferedFlush* bf = fIter.get(); const GrPrimitiveProcessor* primProc = bf->fPrimitiveProcessor.get();
const GrPipeline* pipeline = bf->fPipeline; fGpu->buildProgramDesc(&desc, *primProc, *pipeline, pipeline->descInfo(),
const GrPrimitiveProcessor* primProc = bf->fPrimitiveProcessor.get(); bf->fBatchTracker);
fGpu->buildProgramDesc(&desc, *primProc, *pipeline, pipeline->descInfo(),
bf->fBatchTracker);
GrGpu::DrawArgs args(primProc, pipeline, &desc, &bf->fBatchTracker); GrGpu::DrawArgs args(primProc, pipeline, &desc, &bf->fBatchTracker);
for (int i = 0; i < bf->fDraws.count(); i++) { for (int i = 0; i < bf->fDraws.count(); i++) {
fGpu->draw(args, bf->fDraws[i]); fGpu->draw(args, bf->fDraws[i]);
}
} }
} }

View File

@ -17,9 +17,6 @@
* that render their batch. * that render their batch.
*/ */
class GrIndexBufferAllocPool;
class GrVertexBufferAllocPool;
class GrBatchTarget : public SkNoncopyable { class GrBatchTarget : public SkNoncopyable {
public: public:
GrBatchTarget(GrGpu* gpu, GrBatchTarget(GrGpu* gpu,
@ -29,13 +26,11 @@ public:
, fVertexPool(vpool) , fVertexPool(vpool)
, fIndexPool(ipool) , fIndexPool(ipool)
, fFlushBuffer(kFlushBufferInitialSizeInBytes) , fFlushBuffer(kFlushBufferInitialSizeInBytes)
, fIter(fFlushBuffer) , fIter(fFlushBuffer) {}
, fNumberOfDraws(0) {}
typedef GrDrawTarget::DrawInfo DrawInfo; typedef GrDrawTarget::DrawInfo DrawInfo;
void initDraw(const GrPrimitiveProcessor* primProc, const GrPipeline* pipeline) { void initDraw(const GrPrimitiveProcessor* primProc, const GrPipeline* pipeline) {
GrNEW_APPEND_TO_RECORDER(fFlushBuffer, BufferedFlush, (primProc, pipeline)); GrNEW_APPEND_TO_RECORDER(fFlushBuffer, BufferedFlush, (primProc, pipeline));
fNumberOfDraws++;
} }
void draw(const GrDrawTarget::DrawInfo& draw) { void draw(const GrDrawTarget::DrawInfo& draw) {
@ -44,10 +39,8 @@ public:
// TODO this is temporary until batch is everywhere // TODO this is temporary until batch is everywhere
//void flush(); //void flush();
void resetNumberOfDraws() { fNumberOfDraws = 0; }
int numberOfDraws() const { return fNumberOfDraws; }
void preFlush() { fIter = FlushBuffer::Iter(fFlushBuffer); } void preFlush() { fIter = FlushBuffer::Iter(fFlushBuffer); }
void flushNext(int n); void flushNext();
void postFlush() { SkASSERT(!fIter.next()); fFlushBuffer.reset(); } void postFlush() { SkASSERT(!fIter.next()); fFlushBuffer.reset(); }
// TODO This goes away when everything uses batch // TODO This goes away when everything uses batch
@ -56,8 +49,6 @@ public:
return &fFlushBuffer.back().fBatchTracker; return &fFlushBuffer.back().fBatchTracker;
} }
const GrDrawTargetCaps& caps() const { return *fGpu->caps(); }
GrVertexBufferAllocPool* vertexPool() { return fVertexPool; } GrVertexBufferAllocPool* vertexPool() { return fVertexPool; }
GrIndexBufferAllocPool* indexPool() { return fIndexPool; } GrIndexBufferAllocPool* indexPool() { return fIndexPool; }
@ -71,7 +62,8 @@ private:
struct BufferedFlush { struct BufferedFlush {
BufferedFlush(const GrPrimitiveProcessor* primProc, const GrPipeline* pipeline) BufferedFlush(const GrPrimitiveProcessor* primProc, const GrPipeline* pipeline)
: fPrimitiveProcessor(primProc) : fPrimitiveProcessor(primProc)
, fPipeline(pipeline) {} , fPipeline(pipeline)
, fDraws(kDrawRecorderInitialSizeInBytes) {}
typedef GrPendingProgramElement<const GrPrimitiveProcessor> ProgramPrimitiveProcessor; typedef GrPendingProgramElement<const GrPrimitiveProcessor> ProgramPrimitiveProcessor;
ProgramPrimitiveProcessor fPrimitiveProcessor; ProgramPrimitiveProcessor fPrimitiveProcessor;
const GrPipeline* fPipeline; const GrPipeline* fPipeline;
@ -81,6 +73,7 @@ private:
enum { enum {
kFlushBufferInitialSizeInBytes = 8 * sizeof(BufferedFlush), kFlushBufferInitialSizeInBytes = 8 * sizeof(BufferedFlush),
kDrawRecorderInitialSizeInBytes = 8 * sizeof(DrawInfo),
}; };
typedef GrTRecorder<BufferedFlush, TBufferAlign> FlushBuffer; typedef GrTRecorder<BufferedFlush, TBufferAlign> FlushBuffer;
@ -88,7 +81,6 @@ private:
FlushBuffer fFlushBuffer; FlushBuffer fFlushBuffer;
// TODO this is temporary // TODO this is temporary
FlushBuffer::Iter fIter; FlushBuffer::Iter fIter;
int fNumberOfDraws;
}; };
#endif #endif

View File

@ -473,8 +473,7 @@ void GrInOrderDrawBuffer::onFlush() {
// TODO temporary hack // TODO temporary hack
if (kDrawBatch_Cmd == strip_trace_bit(iter->fType)) { if (kDrawBatch_Cmd == strip_trace_bit(iter->fType)) {
DrawBatch* db = reinterpret_cast<DrawBatch*>(iter.get()); fBatchTarget.flushNext();
fBatchTarget.flushNext(db->fBatch->numberOfDraws());
continue; continue;
} }
@ -647,9 +646,7 @@ void GrInOrderDrawBuffer::recordTraceMarkersIfNecessary() {
void GrInOrderDrawBuffer::closeBatch() { void GrInOrderDrawBuffer::closeBatch() {
if (fDrawBatch) { if (fDrawBatch) {
fBatchTarget.resetNumberOfDraws();
fDrawBatch->execute(this, fPrevState); fDrawBatch->execute(this, fPrevState);
fDrawBatch->fBatch->setNumberOfDraws(fBatchTarget.numberOfDraws());
fDrawBatch = NULL; fDrawBatch = NULL;
} }
} }