blob: d6b5578d8a21d5013ed2adcd5a32894ee13b3302 [file] [log] [blame]
ethannicholas6536ae52016-05-02 12:16:49 -07001/*
2 * Copyright 2016 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#include "GrMSAAPathRenderer.h"
9
robertphillips976f5f02016-06-03 10:59:20 -070010#include "GrAuditTrail.h"
ethannicholas6536ae52016-05-02 12:16:49 -070011#include "GrBatchFlushState.h"
robertphillips976f5f02016-06-03 10:59:20 -070012#include "GrClip.h"
ethannicholas6536ae52016-05-02 12:16:49 -070013#include "GrDefaultGeoProcFactory.h"
csmartdalton02fa32c2016-08-19 13:29:27 -070014#include "GrFixedClip.h"
ethannicholas6536ae52016-05-02 12:16:49 -070015#include "GrPathStencilSettings.h"
16#include "GrPathUtils.h"
bsalomonbb243832016-07-22 07:10:19 -070017#include "GrPipelineBuilder.h"
ethannicholas6536ae52016-05-02 12:16:49 -070018#include "GrMesh.h"
19#include "SkGeometry.h"
20#include "SkTraceEvent.h"
21#include "glsl/GrGLSLGeometryProcessor.h"
22#include "glsl/GrGLSLFragmentShaderBuilder.h"
23#include "glsl/GrGLSLVertexShaderBuilder.h"
24#include "glsl/GrGLSLProgramDataManager.h"
25#include "glsl/GrGLSLUtil.h"
26#include "gl/GrGLVaryingHandler.h"
27#include "batches/GrRectBatchFactory.h"
28#include "batches/GrVertexBatch.h"
29
30static const float kTolerance = 0.5f;
31
32////////////////////////////////////////////////////////////////////////////////
33// Helpers for drawPath
34
bsalomon8acedde2016-06-24 10:42:16 -070035static inline bool single_pass_shape(const GrShape& shape) {
36 if (!shape.inverseFilled()) {
37 return shape.knownToBeConvex();
ethannicholas6536ae52016-05-02 12:16:49 -070038 }
39 return false;
40}
41
bsalomon8acedde2016-06-24 10:42:16 -070042GrPathRenderer::StencilSupport GrMSAAPathRenderer::onGetStencilSupport(const GrShape& shape) const {
43 if (single_pass_shape(shape)) {
ethannicholas6536ae52016-05-02 12:16:49 -070044 return GrPathRenderer::kNoRestriction_StencilSupport;
45 } else {
46 return GrPathRenderer::kStencilOnly_StencilSupport;
47 }
48}
49
50struct MSAALineVertices {
51 struct Vertex {
52 SkPoint fPosition;
53 SkColor fColor;
54 };
55 Vertex* vertices;
56 Vertex* nextVertex;
57#ifdef SK_DEBUG
58 Vertex* verticesEnd;
59#endif
60 uint16_t* indices;
61 uint16_t* nextIndex;
62};
63
64struct MSAAQuadVertices {
65 struct Vertex {
66 SkPoint fPosition;
67 SkPoint fUV;
68 SkColor fColor;
69 };
70 Vertex* vertices;
71 Vertex* nextVertex;
72#ifdef SK_DEBUG
73 Vertex* verticesEnd;
74#endif
75 uint16_t* indices;
76 uint16_t* nextIndex;
77};
78
79static inline void append_contour_edge_indices(uint16_t fanCenterIdx,
80 uint16_t edgeV0Idx,
81 MSAALineVertices& lines) {
82 *(lines.nextIndex++) = fanCenterIdx;
83 *(lines.nextIndex++) = edgeV0Idx;
84 *(lines.nextIndex++) = edgeV0Idx + 1;
85}
86
bungeman06ca8ec2016-06-09 08:01:03 -070087static inline void add_quad(MSAALineVertices& lines, MSAAQuadVertices& quads, const SkPoint pts[],
ethannicholas6536ae52016-05-02 12:16:49 -070088 SkColor color, bool indexed, uint16_t subpathLineIdxStart) {
89 SkASSERT(lines.nextVertex < lines.verticesEnd);
90 *lines.nextVertex = { pts[2], color };
91 if (indexed) {
92 int prevIdx = (uint16_t) (lines.nextVertex - lines.vertices - 1);
93 if (prevIdx > subpathLineIdxStart) {
94 append_contour_edge_indices(subpathLineIdxStart, prevIdx, lines);
95 }
96 }
97 lines.nextVertex++;
98
99 SkASSERT(quads.nextVertex + 2 < quads.verticesEnd);
100 // the texture coordinates are drawn from the Loop-Blinn rendering algorithm
101 *(quads.nextVertex++) = { pts[0], SkPoint::Make(0.0, 0.0), color };
102 *(quads.nextVertex++) = { pts[1], SkPoint::Make(0.5, 0.0), color };
103 *(quads.nextVertex++) = { pts[2], SkPoint::Make(1.0, 1.0), color };
104 if (indexed) {
105 uint16_t offset = (uint16_t) (quads.nextVertex - quads.vertices) - 3;
106 *(quads.nextIndex++) = offset++;
107 *(quads.nextIndex++) = offset++;
108 *(quads.nextIndex++) = offset++;
109 }
110}
111
112class MSAAQuadProcessor : public GrGeometryProcessor {
113public:
114 static GrGeometryProcessor* Create(const SkMatrix& viewMatrix) {
115 return new MSAAQuadProcessor(viewMatrix);
116 }
117
118 virtual ~MSAAQuadProcessor() {}
119
120 const char* name() const override { return "MSAAQuadProcessor"; }
121
122 const Attribute* inPosition() const { return fInPosition; }
123 const Attribute* inUV() const { return fInUV; }
124 const Attribute* inColor() const { return fInColor; }
125 const SkMatrix& viewMatrix() const { return fViewMatrix; }
ethannicholas6536ae52016-05-02 12:16:49 -0700126
127 class GLSLProcessor : public GrGLSLGeometryProcessor {
128 public:
129 GLSLProcessor(const GrGeometryProcessor& qpr) {}
130
131 void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override {
132 const MSAAQuadProcessor& qp = args.fGP.cast<MSAAQuadProcessor>();
133 GrGLSLVertexBuilder* vsBuilder = args.fVertBuilder;
134 GrGLSLVaryingHandler* varyingHandler = args.fVaryingHandler;
135 GrGLSLUniformHandler* uniformHandler = args.fUniformHandler;
136
137 // emit attributes
138 varyingHandler->emitAttributes(qp);
139 varyingHandler->addPassThroughAttribute(qp.inColor(), args.fOutputColor);
140
141 GrGLSLVertToFrag uv(kVec2f_GrSLType);
142 varyingHandler->addVarying("uv", &uv, kHigh_GrSLPrecision);
143 vsBuilder->codeAppendf("%s = %s;", uv.vsOut(), qp.inUV()->fName);
144
145 // Setup position
bungeman06ca8ec2016-06-09 08:01:03 -0700146 this->setupPosition(vsBuilder, uniformHandler, gpArgs, qp.inPosition()->fName,
ethannicholas6536ae52016-05-02 12:16:49 -0700147 qp.viewMatrix(), &fViewMatrixUniform);
148
149 // emit transforms
bungeman06ca8ec2016-06-09 08:01:03 -0700150 this->emitTransforms(vsBuilder, varyingHandler, uniformHandler, gpArgs->fPositionVar,
151 qp.inPosition()->fName, SkMatrix::I(), args.fTransformsIn,
ethannicholas6536ae52016-05-02 12:16:49 -0700152 args.fTransformsOut);
153
154 GrGLSLPPFragmentBuilder* fsBuilder = args.fFragBuilder;
bungeman06ca8ec2016-06-09 08:01:03 -0700155 fsBuilder->codeAppendf("if (%s.x * %s.x >= %s.y) discard;", uv.fsIn(), uv.fsIn(),
ethannicholas6536ae52016-05-02 12:16:49 -0700156 uv.fsIn());
157 fsBuilder->codeAppendf("%s = vec4(1.0);", args.fOutputCoverage);
158 }
159
160 static inline void GenKey(const GrGeometryProcessor& gp,
161 const GrGLSLCaps&,
162 GrProcessorKeyBuilder* b) {
163 const MSAAQuadProcessor& qp = gp.cast<MSAAQuadProcessor>();
164 uint32_t key = 0;
165 key |= qp.viewMatrix().hasPerspective() ? 0x1 : 0x0;
166 key |= qp.viewMatrix().isIdentity() ? 0x2: 0x0;
167 b->add32(key);
168 }
169
170 virtual void setData(const GrGLSLProgramDataManager& pdman,
171 const GrPrimitiveProcessor& gp) override {
172 const MSAAQuadProcessor& qp = gp.cast<MSAAQuadProcessor>();
173 if (!qp.viewMatrix().isIdentity()) {
174 float viewMatrix[3 * 3];
175 GrGLSLGetMatrix<3>(viewMatrix, qp.viewMatrix());
176 pdman.setMatrix3f(fViewMatrixUniform, viewMatrix);
177 }
178 }
179
ethannicholas6536ae52016-05-02 12:16:49 -0700180 private:
181 typedef GrGLSLGeometryProcessor INHERITED;
182
183 UniformHandle fViewMatrixUniform;
184 };
185
186 virtual void getGLSLProcessorKey(const GrGLSLCaps& caps,
187 GrProcessorKeyBuilder* b) const override {
188 GLSLProcessor::GenKey(*this, caps, b);
189 }
190
191 virtual GrGLSLPrimitiveProcessor* createGLSLInstance(const GrGLSLCaps&) const override {
192 return new GLSLProcessor(*this);
193 }
194
195private:
196 MSAAQuadProcessor(const SkMatrix& viewMatrix)
197 : fViewMatrix(viewMatrix) {
198 this->initClassID<MSAAQuadProcessor>();
bsalomon6cb807b2016-08-17 11:33:39 -0700199 fInPosition = &this->addVertexAttrib("inPosition", kVec2f_GrVertexAttribType,
200 kHigh_GrSLPrecision);
201 fInUV = &this->addVertexAttrib("inUV", kVec2f_GrVertexAttribType, kHigh_GrSLPrecision);
202 fInColor = &this->addVertexAttrib("inColor", kVec4ub_GrVertexAttribType);
ethannicholas6536ae52016-05-02 12:16:49 -0700203 this->setSampleShading(1.0f);
204 }
205
206 const Attribute* fInPosition;
207 const Attribute* fInUV;
208 const Attribute* fInColor;
209 SkMatrix fViewMatrix;
bungeman06ca8ec2016-06-09 08:01:03 -0700210
ethannicholas6536ae52016-05-02 12:16:49 -0700211 GR_DECLARE_GEOMETRY_PROCESSOR_TEST;
212
213 typedef GrGeometryProcessor INHERITED;
214};
215
216class MSAAPathBatch : public GrVertexBatch {
217public:
218 DEFINE_BATCH_CLASS_ID
219
bsalomon8e9e45a2016-07-08 09:14:07 -0700220 MSAAPathBatch(GrColor color, const SkPath& path, const SkMatrix& viewMatrix,
221 const SkRect& devBounds)
bsalomon50c56a32016-06-30 12:05:32 -0700222 : INHERITED(ClassID())
223 , fViewMatrix(viewMatrix) {
224 fPaths.emplace_back(PathInfo{color, path});
bsalomon8e9e45a2016-07-08 09:14:07 -0700225 this->setBounds(devBounds, HasAABloat::kNo, IsZeroArea::kNo);
bsalomon50c56a32016-06-30 12:05:32 -0700226 int contourCount;
227 this->computeWorstCasePointCount(path, &contourCount, &fMaxLineVertices, &fMaxQuadVertices);
228 fMaxLineIndices = fMaxLineVertices * 3;
229 fMaxQuadIndices = fMaxQuadVertices * 3;
230 fIsIndexed = contourCount > 1;
ethannicholas6536ae52016-05-02 12:16:49 -0700231 }
232
233 const char* name() const override { return "MSAAPathBatch"; }
234
bungeman06ca8ec2016-06-09 08:01:03 -0700235 void computePipelineOptimizations(GrInitInvariantOutput* color,
ethannicholas6536ae52016-05-02 12:16:49 -0700236 GrInitInvariantOutput* coverage,
237 GrBatchToXPOverrides* overrides) const override {
bsalomon50c56a32016-06-30 12:05:32 -0700238 // When this is called on a batch, there is only one path
239 color->setKnownFourComponents(fPaths[0].fColor);
ethannicholas6536ae52016-05-02 12:16:49 -0700240 coverage->setKnownSingleComponent(0xff);
241 }
242
243 bool isValid() const {
244 return !fIsIndexed || fMaxLineIndices <= SK_MaxU16;
245 }
246
247private:
248 void initBatchTracker(const GrXPOverridesForBatch& overrides) override {
249 // Handle any color overrides
250 if (!overrides.readsColor()) {
bsalomon50c56a32016-06-30 12:05:32 -0700251 fPaths[0].fColor = GrColor_ILLEGAL;
ethannicholas6536ae52016-05-02 12:16:49 -0700252 }
bsalomon50c56a32016-06-30 12:05:32 -0700253 overrides.getOverrideColorIfSet(&fPaths[0].fColor);
ethannicholas6536ae52016-05-02 12:16:49 -0700254 }
255
bsalomon50c56a32016-06-30 12:05:32 -0700256 void computeWorstCasePointCount(const SkPath& path, int* subpaths, int* outLinePointCount,
257 int* outQuadPointCount) const {
ethannicholas6536ae52016-05-02 12:16:49 -0700258 int linePointCount = 0;
259 int quadPointCount = 0;
260 *subpaths = 1;
261
262 bool first = true;
263
264 SkPath::Iter iter(path, false);
265 SkPath::Verb verb;
266
267 SkPoint pts[4];
268 while ((verb = iter.next(pts)) != SkPath::kDone_Verb) {
269 switch (verb) {
270 case SkPath::kLine_Verb:
271 linePointCount += 1;
272 break;
273 case SkPath::kConic_Verb: {
274 SkScalar weight = iter.conicWeight();
275 SkAutoConicToQuads converter;
276 converter.computeQuads(pts, weight, kTolerance);
277 int quadPts = converter.countQuads();
278 linePointCount += quadPts;
279 quadPointCount += 3 * quadPts;
280 }
281 case SkPath::kQuad_Verb:
282 linePointCount += 1;
283 quadPointCount += 3;
284 break;
285 case SkPath::kCubic_Verb: {
286 SkSTArray<15, SkPoint, true> quadPts;
287 GrPathUtils::convertCubicToQuads(pts, kTolerance, &quadPts);
288 int count = quadPts.count();
289 linePointCount += count / 3;
290 quadPointCount += count;
291 break;
292 }
293 case SkPath::kMove_Verb:
294 linePointCount += 1;
295 if (!first) {
296 ++(*subpaths);
297 }
298 break;
299 default:
300 break;
301 }
302 first = false;
303 }
304 *outLinePointCount = linePointCount;
305 *outQuadPointCount = quadPointCount;
306 }
307
308 void onPrepareDraws(Target* target) const override {
309 SkASSERT(this->isValid());
310 if (fMaxLineVertices == 0) {
311 SkASSERT(fMaxQuadVertices == 0);
312 return;
313 }
314
bungeman06ca8ec2016-06-09 08:01:03 -0700315 GrPrimitiveType primitiveType = fIsIndexed ? kTriangles_GrPrimitiveType
ethannicholas6536ae52016-05-02 12:16:49 -0700316 : kTriangleFan_GrPrimitiveType;
317
318 // allocate vertex / index buffers
319 const GrBuffer* lineVertexBuffer;
320 int firstLineVertex;
321 MSAALineVertices lines;
322 size_t lineVertexStride = sizeof(MSAALineVertices::Vertex);
bungeman06ca8ec2016-06-09 08:01:03 -0700323 lines.vertices = (MSAALineVertices::Vertex*) target->makeVertexSpace(lineVertexStride,
ethannicholas6536ae52016-05-02 12:16:49 -0700324 fMaxLineVertices,
bungeman06ca8ec2016-06-09 08:01:03 -0700325 &lineVertexBuffer,
ethannicholas6536ae52016-05-02 12:16:49 -0700326 &firstLineVertex);
327 if (!lines.vertices) {
328 SkDebugf("Could not allocate vertices\n");
329 return;
330 }
331 lines.nextVertex = lines.vertices;
332 SkDEBUGCODE(lines.verticesEnd = lines.vertices + fMaxLineVertices;)
333
334 MSAAQuadVertices quads;
335 size_t quadVertexStride = sizeof(MSAAQuadVertices::Vertex);
336 SkAutoFree quadVertexPtr(sk_malloc_throw(fMaxQuadVertices * quadVertexStride));
337 quads.vertices = (MSAAQuadVertices::Vertex*) quadVertexPtr.get();
338 quads.nextVertex = quads.vertices;
339 SkDEBUGCODE(quads.verticesEnd = quads.vertices + fMaxQuadVertices;)
340
341 const GrBuffer* lineIndexBuffer = nullptr;
342 int firstLineIndex;
343 if (fIsIndexed) {
bungeman06ca8ec2016-06-09 08:01:03 -0700344 lines.indices = target->makeIndexSpace(fMaxLineIndices, &lineIndexBuffer,
ethannicholas6536ae52016-05-02 12:16:49 -0700345 &firstLineIndex);
346 if (!lines.indices) {
347 SkDebugf("Could not allocate indices\n");
348 return;
349 }
350 lines.nextIndex = lines.indices;
351 } else {
352 lines.indices = nullptr;
353 lines.nextIndex = nullptr;
354 }
355
356 SkAutoFree quadIndexPtr;
357 if (fIsIndexed) {
358 quads.indices = (uint16_t*) sk_malloc_throw(fMaxQuadIndices * sizeof(uint16_t));
359 quadIndexPtr.set(quads.indices);
360 quads.nextIndex = quads.indices;
361 } else {
362 quads.indices = nullptr;
363 quads.nextIndex = nullptr;
364 }
365
366 // fill buffers
bsalomon50c56a32016-06-30 12:05:32 -0700367 for (int i = 0; i < fPaths.count(); i++) {
368 const PathInfo& pathInfo = fPaths[i];
ethannicholas6536ae52016-05-02 12:16:49 -0700369
370 if (!this->createGeom(lines,
371 quads,
bsalomon50c56a32016-06-30 12:05:32 -0700372 pathInfo.fPath,
ethannicholas6536ae52016-05-02 12:16:49 -0700373 fViewMatrix,
bsalomon50c56a32016-06-30 12:05:32 -0700374 pathInfo.fColor,
ethannicholas6536ae52016-05-02 12:16:49 -0700375 fIsIndexed)) {
376 return;
377 }
378 }
379 int lineVertexOffset = (int) (lines.nextVertex - lines.vertices);
380 int lineIndexOffset = (int) (lines.nextIndex - lines.indices);
381 SkASSERT(lineVertexOffset <= fMaxLineVertices && lineIndexOffset <= fMaxLineIndices);
382 int quadVertexOffset = (int) (quads.nextVertex - quads.vertices);
383 int quadIndexOffset = (int) (quads.nextIndex - quads.indices);
384 SkASSERT(quadVertexOffset <= fMaxQuadVertices && quadIndexOffset <= fMaxQuadIndices);
385
386 if (lineVertexOffset) {
bungeman06ca8ec2016-06-09 08:01:03 -0700387 sk_sp<GrGeometryProcessor> lineGP;
ethannicholas6536ae52016-05-02 12:16:49 -0700388 {
389 using namespace GrDefaultGeoProcFactory;
bungeman06ca8ec2016-06-09 08:01:03 -0700390 lineGP = GrDefaultGeoProcFactory::Make(Color(Color::kAttribute_Type),
391 Coverage(255),
392 LocalCoords(LocalCoords::kUnused_Type),
393 fViewMatrix);
ethannicholas6536ae52016-05-02 12:16:49 -0700394 }
395 SkASSERT(lineVertexStride == lineGP->getVertexStride());
396
397 GrMesh lineMeshes;
398 if (fIsIndexed) {
bungeman06ca8ec2016-06-09 08:01:03 -0700399 lineMeshes.initIndexed(primitiveType, lineVertexBuffer, lineIndexBuffer,
400 firstLineVertex, firstLineIndex, lineVertexOffset,
ethannicholas6536ae52016-05-02 12:16:49 -0700401 lineIndexOffset);
402 } else {
bungeman06ca8ec2016-06-09 08:01:03 -0700403 lineMeshes.init(primitiveType, lineVertexBuffer, firstLineVertex,
ethannicholas6536ae52016-05-02 12:16:49 -0700404 lineVertexOffset);
405 }
bungeman06ca8ec2016-06-09 08:01:03 -0700406 target->draw(lineGP.get(), lineMeshes);
ethannicholas6536ae52016-05-02 12:16:49 -0700407 }
408
409 if (quadVertexOffset) {
410 SkAutoTUnref<const GrGeometryProcessor> quadGP(MSAAQuadProcessor::Create(fViewMatrix));
411 SkASSERT(quadVertexStride == quadGP->getVertexStride());
412
413 const GrBuffer* quadVertexBuffer;
414 int firstQuadVertex;
bungeman06ca8ec2016-06-09 08:01:03 -0700415 MSAAQuadVertices::Vertex* quadVertices = (MSAAQuadVertices::Vertex*)
ethannicholas6536ae52016-05-02 12:16:49 -0700416 target->makeVertexSpace(quadVertexStride, quadVertexOffset, &quadVertexBuffer,
417 &firstQuadVertex);
418 memcpy(quadVertices, quads.vertices, quadVertexStride * quadVertexOffset);
419 GrMesh quadMeshes;
420 if (fIsIndexed) {
421 const GrBuffer* quadIndexBuffer;
422 int firstQuadIndex;
bungeman06ca8ec2016-06-09 08:01:03 -0700423 uint16_t* quadIndices = (uint16_t*) target->makeIndexSpace(quadIndexOffset,
424 &quadIndexBuffer,
ethannicholas6536ae52016-05-02 12:16:49 -0700425 &firstQuadIndex);
426 memcpy(quadIndices, quads.indices, sizeof(uint16_t) * quadIndexOffset);
bungeman06ca8ec2016-06-09 08:01:03 -0700427 quadMeshes.initIndexed(kTriangles_GrPrimitiveType, quadVertexBuffer,
428 quadIndexBuffer, firstQuadVertex, firstQuadIndex,
ethannicholas6536ae52016-05-02 12:16:49 -0700429 quadVertexOffset, quadIndexOffset);
430 } else {
bungeman06ca8ec2016-06-09 08:01:03 -0700431 quadMeshes.init(kTriangles_GrPrimitiveType, quadVertexBuffer, firstQuadVertex,
ethannicholas6536ae52016-05-02 12:16:49 -0700432 quadVertexOffset);
433 }
434 target->draw(quadGP, quadMeshes);
435 }
436 }
437
ethannicholas6536ae52016-05-02 12:16:49 -0700438 bool onCombineIfPossible(GrBatch* t, const GrCaps& caps) override {
439 MSAAPathBatch* that = t->cast<MSAAPathBatch>();
440 if (!GrPipeline::CanCombine(*this->pipeline(), this->bounds(), *that->pipeline(),
441 that->bounds(), caps)) {
442 return false;
443 }
444
445 if (!fViewMatrix.cheapEqualTo(that->fViewMatrix)) {
446 return false;
447 }
448
bungeman06ca8ec2016-06-09 08:01:03 -0700449 if ((fMaxLineIndices + that->fMaxLineIndices > SK_MaxU16) ||
ethannicholas6536ae52016-05-02 12:16:49 -0700450 (fMaxQuadIndices + that->fMaxQuadIndices > SK_MaxU16)) {
451 return false;
452 }
453
bsalomon50c56a32016-06-30 12:05:32 -0700454 fPaths.push_back_n(that->fPaths.count(), that->fPaths.begin());
bsalomon88cf17d2016-07-08 06:40:56 -0700455 this->joinBounds(*that);
ethannicholas6536ae52016-05-02 12:16:49 -0700456 fIsIndexed = true;
457 fMaxLineVertices += that->fMaxLineVertices;
458 fMaxQuadVertices += that->fMaxQuadVertices;
459 fMaxLineIndices += that->fMaxLineIndices;
460 fMaxQuadIndices += that->fMaxQuadIndices;
461 return true;
462 }
463
464 bool createGeom(MSAALineVertices& lines,
465 MSAAQuadVertices& quads,
466 const SkPath& path,
ethannicholas6536ae52016-05-02 12:16:49 -0700467 const SkMatrix& m,
468 SkColor color,
469 bool isIndexed) const {
470 {
471 uint16_t subpathIdxStart = (uint16_t) (lines.nextVertex - lines.vertices);
472
473 SkPoint pts[4];
474
475 bool first = true;
476 SkPath::Iter iter(path, false);
477
478 bool done = false;
479 while (!done) {
480 SkPath::Verb verb = iter.next(pts);
481 switch (verb) {
482 case SkPath::kMove_Verb:
483 if (!first) {
484 uint16_t currIdx = (uint16_t) (lines.nextVertex - lines.vertices);
485 subpathIdxStart = currIdx;
486 }
487 SkASSERT(lines.nextVertex < lines.verticesEnd);
488 *(lines.nextVertex++) = { pts[0], color };
489 break;
490 case SkPath::kLine_Verb:
491 if (isIndexed) {
492 uint16_t prevIdx = (uint16_t) (lines.nextVertex - lines.vertices - 1);
493 if (prevIdx > subpathIdxStart) {
494 append_contour_edge_indices(subpathIdxStart, prevIdx, lines);
495 }
496 }
497 SkASSERT(lines.nextVertex < lines.verticesEnd);
498 *(lines.nextVertex++) = { pts[1], color };
499 break;
500 case SkPath::kConic_Verb: {
501 SkScalar weight = iter.conicWeight();
502 SkAutoConicToQuads converter;
bsalomon50c56a32016-06-30 12:05:32 -0700503 const SkPoint* quadPts = converter.computeQuads(pts, weight, kTolerance);
ethannicholas6536ae52016-05-02 12:16:49 -0700504 for (int i = 0; i < converter.countQuads(); ++i) {
bungeman06ca8ec2016-06-09 08:01:03 -0700505 add_quad(lines, quads, quadPts + i * 2, color, isIndexed,
ethannicholas6536ae52016-05-02 12:16:49 -0700506 subpathIdxStart);
507 }
508 break;
509 }
510 case SkPath::kQuad_Verb: {
511 add_quad(lines, quads, pts, color, isIndexed, subpathIdxStart);
bungeman06ca8ec2016-06-09 08:01:03 -0700512 break;
ethannicholas6536ae52016-05-02 12:16:49 -0700513 }
514 case SkPath::kCubic_Verb: {
515 SkSTArray<15, SkPoint, true> quadPts;
516 GrPathUtils::convertCubicToQuads(pts, kTolerance, &quadPts);
517 int count = quadPts.count();
518 for (int i = 0; i < count; i += 3) {
519 add_quad(lines, quads, &quadPts[i], color, isIndexed, subpathIdxStart);
520 }
521 break;
522 }
523 case SkPath::kClose_Verb:
524 break;
525 case SkPath::kDone_Verb:
526 done = true;
527 }
528 first = false;
529 }
530 }
531 return true;
532 }
533
bsalomon50c56a32016-06-30 12:05:32 -0700534 struct PathInfo {
535 GrColor fColor;
536 SkPath fPath;
537 };
538
539 SkSTArray<1, PathInfo, true> fPaths;
ethannicholas6536ae52016-05-02 12:16:49 -0700540
541 SkMatrix fViewMatrix;
542 int fMaxLineVertices;
543 int fMaxQuadVertices;
544 int fMaxLineIndices;
545 int fMaxQuadIndices;
546 bool fIsIndexed;
547
548 typedef GrVertexBatch INHERITED;
549};
550
robertphillips976f5f02016-06-03 10:59:20 -0700551bool GrMSAAPathRenderer::internalDrawPath(GrDrawContext* drawContext,
552 const GrPaint& paint,
robertphillipsd2b6d642016-07-21 08:55:08 -0700553 const GrUserStencilSettings& userStencilSettings,
cdalton862cff32016-05-12 15:09:48 -0700554 const GrClip& clip,
ethannicholas6536ae52016-05-02 12:16:49 -0700555 const SkMatrix& viewMatrix,
bsalomon8acedde2016-06-24 10:42:16 -0700556 const GrShape& shape,
ethannicholas6536ae52016-05-02 12:16:49 -0700557 bool stencilOnly) {
bsalomon8acedde2016-06-24 10:42:16 -0700558 SkASSERT(shape.style().isSimpleFill());
559 SkPath path;
560 shape.asPath(&path);
561
robertphillips8e375302016-07-11 10:43:58 -0700562 static const int kMaxNumPasses = 2;
563
cdalton93a379b2016-05-11 13:58:08 -0700564 int passCount = 0;
robertphillips8e375302016-07-11 10:43:58 -0700565 const GrUserStencilSettings* passes[kMaxNumPasses];
cdalton93a379b2016-05-11 13:58:08 -0700566 bool reverse = false;
567 bool lastPassIsBounds;
ethannicholas6536ae52016-05-02 12:16:49 -0700568
bsalomon8acedde2016-06-24 10:42:16 -0700569 if (single_pass_shape(shape)) {
ethannicholas6536ae52016-05-02 12:16:49 -0700570 passCount = 1;
571 if (stencilOnly) {
572 passes[0] = &gDirectToStencil;
573 } else {
robertphillipsd2b6d642016-07-21 08:55:08 -0700574 passes[0] = &userStencilSettings;
ethannicholas6536ae52016-05-02 12:16:49 -0700575 }
ethannicholas6536ae52016-05-02 12:16:49 -0700576 lastPassIsBounds = false;
577 } else {
578 switch (path.getFillType()) {
579 case SkPath::kInverseEvenOdd_FillType:
580 reverse = true;
581 // fallthrough
582 case SkPath::kEvenOdd_FillType:
583 passes[0] = &gEOStencilPass;
584 if (stencilOnly) {
585 passCount = 1;
586 lastPassIsBounds = false;
587 } else {
588 passCount = 2;
589 lastPassIsBounds = true;
590 if (reverse) {
591 passes[1] = &gInvEOColorPass;
592 } else {
593 passes[1] = &gEOColorPass;
594 }
595 }
ethannicholas6536ae52016-05-02 12:16:49 -0700596 break;
597
598 case SkPath::kInverseWinding_FillType:
599 reverse = true;
600 // fallthrough
601 case SkPath::kWinding_FillType:
602 passes[0] = &gWindStencilSeparateWithWrap;
603 passCount = 2;
ethannicholas6536ae52016-05-02 12:16:49 -0700604 if (stencilOnly) {
605 lastPassIsBounds = false;
robertphillips8e375302016-07-11 10:43:58 -0700606 passCount = 1;
ethannicholas6536ae52016-05-02 12:16:49 -0700607 } else {
608 lastPassIsBounds = true;
ethannicholas6536ae52016-05-02 12:16:49 -0700609 if (reverse) {
robertphillips8e375302016-07-11 10:43:58 -0700610 passes[1] = &gInvWindColorPass;
ethannicholas6536ae52016-05-02 12:16:49 -0700611 } else {
robertphillips8e375302016-07-11 10:43:58 -0700612 passes[1] = &gWindColorPass;
ethannicholas6536ae52016-05-02 12:16:49 -0700613 }
614 }
615 break;
616 default:
617 SkDEBUGFAIL("Unknown path fFill!");
618 return false;
619 }
620 }
621
622 SkRect devBounds;
robertphillips976f5f02016-06-03 10:59:20 -0700623 GetPathDevBounds(path, drawContext->width(), drawContext->height(), viewMatrix, &devBounds);
ethannicholas6536ae52016-05-02 12:16:49 -0700624
robertphillips8e375302016-07-11 10:43:58 -0700625 SkASSERT(passCount <= kMaxNumPasses);
626
ethannicholas6536ae52016-05-02 12:16:49 -0700627 for (int p = 0; p < passCount; ++p) {
ethannicholas6536ae52016-05-02 12:16:49 -0700628 if (lastPassIsBounds && (p == passCount-1)) {
ethannicholas6536ae52016-05-02 12:16:49 -0700629 SkRect bounds;
630 SkMatrix localMatrix = SkMatrix::I();
631 if (reverse) {
ethannicholas6536ae52016-05-02 12:16:49 -0700632 // draw over the dev bounds (which will be the whole dst surface for inv fill).
633 bounds = devBounds;
634 SkMatrix vmi;
635 // mapRect through persp matrix may not be correct
636 if (!viewMatrix.hasPerspective() && viewMatrix.invert(&vmi)) {
637 vmi.mapRect(&bounds);
638 } else {
639 if (!viewMatrix.invert(&localMatrix)) {
640 return false;
641 }
642 }
643 } else {
644 bounds = path.getBounds();
645 }
646 const SkMatrix& viewM = (reverse && viewMatrix.hasPerspective()) ? SkMatrix::I() :
647 viewMatrix;
648 SkAutoTUnref<GrDrawBatch> batch(
robertphillips3950f0d2016-07-07 07:33:13 -0700649 GrRectBatchFactory::CreateNonAAFill(paint.getColor(), viewM, bounds, nullptr,
ethannicholas6536ae52016-05-02 12:16:49 -0700650 &localMatrix));
robertphillips976f5f02016-06-03 10:59:20 -0700651
bsalomonbb243832016-07-22 07:10:19 -0700652 GrPipelineBuilder pipelineBuilder(paint, drawContext->mustUseHWAA(paint));
653 pipelineBuilder.setUserStencil(passes[p]);
654
655 drawContext->drawBatch(pipelineBuilder, clip, batch);
robertphillips976f5f02016-06-03 10:59:20 -0700656 } else {
bsalomon88cf17d2016-07-08 06:40:56 -0700657 SkAutoTUnref<MSAAPathBatch> batch(new MSAAPathBatch(paint.getColor(), path,
bsalomon8e9e45a2016-07-08 09:14:07 -0700658 viewMatrix, devBounds));
robertphillips976f5f02016-06-03 10:59:20 -0700659 if (!batch->isValid()) {
ethannicholas6536ae52016-05-02 12:16:49 -0700660 return false;
661 }
robertphillips976f5f02016-06-03 10:59:20 -0700662
bsalomonbb243832016-07-22 07:10:19 -0700663 GrPipelineBuilder pipelineBuilder(paint, drawContext->mustUseHWAA(paint));
664 pipelineBuilder.setUserStencil(passes[p]);
robertphillips976f5f02016-06-03 10:59:20 -0700665 if (passCount > 1) {
bsalomonbb243832016-07-22 07:10:19 -0700666 pipelineBuilder.setDisableColorXPFactory();
robertphillips976f5f02016-06-03 10:59:20 -0700667 }
668
bsalomonbb243832016-07-22 07:10:19 -0700669 drawContext->drawBatch(pipelineBuilder, clip, batch);
ethannicholas6536ae52016-05-02 12:16:49 -0700670 }
671 }
672 return true;
673}
674
675bool GrMSAAPathRenderer::onCanDrawPath(const CanDrawPathArgs& args) const {
bsalomonee432412016-06-27 07:18:18 -0700676 // This path renderer only fills and relies on MSAA for antialiasing. Stroked shapes are
677 // handled by passing on the original shape and letting the caller compute the stroked shape
678 // which will have a fill style.
679 return args.fShape->style().isSimpleFill() && !args.fAntiAlias;
ethannicholas6536ae52016-05-02 12:16:49 -0700680}
681
682bool GrMSAAPathRenderer::onDrawPath(const DrawPathArgs& args) {
robertphillips976f5f02016-06-03 10:59:20 -0700683 GR_AUDIT_TRAIL_AUTO_FRAME(args.fDrawContext->auditTrail(),
684 "GrMSAAPathRenderer::onDrawPath");
bsalomon8acedde2016-06-24 10:42:16 -0700685 SkTLazy<GrShape> tmpShape;
686 const GrShape* shape = args.fShape;
687 if (shape->style().applies()) {
bsalomon6663acf2016-05-10 09:14:17 -0700688 SkScalar styleScale = GrStyle::MatrixToScaleFactor(*args.fViewMatrix);
bsalomon8acedde2016-06-24 10:42:16 -0700689 tmpShape.init(args.fShape->applyStyle(GrStyle::Apply::kPathEffectAndStrokeRec, styleScale));
690 shape = tmpShape.get();
ethannicholas6536ae52016-05-02 12:16:49 -0700691 }
robertphillips976f5f02016-06-03 10:59:20 -0700692 return this->internalDrawPath(args.fDrawContext,
693 *args.fPaint,
robertphillipsd2b6d642016-07-21 08:55:08 -0700694 *args.fUserStencilSettings,
cdalton862cff32016-05-12 15:09:48 -0700695 *args.fClip,
ethannicholas6536ae52016-05-02 12:16:49 -0700696 *args.fViewMatrix,
bsalomon8acedde2016-06-24 10:42:16 -0700697 *shape,
ethannicholas6536ae52016-05-02 12:16:49 -0700698 false);
699}
700
701void GrMSAAPathRenderer::onStencilPath(const StencilPathArgs& args) {
robertphillips976f5f02016-06-03 10:59:20 -0700702 GR_AUDIT_TRAIL_AUTO_FRAME(args.fDrawContext->auditTrail(),
703 "GrMSAAPathRenderer::onStencilPath");
bsalomon8acedde2016-06-24 10:42:16 -0700704 SkASSERT(args.fShape->style().isSimpleFill());
705 SkASSERT(!args.fShape->mayBeInverseFilledAfterStyling());
robertphillips976f5f02016-06-03 10:59:20 -0700706
707 GrPaint paint;
bungeman06ca8ec2016-06-09 08:01:03 -0700708 paint.setXPFactory(GrDisableColorXPFactory::Make());
robertphillips976f5f02016-06-03 10:59:20 -0700709 paint.setAntiAlias(args.fIsAA);
710
robertphillipsd2b6d642016-07-21 08:55:08 -0700711 this->internalDrawPath(args.fDrawContext, paint, GrUserStencilSettings::kUnused, *args.fClip,
robertphillips3950f0d2016-07-07 07:33:13 -0700712 *args.fViewMatrix, *args.fShape, true);
ethannicholas6536ae52016-05-02 12:16:49 -0700713}
714
715///////////////////////////////////////////////////////////////////////////////////////////////////