blob: 785dd2c919ac6d3aa898595ae0c053c5d6100b83 [file] [log] [blame]
Chris Dalton1a325d22017-07-14 15:17:41 -06001/*
2 * Copyright 2017 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
Chris Dalton383a2ef2018-01-08 17:21:41 -05008#ifndef GrCCPathProcessor_DEFINED
9#define GrCCPathProcessor_DEFINED
Chris Dalton1a325d22017-07-14 15:17:41 -060010
Chris Dalton27059d32018-01-23 14:06:50 -070011#include "GrCaps.h"
Chris Dalton1a325d22017-07-14 15:17:41 -060012#include "GrGeometryProcessor.h"
13#include "SkPath.h"
14#include <array>
15
Chris Dalton4da70192018-06-18 09:51:36 -060016class GrCCPathCacheEntry;
17class GrCCPerFlushResources;
Chris Dalton1a325d22017-07-14 15:17:41 -060018class GrOnFlushResourceProvider;
Chris Daltond925f2d2018-05-07 19:19:06 -060019class GrOpFlushState;
20class GrPipeline;
Chris Dalton1a325d22017-07-14 15:17:41 -060021
22/**
Chris Dalton383a2ef2018-01-08 17:21:41 -050023 * This class draws AA paths using the coverage count masks produced by GrCCCoverageProcessor.
Chris Dalton1a325d22017-07-14 15:17:41 -060024 *
25 * Paths are drawn as bloated octagons, and coverage is derived from the coverage count mask and
26 * fill rule.
27 *
Chris Daltond925f2d2018-05-07 19:19:06 -060028 * To draw paths, the caller must set up an instance buffer as detailed below, then call drawPaths()
29 * providing its own instance buffer alongside the buffers found by calling FindIndexBuffer/
30 * FindVertexBuffer.
Chris Dalton1a325d22017-07-14 15:17:41 -060031 */
Chris Dalton383a2ef2018-01-08 17:21:41 -050032class GrCCPathProcessor : public GrGeometryProcessor {
Chris Dalton1a325d22017-07-14 15:17:41 -060033public:
Chris Dalton1a325d22017-07-14 15:17:41 -060034 enum class InstanceAttribs {
35 kDevBounds,
36 kDevBounds45,
Chris Dalton9414c962018-06-14 10:14:50 -060037 kDevToAtlasOffset,
Chris Dalton1a325d22017-07-14 15:17:41 -060038 kColor
39 };
40 static constexpr int kNumInstanceAttribs = 1 + (int)InstanceAttribs::kColor;
41
Chris Dalton4da70192018-06-18 09:51:36 -060042 // Helper to offset the 45-degree bounding box returned by GrCCPathParser::parsePath().
43 static SkRect MakeOffset45(const SkRect& devBounds45, float dx, float dy) {
44 // devBounds45 is in "| 1 -1 | * devCoords" space.
45 // | 1 1 |
46 return devBounds45.makeOffset(dx - dy, dx + dy);
47 }
48
49 enum class DoEvenOddFill : bool {
50 kNo = false,
51 kYes = true
52 };
53
Chris Dalton1a325d22017-07-14 15:17:41 -060054 struct Instance {
Chris Dalton9414c962018-06-14 10:14:50 -060055 SkRect fDevBounds; // "right < left" indicates even-odd fill type.
56 SkRect fDevBounds45; // Bounding box in "| 1 -1 | * devCoords" space.
57 // | 1 1 |
58 SkIVector fDevToAtlasOffset; // Translation from device space to location in atlas.
Chris Dalton4da70192018-06-18 09:51:36 -060059 GrColor fColor;
Chris Dalton1a325d22017-07-14 15:17:41 -060060
Chris Dalton4da70192018-06-18 09:51:36 -060061 void set(const SkRect& devBounds, const SkRect& devBounds45,
62 const SkIVector& devToAtlasOffset, GrColor, DoEvenOddFill = DoEvenOddFill::kNo);
63 void set(const GrCCPathCacheEntry&, const SkIVector& shift, GrColor,
64 DoEvenOddFill = DoEvenOddFill::kNo);
Chris Dalton1a325d22017-07-14 15:17:41 -060065 };
66
Chris Dalton9414c962018-06-14 10:14:50 -060067 GR_STATIC_ASSERT(4 * 11 == sizeof(Instance));
Chris Dalton1a325d22017-07-14 15:17:41 -060068
Chris Dalton5d2de082017-12-19 10:40:23 -070069 static sk_sp<const GrBuffer> FindVertexBuffer(GrOnFlushResourceProvider*);
Chris Dalton27059d32018-01-23 14:06:50 -070070 static sk_sp<const GrBuffer> FindIndexBuffer(GrOnFlushResourceProvider*);
Chris Dalton5d2de082017-12-19 10:40:23 -070071
Chris Daltondaef06a2018-05-23 17:11:09 -060072 GrCCPathProcessor(GrResourceProvider*, sk_sp<GrTextureProxy> atlas,
Chris Dalton1c548942018-05-22 13:09:48 -060073 const SkMatrix& viewMatrixIfUsingLocalCoords = SkMatrix::I());
Chris Dalton1a325d22017-07-14 15:17:41 -060074
Chris Dalton383a2ef2018-01-08 17:21:41 -050075 const char* name() const override { return "GrCCPathProcessor"; }
Robert Phillipse44ef102017-07-21 15:37:19 -040076 const GrSurfaceProxy* atlasProxy() const { return fAtlasAccess.proxy(); }
Chris Dalton1a325d22017-07-14 15:17:41 -060077 const GrTexture* atlas() const { return fAtlasAccess.peekTexture(); }
Chris Dalton1c548942018-05-22 13:09:48 -060078 const SkMatrix& localMatrix() const { return fLocalMatrix; }
Chris Dalton1a325d22017-07-14 15:17:41 -060079 const Attribute& getInstanceAttrib(InstanceAttribs attribID) const {
Mike Klein5045e502018-06-19 01:40:57 +000080 const Attribute& attrib = this->getAttrib((int)attribID);
81 SkASSERT(Attribute::InputRate::kPerInstance == attrib.inputRate());
82 return attrib;
Chris Dalton1a325d22017-07-14 15:17:41 -060083 }
Mike Klein5045e502018-06-19 01:40:57 +000084 const Attribute& getEdgeNormsAttrib() const {
85 SkASSERT(1 + kNumInstanceAttribs == this->numAttribs());
86 const Attribute& attrib = this->getAttrib(kNumInstanceAttribs);
87 SkASSERT(Attribute::InputRate::kPerVertex == attrib.inputRate());
88 return attrib;
89 }
Chris Dalton1a325d22017-07-14 15:17:41 -060090
Chris Daltondaef06a2018-05-23 17:11:09 -060091 void getGLSLProcessorKey(const GrShaderCaps&, GrProcessorKeyBuilder*) const override {}
Chris Dalton1a325d22017-07-14 15:17:41 -060092 GrGLSLPrimitiveProcessor* createGLSLInstance(const GrShaderCaps&) const override;
93
Chris Dalton4da70192018-06-18 09:51:36 -060094 void drawPaths(GrOpFlushState*, const GrPipeline&, const GrCCPerFlushResources&,
95 int baseInstance, int endInstance, const SkRect& bounds) const;
Chris Daltond925f2d2018-05-07 19:19:06 -060096
Chris Dalton1a325d22017-07-14 15:17:41 -060097private:
Chris Daltona3e92712017-12-04 11:45:51 -070098 const TextureSampler fAtlasAccess;
Chris Dalton1c548942018-05-22 13:09:48 -060099 SkMatrix fLocalMatrix;
Chris Dalton1a325d22017-07-14 15:17:41 -0600100
101 typedef GrGeometryProcessor INHERITED;
102};
103
Chris Dalton4da70192018-06-18 09:51:36 -0600104inline void GrCCPathProcessor::Instance::set(const SkRect& devBounds, const SkRect& devBounds45,
105 const SkIVector& devToAtlasOffset, GrColor color,
106 DoEvenOddFill doEvenOddFill) {
107 if (DoEvenOddFill::kYes == doEvenOddFill) {
Chris Daltondaef06a2018-05-23 17:11:09 -0600108 // "right < left" indicates even-odd fill type.
109 fDevBounds.setLTRB(devBounds.fRight, devBounds.fTop, devBounds.fLeft, devBounds.fBottom);
110 } else {
Chris Daltondaef06a2018-05-23 17:11:09 -0600111 fDevBounds = devBounds;
112 }
113 fDevBounds45 = devBounds45;
Chris Dalton9414c962018-06-14 10:14:50 -0600114 fDevToAtlasOffset = devToAtlasOffset;
Chris Daltondaef06a2018-05-23 17:11:09 -0600115 fColor = color;
116}
117
Chris Dalton1a325d22017-07-14 15:17:41 -0600118#endif