blob: ac97fd848a4751444eb21daf224d6119c76d56de [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
Brian Salomon49348902018-06-26 09:12:38 -040011#include <array>
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/core/SkPath.h"
13#include "src/gpu/GrCaps.h"
14#include "src/gpu/GrGeometryProcessor.h"
15#include "src/gpu/GrPipeline.h"
Chris Dalton6a5317a2019-07-12 09:55:52 -060016#include "src/gpu/ccpr/GrCCAtlas.h"
Chris Dalton8610e9c2019-05-09 11:07:10 -060017#include "src/gpu/ccpr/GrOctoBounds.h"
Chris Dalton1a325d22017-07-14 15:17:41 -060018
Chris Dalton4da70192018-06-18 09:51:36 -060019class GrCCPathCacheEntry;
20class GrCCPerFlushResources;
Chris Dalton1a325d22017-07-14 15:17:41 -060021class GrOnFlushResourceProvider;
Chris Daltond925f2d2018-05-07 19:19:06 -060022class GrOpFlushState;
Chris Dalton1a325d22017-07-14 15:17:41 -060023
24/**
Chris Dalton383a2ef2018-01-08 17:21:41 -050025 * This class draws AA paths using the coverage count masks produced by GrCCCoverageProcessor.
Chris Dalton1a325d22017-07-14 15:17:41 -060026 *
27 * Paths are drawn as bloated octagons, and coverage is derived from the coverage count mask and
28 * fill rule.
29 *
Chris Daltond925f2d2018-05-07 19:19:06 -060030 * To draw paths, the caller must set up an instance buffer as detailed below, then call drawPaths()
31 * providing its own instance buffer alongside the buffers found by calling FindIndexBuffer/
32 * FindVertexBuffer.
Chris Dalton1a325d22017-07-14 15:17:41 -060033 */
Chris Dalton383a2ef2018-01-08 17:21:41 -050034class GrCCPathProcessor : public GrGeometryProcessor {
Chris Dalton1a325d22017-07-14 15:17:41 -060035public:
Chris Dalton4da70192018-06-18 09:51:36 -060036 enum class DoEvenOddFill : bool {
37 kNo = false,
38 kYes = true
39 };
40
Chris Dalton1a325d22017-07-14 15:17:41 -060041 struct Instance {
Chris Dalton9414c962018-06-14 10:14:50 -060042 SkRect fDevBounds; // "right < left" indicates even-odd fill type.
Chris Dalton8610e9c2019-05-09 11:07:10 -060043 SkRect fDevBounds45; // Bounding box in "| 1 -1 | * devCoords" space. See GrOctoBounds.
Chris Dalton9414c962018-06-14 10:14:50 -060044 // | 1 1 |
45 SkIVector fDevToAtlasOffset; // Translation from device space to location in atlas.
Brian Osmanc6444d22019-01-09 16:30:12 -050046 uint64_t fColor; // Color always stored as 4 x fp16
Chris Dalton1a325d22017-07-14 15:17:41 -060047
Chris Dalton5b5403e2019-06-05 11:54:39 -060048 void set(const GrOctoBounds&, const SkIVector& devToAtlasOffset, uint64_t, DoEvenOddFill);
49 void set(const GrCCPathCacheEntry&, const SkIVector& shift, uint64_t, DoEvenOddFill);
Chris Dalton1a325d22017-07-14 15:17:41 -060050 };
51
Brian Osmanc6444d22019-01-09 16:30:12 -050052 GR_STATIC_ASSERT(4 * 12 == sizeof(Instance));
Chris Dalton1a325d22017-07-14 15:17:41 -060053
Brian Salomondbf70722019-02-07 11:31:24 -050054 static sk_sp<const GrGpuBuffer> FindVertexBuffer(GrOnFlushResourceProvider*);
55 static sk_sp<const GrGpuBuffer> FindIndexBuffer(GrOnFlushResourceProvider*);
Chris Dalton5d2de082017-12-19 10:40:23 -070056
Chris Dalton6a5317a2019-07-12 09:55:52 -060057 enum class CoverageMode : bool {
58 kCoverageCount,
59 kLiteral
60 };
61
62 static CoverageMode GetCoverageMode(GrCCAtlas::CoverageType coverageType) {
63 return (GrCCAtlas::CoverageType::kFP16_CoverageCount == coverageType)
64 ? CoverageMode::kCoverageCount
65 : CoverageMode::kLiteral;
66 }
67
68 GrCCPathProcessor(
69 CoverageMode, const GrTexture* atlasTexture, const GrSwizzle&,
70 GrSurfaceOrigin atlasOrigin,
71 const SkMatrix& viewMatrixIfUsingLocalCoords = SkMatrix::I());
Chris Dalton1a325d22017-07-14 15:17:41 -060072
Chris Dalton383a2ef2018-01-08 17:21:41 -050073 const char* name() const override { return "GrCCPathProcessor"; }
Chris Dalton6a5317a2019-07-12 09:55:52 -060074 void getGLSLProcessorKey(const GrShaderCaps&, GrProcessorKeyBuilder* b) const override {
75 b->add32((uint32_t)fCoverageMode);
76 }
Chris Dalton1a325d22017-07-14 15:17:41 -060077 GrGLSLPrimitiveProcessor* createGLSLInstance(const GrShaderCaps&) const override;
78
Brian Salomon49348902018-06-26 09:12:38 -040079 void drawPaths(GrOpFlushState*, const GrPipeline&, const GrPipeline::FixedDynamicState*,
80 const GrCCPerFlushResources&, int baseInstance, int endInstance,
81 const SkRect& bounds) const;
Chris Daltond925f2d2018-05-07 19:19:06 -060082
Chris Dalton1a325d22017-07-14 15:17:41 -060083private:
Brian Salomonf7dcd762018-07-30 14:48:15 -040084 const TextureSampler& onTextureSampler(int) const override { return fAtlasAccess; }
Brian Salomon92be2f72018-06-19 14:33:47 -040085
Chris Dalton6a5317a2019-07-12 09:55:52 -060086 const CoverageMode fCoverageMode;
Chris Daltona3e92712017-12-04 11:45:51 -070087 const TextureSampler fAtlasAccess;
Brian Salomon7eae3e02018-08-07 14:02:38 +000088 SkISize fAtlasSize;
89 GrSurfaceOrigin fAtlasOrigin;
90
Chris Dalton1c548942018-05-22 13:09:48 -060091 SkMatrix fLocalMatrix;
Chris Dalton377b18b2019-04-18 13:33:50 -060092 static constexpr Attribute kInstanceAttribs[] = {
Brian Osmand4c29702018-09-14 16:16:55 -040093 {"devbounds", kFloat4_GrVertexAttribType, kFloat4_GrSLType},
94 {"devbounds45", kFloat4_GrVertexAttribType, kFloat4_GrSLType},
95 {"dev_to_atlas_offset", kInt2_GrVertexAttribType, kInt2_GrSLType},
Brian Osmanc6444d22019-01-09 16:30:12 -050096 {"color", kHalf4_GrVertexAttribType, kHalf4_GrSLType}
Brian Salomon92be2f72018-06-19 14:33:47 -040097 };
Chris Dalton377b18b2019-04-18 13:33:50 -060098 static constexpr int kColorAttribIdx = 3;
99 static constexpr Attribute kCornersAttrib =
100 {"corners", kFloat4_GrVertexAttribType, kFloat4_GrSLType};
101
102 class Impl;
Chris Dalton1a325d22017-07-14 15:17:41 -0600103
104 typedef GrGeometryProcessor INHERITED;
105};
106
Chris Dalton377b18b2019-04-18 13:33:50 -0600107inline void GrCCPathProcessor::Instance::set(
Chris Dalton8610e9c2019-05-09 11:07:10 -0600108 const GrOctoBounds& octoBounds, const SkIVector& devToAtlasOffset, uint64_t color,
109 DoEvenOddFill doEvenOddFill) {
Chris Dalton377b18b2019-04-18 13:33:50 -0600110 if (DoEvenOddFill::kNo == doEvenOddFill) {
111 // We cover "nonzero" paths with clockwise triangles, which is the default result from
Chris Dalton8610e9c2019-05-09 11:07:10 -0600112 // normal octo bounds.
113 fDevBounds = octoBounds.bounds();
114 fDevBounds45 = octoBounds.bounds45();
Chris Dalton377b18b2019-04-18 13:33:50 -0600115 } else {
Chris Dalton8610e9c2019-05-09 11:07:10 -0600116 // We cover "even/odd" paths with counterclockwise triangles. Here we reorder the bounding
117 // box vertices so the output is flipped horizontally.
118 fDevBounds.setLTRB(
119 octoBounds.right(), octoBounds.top(), octoBounds.left(), octoBounds.bottom());
Chris Dalton377b18b2019-04-18 13:33:50 -0600120 fDevBounds45.setLTRB(
Chris Dalton8610e9c2019-05-09 11:07:10 -0600121 octoBounds.bottom45(), octoBounds.right45(), octoBounds.top45(),
122 octoBounds.left45());
Chris Daltondaef06a2018-05-23 17:11:09 -0600123 }
Chris Dalton9414c962018-06-14 10:14:50 -0600124 fDevToAtlasOffset = devToAtlasOffset;
Chris Daltondaef06a2018-05-23 17:11:09 -0600125 fColor = color;
126}
127
Chris Dalton1a325d22017-07-14 15:17:41 -0600128#endif