Make GrAAConvexPathRenderer use vertex color

Change-Id: Iea2902a73fad9493d69b24e8bc88cdafd6aea9fc
Reviewed-on: https://skia-review.googlesource.com/20060
Reviewed-by: Robert Phillips <robertphillips@google.com>
Commit-Queue: Brian Salomon <bsalomon@google.com>
This commit is contained in:
Brian Salomon 2017-06-15 17:09:36 -04:00 committed by Skia Commit-Bot
parent 7292231905
commit 60fb0b2548
2 changed files with 70 additions and 78 deletions

View File

@ -335,6 +335,7 @@ static bool get_segments(const SkPath& path,
struct QuadVertex { struct QuadVertex {
SkPoint fPos; SkPoint fPos;
GrColor fColor;
SkPoint fUV; SkPoint fUV;
SkScalar fD0; SkScalar fD0;
SkScalar fD1; SkScalar fD1;
@ -348,11 +349,12 @@ struct Draw {
typedef SkTArray<Draw, true> DrawArray; typedef SkTArray<Draw, true> DrawArray;
static void create_vertices(const SegmentArray& segments, static void create_vertices(const SegmentArray& segments,
const SkPoint& fanPt, const SkPoint& fanPt,
DrawArray* draws, GrColor color,
QuadVertex* verts, DrawArray* draws,
uint16_t* idxs) { QuadVertex* verts,
uint16_t* idxs) {
Draw* draw = &draws->push_back(); Draw* draw = &draws->push_back();
// alias just to make vert/index assignments easier to read. // alias just to make vert/index assignments easier to read.
int* v = &draw->fVertexCnt; int* v = &draw->fVertexCnt;
@ -385,6 +387,10 @@ static void create_vertices(const SegmentArray& segments,
verts[*v + 1].fPos = verts[*v + 0].fPos + sega.endNorm(); verts[*v + 1].fPos = verts[*v + 0].fPos + sega.endNorm();
verts[*v + 2].fPos = verts[*v + 0].fPos + segb.fMid; verts[*v + 2].fPos = verts[*v + 0].fPos + segb.fMid;
verts[*v + 3].fPos = verts[*v + 0].fPos + segb.fNorms[0]; verts[*v + 3].fPos = verts[*v + 0].fPos + segb.fNorms[0];
verts[*v + 0].fColor = color;
verts[*v + 1].fColor = color;
verts[*v + 2].fColor = color;
verts[*v + 3].fColor = color;
verts[*v + 0].fUV.set(0,0); verts[*v + 0].fUV.set(0,0);
verts[*v + 1].fUV.set(0,-SK_Scalar1); verts[*v + 1].fUV.set(0,-SK_Scalar1);
verts[*v + 2].fUV.set(0,-SK_Scalar1); verts[*v + 2].fUV.set(0,-SK_Scalar1);
@ -412,6 +418,12 @@ static void create_vertices(const SegmentArray& segments,
verts[*v + 3].fPos = verts[*v + 1].fPos + segb.fNorms[0]; verts[*v + 3].fPos = verts[*v + 1].fPos + segb.fNorms[0];
verts[*v + 4].fPos = verts[*v + 2].fPos + segb.fNorms[0]; verts[*v + 4].fPos = verts[*v + 2].fPos + segb.fNorms[0];
verts[*v + 0].fColor = color;
verts[*v + 1].fColor = color;
verts[*v + 2].fColor = color;
verts[*v + 3].fColor = color;
verts[*v + 4].fColor = color;
// we draw the line edge as a degenerate quad (u is 0, v is the // we draw the line edge as a degenerate quad (u is 0, v is the
// signed distance to the edge) // signed distance to the edge)
SkScalar dist = fanPt.distanceToLineBetween(verts[*v + 1].fPos, SkScalar dist = fanPt.distanceToLineBetween(verts[*v + 1].fPos,
@ -463,6 +475,13 @@ static void create_vertices(const SegmentArray& segments,
verts[*v + 4].fPos = qpts[2] + segb.fNorms[1]; verts[*v + 4].fPos = qpts[2] + segb.fNorms[1];
verts[*v + 5].fPos = qpts[1] + midVec; verts[*v + 5].fPos = qpts[1] + midVec;
verts[*v + 0].fColor = color;
verts[*v + 1].fColor = color;
verts[*v + 2].fColor = color;
verts[*v + 3].fColor = color;
verts[*v + 4].fColor = color;
verts[*v + 5].fColor = color;
SkScalar c = segb.fNorms[0].dot(qpts[0]); SkScalar c = segb.fNorms[0].dot(qpts[0]);
verts[*v + 0].fD0 = -segb.fNorms[0].dot(fanPt) + c; verts[*v + 0].fD0 = -segb.fNorms[0].dot(fanPt) + c;
verts[*v + 1].fD0 = 0.f; verts[*v + 1].fD0 = 0.f;
@ -480,7 +499,7 @@ static void create_vertices(const SegmentArray& segments,
verts[*v + 5].fD1 = -SK_ScalarMax/100; verts[*v + 5].fD1 = -SK_ScalarMax/100;
GrPathUtils::QuadUVMatrix toUV(qpts); GrPathUtils::QuadUVMatrix toUV(qpts);
toUV.apply<6, sizeof(QuadVertex), sizeof(SkPoint)>(verts + *v); toUV.apply<6, sizeof(QuadVertex), offsetof(QuadVertex, fUV)>(verts + *v);
idxs[*i + 0] = *v + 3; idxs[*i + 0] = *v + 3;
idxs[*i + 1] = *v + 1; idxs[*i + 1] = *v + 1;
@ -525,25 +544,17 @@ static void create_vertices(const SegmentArray& segments,
class QuadEdgeEffect : public GrGeometryProcessor { class QuadEdgeEffect : public GrGeometryProcessor {
public: public:
static sk_sp<GrGeometryProcessor> Make(const SkMatrix& localMatrix, bool usesLocalCoords) {
static sk_sp<GrGeometryProcessor> Make(GrColor color, const SkMatrix& localMatrix, return sk_sp<GrGeometryProcessor>(new QuadEdgeEffect(localMatrix, usesLocalCoords));
bool usesLocalCoords) {
return sk_sp<GrGeometryProcessor>(new QuadEdgeEffect(color, localMatrix, usesLocalCoords));
} }
~QuadEdgeEffect() override {} ~QuadEdgeEffect() override {}
const char* name() const override { return "QuadEdge"; } const char* name() const override { return "QuadEdge"; }
const Attribute* inPosition() const { return fInPosition; }
const Attribute* inQuadEdge() const { return fInQuadEdge; }
GrColor color() const { return fColor; }
const SkMatrix& localMatrix() const { return fLocalMatrix; }
bool usesLocalCoords() const { return fUsesLocalCoords; }
class GLSLProcessor : public GrGLSLGeometryProcessor { class GLSLProcessor : public GrGLSLGeometryProcessor {
public: public:
GLSLProcessor() : fColor(GrColor_ILLEGAL) {} GLSLProcessor() {}
void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override { void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override {
const QuadEdgeEffect& qe = args.fGP.cast<QuadEdgeEffect>(); const QuadEdgeEffect& qe = args.fGP.cast<QuadEdgeEffect>();
@ -556,23 +567,23 @@ public:
GrGLSLVertToFrag v(kVec4f_GrSLType); GrGLSLVertToFrag v(kVec4f_GrSLType);
varyingHandler->addVarying("QuadEdge", &v); varyingHandler->addVarying("QuadEdge", &v);
vertBuilder->codeAppendf("%s = %s;", v.vsOut(), qe.inQuadEdge()->fName); vertBuilder->codeAppendf("%s = %s;", v.vsOut(), qe.fInQuadEdge->fName);
// Setup pass through color
varyingHandler->addPassThroughAttribute(qe.fInColor, args.fOutputColor);
GrGLSLPPFragmentBuilder* fragBuilder = args.fFragBuilder; GrGLSLPPFragmentBuilder* fragBuilder = args.fFragBuilder;
// Setup pass through color
this->setupUniformColor(fragBuilder, uniformHandler, args.fOutputColor,
&fColorUniform);
// Setup position // Setup position
this->setupPosition(vertBuilder, gpArgs, qe.inPosition()->fName); this->setupPosition(vertBuilder, gpArgs, qe.fInPosition->fName);
// emit transforms // emit transforms
this->emitTransforms(vertBuilder, this->emitTransforms(vertBuilder,
varyingHandler, varyingHandler,
uniformHandler, uniformHandler,
gpArgs->fPositionVar, gpArgs->fPositionVar,
qe.inPosition()->fName, qe.fInPosition->fName,
qe.localMatrix(), qe.fLocalMatrix,
args.fFPCoordTransformHandler); args.fFPCoordTransformHandler);
fragBuilder->codeAppendf("float edgeAlpha;"); fragBuilder->codeAppendf("float edgeAlpha;");
@ -600,26 +611,17 @@ public:
const GrShaderCaps&, const GrShaderCaps&,
GrProcessorKeyBuilder* b) { GrProcessorKeyBuilder* b) {
const QuadEdgeEffect& qee = gp.cast<QuadEdgeEffect>(); const QuadEdgeEffect& qee = gp.cast<QuadEdgeEffect>();
b->add32(SkToBool(qee.usesLocalCoords() && qee.localMatrix().hasPerspective())); b->add32(SkToBool(qee.fUsesLocalCoords && qee.fLocalMatrix.hasPerspective()));
} }
void setData(const GrGLSLProgramDataManager& pdman, void setData(const GrGLSLProgramDataManager& pdman,
const GrPrimitiveProcessor& gp, const GrPrimitiveProcessor& gp,
FPCoordTransformIter&& transformIter) override { FPCoordTransformIter&& transformIter) override {
const QuadEdgeEffect& qe = gp.cast<QuadEdgeEffect>(); const QuadEdgeEffect& qe = gp.cast<QuadEdgeEffect>();
if (qe.color() != fColor) {
float c[4];
GrColorToRGBAFloat(qe.color(), c);
pdman.set4fv(fColorUniform, 1, c);
fColor = qe.color();
}
this->setTransformDataHelper(qe.fLocalMatrix, pdman, &transformIter); this->setTransformDataHelper(qe.fLocalMatrix, pdman, &transformIter);
} }
private: private:
GrColor fColor;
UniformHandle fColorUniform;
typedef GrGLSLGeometryProcessor INHERITED; typedef GrGLSLGeometryProcessor INHERITED;
}; };
@ -632,18 +634,17 @@ public:
} }
private: private:
QuadEdgeEffect(GrColor color, const SkMatrix& localMatrix, bool usesLocalCoords) QuadEdgeEffect(const SkMatrix& localMatrix, bool usesLocalCoords)
: fColor(color) : fLocalMatrix(localMatrix), fUsesLocalCoords(usesLocalCoords) {
, fLocalMatrix(localMatrix)
, fUsesLocalCoords(usesLocalCoords) {
this->initClassID<QuadEdgeEffect>(); this->initClassID<QuadEdgeEffect>();
fInPosition = &this->addVertexAttrib("inPosition", kVec2f_GrVertexAttribType); fInPosition = &this->addVertexAttrib("inPosition", kVec2f_GrVertexAttribType);
fInColor = &this->addVertexAttrib("inColor", kVec4ub_GrVertexAttribType);
fInQuadEdge = &this->addVertexAttrib("inQuadEdge", kVec4f_GrVertexAttribType); fInQuadEdge = &this->addVertexAttrib("inQuadEdge", kVec4f_GrVertexAttribType);
} }
const Attribute* fInPosition; const Attribute* fInPosition;
const Attribute* fInQuadEdge; const Attribute* fInQuadEdge;
GrColor fColor; const Attribute* fInColor;
SkMatrix fLocalMatrix; SkMatrix fLocalMatrix;
bool fUsesLocalCoords; bool fUsesLocalCoords;
@ -658,9 +659,7 @@ GR_DEFINE_GEOMETRY_PROCESSOR_TEST(QuadEdgeEffect);
sk_sp<GrGeometryProcessor> QuadEdgeEffect::TestCreate(GrProcessorTestData* d) { sk_sp<GrGeometryProcessor> QuadEdgeEffect::TestCreate(GrProcessorTestData* d) {
// Doesn't work without derivative instructions. // Doesn't work without derivative instructions.
return d->caps()->shaderCaps()->shaderDerivativeSupport() return d->caps()->shaderCaps()->shaderDerivativeSupport()
? QuadEdgeEffect::Make(GrRandomColor(d->fRandom), ? QuadEdgeEffect::Make(GrTest::TestMatrix(d->fRandom), d->fRandom->nextBool())
GrTest::TestMatrix(d->fRandom),
d->fRandom->nextBool())
: nullptr; : nullptr;
} }
#endif #endif
@ -674,12 +673,12 @@ bool GrAAConvexPathRenderer::onCanDrawPath(const CanDrawPathArgs& args) const {
} }
// extract the result vertices and indices from the GrAAConvexTessellator // extract the result vertices and indices from the GrAAConvexTessellator
static void extract_verts(const GrAAConvexTessellator& tess, static void extract_lines_only_verts(const GrAAConvexTessellator& tess,
void* vertices, void* vertices,
size_t vertexStride, size_t vertexStride,
GrColor color, GrColor color,
uint16_t* idxs, uint16_t* idxs,
bool tweakAlphaForCoverage) { bool tweakAlphaForCoverage) {
intptr_t verts = reinterpret_cast<intptr_t>(vertices); intptr_t verts = reinterpret_cast<intptr_t>(vertices);
for (int i = 0; i < tess.numPts(); ++i) { for (int i = 0; i < tess.numPts(); ++i) {
@ -706,9 +705,9 @@ static void extract_verts(const GrAAConvexTessellator& tess,
} }
} }
static sk_sp<GrGeometryProcessor> create_fill_gp(bool tweakAlphaForCoverage, static sk_sp<GrGeometryProcessor> make_lines_only_gp(bool tweakAlphaForCoverage,
const SkMatrix& viewMatrix, const SkMatrix& viewMatrix,
bool usesLocalCoords) { bool usesLocalCoords) {
using namespace GrDefaultGeoProcFactory; using namespace GrDefaultGeoProcFactory;
Coverage::Type coverageType; Coverage::Type coverageType;
@ -740,10 +739,8 @@ public:
AAConvexPathOp(const Helper::MakeArgs& helperArgs, GrColor color, const SkMatrix& viewMatrix, AAConvexPathOp(const Helper::MakeArgs& helperArgs, GrColor color, const SkMatrix& viewMatrix,
const SkPath& path, const GrUserStencilSettings* stencilSettings) const SkPath& path, const GrUserStencilSettings* stencilSettings)
: INHERITED(ClassID()) : INHERITED(ClassID()), fHelper(helperArgs, GrAAType::kCoverage, stencilSettings) {
, fHelper(helperArgs, GrAAType::kCoverage, stencilSettings) fPaths.emplace_back(PathData{viewMatrix, path, color});
, fColor(color) {
fPaths.emplace_back(PathData{viewMatrix, path});
this->setTransformedBounds(path.getBounds(), viewMatrix, HasAABloat::kYes, IsZeroArea::kNo); this->setTransformedBounds(path.getBounds(), viewMatrix, HasAABloat::kYes, IsZeroArea::kNo);
fLinesOnly = SkPath::kLine_SegmentMask == path.getSegmentMasks(); fLinesOnly = SkPath::kLine_SegmentMask == path.getSegmentMasks();
} }
@ -752,7 +749,7 @@ public:
SkString dumpInfo() const override { SkString dumpInfo() const override {
SkString string; SkString string;
string.appendf("Color: 0x%08x, Count: %d\n", fColor, fPaths.count()); string.appendf("Count: %d\n", fPaths.count());
string += fHelper.dumpInfo(); string += fHelper.dumpInfo();
string += INHERITED::dumpInfo(); string += INHERITED::dumpInfo();
return string; return string;
@ -762,16 +759,15 @@ public:
bool xpRequiresDstTexture(const GrCaps& caps, const GrAppliedClip* clip) override { bool xpRequiresDstTexture(const GrCaps& caps, const GrAppliedClip* clip) override {
return fHelper.xpRequiresDstTexture(caps, clip, GrProcessorAnalysisCoverage::kSingleChannel, return fHelper.xpRequiresDstTexture(caps, clip, GrProcessorAnalysisCoverage::kSingleChannel,
&fColor); &fPaths.back().fColor);
} }
private: private:
void prepareLinesOnlyDraws(Target* target) const { void prepareLinesOnlyDraws(Target* target) const {
bool canTweakAlphaForCoverage = fHelper.compatibleWithAlphaAsCoverage(); // Setup GrGeometryProcessor
sk_sp<GrGeometryProcessor> gp(make_lines_only_gp(fHelper.compatibleWithAlphaAsCoverage(),
// Setup GrGeometryProcessor. fPaths.back().fViewMatrix,
sk_sp<GrGeometryProcessor> gp(create_fill_gp( fHelper.usesLocalCoords()));
canTweakAlphaForCoverage, fPaths.back().fViewMatrix, fHelper.usesLocalCoords()));
if (!gp) { if (!gp) {
SkDebugf("Could not create GrGeometryProcessor\n"); SkDebugf("Could not create GrGeometryProcessor\n");
return; return;
@ -779,9 +775,10 @@ private:
size_t vertexStride = gp->getVertexStride(); size_t vertexStride = gp->getVertexStride();
SkASSERT(canTweakAlphaForCoverage ? SkASSERT(fHelper.compatibleWithAlphaAsCoverage()
vertexStride == sizeof(GrDefaultGeoProcFactory::PositionColorAttr) : ? vertexStride == sizeof(GrDefaultGeoProcFactory::PositionColorAttr)
vertexStride == sizeof(GrDefaultGeoProcFactory::PositionColorCoverageAttr)); : vertexStride ==
sizeof(GrDefaultGeoProcFactory::PositionColorCoverageAttr));
GrAAConvexTessellator tess; GrAAConvexTessellator tess;
@ -815,7 +812,8 @@ private:
return; return;
} }
extract_verts(tess, verts, vertexStride, fColor, idxs, canTweakAlphaForCoverage); extract_lines_only_verts(tess, verts, vertexStride, args.fColor, idxs,
fHelper.compatibleWithAlphaAsCoverage());
GrMesh mesh(GrPrimitiveType::kTriangles); GrMesh mesh(GrPrimitiveType::kTriangles);
mesh.setIndexed(indexBuffer, tess.numIndices(), firstIndex, 0, tess.numPts() - 1); mesh.setIndexed(indexBuffer, tess.numIndices(), firstIndex, 0, tess.numPts() - 1);
@ -842,7 +840,7 @@ private:
// Setup GrGeometryProcessor // Setup GrGeometryProcessor
sk_sp<GrGeometryProcessor> quadProcessor( sk_sp<GrGeometryProcessor> quadProcessor(
QuadEdgeEffect::Make(fColor, invert, fHelper.usesLocalCoords())); QuadEdgeEffect::Make(invert, fHelper.usesLocalCoords()));
// TODO generate all segments for all paths and use one vertex buffer // TODO generate all segments for all paths and use one vertex buffer
for (int i = 0; i < instanceCount; i++) { for (int i = 0; i < instanceCount; i++) {
@ -900,7 +898,7 @@ private:
} }
SkSTArray<kPreallocDrawCnt, Draw, true> draws; SkSTArray<kPreallocDrawCnt, Draw, true> draws;
create_vertices(segments, fanPt, &draws, verts, idxs); create_vertices(segments, fanPt, args.fColor, &draws, verts, idxs);
GrMesh mesh(GrPrimitiveType::kTriangles); GrMesh mesh(GrPrimitiveType::kTriangles);
@ -920,11 +918,6 @@ private:
if (!fHelper.isCompatible(that->fHelper, caps, this->bounds(), that->bounds())) { if (!fHelper.isCompatible(that->fHelper, caps, this->bounds(), that->bounds())) {
return false; return false;
} }
if (fColor != that->fColor) {
return false;
}
if (fHelper.usesLocalCoords() && if (fHelper.usesLocalCoords() &&
!fPaths[0].fViewMatrix.cheapEqualTo(that->fPaths[0].fViewMatrix)) { !fPaths[0].fViewMatrix.cheapEqualTo(that->fPaths[0].fViewMatrix)) {
return false; return false;
@ -934,7 +927,6 @@ private:
return false; return false;
} }
fPaths.push_back_n(that->fPaths.count(), that->fPaths.begin()); fPaths.push_back_n(that->fPaths.count(), that->fPaths.begin());
this->joinBounds(*that); this->joinBounds(*that);
return true; return true;
@ -943,11 +935,11 @@ private:
struct PathData { struct PathData {
SkMatrix fViewMatrix; SkMatrix fViewMatrix;
SkPath fPath; SkPath fPath;
GrColor fColor;
}; };
Helper fHelper; Helper fHelper;
SkSTArray<1, PathData, true> fPaths; SkSTArray<1, PathData, true> fPaths;
GrColor fColor;
bool fLinesOnly; bool fLinesOnly;
typedef GrMeshDrawOp INHERITED; typedef GrMeshDrawOp INHERITED;

View File

@ -100,9 +100,9 @@ static void extract_verts(const GrAAConvexTessellator& tess,
} }
} }
static sk_sp<GrGeometryProcessor> create_fill_gp(bool tweakAlphaForCoverage, static sk_sp<GrGeometryProcessor> create_lines_only_gp(bool tweakAlphaForCoverage,
const SkMatrix& viewMatrix, const SkMatrix& viewMatrix,
bool usesLocalCoords) { bool usesLocalCoords) {
using namespace GrDefaultGeoProcFactory; using namespace GrDefaultGeoProcFactory;
Coverage::Type coverageType; Coverage::Type coverageType;
@ -217,7 +217,7 @@ private:
bool canTweakAlphaForCoverage = this->canTweakAlphaForCoverage(); bool canTweakAlphaForCoverage = this->canTweakAlphaForCoverage();
// Setup GrGeometryProcessor // Setup GrGeometryProcessor
sk_sp<GrGeometryProcessor> gp(create_fill_gp( sk_sp<GrGeometryProcessor> gp(create_lines_only_gp(
canTweakAlphaForCoverage, this->viewMatrix(), this->usesLocalCoords())); canTweakAlphaForCoverage, this->viewMatrix(), this->usesLocalCoords()));
if (!gp) { if (!gp) {
SkDebugf("Couldn't create a GrGeometryProcessor\n"); SkDebugf("Couldn't create a GrGeometryProcessor\n");