/* * Copyright 2017 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #ifndef GrCCPathProcessor_DEFINED #define GrCCPathProcessor_DEFINED #include "GrCaps.h" #include "GrGeometryProcessor.h" #include "SkPath.h" #include <array> class GrOnFlushResourceProvider; /** * This class draws AA paths using the coverage count masks produced by GrCCCoverageProcessor. * * Paths are drawn as bloated octagons, and coverage is derived from the coverage count mask and * fill rule. * * The caller must set up an instance buffer as detailed below, then draw indexed-instanced * meshes using the buffers and parameters provided by this class. */ class GrCCPathProcessor : public GrGeometryProcessor { public: enum class InstanceAttribs { kDevBounds, kDevBounds45, kViewMatrix, // FIXME: This causes a lot of duplication. It could move to a texel buffer. kViewTranslate, kAtlasOffset, kColor }; static constexpr int kNumInstanceAttribs = 1 + (int)InstanceAttribs::kColor; struct Instance { SkRect fDevBounds; SkRect fDevBounds45; // Bounding box in "| 1 -1 | * devCoords" space. // | 1 1 | std::array<float, 4> fViewMatrix; // {kScaleX, kSkewy, kSkewX, kScaleY} std::array<float, 2> fViewTranslate; std::array<int16_t, 2> fAtlasOffset; uint32_t fColor; GR_STATIC_ASSERT(SK_SCALAR_IS_FLOAT); }; GR_STATIC_ASSERT(4 * 16 == sizeof(Instance)); static GrPrimitiveType MeshPrimitiveType(const GrCaps& caps) { return caps.usePrimitiveRestart() ? GrPrimitiveType::kTriangleStrip : GrPrimitiveType::kTriangles; } static sk_sp<const GrBuffer> FindVertexBuffer(GrOnFlushResourceProvider*); static sk_sp<const GrBuffer> FindIndexBuffer(GrOnFlushResourceProvider*); static int NumIndicesPerInstance(const GrCaps&); GrCCPathProcessor(GrResourceProvider*, sk_sp<GrTextureProxy> atlas, SkPath::FillType); const char* name() const override { return "GrCCPathProcessor"; } const GrSurfaceProxy* atlasProxy() const { return fAtlasAccess.proxy(); } const GrTexture* atlas() const { return fAtlasAccess.peekTexture(); } SkPath::FillType fillType() const { return fFillType; } const Attribute& getInstanceAttrib(InstanceAttribs attribID) const { const Attribute& attrib = this->getAttrib((int)attribID); SkASSERT(Attribute::InputRate::kPerInstance == attrib.fInputRate); return attrib; } const Attribute& getEdgeNormsAttrib() const { SkASSERT(1 + kNumInstanceAttribs == this->numAttribs()); const Attribute& attrib = this->getAttrib(kNumInstanceAttribs); SkASSERT(Attribute::InputRate::kPerVertex == attrib.fInputRate); return attrib; } void getGLSLProcessorKey(const GrShaderCaps&, GrProcessorKeyBuilder*) const override; GrGLSLPrimitiveProcessor* createGLSLInstance(const GrShaderCaps&) const override; private: const SkPath::FillType fFillType; const TextureSampler fAtlasAccess; typedef GrGeometryProcessor INHERITED; }; #endif