blob: e2f12cfb2ae4d74020e8e9eaba7ddeb0ab77b777 [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"
robertphillips976f5f02016-06-03 10:59:20 -070011#include "GrClip.h"
ethannicholas6536ae52016-05-02 12:16:49 -070012#include "GrDefaultGeoProcFactory.h"
csmartdalton02fa32c2016-08-19 13:29:27 -070013#include "GrFixedClip.h"
Brian Salomondad29232016-12-01 16:40:24 -050014#include "GrMesh.h"
Brian Salomon742e31d2016-12-07 17:06:19 -050015#include "GrOpFlushState.h"
ethannicholas6536ae52016-05-02 12:16:49 -070016#include "GrPathStencilSettings.h"
17#include "GrPathUtils.h"
bsalomonbb243832016-07-22 07:10:19 -070018#include "GrPipelineBuilder.h"
Hal Canary95e3c052017-01-11 12:44:43 -050019#include "SkAutoMalloc.h"
ethannicholas6536ae52016-05-02 12:16:49 -070020#include "SkGeometry.h"
21#include "SkTraceEvent.h"
Brian Salomondad29232016-12-01 16:40:24 -050022#include "gl/GrGLVaryingHandler.h"
ethannicholas6536ae52016-05-02 12:16:49 -070023#include "glsl/GrGLSLFragmentShaderBuilder.h"
Brian Salomondad29232016-12-01 16:40:24 -050024#include "glsl/GrGLSLGeometryProcessor.h"
ethannicholas6536ae52016-05-02 12:16:49 -070025#include "glsl/GrGLSLProgramDataManager.h"
26#include "glsl/GrGLSLUtil.h"
Brian Salomondad29232016-12-01 16:40:24 -050027#include "glsl/GrGLSLVertexShaderBuilder.h"
Brian Salomon89527432016-12-16 09:52:16 -050028#include "ops/GrMeshDrawOp.h"
Brian Salomonac70f842017-05-08 10:43:33 -040029#include "ops/GrNonAAFillRectOp.h"
ethannicholas6536ae52016-05-02 12:16:49 -070030
31static const float kTolerance = 0.5f;
32
33////////////////////////////////////////////////////////////////////////////////
34// Helpers for drawPath
35
bsalomon8acedde2016-06-24 10:42:16 -070036static inline bool single_pass_shape(const GrShape& shape) {
37 if (!shape.inverseFilled()) {
38 return shape.knownToBeConvex();
ethannicholas6536ae52016-05-02 12:16:49 -070039 }
40 return false;
41}
42
bsalomon8acedde2016-06-24 10:42:16 -070043GrPathRenderer::StencilSupport GrMSAAPathRenderer::onGetStencilSupport(const GrShape& shape) const {
44 if (single_pass_shape(shape)) {
ethannicholas6536ae52016-05-02 12:16:49 -070045 return GrPathRenderer::kNoRestriction_StencilSupport;
46 } else {
47 return GrPathRenderer::kStencilOnly_StencilSupport;
48 }
49}
50
51struct MSAALineVertices {
52 struct Vertex {
53 SkPoint fPosition;
54 SkColor fColor;
55 };
56 Vertex* vertices;
57 Vertex* nextVertex;
58#ifdef SK_DEBUG
59 Vertex* verticesEnd;
60#endif
61 uint16_t* indices;
62 uint16_t* nextIndex;
63};
64
65struct MSAAQuadVertices {
66 struct Vertex {
67 SkPoint fPosition;
68 SkPoint fUV;
69 SkColor fColor;
70 };
71 Vertex* vertices;
72 Vertex* nextVertex;
73#ifdef SK_DEBUG
74 Vertex* verticesEnd;
75#endif
76 uint16_t* indices;
77 uint16_t* nextIndex;
78};
79
80static inline void append_contour_edge_indices(uint16_t fanCenterIdx,
81 uint16_t edgeV0Idx,
82 MSAALineVertices& lines) {
83 *(lines.nextIndex++) = fanCenterIdx;
84 *(lines.nextIndex++) = edgeV0Idx;
85 *(lines.nextIndex++) = edgeV0Idx + 1;
86}
87
bungeman06ca8ec2016-06-09 08:01:03 -070088static inline void add_quad(MSAALineVertices& lines, MSAAQuadVertices& quads, const SkPoint pts[],
ethannicholas6536ae52016-05-02 12:16:49 -070089 SkColor color, bool indexed, uint16_t subpathLineIdxStart) {
90 SkASSERT(lines.nextVertex < lines.verticesEnd);
91 *lines.nextVertex = { pts[2], color };
92 if (indexed) {
93 int prevIdx = (uint16_t) (lines.nextVertex - lines.vertices - 1);
94 if (prevIdx > subpathLineIdxStart) {
95 append_contour_edge_indices(subpathLineIdxStart, prevIdx, lines);
96 }
97 }
98 lines.nextVertex++;
99
100 SkASSERT(quads.nextVertex + 2 < quads.verticesEnd);
101 // the texture coordinates are drawn from the Loop-Blinn rendering algorithm
102 *(quads.nextVertex++) = { pts[0], SkPoint::Make(0.0, 0.0), color };
103 *(quads.nextVertex++) = { pts[1], SkPoint::Make(0.5, 0.0), color };
104 *(quads.nextVertex++) = { pts[2], SkPoint::Make(1.0, 1.0), color };
105 if (indexed) {
106 uint16_t offset = (uint16_t) (quads.nextVertex - quads.vertices) - 3;
107 *(quads.nextIndex++) = offset++;
108 *(quads.nextIndex++) = offset++;
109 *(quads.nextIndex++) = offset++;
110 }
111}
112
113class MSAAQuadProcessor : public GrGeometryProcessor {
114public:
115 static GrGeometryProcessor* Create(const SkMatrix& viewMatrix) {
116 return new MSAAQuadProcessor(viewMatrix);
117 }
118
Brian Salomond3b65972017-03-22 12:05:03 -0400119 ~MSAAQuadProcessor() override {}
ethannicholas6536ae52016-05-02 12:16:49 -0700120
121 const char* name() const override { return "MSAAQuadProcessor"; }
122
123 const Attribute* inPosition() const { return fInPosition; }
124 const Attribute* inUV() const { return fInUV; }
125 const Attribute* inColor() const { return fInColor; }
126 const SkMatrix& viewMatrix() const { return fViewMatrix; }
ethannicholas6536ae52016-05-02 12:16:49 -0700127
128 class GLSLProcessor : public GrGLSLGeometryProcessor {
129 public:
130 GLSLProcessor(const GrGeometryProcessor& qpr) {}
131
132 void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override {
133 const MSAAQuadProcessor& qp = args.fGP.cast<MSAAQuadProcessor>();
134 GrGLSLVertexBuilder* vsBuilder = args.fVertBuilder;
135 GrGLSLVaryingHandler* varyingHandler = args.fVaryingHandler;
136 GrGLSLUniformHandler* uniformHandler = args.fUniformHandler;
137
138 // emit attributes
139 varyingHandler->emitAttributes(qp);
140 varyingHandler->addPassThroughAttribute(qp.inColor(), args.fOutputColor);
141
142 GrGLSLVertToFrag uv(kVec2f_GrSLType);
143 varyingHandler->addVarying("uv", &uv, kHigh_GrSLPrecision);
144 vsBuilder->codeAppendf("%s = %s;", uv.vsOut(), qp.inUV()->fName);
145
146 // Setup position
bungeman06ca8ec2016-06-09 08:01:03 -0700147 this->setupPosition(vsBuilder, uniformHandler, gpArgs, qp.inPosition()->fName,
ethannicholas6536ae52016-05-02 12:16:49 -0700148 qp.viewMatrix(), &fViewMatrixUniform);
149
150 // emit transforms
bungeman06ca8ec2016-06-09 08:01:03 -0700151 this->emitTransforms(vsBuilder, varyingHandler, uniformHandler, gpArgs->fPositionVar,
bsalomona624bf32016-09-20 09:12:47 -0700152 qp.inPosition()->fName, SkMatrix::I(),
153 args.fFPCoordTransformHandler);
ethannicholas6536ae52016-05-02 12:16:49 -0700154
155 GrGLSLPPFragmentBuilder* fsBuilder = args.fFragBuilder;
bungeman06ca8ec2016-06-09 08:01:03 -0700156 fsBuilder->codeAppendf("if (%s.x * %s.x >= %s.y) discard;", uv.fsIn(), uv.fsIn(),
ethannicholas6536ae52016-05-02 12:16:49 -0700157 uv.fsIn());
158 fsBuilder->codeAppendf("%s = vec4(1.0);", args.fOutputCoverage);
159 }
160
161 static inline void GenKey(const GrGeometryProcessor& gp,
Brian Salomon94efbf52016-11-29 13:43:05 -0500162 const GrShaderCaps&,
ethannicholas6536ae52016-05-02 12:16:49 -0700163 GrProcessorKeyBuilder* b) {
164 const MSAAQuadProcessor& qp = gp.cast<MSAAQuadProcessor>();
165 uint32_t key = 0;
166 key |= qp.viewMatrix().hasPerspective() ? 0x1 : 0x0;
167 key |= qp.viewMatrix().isIdentity() ? 0x2: 0x0;
168 b->add32(key);
169 }
170
bsalomona624bf32016-09-20 09:12:47 -0700171 void setData(const GrGLSLProgramDataManager& pdman, const GrPrimitiveProcessor& gp,
172 FPCoordTransformIter&& transformIter) override {
ethannicholas6536ae52016-05-02 12:16:49 -0700173 const MSAAQuadProcessor& qp = gp.cast<MSAAQuadProcessor>();
174 if (!qp.viewMatrix().isIdentity()) {
175 float viewMatrix[3 * 3];
176 GrGLSLGetMatrix<3>(viewMatrix, qp.viewMatrix());
177 pdman.setMatrix3f(fViewMatrixUniform, viewMatrix);
178 }
bsalomona624bf32016-09-20 09:12:47 -0700179 this->setTransformDataHelper(SkMatrix::I(), pdman, &transformIter);
ethannicholas6536ae52016-05-02 12:16:49 -0700180 }
181
ethannicholas6536ae52016-05-02 12:16:49 -0700182 private:
183 typedef GrGLSLGeometryProcessor INHERITED;
184
185 UniformHandle fViewMatrixUniform;
186 };
187
Brian Salomon94efbf52016-11-29 13:43:05 -0500188 virtual void getGLSLProcessorKey(const GrShaderCaps& caps,
ethannicholas6536ae52016-05-02 12:16:49 -0700189 GrProcessorKeyBuilder* b) const override {
190 GLSLProcessor::GenKey(*this, caps, b);
191 }
192
Brian Salomon94efbf52016-11-29 13:43:05 -0500193 virtual GrGLSLPrimitiveProcessor* createGLSLInstance(const GrShaderCaps&) const override {
ethannicholas6536ae52016-05-02 12:16:49 -0700194 return new GLSLProcessor(*this);
195 }
196
197private:
198 MSAAQuadProcessor(const SkMatrix& viewMatrix)
199 : fViewMatrix(viewMatrix) {
200 this->initClassID<MSAAQuadProcessor>();
bsalomon6cb807b2016-08-17 11:33:39 -0700201 fInPosition = &this->addVertexAttrib("inPosition", kVec2f_GrVertexAttribType,
202 kHigh_GrSLPrecision);
203 fInUV = &this->addVertexAttrib("inUV", kVec2f_GrVertexAttribType, kHigh_GrSLPrecision);
204 fInColor = &this->addVertexAttrib("inColor", kVec4ub_GrVertexAttribType);
ethannicholas6536ae52016-05-02 12:16:49 -0700205 this->setSampleShading(1.0f);
206 }
207
208 const Attribute* fInPosition;
209 const Attribute* fInUV;
210 const Attribute* fInColor;
211 SkMatrix fViewMatrix;
bungeman06ca8ec2016-06-09 08:01:03 -0700212
ethannicholas6536ae52016-05-02 12:16:49 -0700213 GR_DECLARE_GEOMETRY_PROCESSOR_TEST;
214
215 typedef GrGeometryProcessor INHERITED;
216};
217
Brian Salomond3ccb0a2017-04-03 10:38:00 -0400218class MSAAPathOp final : public GrLegacyMeshDrawOp {
ethannicholas6536ae52016-05-02 12:16:49 -0700219public:
Brian Salomon25a88092016-12-01 09:36:50 -0500220 DEFINE_OP_CLASS_ID
Brian Salomond3ccb0a2017-04-03 10:38:00 -0400221 static std::unique_ptr<GrLegacyMeshDrawOp> Make(GrColor color, const SkPath& path,
222 const SkMatrix& viewMatrix,
223 const SkRect& devBounds) {
bsalomon50c56a32016-06-30 12:05:32 -0700224 int contourCount;
Brian Salomon780dad12016-12-15 18:08:40 -0500225 int maxLineVertices;
226 int maxQuadVertices;
Brian Salomon32ebaba2017-03-30 10:22:28 -0400227 ComputeWorstCasePointCount(path, viewMatrix, &contourCount, &maxLineVertices,
228 &maxQuadVertices);
Brian Salomon780dad12016-12-15 18:08:40 -0500229 bool isIndexed = contourCount > 1;
230 if (isIndexed &&
231 (maxLineVertices > kMaxIndexedVertexCnt || maxQuadVertices > kMaxIndexedVertexCnt)) {
232 return nullptr;
233 }
234
Brian Salomond3ccb0a2017-04-03 10:38:00 -0400235 return std::unique_ptr<GrLegacyMeshDrawOp>(new MSAAPathOp(
Brian Salomonf8334782017-01-03 09:42:58 -0500236 color, path, viewMatrix, devBounds, maxLineVertices, maxQuadVertices, isIndexed));
ethannicholas6536ae52016-05-02 12:16:49 -0700237 }
238
Brian Salomon780dad12016-12-15 18:08:40 -0500239 const char* name() const override { return "MSAAPathOp"; }
ethannicholas6536ae52016-05-02 12:16:49 -0700240
Brian Salomon7c3e7182016-12-01 09:35:30 -0500241 SkString dumpInfo() const override {
242 SkString string;
243 string.appendf("Indexed: %d\n", fIsIndexed);
244 for (const auto& path : fPaths) {
245 string.appendf("Color: 0x%08x\n", path.fColor);
246 }
247 string.append(DumpPipelineInfo(*this->pipeline()));
248 string.append(INHERITED::dumpInfo());
249 return string;
250 }
251
Brian Salomon780dad12016-12-15 18:08:40 -0500252private:
253 MSAAPathOp(GrColor color, const SkPath& path, const SkMatrix& viewMatrix,
254 const SkRect& devBounds, int maxLineVertices, int maxQuadVertices, bool isIndexed)
255 : INHERITED(ClassID())
256 , fViewMatrix(viewMatrix)
257 , fMaxLineVertices(maxLineVertices)
258 , fMaxQuadVertices(maxQuadVertices)
259 , fIsIndexed(isIndexed) {
260 fPaths.emplace_back(PathInfo{color, path});
261 this->setBounds(devBounds, HasAABloat::kNo, IsZeroArea::kNo);
ethannicholas6536ae52016-05-02 12:16:49 -0700262 }
263
Brian Salomona811b122017-03-30 08:21:32 -0400264 void getProcessorAnalysisInputs(GrProcessorAnalysisColor* color,
265 GrProcessorAnalysisCoverage* coverage) const override {
Brian Salomonc0b642c2017-03-27 13:09:36 -0400266 color->setToConstant(fPaths[0].fColor);
Brian Salomona811b122017-03-30 08:21:32 -0400267 *coverage = GrProcessorAnalysisCoverage::kNone;
Brian Salomon92aee3d2016-12-21 09:20:25 -0500268 }
269
Brian Salomone7d30482017-03-29 12:09:15 -0400270 void applyPipelineOptimizations(const PipelineOptimizations& optimizations) override {
Brian Salomon92aee3d2016-12-21 09:20:25 -0500271 optimizations.getOverrideColorIfSet(&fPaths[0].fColor);
ethannicholas6536ae52016-05-02 12:16:49 -0700272 }
273
Brian Salomon32ebaba2017-03-30 10:22:28 -0400274 static void ComputeWorstCasePointCount(const SkPath& path, const SkMatrix& m, int* subpaths,
Brian Salomon780dad12016-12-15 18:08:40 -0500275 int* outLinePointCount, int* outQuadPointCount) {
Brian Salomon32ebaba2017-03-30 10:22:28 -0400276 SkScalar tolerance = GrPathUtils::scaleToleranceToSrc(kTolerance, m, path.getBounds());
ethannicholas6536ae52016-05-02 12:16:49 -0700277 int linePointCount = 0;
278 int quadPointCount = 0;
279 *subpaths = 1;
280
281 bool first = true;
282
bsalomon8eb43e52016-09-21 07:47:34 -0700283 SkPath::Iter iter(path, true);
ethannicholas6536ae52016-05-02 12:16:49 -0700284 SkPath::Verb verb;
285
286 SkPoint pts[4];
287 while ((verb = iter.next(pts)) != SkPath::kDone_Verb) {
288 switch (verb) {
289 case SkPath::kLine_Verb:
290 linePointCount += 1;
291 break;
292 case SkPath::kConic_Verb: {
293 SkScalar weight = iter.conicWeight();
294 SkAutoConicToQuads converter;
Brian Salomon32ebaba2017-03-30 10:22:28 -0400295 converter.computeQuads(pts, weight, tolerance);
ethannicholas6536ae52016-05-02 12:16:49 -0700296 int quadPts = converter.countQuads();
297 linePointCount += quadPts;
298 quadPointCount += 3 * quadPts;
299 }
300 case SkPath::kQuad_Verb:
301 linePointCount += 1;
302 quadPointCount += 3;
303 break;
304 case SkPath::kCubic_Verb: {
305 SkSTArray<15, SkPoint, true> quadPts;
Brian Salomon32ebaba2017-03-30 10:22:28 -0400306 GrPathUtils::convertCubicToQuads(pts, tolerance, &quadPts);
ethannicholas6536ae52016-05-02 12:16:49 -0700307 int count = quadPts.count();
308 linePointCount += count / 3;
309 quadPointCount += count;
310 break;
311 }
312 case SkPath::kMove_Verb:
313 linePointCount += 1;
314 if (!first) {
315 ++(*subpaths);
316 }
317 break;
318 default:
319 break;
320 }
321 first = false;
322 }
323 *outLinePointCount = linePointCount;
324 *outQuadPointCount = quadPointCount;
325 }
326
327 void onPrepareDraws(Target* target) const override {
ethannicholas6536ae52016-05-02 12:16:49 -0700328 if (fMaxLineVertices == 0) {
329 SkASSERT(fMaxQuadVertices == 0);
330 return;
331 }
332
bungeman06ca8ec2016-06-09 08:01:03 -0700333 GrPrimitiveType primitiveType = fIsIndexed ? kTriangles_GrPrimitiveType
ethannicholas6536ae52016-05-02 12:16:49 -0700334 : kTriangleFan_GrPrimitiveType;
335
336 // allocate vertex / index buffers
337 const GrBuffer* lineVertexBuffer;
338 int firstLineVertex;
339 MSAALineVertices lines;
340 size_t lineVertexStride = sizeof(MSAALineVertices::Vertex);
bungeman06ca8ec2016-06-09 08:01:03 -0700341 lines.vertices = (MSAALineVertices::Vertex*) target->makeVertexSpace(lineVertexStride,
ethannicholas6536ae52016-05-02 12:16:49 -0700342 fMaxLineVertices,
bungeman06ca8ec2016-06-09 08:01:03 -0700343 &lineVertexBuffer,
ethannicholas6536ae52016-05-02 12:16:49 -0700344 &firstLineVertex);
345 if (!lines.vertices) {
346 SkDebugf("Could not allocate vertices\n");
347 return;
348 }
349 lines.nextVertex = lines.vertices;
350 SkDEBUGCODE(lines.verticesEnd = lines.vertices + fMaxLineVertices;)
351
352 MSAAQuadVertices quads;
353 size_t quadVertexStride = sizeof(MSAAQuadVertices::Vertex);
Hal Canary95e3c052017-01-11 12:44:43 -0500354 SkAutoMalloc quadVertexPtr(fMaxQuadVertices * quadVertexStride);
ethannicholas6536ae52016-05-02 12:16:49 -0700355 quads.vertices = (MSAAQuadVertices::Vertex*) quadVertexPtr.get();
356 quads.nextVertex = quads.vertices;
357 SkDEBUGCODE(quads.verticesEnd = quads.vertices + fMaxQuadVertices;)
358
359 const GrBuffer* lineIndexBuffer = nullptr;
360 int firstLineIndex;
361 if (fIsIndexed) {
Brian Salomon780dad12016-12-15 18:08:40 -0500362 lines.indices =
363 target->makeIndexSpace(3 * fMaxLineVertices, &lineIndexBuffer, &firstLineIndex);
ethannicholas6536ae52016-05-02 12:16:49 -0700364 if (!lines.indices) {
365 SkDebugf("Could not allocate indices\n");
366 return;
367 }
368 lines.nextIndex = lines.indices;
369 } else {
370 lines.indices = nullptr;
371 lines.nextIndex = nullptr;
372 }
373
374 SkAutoFree quadIndexPtr;
375 if (fIsIndexed) {
Brian Salomon780dad12016-12-15 18:08:40 -0500376 quads.indices = (uint16_t*)sk_malloc_throw(3 * fMaxQuadVertices * sizeof(uint16_t));
Hal Canary95e3c052017-01-11 12:44:43 -0500377 quadIndexPtr.reset(quads.indices);
ethannicholas6536ae52016-05-02 12:16:49 -0700378 quads.nextIndex = quads.indices;
379 } else {
380 quads.indices = nullptr;
381 quads.nextIndex = nullptr;
382 }
ethannicholas6536ae52016-05-02 12:16:49 -0700383 // fill buffers
bsalomon50c56a32016-06-30 12:05:32 -0700384 for (int i = 0; i < fPaths.count(); i++) {
385 const PathInfo& pathInfo = fPaths[i];
ethannicholas6536ae52016-05-02 12:16:49 -0700386 if (!this->createGeom(lines,
387 quads,
bsalomon50c56a32016-06-30 12:05:32 -0700388 pathInfo.fPath,
ethannicholas6536ae52016-05-02 12:16:49 -0700389 fViewMatrix,
bsalomon50c56a32016-06-30 12:05:32 -0700390 pathInfo.fColor,
ethannicholas6536ae52016-05-02 12:16:49 -0700391 fIsIndexed)) {
392 return;
393 }
394 }
395 int lineVertexOffset = (int) (lines.nextVertex - lines.vertices);
396 int lineIndexOffset = (int) (lines.nextIndex - lines.indices);
Brian Salomon780dad12016-12-15 18:08:40 -0500397 SkASSERT(lineVertexOffset <= fMaxLineVertices && lineIndexOffset <= 3 * fMaxLineVertices);
ethannicholas6536ae52016-05-02 12:16:49 -0700398 int quadVertexOffset = (int) (quads.nextVertex - quads.vertices);
399 int quadIndexOffset = (int) (quads.nextIndex - quads.indices);
Brian Salomon780dad12016-12-15 18:08:40 -0500400 SkASSERT(quadVertexOffset <= fMaxQuadVertices && quadIndexOffset <= 3 * fMaxQuadVertices);
ethannicholas6536ae52016-05-02 12:16:49 -0700401
402 if (lineVertexOffset) {
bungeman06ca8ec2016-06-09 08:01:03 -0700403 sk_sp<GrGeometryProcessor> lineGP;
ethannicholas6536ae52016-05-02 12:16:49 -0700404 {
405 using namespace GrDefaultGeoProcFactory;
Brian Salomon3de0aee2017-01-29 09:34:17 -0500406 lineGP = GrDefaultGeoProcFactory::Make(Color(Color::kPremulGrColorAttribute_Type),
Brian Salomon8c852be2017-01-04 10:44:42 -0500407 Coverage::kSolid_Type,
bungeman06ca8ec2016-06-09 08:01:03 -0700408 LocalCoords(LocalCoords::kUnused_Type),
409 fViewMatrix);
ethannicholas6536ae52016-05-02 12:16:49 -0700410 }
411 SkASSERT(lineVertexStride == lineGP->getVertexStride());
412
413 GrMesh lineMeshes;
Chris Daltonff926502017-05-03 14:36:54 -0400414 lineMeshes.fPrimitiveType = primitiveType;
ethannicholas6536ae52016-05-02 12:16:49 -0700415 if (fIsIndexed) {
Chris Daltonff926502017-05-03 14:36:54 -0400416 lineMeshes.fIndexBuffer.reset(lineIndexBuffer);
417 lineMeshes.fIndexCount = lineIndexOffset;
418 lineMeshes.fBaseIndex = firstLineIndex;
ethannicholas6536ae52016-05-02 12:16:49 -0700419 }
Chris Daltonff926502017-05-03 14:36:54 -0400420 lineMeshes.fVertexBuffer.reset(lineVertexBuffer);
421 lineMeshes.fVertexCount = lineVertexOffset;
422 lineMeshes.fBaseVertex = firstLineVertex;
423
Brian Salomon2a55c8e2017-05-09 09:57:19 -0400424 // We can get line vertices from path moveTos with no actual segments and thus no index
425 // count. We assert that indexed draws contain a positive index count, so bail here in
426 // that case.
427 if (!fIsIndexed || lineIndexOffset) {
428 target->draw(lineGP.get(), this->pipeline(), lineMeshes);
429 }
ethannicholas6536ae52016-05-02 12:16:49 -0700430 }
431
432 if (quadVertexOffset) {
Hal Canary144caf52016-11-07 17:57:18 -0500433 sk_sp<const GrGeometryProcessor> quadGP(MSAAQuadProcessor::Create(fViewMatrix));
ethannicholas6536ae52016-05-02 12:16:49 -0700434 SkASSERT(quadVertexStride == quadGP->getVertexStride());
435
436 const GrBuffer* quadVertexBuffer;
437 int firstQuadVertex;
bungeman06ca8ec2016-06-09 08:01:03 -0700438 MSAAQuadVertices::Vertex* quadVertices = (MSAAQuadVertices::Vertex*)
ethannicholas6536ae52016-05-02 12:16:49 -0700439 target->makeVertexSpace(quadVertexStride, quadVertexOffset, &quadVertexBuffer,
440 &firstQuadVertex);
441 memcpy(quadVertices, quads.vertices, quadVertexStride * quadVertexOffset);
442 GrMesh quadMeshes;
Chris Daltonff926502017-05-03 14:36:54 -0400443 quadMeshes.fPrimitiveType = kTriangles_GrPrimitiveType;
ethannicholas6536ae52016-05-02 12:16:49 -0700444 if (fIsIndexed) {
445 const GrBuffer* quadIndexBuffer;
bungeman06ca8ec2016-06-09 08:01:03 -0700446 uint16_t* quadIndices = (uint16_t*) target->makeIndexSpace(quadIndexOffset,
447 &quadIndexBuffer,
Chris Daltonff926502017-05-03 14:36:54 -0400448 &quadMeshes.fBaseIndex);
449 quadMeshes.fIndexBuffer.reset(quadIndexBuffer);
450 quadMeshes.fIndexCount = quadIndexOffset;
ethannicholas6536ae52016-05-02 12:16:49 -0700451 memcpy(quadIndices, quads.indices, sizeof(uint16_t) * quadIndexOffset);
ethannicholas6536ae52016-05-02 12:16:49 -0700452 }
Chris Daltonff926502017-05-03 14:36:54 -0400453 quadMeshes.fVertexBuffer.reset(quadVertexBuffer);
454 quadMeshes.fVertexCount = quadVertexOffset;
455 quadMeshes.fBaseVertex = firstQuadVertex;
456
Brian Salomond3ccb0a2017-04-03 10:38:00 -0400457 target->draw(quadGP.get(), this->pipeline(), quadMeshes);
ethannicholas6536ae52016-05-02 12:16:49 -0700458 }
459 }
460
Brian Salomon25a88092016-12-01 09:36:50 -0500461 bool onCombineIfPossible(GrOp* t, const GrCaps& caps) override {
Brian Salomon780dad12016-12-15 18:08:40 -0500462 MSAAPathOp* that = t->cast<MSAAPathOp>();
ethannicholas6536ae52016-05-02 12:16:49 -0700463 if (!GrPipeline::CanCombine(*this->pipeline(), this->bounds(), *that->pipeline(),
Brian Salomon9e50f7b2017-03-06 12:02:34 -0500464 that->bounds(), caps)) {
ethannicholas6536ae52016-05-02 12:16:49 -0700465 return false;
466 }
467
Jim Van Verth9d01fbc2017-02-22 14:50:52 -0500468 if (this->bounds().intersects(that->bounds())) {
469 return false;
470 }
471
ethannicholas6536ae52016-05-02 12:16:49 -0700472 if (!fViewMatrix.cheapEqualTo(that->fViewMatrix)) {
473 return false;
474 }
475
Brian Salomon780dad12016-12-15 18:08:40 -0500476 // If we grow to include 2+ paths we will be indexed.
477 if (((fMaxLineVertices + that->fMaxLineVertices) > kMaxIndexedVertexCnt) ||
478 ((fMaxQuadVertices + that->fMaxQuadVertices) > kMaxIndexedVertexCnt)) {
ethannicholas6536ae52016-05-02 12:16:49 -0700479 return false;
480 }
481
bsalomon50c56a32016-06-30 12:05:32 -0700482 fPaths.push_back_n(that->fPaths.count(), that->fPaths.begin());
bsalomon88cf17d2016-07-08 06:40:56 -0700483 this->joinBounds(*that);
ethannicholas6536ae52016-05-02 12:16:49 -0700484 fIsIndexed = true;
485 fMaxLineVertices += that->fMaxLineVertices;
486 fMaxQuadVertices += that->fMaxQuadVertices;
ethannicholas6536ae52016-05-02 12:16:49 -0700487 return true;
488 }
489
490 bool createGeom(MSAALineVertices& lines,
491 MSAAQuadVertices& quads,
492 const SkPath& path,
ethannicholas6536ae52016-05-02 12:16:49 -0700493 const SkMatrix& m,
494 SkColor color,
495 bool isIndexed) const {
496 {
Brian Salomon32ebaba2017-03-30 10:22:28 -0400497 const SkScalar tolerance = GrPathUtils::scaleToleranceToSrc(kTolerance, m,
498 path.getBounds());
ethannicholas6536ae52016-05-02 12:16:49 -0700499 uint16_t subpathIdxStart = (uint16_t) (lines.nextVertex - lines.vertices);
500
501 SkPoint pts[4];
502
503 bool first = true;
bsalomon8eb43e52016-09-21 07:47:34 -0700504 SkPath::Iter iter(path, true);
ethannicholas6536ae52016-05-02 12:16:49 -0700505
506 bool done = false;
507 while (!done) {
508 SkPath::Verb verb = iter.next(pts);
509 switch (verb) {
510 case SkPath::kMove_Verb:
511 if (!first) {
512 uint16_t currIdx = (uint16_t) (lines.nextVertex - lines.vertices);
513 subpathIdxStart = currIdx;
514 }
515 SkASSERT(lines.nextVertex < lines.verticesEnd);
516 *(lines.nextVertex++) = { pts[0], color };
517 break;
518 case SkPath::kLine_Verb:
519 if (isIndexed) {
520 uint16_t prevIdx = (uint16_t) (lines.nextVertex - lines.vertices - 1);
521 if (prevIdx > subpathIdxStart) {
522 append_contour_edge_indices(subpathIdxStart, prevIdx, lines);
523 }
524 }
525 SkASSERT(lines.nextVertex < lines.verticesEnd);
526 *(lines.nextVertex++) = { pts[1], color };
527 break;
528 case SkPath::kConic_Verb: {
529 SkScalar weight = iter.conicWeight();
530 SkAutoConicToQuads converter;
Brian Salomon32ebaba2017-03-30 10:22:28 -0400531 const SkPoint* quadPts = converter.computeQuads(pts, weight, tolerance);
ethannicholas6536ae52016-05-02 12:16:49 -0700532 for (int i = 0; i < converter.countQuads(); ++i) {
bungeman06ca8ec2016-06-09 08:01:03 -0700533 add_quad(lines, quads, quadPts + i * 2, color, isIndexed,
ethannicholas6536ae52016-05-02 12:16:49 -0700534 subpathIdxStart);
535 }
536 break;
537 }
538 case SkPath::kQuad_Verb: {
539 add_quad(lines, quads, pts, color, isIndexed, subpathIdxStart);
bungeman06ca8ec2016-06-09 08:01:03 -0700540 break;
ethannicholas6536ae52016-05-02 12:16:49 -0700541 }
542 case SkPath::kCubic_Verb: {
543 SkSTArray<15, SkPoint, true> quadPts;
Brian Salomon32ebaba2017-03-30 10:22:28 -0400544 GrPathUtils::convertCubicToQuads(pts, tolerance, &quadPts);
ethannicholas6536ae52016-05-02 12:16:49 -0700545 int count = quadPts.count();
546 for (int i = 0; i < count; i += 3) {
547 add_quad(lines, quads, &quadPts[i], color, isIndexed, subpathIdxStart);
548 }
549 break;
550 }
551 case SkPath::kClose_Verb:
552 break;
553 case SkPath::kDone_Verb:
554 done = true;
555 }
556 first = false;
557 }
558 }
559 return true;
560 }
561
Brian Salomon780dad12016-12-15 18:08:40 -0500562 // Lines and quads may render with an index buffer. However, we don't have any support for
563 // overflowing the max index.
564 static constexpr int kMaxIndexedVertexCnt = SK_MaxU16 / 3;
bsalomon50c56a32016-06-30 12:05:32 -0700565 struct PathInfo {
566 GrColor fColor;
567 SkPath fPath;
568 };
569
570 SkSTArray<1, PathInfo, true> fPaths;
ethannicholas6536ae52016-05-02 12:16:49 -0700571
572 SkMatrix fViewMatrix;
573 int fMaxLineVertices;
574 int fMaxQuadVertices;
ethannicholas6536ae52016-05-02 12:16:49 -0700575 bool fIsIndexed;
576
Brian Salomond3ccb0a2017-04-03 10:38:00 -0400577 typedef GrLegacyMeshDrawOp INHERITED;
ethannicholas6536ae52016-05-02 12:16:49 -0700578};
579
Brian Osman11052242016-10-27 14:47:55 -0400580bool GrMSAAPathRenderer::internalDrawPath(GrRenderTargetContext* renderTargetContext,
Brian Salomon82f44312017-01-11 13:42:54 -0500581 GrPaint&& paint,
Brian Salomon0e8fc8b2016-12-09 15:10:07 -0500582 GrAAType aaType,
robertphillipsd2b6d642016-07-21 08:55:08 -0700583 const GrUserStencilSettings& userStencilSettings,
cdalton862cff32016-05-12 15:09:48 -0700584 const GrClip& clip,
ethannicholas6536ae52016-05-02 12:16:49 -0700585 const SkMatrix& viewMatrix,
bsalomon8acedde2016-06-24 10:42:16 -0700586 const GrShape& shape,
ethannicholas6536ae52016-05-02 12:16:49 -0700587 bool stencilOnly) {
bsalomon8acedde2016-06-24 10:42:16 -0700588 SkASSERT(shape.style().isSimpleFill());
589 SkPath path;
590 shape.asPath(&path);
591
Brian Salomon82f44312017-01-11 13:42:54 -0500592 const GrUserStencilSettings* passes[2] = {nullptr, nullptr};
cdalton93a379b2016-05-11 13:58:08 -0700593 bool reverse = false;
ethannicholas6536ae52016-05-02 12:16:49 -0700594
bsalomon8acedde2016-06-24 10:42:16 -0700595 if (single_pass_shape(shape)) {
ethannicholas6536ae52016-05-02 12:16:49 -0700596 if (stencilOnly) {
597 passes[0] = &gDirectToStencil;
598 } else {
robertphillipsd2b6d642016-07-21 08:55:08 -0700599 passes[0] = &userStencilSettings;
ethannicholas6536ae52016-05-02 12:16:49 -0700600 }
ethannicholas6536ae52016-05-02 12:16:49 -0700601 } else {
602 switch (path.getFillType()) {
603 case SkPath::kInverseEvenOdd_FillType:
604 reverse = true;
605 // fallthrough
606 case SkPath::kEvenOdd_FillType:
607 passes[0] = &gEOStencilPass;
Brian Salomon82f44312017-01-11 13:42:54 -0500608 if (!stencilOnly) {
609 passes[1] = reverse ? &gInvEOColorPass : &gEOColorPass;
ethannicholas6536ae52016-05-02 12:16:49 -0700610 }
ethannicholas6536ae52016-05-02 12:16:49 -0700611 break;
612
613 case SkPath::kInverseWinding_FillType:
614 reverse = true;
615 // fallthrough
616 case SkPath::kWinding_FillType:
Brian Salomon15b25092017-05-08 11:10:53 -0400617 passes[0] = &gWindStencilPass;
Brian Salomon82f44312017-01-11 13:42:54 -0500618 if (!stencilOnly) {
619 passes[1] = reverse ? &gInvWindColorPass : &gWindColorPass;
ethannicholas6536ae52016-05-02 12:16:49 -0700620 }
621 break;
622 default:
623 SkDEBUGFAIL("Unknown path fFill!");
624 return false;
625 }
626 }
627
628 SkRect devBounds;
Brian Osman11052242016-10-27 14:47:55 -0400629 GetPathDevBounds(path, renderTargetContext->width(), renderTargetContext->height(), viewMatrix,
630 &devBounds);
ethannicholas6536ae52016-05-02 12:16:49 -0700631
Brian Salomond4652ca2017-01-13 12:11:36 -0500632 SkASSERT(passes[0]);
633 { // First pass
Brian Salomond3ccb0a2017-04-03 10:38:00 -0400634 std::unique_ptr<GrLegacyMeshDrawOp> op =
Brian Salomond4652ca2017-01-13 12:11:36 -0500635 MSAAPathOp::Make(paint.getColor(), path, viewMatrix, devBounds);
636 if (!op) {
637 return false;
638 }
639 bool firstPassIsStencil = stencilOnly || passes[1];
640 // If we have a cover pass then we ignore the paint in the first pass and apply it in the
641 // second.
642 GrPaint::MoveOrNew firstPassPaint(paint, firstPassIsStencil);
643 if (firstPassIsStencil) {
644 firstPassPaint.paint().setXPFactory(GrDisableColorXPFactory::Get());
645 }
646 GrPipelineBuilder pipelineBuilder(std::move(firstPassPaint), aaType);
647 pipelineBuilder.setUserStencil(passes[0]);
Brian Salomone14bd802017-04-04 15:13:25 -0400648 renderTargetContext->addLegacyMeshDrawOp(std::move(pipelineBuilder), clip, std::move(op));
Brian Salomon82f44312017-01-11 13:42:54 -0500649 }
robertphillips8e375302016-07-11 10:43:58 -0700650
Brian Salomon82f44312017-01-11 13:42:54 -0500651 if (passes[1]) {
652 SkRect bounds;
653 SkMatrix localMatrix = SkMatrix::I();
654 if (reverse) {
655 // draw over the dev bounds (which will be the whole dst surface for inv fill).
656 bounds = devBounds;
657 SkMatrix vmi;
658 // mapRect through persp matrix may not be correct
659 if (!viewMatrix.hasPerspective() && viewMatrix.invert(&vmi)) {
660 vmi.mapRect(&bounds);
ethannicholas6536ae52016-05-02 12:16:49 -0700661 } else {
Brian Salomon82f44312017-01-11 13:42:54 -0500662 if (!viewMatrix.invert(&localMatrix)) {
663 return false;
664 }
ethannicholas6536ae52016-05-02 12:16:49 -0700665 }
robertphillips976f5f02016-06-03 10:59:20 -0700666 } else {
Brian Salomon82f44312017-01-11 13:42:54 -0500667 bounds = path.getBounds();
ethannicholas6536ae52016-05-02 12:16:49 -0700668 }
Brian Salomon82f44312017-01-11 13:42:54 -0500669 const SkMatrix& viewM =
670 (reverse && viewMatrix.hasPerspective()) ? SkMatrix::I() : viewMatrix;
Brian Salomonac70f842017-05-08 10:43:33 -0400671 renderTargetContext->addDrawOp(
672 clip,
673 GrNonAAFillRectOp::Make(std::move(paint), viewM, bounds, nullptr, &localMatrix,
674 aaType, passes[1]));
ethannicholas6536ae52016-05-02 12:16:49 -0700675 }
676 return true;
677}
678
679bool GrMSAAPathRenderer::onCanDrawPath(const CanDrawPathArgs& args) const {
Eric Karl5c779752017-05-08 12:02:07 -0700680 // If we aren't a single_pass_shape, we require stencil buffers.
681 if (!single_pass_shape(*args.fShape) && args.fCaps->avoidStencilBuffers()) {
682 return false;
683 }
bsalomonee432412016-06-27 07:18:18 -0700684 // This path renderer only fills and relies on MSAA for antialiasing. Stroked shapes are
685 // handled by passing on the original shape and letting the caller compute the stroked shape
686 // which will have a fill style.
Brian Salomon0e8fc8b2016-12-09 15:10:07 -0500687 return args.fShape->style().isSimpleFill() && (GrAAType::kCoverage != args.fAAType);
ethannicholas6536ae52016-05-02 12:16:49 -0700688}
689
690bool GrMSAAPathRenderer::onDrawPath(const DrawPathArgs& args) {
Brian Osman11052242016-10-27 14:47:55 -0400691 GR_AUDIT_TRAIL_AUTO_FRAME(args.fRenderTargetContext->auditTrail(),
robertphillips976f5f02016-06-03 10:59:20 -0700692 "GrMSAAPathRenderer::onDrawPath");
bsalomon8acedde2016-06-24 10:42:16 -0700693 SkTLazy<GrShape> tmpShape;
694 const GrShape* shape = args.fShape;
695 if (shape->style().applies()) {
bsalomon6663acf2016-05-10 09:14:17 -0700696 SkScalar styleScale = GrStyle::MatrixToScaleFactor(*args.fViewMatrix);
bsalomon8acedde2016-06-24 10:42:16 -0700697 tmpShape.init(args.fShape->applyStyle(GrStyle::Apply::kPathEffectAndStrokeRec, styleScale));
698 shape = tmpShape.get();
ethannicholas6536ae52016-05-02 12:16:49 -0700699 }
Brian Osman11052242016-10-27 14:47:55 -0400700 return this->internalDrawPath(args.fRenderTargetContext,
Brian Salomon82f44312017-01-11 13:42:54 -0500701 std::move(args.fPaint),
Brian Salomon0e8fc8b2016-12-09 15:10:07 -0500702 args.fAAType,
robertphillipsd2b6d642016-07-21 08:55:08 -0700703 *args.fUserStencilSettings,
cdalton862cff32016-05-12 15:09:48 -0700704 *args.fClip,
ethannicholas6536ae52016-05-02 12:16:49 -0700705 *args.fViewMatrix,
bsalomon8acedde2016-06-24 10:42:16 -0700706 *shape,
ethannicholas6536ae52016-05-02 12:16:49 -0700707 false);
708}
709
710void GrMSAAPathRenderer::onStencilPath(const StencilPathArgs& args) {
Brian Osman11052242016-10-27 14:47:55 -0400711 GR_AUDIT_TRAIL_AUTO_FRAME(args.fRenderTargetContext->auditTrail(),
robertphillips976f5f02016-06-03 10:59:20 -0700712 "GrMSAAPathRenderer::onStencilPath");
bsalomon8acedde2016-06-24 10:42:16 -0700713 SkASSERT(args.fShape->style().isSimpleFill());
714 SkASSERT(!args.fShape->mayBeInverseFilledAfterStyling());
robertphillips976f5f02016-06-03 10:59:20 -0700715
716 GrPaint paint;
Brian Salomona1633922017-01-09 11:46:10 -0500717 paint.setXPFactory(GrDisableColorXPFactory::Get());
robertphillips976f5f02016-06-03 10:59:20 -0700718
Brian Salomon82f44312017-01-11 13:42:54 -0500719 this->internalDrawPath(args.fRenderTargetContext, std::move(paint), args.fAAType,
Brian Salomon0e8fc8b2016-12-09 15:10:07 -0500720 GrUserStencilSettings::kUnused, *args.fClip, *args.fViewMatrix,
721 *args.fShape, true);
ethannicholas6536ae52016-05-02 12:16:49 -0700722}
723
724///////////////////////////////////////////////////////////////////////////////////////////////////