blob: 0a1a7a1ae92265787eebb57bdf01ec1b6b4f8f3c [file] [log] [blame]
joshualitta751c972015-11-20 13:37:32 -08001/*
2 * Copyright 2015 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
Brian Salomon344ec422016-12-15 10:58:41 -05008#ifndef GrAtlasTextOp_DEFINED
9#define GrAtlasTextOp_DEFINED
joshualitta751c972015-11-20 13:37:32 -080010
Brian Salomon89527432016-12-16 09:52:16 -050011#include "ops/GrMeshDrawOp.h"
joshualitte8042922015-12-11 06:11:21 -080012#include "text/GrAtlasTextContext.h"
13#include "text/GrDistanceFieldAdjustTable.h"
joshualitta751c972015-11-20 13:37:32 -080014
Brian Salomon44acb5b2017-07-18 19:59:24 -040015class GrAtlasTextOp final : public GrMeshDrawOp {
joshualitta751c972015-11-20 13:37:32 -080016public:
Brian Salomon25a88092016-12-01 09:36:50 -050017 DEFINE_OP_CLASS_ID
joshualitta751c972015-11-20 13:37:32 -080018
Brian Salomonf8334782017-01-03 09:42:58 -050019 ~GrAtlasTextOp() override {
20 for (int i = 0; i < fGeoCount; i++) {
21 fGeoData[i].fBlob->unref();
22 }
23 }
24
joshualitt3660d532015-12-07 11:32:50 -080025 static const int kVerticesPerGlyph = GrAtlasTextBlob::kVerticesPerGlyph;
joshualitta751c972015-11-20 13:37:32 -080026 static const int kIndicesPerGlyph = 6;
27
joshualitta751c972015-11-20 13:37:32 -080028 typedef GrAtlasTextBlob Blob;
joshualitta751c972015-11-20 13:37:32 -080029 struct Geometry {
joshualitt8e0ef292016-02-19 14:13:03 -080030 SkMatrix fViewMatrix;
joshualitta751c972015-11-20 13:37:32 -080031 Blob* fBlob;
joshualitt8e0ef292016-02-19 14:13:03 -080032 SkScalar fX;
33 SkScalar fY;
joshualitta751c972015-11-20 13:37:32 -080034 int fRun;
35 int fSubRun;
36 GrColor fColor;
joshualitta751c972015-11-20 13:37:32 -080037 };
38
Brian Salomon44acb5b2017-07-18 19:59:24 -040039 static std::unique_ptr<GrAtlasTextOp> MakeBitmap(GrPaint&& paint, GrMaskFormat maskFormat,
40 int glyphCount, GrAtlasGlyphCache* fontCache) {
41 std::unique_ptr<GrAtlasTextOp> op(new GrAtlasTextOp(std::move(paint)));
joshualitta751c972015-11-20 13:37:32 -080042
Brian Salomon344ec422016-12-15 10:58:41 -050043 op->fFontCache = fontCache;
joshualitta751c972015-11-20 13:37:32 -080044 switch (maskFormat) {
45 case kA8_GrMaskFormat:
Brian Salomon344ec422016-12-15 10:58:41 -050046 op->fMaskType = kGrayscaleCoverageMask_MaskType;
joshualitta751c972015-11-20 13:37:32 -080047 break;
48 case kA565_GrMaskFormat:
Brian Salomon344ec422016-12-15 10:58:41 -050049 op->fMaskType = kLCDCoverageMask_MaskType;
joshualitta751c972015-11-20 13:37:32 -080050 break;
51 case kARGB_GrMaskFormat:
Brian Salomon344ec422016-12-15 10:58:41 -050052 op->fMaskType = kColorBitmapMask_MaskType;
joshualitta751c972015-11-20 13:37:32 -080053 break;
54 }
Brian Salomon344ec422016-12-15 10:58:41 -050055 op->fNumGlyphs = glyphCount;
56 op->fGeoCount = 1;
Jim Van Verthbc2cdd12017-06-08 11:14:35 -040057 op->fLuminanceColor = 0;
Brian Salomon344ec422016-12-15 10:58:41 -050058 op->fFontCache = fontCache;
Brian Salomon344ec422016-12-15 10:58:41 -050059 return op;
joshualitta751c972015-11-20 13:37:32 -080060 }
61
Brian Salomonf8334782017-01-03 09:42:58 -050062 static std::unique_ptr<GrAtlasTextOp> MakeDistanceField(
Brian Salomon44acb5b2017-07-18 19:59:24 -040063 GrPaint&& paint, int glyphCount, GrAtlasGlyphCache* fontCache,
Brian Salomon344ec422016-12-15 10:58:41 -050064 const GrDistanceFieldAdjustTable* distanceAdjustTable,
Jim Van Verth90e89b32017-07-06 16:36:55 -040065 bool useGammaCorrectDistanceTable, SkColor luminanceColor, bool isLCD, bool useBGR,
66 bool isAntiAliased) {
Brian Salomon44acb5b2017-07-18 19:59:24 -040067 std::unique_ptr<GrAtlasTextOp> op(new GrAtlasTextOp(std::move(paint)));
joshualitta751c972015-11-20 13:37:32 -080068
Brian Salomon344ec422016-12-15 10:58:41 -050069 op->fFontCache = fontCache;
Jim Van Verth90e89b32017-07-06 16:36:55 -040070 op->fMaskType = !isAntiAliased ? kAliasedDistanceField_MaskType
71 : isLCD ? (useBGR ? kLCDBGRDistanceField_MaskType
72 : kLCDDistanceField_MaskType)
73 : kGrayscaleDistanceField_MaskType;
Brian Salomon344ec422016-12-15 10:58:41 -050074 op->fDistanceAdjustTable.reset(SkRef(distanceAdjustTable));
75 op->fUseGammaCorrectDistanceTable = useGammaCorrectDistanceTable;
Jim Van Verthbc2cdd12017-06-08 11:14:35 -040076 op->fLuminanceColor = luminanceColor;
Brian Salomon344ec422016-12-15 10:58:41 -050077 op->fNumGlyphs = glyphCount;
78 op->fGeoCount = 1;
79 return op;
joshualitta751c972015-11-20 13:37:32 -080080 }
81
Brian Salomon344ec422016-12-15 10:58:41 -050082 // To avoid even the initial copy of the struct, we have a getter for the first item which
83 // is used to seed the op with its initial geometry. After seeding, the client should call
84 // init() so the op can initialize itself
joshualitta751c972015-11-20 13:37:32 -080085 Geometry& geometry() { return fGeoData[0]; }
86
87 void init() {
88 const Geometry& geo = fGeoData[0];
Brian Salomon344ec422016-12-15 10:58:41 -050089 fColor = geo.fColor;
bsalomon88cf17d2016-07-08 06:40:56 -070090 SkRect bounds;
91 geo.fBlob->computeSubRunBounds(&bounds, geo.fRun, geo.fSubRun, geo.fViewMatrix, geo.fX,
joshualitt8e0ef292016-02-19 14:13:03 -080092 geo.fY);
bsalomon88cf17d2016-07-08 06:40:56 -070093 // We don't have tight bounds on the glyph paths in device space. For the purposes of bounds
94 // we treat this as a set of non-AA rects rendered with a texture.
95 this->setBounds(bounds, HasAABloat::kNo, IsZeroArea::kNo);
joshualitta751c972015-11-20 13:37:32 -080096 }
97
Brian Salomon344ec422016-12-15 10:58:41 -050098 const char* name() const override { return "AtlasTextOp"; }
joshualitta751c972015-11-20 13:37:32 -080099
Robert Phillipsf1748f52017-09-14 14:11:24 -0400100 void visitProxies(const VisitProxyFunc& func) const override {
Robert Phillipsb493eeb2017-09-13 13:10:52 -0400101 fProcessors.visitProxies(func);
102
103 const sk_sp<GrTextureProxy>* proxies = fFontCache->getProxies(this->maskFormat());
104 for (int i = 0; i < kMaxTextures; ++i) {
105 if (proxies[i]) {
106 func(proxies[i].get());
107 }
108 }
109 }
110
joshualitta751c972015-11-20 13:37:32 -0800111 SkString dumpInfo() const override;
112
Brian Salomon44acb5b2017-07-18 19:59:24 -0400113 FixedFunctionFlags fixedFunctionFlags() const override;
114
115 RequiresDstTexture finalize(const GrCaps& caps, const GrAppliedClip* clip) override;
116
joshualitta751c972015-11-20 13:37:32 -0800117private:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400118 GrAtlasTextOp(GrPaint&& paint)
119 : INHERITED(ClassID())
120 , fColor(paint.getColor())
121 , fSRGBFlags(GrPipeline::SRGBFlagsFromPaint(paint))
122 , fProcessors(std::move(paint)) {}
joshualitta751c972015-11-20 13:37:32 -0800123
124 struct FlushInfo {
Brian Salomon344ec422016-12-15 10:58:41 -0500125 sk_sp<const GrBuffer> fVertexBuffer;
126 sk_sp<const GrBuffer> fIndexBuffer;
Hal Canary144caf52016-11-07 17:57:18 -0500127 sk_sp<GrGeometryProcessor> fGeometryProcessor;
Brian Salomon44acb5b2017-07-18 19:59:24 -0400128 const GrPipeline* fPipeline;
Brian Salomon344ec422016-12-15 10:58:41 -0500129 int fGlyphsToFlush;
130 int fVertexOffset;
joshualitta751c972015-11-20 13:37:32 -0800131 };
132
Brian Salomon91326c32017-08-09 16:02:19 -0400133 void onPrepareDraws(Target*) override;
joshualitta751c972015-11-20 13:37:32 -0800134
joshualitta751c972015-11-20 13:37:32 -0800135 GrMaskFormat maskFormat() const {
136 switch (fMaskType) {
137 case kLCDCoverageMask_MaskType:
138 return kA565_GrMaskFormat;
139 case kColorBitmapMask_MaskType:
140 return kARGB_GrMaskFormat;
141 case kGrayscaleCoverageMask_MaskType:
Jim Van Verth90e89b32017-07-06 16:36:55 -0400142 case kAliasedDistanceField_MaskType:
joshualitta751c972015-11-20 13:37:32 -0800143 case kGrayscaleDistanceField_MaskType:
144 case kLCDDistanceField_MaskType:
Jim Van Verth90e89b32017-07-06 16:36:55 -0400145 case kLCDBGRDistanceField_MaskType:
joshualitta751c972015-11-20 13:37:32 -0800146 return kA8_GrMaskFormat;
147 }
Brian Salomon344ec422016-12-15 10:58:41 -0500148 return kA8_GrMaskFormat; // suppress warning
joshualitta751c972015-11-20 13:37:32 -0800149 }
150
151 bool usesDistanceFields() const {
Jim Van Verth90e89b32017-07-06 16:36:55 -0400152 return kAliasedDistanceField_MaskType == fMaskType ||
153 kGrayscaleDistanceField_MaskType == fMaskType ||
154 kLCDDistanceField_MaskType == fMaskType ||
155 kLCDBGRDistanceField_MaskType == fMaskType;
joshualitta751c972015-11-20 13:37:32 -0800156 }
157
158 bool isLCD() const {
Jim Van Verth90e89b32017-07-06 16:36:55 -0400159 return kLCDCoverageMask_MaskType == fMaskType ||
160 kLCDDistanceField_MaskType == fMaskType ||
161 kLCDBGRDistanceField_MaskType == fMaskType;
joshualitta751c972015-11-20 13:37:32 -0800162 }
163
Brian Salomone5b399e2017-07-19 13:50:54 -0400164 inline void flush(GrMeshDrawOp::Target* target, FlushInfo* flushInfo) const;
joshualitta751c972015-11-20 13:37:32 -0800165
Brian Salomon344ec422016-12-15 10:58:41 -0500166 GrColor color() const { return fColor; }
joshualitt8e0ef292016-02-19 14:13:03 -0800167 const SkMatrix& viewMatrix() const { return fGeoData[0].fViewMatrix; }
Brian Salomon344ec422016-12-15 10:58:41 -0500168 bool usesLocalCoords() const { return fUsesLocalCoords; }
169 int numGlyphs() const { return fNumGlyphs; }
joshualitta751c972015-11-20 13:37:32 -0800170
Brian Salomon25a88092016-12-01 09:36:50 -0500171 bool onCombineIfPossible(GrOp* t, const GrCaps& caps) override;
joshualitta751c972015-11-20 13:37:32 -0800172
Jim Van Vertha950b632017-09-12 11:54:11 -0400173 static constexpr auto kMaxTextures = 4;
174
joshualitta751c972015-11-20 13:37:32 -0800175 // TODO just use class params
Robert Phillipsfbcef6e2017-06-15 12:07:18 -0400176 sk_sp<GrGeometryProcessor> setupDfProcessor(const SkMatrix& viewMatrix, SkColor luminanceColor,
Jim Van Vertha950b632017-09-12 11:54:11 -0400177 GrColor color,
178 const sk_sp<GrTextureProxy> [kMaxTextures]) const;
joshualitta751c972015-11-20 13:37:32 -0800179
joshualitta751c972015-11-20 13:37:32 -0800180
joshualitta751c972015-11-20 13:37:32 -0800181 // The minimum number of Geometry we will try to allocate.
182 enum { kMinGeometryAllocated = 4 };
joshualitta751c972015-11-20 13:37:32 -0800183
184 enum MaskType {
185 kGrayscaleCoverageMask_MaskType,
186 kLCDCoverageMask_MaskType,
187 kColorBitmapMask_MaskType,
Jim Van Verth90e89b32017-07-06 16:36:55 -0400188 kAliasedDistanceField_MaskType,
joshualitta751c972015-11-20 13:37:32 -0800189 kGrayscaleDistanceField_MaskType,
190 kLCDDistanceField_MaskType,
Jim Van Verth90e89b32017-07-06 16:36:55 -0400191 kLCDBGRDistanceField_MaskType,
Brian Salomon44acb5b2017-07-18 19:59:24 -0400192 };
joshualitta751c972015-11-20 13:37:32 -0800193
Brian Salomon44acb5b2017-07-18 19:59:24 -0400194 SkAutoSTMalloc<kMinGeometryAllocated, Geometry> fGeoData;
195 GrColor fColor;
196 uint32_t fSRGBFlags;
197 GrProcessorSet fProcessors;
198 bool fUsesLocalCoords;
199 bool fCanCombineOnTouchOrOverlap;
200 int fGeoCount;
201 int fNumGlyphs;
202 MaskType fMaskType;
Brian Salomonf856fd12016-12-16 14:24:34 -0500203 GrAtlasGlyphCache* fFontCache;
joshualitta751c972015-11-20 13:37:32 -0800204 // Distance field properties
Hal Canary144caf52016-11-07 17:57:18 -0500205 sk_sp<const GrDistanceFieldAdjustTable> fDistanceAdjustTable;
Jim Van Verthbc2cdd12017-06-08 11:14:35 -0400206 SkColor fLuminanceColor;
brianosmanb461d342016-04-13 13:10:14 -0700207 bool fUseGammaCorrectDistanceTable;
joshualitta751c972015-11-20 13:37:32 -0800208
Brian Salomon344ec422016-12-15 10:58:41 -0500209 friend class GrBlobRegenHelper; // Needs to trigger flushes
joshualittddd22d82016-02-16 06:47:52 -0800210
Brian Salomon44acb5b2017-07-18 19:59:24 -0400211 typedef GrMeshDrawOp INHERITED;
joshualitta751c972015-11-20 13:37:32 -0800212};
213
joshualittddd22d82016-02-16 06:47:52 -0800214/*
215 * A simple helper class to abstract the interface GrAtlasTextBlob needs to regenerate itself.
216 * It'd be nicer if this was nested, but we need to forward declare it in GrAtlasTextBlob.h
217 */
218class GrBlobRegenHelper {
219public:
Brian Salomone5b399e2017-07-19 13:50:54 -0400220 GrBlobRegenHelper(const GrAtlasTextOp* op, GrMeshDrawOp::Target* target,
Brian Salomon344ec422016-12-15 10:58:41 -0500221 GrAtlasTextOp::FlushInfo* flushInfo)
222 : fOp(op), fTarget(target), fFlushInfo(flushInfo) {}
joshualittddd22d82016-02-16 06:47:52 -0800223
224 void flush();
225
Brian Salomon344ec422016-12-15 10:58:41 -0500226 void incGlyphCount(int glyphCount = 1) { fFlushInfo->fGlyphsToFlush += glyphCount; }
joshualittddd22d82016-02-16 06:47:52 -0800227
228private:
Brian Salomon344ec422016-12-15 10:58:41 -0500229 const GrAtlasTextOp* fOp;
Brian Salomone5b399e2017-07-19 13:50:54 -0400230 GrMeshDrawOp::Target* fTarget;
Brian Salomon344ec422016-12-15 10:58:41 -0500231 GrAtlasTextOp::FlushInfo* fFlushInfo;
joshualittddd22d82016-02-16 06:47:52 -0800232};
233
joshualitta751c972015-11-20 13:37:32 -0800234#endif