blob: 81d6d65bba91fe8a334b568274dae9cf653c90b9 [file] [log] [blame]
joshualitt374b2f72015-07-21 08:05:03 -07001/*
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
Herb Derby86240592018-05-24 16:12:31 -04008#ifndef GrTextBlob_DEFINED
9#define GrTextBlob_DEFINED
joshualitt374b2f72015-07-21 08:05:03 -070010
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/core/SkPathEffect.h"
12#include "include/core/SkPoint3.h"
13#include "include/core/SkSurfaceProps.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050014#include "src/core/SkDescriptor.h"
15#include "src/core/SkMaskFilterBase.h"
16#include "src/core/SkOpts.h"
17#include "src/core/SkRectPriv.h"
18#include "src/core/SkStrikeCache.h"
Herb Derbye7efd082019-05-28 11:30:33 -040019#include "src/core/SkStrikeSpec.h"
Ben Wagner729a23f2019-05-17 16:29:34 -040020#include "src/core/SkTInternalLList.h"
Greg Danielf91aeb22019-06-18 09:58:02 -040021#include "src/gpu/GrColor.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050022#include "src/gpu/GrDrawOpAtlas.h"
23#include "src/gpu/text/GrStrikeCache.h"
24#include "src/gpu/text/GrTextContext.h"
25#include "src/gpu/text/GrTextTarget.h"
joshualitt374b2f72015-07-21 08:05:03 -070026
Robert Phillipsc4039ea2018-03-01 11:36:45 -050027class GrAtlasManager;
joshualitt2e2202e2015-12-10 11:22:08 -080028struct GrDistanceFieldAdjustTable;
Robert Phillipsc4039ea2018-03-01 11:36:45 -050029struct GrGlyph;
Robert Phillipsc4039ea2018-03-01 11:36:45 -050030
joshualitt2e2202e2015-12-10 11:22:08 -080031class SkTextBlob;
32class SkTextBlobRunIterator;
33
Herb Derby26cbe512018-05-24 14:39:01 -040034// With this flag enabled, the GrTextContext will, as a sanity check, regenerate every blob
joshualitt259fbf12015-07-21 11:39:34 -070035// that comes in to verify the integrity of its cache
joshualitt2f2ee832016-02-10 08:52:24 -080036#define CACHE_SANITY_CHECK 0
joshualitt259fbf12015-07-21 11:39:34 -070037
joshualitt374b2f72015-07-21 08:05:03 -070038/*
Herb Derby86240592018-05-24 16:12:31 -040039 * A GrTextBlob contains a fully processed SkTextBlob, suitable for nearly immediate drawing
joshualitt374b2f72015-07-21 08:05:03 -070040 * on the GPU. These are initially created with valid positions and colors, but invalid
Herb Derby86240592018-05-24 16:12:31 -040041 * texture coordinates. The GrTextBlob itself has a few Blob-wide properties, and also
joshualitt374b2f72015-07-21 08:05:03 -070042 * consists of a number of runs. Runs inside a blob are flushed individually so they can be
43 * reordered.
44 *
Herb Derby86240592018-05-24 16:12:31 -040045 * The only thing(aside from a memcopy) required to flush a GrTextBlob is to ensure that
joshualitt374b2f72015-07-21 08:05:03 -070046 * the GrAtlas will not evict anything the Blob needs.
47 *
48 * Note: This struct should really be named GrCachedAtasTextBlob, but that is too verbose.
joshualitt259fbf12015-07-21 11:39:34 -070049 *
50 * *WARNING* If you add new fields to this struct, then you may need to to update AssertEqual
joshualitt374b2f72015-07-21 08:05:03 -070051 */
Herb Derbyac962c92019-03-08 12:31:47 -050052class GrTextBlob : public SkNVRefCnt<GrTextBlob>, public SkGlyphRunPainterInterface {
Herb Derbydc214c22018-11-08 13:31:39 -050053 struct Run;
joshualitt2e2202e2015-12-10 11:22:08 -080054public:
Herb Derby86240592018-05-24 16:12:31 -040055 SK_DECLARE_INTERNAL_LLIST_INTERFACE(GrTextBlob);
joshualitt374b2f72015-07-21 08:05:03 -070056
Brian Salomon18923f92017-11-06 16:26:02 -050057 class VertexRegenerator;
58
Herb Derbya00da612019-03-04 17:10:01 -050059 void generateFromGlyphRunList(const GrShaderCaps& shaderCaps,
Herb Derbydc214c22018-11-08 13:31:39 -050060 const GrTextContext::Options& options,
61 const SkPaint& paint,
Herb Derbydc214c22018-11-08 13:31:39 -050062 SkScalerContextFlags scalerContextFlags,
63 const SkMatrix& viewMatrix,
64 const SkSurfaceProps& props,
65 const SkGlyphRunList& glyphRunList,
66 SkGlyphRunListPainter* glyphPainter);
67
Herb Derbya00da612019-03-04 17:10:01 -050068 static sk_sp<GrTextBlob> Make(
69 int glyphCount,
70 int runCount,
71 GrColor color,
72 GrStrikeCache* strikeCache);
joshualitt323c2eb2016-01-20 06:48:47 -080073
Brian Salomon5c6ac642017-12-19 11:09:32 -050074 /**
75 * We currently force regeneration of a blob if old or new matrix differ in having perspective.
76 * If we ever change that then the key must contain the perspectiveness when there are distance
77 * fields as perspective distance field use 3 component vertex positions and non-perspective
78 * uses 2.
79 */
joshualitt323c2eb2016-01-20 06:48:47 -080080 struct Key {
81 Key() {
82 sk_bzero(this, sizeof(Key));
83 }
84 uint32_t fUniqueID;
85 // Color may affect the gamma of the mask we generate, but in a fairly limited way.
86 // Each color is assigned to on of a fixed number of buckets based on its
87 // luminance. For each luminance bucket there is a "canonical color" that
88 // represents the bucket. This functionality is currently only supported for A8
89 SkColor fCanonicalColor;
90 SkPaint::Style fStyle;
91 SkPixelGeometry fPixelGeometry;
92 bool fHasBlur;
brianosman8d7ffce2016-04-21 08:29:06 -070093 uint32_t fScalerContextFlags;
joshualitt323c2eb2016-01-20 06:48:47 -080094
95 bool operator==(const Key& other) const {
96 return 0 == memcmp(this, &other, sizeof(Key));
97 }
98 };
99
Herb Derby86240592018-05-24 16:12:31 -0400100 void setupKey(const GrTextBlob::Key& key,
Mike Reed80747ef2018-01-23 15:29:32 -0500101 const SkMaskFilterBase::BlurRec& blurRec,
joshualitt92303772016-02-10 11:55:52 -0800102 const SkPaint& paint) {
103 fKey = key;
104 if (key.fHasBlur) {
105 fBlurRec = blurRec;
106 }
107 if (key.fStyle != SkPaint::kFill_Style) {
108 fStrokeInfo.fFrameWidth = paint.getStrokeWidth();
109 fStrokeInfo.fMiterLimit = paint.getStrokeMiter();
110 fStrokeInfo.fJoin = paint.getStrokeJoin();
111 }
112 }
113
Herb Derby86240592018-05-24 16:12:31 -0400114 static const Key& GetKey(const GrTextBlob& blob) {
joshualitt323c2eb2016-01-20 06:48:47 -0800115 return blob.fKey;
116 }
117
118 static uint32_t Hash(const Key& key) {
mtklein4e976072016-08-08 09:06:27 -0700119 return SkOpts::hash(&key, sizeof(Key));
joshualitt323c2eb2016-01-20 06:48:47 -0800120 }
121
122 void operator delete(void* p) {
Herb Derbyb12175f2018-05-23 16:38:09 -0400123 ::operator delete(p);
joshualitt323c2eb2016-01-20 06:48:47 -0800124 }
Herb Derbyb12175f2018-05-23 16:38:09 -0400125
joshualitt323c2eb2016-01-20 06:48:47 -0800126 void* operator new(size_t) {
Ben Wagnerb4aab9a2017-08-16 10:53:04 -0400127 SK_ABORT("All blobs are created by placement new.");
joshualitt323c2eb2016-01-20 06:48:47 -0800128 return sk_malloc_throw(0);
129 }
130
131 void* operator new(size_t, void* p) { return p; }
joshualitt323c2eb2016-01-20 06:48:47 -0800132
133 bool hasDistanceField() const { return SkToBool(fTextType & kHasDistanceField_TextType); }
134 bool hasBitmap() const { return SkToBool(fTextType & kHasBitmap_TextType); }
135 void setHasDistanceField() { fTextType |= kHasDistanceField_TextType; }
136 void setHasBitmap() { fTextType |= kHasBitmap_TextType; }
137
Herb Derby2bb343c2018-11-08 15:03:48 -0500138 int runCountLimit() const { return fRunCountLimit; }
joshualittddd22d82016-02-16 06:47:52 -0800139
Herb Derby2bb343c2018-11-08 15:03:48 -0500140 Run* pushBackRun() {
141 SkASSERT(fRunCount < fRunCountLimit);
Herb Derby6dff60e2018-11-12 15:45:49 -0500142
143 // If there is more run, then connect up the subruns.
Herb Derby2bb343c2018-11-08 15:03:48 -0500144 if (fRunCount > 0) {
Herb Derby69ff8952018-11-12 11:39:12 -0500145 SubRun& newRun = fRuns[fRunCount].fSubRunInfo.back();
146 SubRun& lastRun = fRuns[fRunCount - 1].fSubRunInfo.back();
joshualitt323c2eb2016-01-20 06:48:47 -0800147 newRun.setAsSuccessor(lastRun);
148 }
joshualitt323c2eb2016-01-20 06:48:47 -0800149
Herb Derby2bb343c2018-11-08 15:03:48 -0500150 fRunCount++;
Herb Derbyc72594a2019-03-05 17:39:38 -0500151 return this->currentRun();
Jim Van Verth89737de2018-02-06 21:30:20 +0000152 }
153
Herb Derby78540b92019-02-19 14:42:24 -0500154 void setMinAndMaxScale(SkScalar scaledMin, SkScalar scaledMax) {
joshualitt323c2eb2016-01-20 06:48:47 -0800155 // we init fMaxMinScale and fMinMaxScale in the constructor
Herb Derby78540b92019-02-19 14:42:24 -0500156 fMaxMinScale = SkMaxScalar(scaledMin, fMaxMinScale);
157 fMinMaxScale = SkMinScalar(scaledMax, fMinMaxScale);
joshualitt323c2eb2016-01-20 06:48:47 -0800158 }
159
Jim Van Verthb515ae72018-05-23 16:44:55 -0400160 static size_t GetVertexStride(GrMaskFormat maskFormat, bool hasWCoord) {
joshualitt323c2eb2016-01-20 06:48:47 -0800161 switch (maskFormat) {
162 case kA8_GrMaskFormat:
Jim Van Verthb515ae72018-05-23 16:44:55 -0400163 return hasWCoord ? kGrayTextDFPerspectiveVASize : kGrayTextVASize;
joshualitt323c2eb2016-01-20 06:48:47 -0800164 case kARGB_GrMaskFormat:
Jim Van Verthb515ae72018-05-23 16:44:55 -0400165 return hasWCoord ? kColorTextPerspectiveVASize : kColorTextVASize;
joshualitt323c2eb2016-01-20 06:48:47 -0800166 default:
Jim Van Verthb515ae72018-05-23 16:44:55 -0400167 SkASSERT(!hasWCoord);
joshualitt323c2eb2016-01-20 06:48:47 -0800168 return kLCDTextVASize;
169 }
170 }
171
Herb Derby0edb2142018-10-16 17:04:11 -0400172 bool mustRegenerate(const SkPaint&, bool, const SkMaskFilterBase::BlurRec& blurRec,
joshualitt323c2eb2016-01-20 06:48:47 -0800173 const SkMatrix& viewMatrix, SkScalar x, SkScalar y);
174
Herb Derbyc1b482c2018-08-09 15:02:27 -0400175 void flush(GrTextTarget*, const SkSurfaceProps& props,
Jim Van Verth54d9c882018-02-08 16:14:48 -0500176 const GrDistanceFieldAdjustTable* distanceAdjustTable,
Brian Osmancf860852018-10-31 14:04:39 -0400177 const SkPaint& paint, const SkPMColor4f& filteredColor, const GrClip& clip,
Robert Phillipse4643cc2018-08-14 13:01:29 -0400178 const SkMatrix& viewMatrix, SkScalar x, SkScalar y);
joshualitt323c2eb2016-01-20 06:48:47 -0800179
joshualitt8e0ef292016-02-19 14:13:03 -0800180 void computeSubRunBounds(SkRect* outBounds, int runIndex, int subRunIndex,
Jim Van Verth70276912018-06-01 13:46:46 -0400181 const SkMatrix& viewMatrix, SkScalar x, SkScalar y,
182 bool needsGlyphTransform) {
joshualittbc811112016-02-11 12:42:02 -0800183 // We don't yet position distance field text on the cpu, so we have to map the vertex bounds
184 // into device space.
185 // We handle vertex bounds differently for distance field text and bitmap text because
186 // the vertex bounds of bitmap text are in device space. If we are flushing multiple runs
187 // from one blob then we are going to pay the price here of mapping the rect for each run.
188 const Run& run = fRuns[runIndex];
Herb Derby69ff8952018-11-12 11:39:12 -0500189 const SubRun& subRun = run.fSubRunInfo[subRunIndex];
joshualittbc811112016-02-11 12:42:02 -0800190 *outBounds = subRun.vertexBounds();
Jim Van Verth70276912018-06-01 13:46:46 -0400191 if (needsGlyphTransform) {
joshualittbc811112016-02-11 12:42:02 -0800192 // Distance field text is positioned with the (X,Y) as part of the glyph position,
193 // and currently the view matrix is applied on the GPU
joshualitt8e0ef292016-02-19 14:13:03 -0800194 outBounds->offset(x - fInitialX, y - fInitialY);
195 viewMatrix.mapRect(outBounds);
joshualittbc811112016-02-11 12:42:02 -0800196 } else {
197 // Bitmap text is fully positioned on the CPU, and offset by an (X,Y) translate in
198 // device space.
199 SkMatrix boundsMatrix = fInitialViewMatrixInverse;
200
201 boundsMatrix.postTranslate(-fInitialX, -fInitialY);
202
joshualitt8e0ef292016-02-19 14:13:03 -0800203 boundsMatrix.postTranslate(x, y);
joshualittbc811112016-02-11 12:42:02 -0800204
joshualitt8e0ef292016-02-19 14:13:03 -0800205 boundsMatrix.postConcat(viewMatrix);
joshualittbc811112016-02-11 12:42:02 -0800206 boundsMatrix.mapRect(outBounds);
207
208 // Due to floating point numerical inaccuracies, we have to round out here
209 outBounds->roundOut(outBounds);
210 }
211 }
212
joshualitt323c2eb2016-01-20 06:48:47 -0800213 // position + local coord
214 static const size_t kColorTextVASize = sizeof(SkPoint) + sizeof(SkIPoint16);
Jim Van Verthb515ae72018-05-23 16:44:55 -0400215 static const size_t kColorTextPerspectiveVASize = sizeof(SkPoint3) + sizeof(SkIPoint16);
joshualitt323c2eb2016-01-20 06:48:47 -0800216 static const size_t kGrayTextVASize = sizeof(SkPoint) + sizeof(GrColor) + sizeof(SkIPoint16);
Brian Salomon5c6ac642017-12-19 11:09:32 -0500217 static const size_t kGrayTextDFPerspectiveVASize =
218 sizeof(SkPoint3) + sizeof(GrColor) + sizeof(SkIPoint16);
joshualitt323c2eb2016-01-20 06:48:47 -0800219 static const size_t kLCDTextVASize = kGrayTextVASize;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500220 static const size_t kMaxVASize = kGrayTextDFPerspectiveVASize;
joshualitt323c2eb2016-01-20 06:48:47 -0800221 static const int kVerticesPerGlyph = 4;
222
Herb Derby86240592018-05-24 16:12:31 -0400223 static void AssertEqual(const GrTextBlob&, const GrTextBlob&);
joshualitt323c2eb2016-01-20 06:48:47 -0800224
225 // The color here is the GrPaint color, and it is used to determine whether we
226 // have to regenerate LCD text blobs.
227 // We use this color vs the SkPaint color because it has the colorfilter applied.
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400228 void initReusableBlob(SkColor luminanceColor, const SkMatrix& viewMatrix,
229 SkScalar x, SkScalar y) {
Jim Van Verthbc2cdd12017-06-08 11:14:35 -0400230 fLuminanceColor = luminanceColor;
joshualitt7481e752016-01-22 06:08:48 -0800231 this->setupViewMatrix(viewMatrix, x, y);
joshualitt323c2eb2016-01-20 06:48:47 -0800232 }
233
joshualitt7481e752016-01-22 06:08:48 -0800234 void initThrowawayBlob(const SkMatrix& viewMatrix, SkScalar x, SkScalar y) {
235 this->setupViewMatrix(viewMatrix, x, y);
joshualitt323c2eb2016-01-20 06:48:47 -0800236 }
237
joshualitt92303772016-02-10 11:55:52 -0800238 const Key& key() const { return fKey; }
239
Herb Derbyb12175f2018-05-23 16:38:09 -0400240 size_t size() const { return fSize; }
241
Herb Derbyac962c92019-03-08 12:31:47 -0500242 ~GrTextBlob() override {
Herb Derby2bb343c2018-11-08 15:03:48 -0500243 for (int i = 0; i < fRunCountLimit; i++) {
joshualitt92303772016-02-10 11:55:52 -0800244 fRuns[i].~Run();
245 }
246 }
247
joshualittbc811112016-02-11 12:42:02 -0800248 ////////////////////////////////////////////////////////////////////////////////////////////////
249 // Internal test methods
Jim Van Verth56c37142017-10-31 14:44:25 -0400250 std::unique_ptr<GrDrawOp> test_makeOp(int glyphCount, uint16_t run, uint16_t subRun,
Brian Salomon44acb5b2017-07-18 19:59:24 -0400251 const SkMatrix& viewMatrix, SkScalar x, SkScalar y,
Brian Osmancf860852018-10-31 14:04:39 -0400252 const SkPaint& paint, const SkPMColor4f& filteredColor,
Herb Derbybc6f9c92018-08-08 13:58:45 -0400253 const SkSurfaceProps&, const GrDistanceFieldAdjustTable*,
Herb Derbyc1b482c2018-08-09 15:02:27 -0400254 GrTextTarget*);
joshualittbc811112016-02-11 12:42:02 -0800255
joshualitt323c2eb2016-01-20 06:48:47 -0800256private:
Herb Derbya00da612019-03-04 17:10:01 -0500257 GrTextBlob(GrStrikeCache* strikeCache) : fStrikeCache{strikeCache} { }
joshualitt92303772016-02-10 11:55:52 -0800258
joshualitt8e0ef292016-02-19 14:13:03 -0800259 // This function will only be called when we are generating a blob from scratch. We record the
joshualitt7481e752016-01-22 06:08:48 -0800260 // initial view matrix and initial offsets(x,y), because we record vertex bounds relative to
261 // these numbers. When blobs are reused with new matrices, we need to return to model space so
262 // we can update the vertex bounds appropriately.
263 void setupViewMatrix(const SkMatrix& viewMatrix, SkScalar x, SkScalar y) {
joshualitt8e0ef292016-02-19 14:13:03 -0800264 fInitialViewMatrix = viewMatrix;
joshualitt7481e752016-01-22 06:08:48 -0800265 if (!viewMatrix.invert(&fInitialViewMatrixInverse)) {
266 fInitialViewMatrixInverse = SkMatrix::I();
joshualitt7481e752016-01-22 06:08:48 -0800267 }
joshualitt8e0ef292016-02-19 14:13:03 -0800268 fInitialX = x;
269 fInitialY = y;
270
271 // make sure all initial subruns have the correct VM and X/Y applied
Herb Derby2bb343c2018-11-08 15:03:48 -0500272 for (int i = 0; i < fRunCountLimit; i++) {
joshualitt8e0ef292016-02-19 14:13:03 -0800273 fRuns[i].fSubRunInfo[0].init(fInitialViewMatrix, x, y);
274 }
joshualitt7481e752016-01-22 06:08:48 -0800275 }
276
Herb Derby69ff8952018-11-12 11:39:12 -0500277 class SubRun {
278 public:
Herb Derby36a54c12019-06-06 10:50:56 -0400279 SubRun(Run* run, const SkStrikeSpec& strikeSpec, GrColor color)
Herb Derby5424a5e2018-11-14 12:04:38 -0500280 : fColor{color}
281 , fRun{run}
Herb Derbye7efd082019-05-28 11:30:33 -0400282 , fStrikeSpec{strikeSpec} {}
Herb Derby69ff8952018-11-12 11:39:12 -0500283
Herb Derby5424a5e2018-11-14 12:04:38 -0500284 // When used with emplace_back, this constructs a SubRun from the last SubRun in an array.
285 //SubRun(SkSTArray<1, SubRun>* subRunList)
286 // : fColor{subRunList->fromBack(1).fColor} { }
287
Herb Derby438ea542018-12-19 18:25:11 -0500288 void appendGlyph(GrGlyph* glyph, SkRect dstRect);
Herb Derby96a17092018-11-12 12:45:21 -0500289
Herb Derby69ff8952018-11-12 11:39:12 -0500290 // TODO when this object is more internal, drop the privacy
291 void resetBulkUseToken() { fBulkUseToken.reset(); }
292 GrDrawOpAtlas::BulkUseTokenUpdater* bulkUseToken() { return &fBulkUseToken; }
293 void setStrike(sk_sp<GrTextStrike> strike) { fStrike = std::move(strike); }
294 GrTextStrike* strike() const { return fStrike.get(); }
295 sk_sp<GrTextStrike> refStrike() const { return fStrike; }
296
297 void setAtlasGeneration(uint64_t atlasGeneration) { fAtlasGeneration = atlasGeneration;}
298 uint64_t atlasGeneration() const { return fAtlasGeneration; }
299
300 size_t byteCount() const { return fVertexEndIndex - fVertexStartIndex; }
301 size_t vertexStartIndex() const { return fVertexStartIndex; }
302 size_t vertexEndIndex() const { return fVertexEndIndex; }
Herb Derby69ff8952018-11-12 11:39:12 -0500303
304 uint32_t glyphCount() const { return fGlyphEndIndex - fGlyphStartIndex; }
305 uint32_t glyphStartIndex() const { return fGlyphStartIndex; }
306 uint32_t glyphEndIndex() const { return fGlyphEndIndex; }
Herb Derby69ff8952018-11-12 11:39:12 -0500307 void setColor(GrColor color) { fColor = color; }
308 GrColor color() const { return fColor; }
309 void setMaskFormat(GrMaskFormat format) { fMaskFormat = format; }
310 GrMaskFormat maskFormat() const { return fMaskFormat; }
311
312 void setAsSuccessor(const SubRun& prev) {
313 fGlyphStartIndex = prev.glyphEndIndex();
Herb Derby6dff60e2018-11-12 15:45:49 -0500314 fGlyphEndIndex = fGlyphStartIndex;
Herb Derby69ff8952018-11-12 11:39:12 -0500315
316 fVertexStartIndex = prev.vertexEndIndex();
Herb Derby6dff60e2018-11-12 15:45:49 -0500317 fVertexEndIndex = fVertexStartIndex;
Herb Derby69ff8952018-11-12 11:39:12 -0500318
319 // copy over viewmatrix settings
320 this->init(prev.fCurrentViewMatrix, prev.fX, prev.fY);
321 }
322
323 const SkRect& vertexBounds() const { return fVertexBounds; }
324 void joinGlyphBounds(const SkRect& glyphBounds) {
325 fVertexBounds.joinNonEmptyArg(glyphBounds);
326 }
327
328 void init(const SkMatrix& viewMatrix, SkScalar x, SkScalar y) {
329 fCurrentViewMatrix = viewMatrix;
330 fX = x;
331 fY = y;
332 }
333
334 // This function assumes the translation will be applied before it is called again
335 void computeTranslation(const SkMatrix& viewMatrix, SkScalar x, SkScalar y,
336 SkScalar* transX, SkScalar* transY);
337
338 // df properties
herbc5df7cb2018-11-15 17:08:26 -0500339 void setDrawAsDistanceFields() { fFlags.drawAsSdf = true; }
340 bool drawAsDistanceFields() const { return fFlags.drawAsSdf; }
341 void setUseLCDText(bool useLCDText) { fFlags.useLCDText = useLCDText; }
342 bool hasUseLCDText() const { return fFlags.useLCDText; }
343 void setAntiAliased(bool antiAliased) { fFlags.antiAliased = antiAliased; }
344 bool isAntiAliased() const { return fFlags.antiAliased; }
345 void setHasWCoord(bool hasW) { fFlags.hasWCoord = hasW; }
346 bool hasWCoord() const { return fFlags.hasWCoord; }
347 void setNeedsTransform(bool needsTransform) { fFlags.needsTransform = needsTransform; }
348 bool needsTransform() const { return fFlags.needsTransform; }
Herb Derby5f7b0142018-12-14 16:47:45 -0500349 void setFallback() { fFlags.argbFallback = true; }
350 bool isFallback() { return fFlags.argbFallback; }
Herb Derby317adf72018-11-16 17:29:29 -0500351
Herb Derby36a54c12019-06-06 10:50:56 -0400352 const SkStrikeSpec& strikeSpec() const { return fStrikeSpec; }
Herb Derby69ff8952018-11-12 11:39:12 -0500353
354 private:
Herb Derby69ff8952018-11-12 11:39:12 -0500355 GrDrawOpAtlas::BulkUseTokenUpdater fBulkUseToken;
356 sk_sp<GrTextStrike> fStrike;
357 SkMatrix fCurrentViewMatrix;
Herb Derby6dff60e2018-11-12 15:45:49 -0500358 SkRect fVertexBounds = SkRectPriv::MakeLargestInverted();
359 uint64_t fAtlasGeneration{GrDrawOpAtlas::kInvalidAtlasGeneration};
360 size_t fVertexStartIndex{0};
361 size_t fVertexEndIndex{0};
362 uint32_t fGlyphStartIndex{0};
363 uint32_t fGlyphEndIndex{0};
Herb Derby69ff8952018-11-12 11:39:12 -0500364 SkScalar fX;
365 SkScalar fY;
Herb Derby6dff60e2018-11-12 15:45:49 -0500366 GrColor fColor{GrColor_ILLEGAL};
367 GrMaskFormat fMaskFormat{kA8_GrMaskFormat};
herbc5df7cb2018-11-15 17:08:26 -0500368 struct {
369 bool drawAsSdf:1;
370 bool useLCDText:1;
371 bool antiAliased:1;
372 bool hasWCoord:1;
373 bool needsTransform:1;
Herb Derby317adf72018-11-16 17:29:29 -0500374 bool argbFallback:1;
375 } fFlags{false, false, false, false, false, false};
Herb Derbyf7d5d742018-11-16 13:24:32 -0500376 Run* const fRun;
Herb Derby36a54c12019-06-06 10:50:56 -0400377 const SkStrikeSpec& fStrikeSpec;
Herb Derby69ff8952018-11-12 11:39:12 -0500378 }; // SubRunInfo
379
joshualitt374b2f72015-07-21 08:05:03 -0700380 /*
381 * Each Run inside of the blob can have its texture coordinates regenerated if required.
382 * To determine if regeneration is necessary, fAtlasGeneration is used. If there have been
383 * any evictions inside of the atlas, then we will simply regenerate Runs. We could track
384 * this at a more fine grained level, but its not clear if this is worth it, as evictions
385 * should be fairly rare.
386 *
387 * One additional point, each run can contain glyphs with any of the three mask formats.
388 * We call these SubRuns. Because a subrun must be a contiguous range, we have to create
389 * a new subrun each time the mask format changes in a run. In theory, a run can have as
390 * many SubRuns as it has glyphs, ie if a run alternates between color emoji and A8. In
391 * practice, the vast majority of runs have only a single subrun.
392 *
Herb Derby26cbe512018-05-24 14:39:01 -0400393 * Finally, for runs where the entire thing is too large for the GrTextContext to
Jim Van Verthf4c13162018-01-11 16:40:24 -0500394 * handle, we have a bit to mark the run as flushable via rendering as paths or as scaled
395 * glyphs. It would be a bit expensive to figure out ahead of time whether or not a run
joshualitt374b2f72015-07-21 08:05:03 -0700396 * can flush in this manner, so we always allocate vertices for the run, regardless of
397 * whether or not it is too large. The benefit of this strategy is that we can always reuse
398 * a blob allocation regardless of viewmatrix changes. We could store positions for these
Jim Van Verthf4c13162018-01-11 16:40:24 -0500399 * glyphs, however, it's not clear if this is a win because we'd still have to either go to the
joshualitt374b2f72015-07-21 08:05:03 -0700400 * glyph cache to get the path at flush time, or hold onto the path in the cache, which
401 * would greatly increase the memory of these cached items.
402 */
403 struct Run {
Herb Derby5424a5e2018-11-14 12:04:38 -0500404 explicit Run(GrTextBlob* blob, GrColor color)
405 : fBlob{blob}, fColor{color} {
joshualitt374b2f72015-07-21 08:05:03 -0700406 // To ensure we always have one subrun, we push back a fresh run here
Herb Derbye7efd082019-05-28 11:30:33 -0400407 fSubRunInfo.emplace_back(this, fStrikeSpec, color);
joshualitt374b2f72015-07-21 08:05:03 -0700408 }
joshualitt374b2f72015-07-21 08:05:03 -0700409
Herb Derby2bb343c2018-11-08 15:03:48 -0500410 // sets the last subrun of runIndex to use w values
411 void setSubRunHasW(bool hasWCoord) {
Herb Derby69ff8952018-11-12 11:39:12 -0500412 SubRun& subRun = this->fSubRunInfo.back();
Herb Derby2bb343c2018-11-08 15:03:48 -0500413 subRun.setHasWCoord(hasWCoord);
414 }
415
416 // inits the override descriptor on the current run. All following subruns must use this
417 // descriptor
Herb Derby317adf72018-11-16 17:29:29 -0500418 SubRun* initARGBFallback() {
Herb Derby36a54c12019-06-06 10:50:56 -0400419 fFallbackStrikeSpec.reset(new SkStrikeSpec{});
Herb Derby2bb343c2018-11-08 15:03:48 -0500420 // Push back a new subrun to fill and set the override descriptor
Herb Derbye7efd082019-05-28 11:30:33 -0400421 SubRun* subRun = this->pushBackSubRun(*fFallbackStrikeSpec, fColor);
Herb Derby317adf72018-11-16 17:29:29 -0500422 subRun->setMaskFormat(kARGB_GrMaskFormat);
423 subRun->setFallback();
424 return subRun;
Herb Derby2bb343c2018-11-08 15:03:48 -0500425 }
426
427 // Appends a glyph to the blob as a path only.
428 void appendPathGlyph(
429 const SkPath& path, SkPoint position, SkScalar scale, bool preTransformed);
430
Herb Derby438ea542018-12-19 18:25:11 -0500431 // Append a glyph to the sub run taking care to switch the glyph if needed.
432 void switchSubRunIfNeededAndAppendGlyph(GrGlyph* glyph,
433 const sk_sp<GrTextStrike>& strike,
434 const SkRect& destRect,
435 bool needsTransform);
436
437 // Used when the glyph in the cache has the CTM already applied, therefore no transform
438 // is needed during rendering.
439 void appendDeviceSpaceGlyph(const sk_sp<GrTextStrike>& strike,
440 const SkGlyph& skGlyph,
441 SkPoint origin);
442
443 // The glyph is oriented upright in the cache and needs to be transformed onto the screen.
444 void appendSourceSpaceGlyph(const sk_sp<GrTextStrike>& strike,
445 const SkGlyph& skGlyph,
446 SkPoint origin,
447 SkScalar textScale);
Herb Derby2bb343c2018-11-08 15:03:48 -0500448
Herb Derby36a54c12019-06-06 10:50:56 -0400449 void setupFont(const SkStrikeSpec& strikeSpec);
Herb Derby2bb343c2018-11-08 15:03:48 -0500450
Mike Reed48b958b2018-12-03 13:09:02 -0500451 void setRunFontAntiAlias(bool aa) {
452 fAntiAlias = aa;
Herb Derby2bb343c2018-11-08 15:03:48 -0500453 }
454
455 // sets the last subrun of runIndex to use distance field text
456 void setSubRunHasDistanceFields(bool hasLCD, bool isAntiAlias, bool hasWCoord) {
Herb Derby69ff8952018-11-12 11:39:12 -0500457 SubRun& subRun = fSubRunInfo.back();
Herb Derby2bb343c2018-11-08 15:03:48 -0500458 subRun.setUseLCDText(hasLCD);
459 subRun.setAntiAliased(isAntiAlias);
460 subRun.setDrawAsDistanceFields();
461 subRun.setHasWCoord(hasWCoord);
462 }
463
Herb Derby36a54c12019-06-06 10:50:56 -0400464 SubRun* pushBackSubRun(const SkStrikeSpec& desc, GrColor color) {
joshualitt374b2f72015-07-21 08:05:03 -0700465 // Forward glyph / vertex information to seed the new sub run
Herb Derby5424a5e2018-11-14 12:04:38 -0500466 SubRun& newSubRun = fSubRunInfo.emplace_back(this, desc, color);
467
Herb Derby69ff8952018-11-12 11:39:12 -0500468 const SubRun& prevSubRun = fSubRunInfo.fromBack(1);
joshualitte43e3bd2015-07-29 11:10:38 -0700469
Herb Derby5424a5e2018-11-14 12:04:38 -0500470 // Forward glyph / vertex information to seed the new sub run
joshualitt18b072d2015-12-07 12:26:12 -0800471 newSubRun.setAsSuccessor(prevSubRun);
Herb Derby317adf72018-11-16 17:29:29 -0500472 return &newSubRun;
joshualitt374b2f72015-07-21 08:05:03 -0700473 }
Jim Van Verth54d9c882018-02-08 16:14:48 -0500474
475 // Any glyphs that can't be rendered with the base or override descriptor
476 // are rendered as paths
477 struct PathGlyph {
478 PathGlyph(const SkPath& path, SkScalar x, SkScalar y, SkScalar scale, bool preXformed)
479 : fPath(path)
480 , fX(x)
481 , fY(y)
482 , fScale(scale)
483 , fPreTransformed(preXformed) {}
484 SkPath fPath;
485 SkScalar fX;
486 SkScalar fY;
487 SkScalar fScale;
488 bool fPreTransformed;
489 };
490
Herb Derbyf7d5d742018-11-16 13:24:32 -0500491 SkSTArray<1, SubRun> fSubRunInfo;
Herb Derby36a54c12019-06-06 10:50:56 -0400492 SkStrikeSpec fStrikeSpec;
Herb Derbyf7d5d742018-11-16 13:24:32 -0500493
494 // Distance field text cannot draw coloremoji, and so has to fall back. However,
495 // though the distance field text and the coloremoji may share the same run, they
Herb Derbye7efd082019-05-28 11:30:33 -0400496 // will have different descriptors. If fFallbackStrikeSpec is non-nullptr, then it
Herb Derbyf7d5d742018-11-16 13:24:32 -0500497 // will be used in place of the run's descriptor to regen texture coords
Herb Derby36a54c12019-06-06 10:50:56 -0400498 std::unique_ptr<SkStrikeSpec> fFallbackStrikeSpec;
Herb Derbyf7d5d742018-11-16 13:24:32 -0500499
Jim Van Verth54d9c882018-02-08 16:14:48 -0500500 SkTArray<PathGlyph> fPathGlyphs;
501
Mike Reed48b958b2018-12-03 13:09:02 -0500502 bool fAntiAlias{false}; // needed mainly for rendering paths
Herb Derbyf7d5d742018-11-16 13:24:32 -0500503 bool fInitialized{false};
504
505 GrTextBlob* const fBlob;
Herb Derby5424a5e2018-11-14 12:04:38 -0500506 GrColor fColor;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400507 }; // Run
joshualitt374b2f72015-07-21 08:05:03 -0700508
Herb Derbyc72594a2019-03-05 17:39:38 -0500509 std::unique_ptr<GrAtlasTextOp> makeOp(
Herb Derby69ff8952018-11-12 11:39:12 -0500510 const SubRun& info, int glyphCount, uint16_t run, uint16_t subRun,
Brian Salomonf18b1d82017-10-27 11:30:49 -0400511 const SkMatrix& viewMatrix, SkScalar x, SkScalar y, const SkIRect& clipRect,
Brian Osmancf860852018-10-31 14:04:39 -0400512 const SkPaint& paint, const SkPMColor4f& filteredColor, const SkSurfaceProps&,
Herb Derbyc1b482c2018-08-09 15:02:27 -0400513 const GrDistanceFieldAdjustTable*, GrTextTarget*);
joshualitt323c2eb2016-01-20 06:48:47 -0800514
Herb Derbyc72594a2019-03-05 17:39:38 -0500515 // currentRun, startRun, and the process* calls are all used by the SkGlyphRunPainter, and
516 // live in SkGlyphRunPainter.cpp file.
517 Run* currentRun();
Herb Derbyc72594a2019-03-05 17:39:38 -0500518
Herb Derbyac962c92019-03-08 12:31:47 -0500519 void startRun(const SkGlyphRun& glyphRun, bool useSDFT) override;
Herb Derbyc72594a2019-03-05 17:39:38 -0500520
Herb Derby44b35192019-03-12 16:17:03 -0400521 void processDeviceMasks(SkSpan<const SkGlyphPos> masks,
Herb Derby36a54c12019-06-06 10:50:56 -0400522 const SkStrikeSpec& strikeSpec) override;
Herb Derbyc72594a2019-03-05 17:39:38 -0500523
Herb Derby44b35192019-03-12 16:17:03 -0400524 void processSourcePaths(SkSpan<const SkGlyphPos> paths,
Herb Derby36a54c12019-06-06 10:50:56 -0400525 const SkStrikeSpec& strikeSpec) override;
Herb Derbyac962c92019-03-08 12:31:47 -0500526
Herb Derby44b35192019-03-12 16:17:03 -0400527 void processDevicePaths(SkSpan<const SkGlyphPos> paths) override;
Herb Derbyac962c92019-03-08 12:31:47 -0500528
Herb Derby44b35192019-03-12 16:17:03 -0400529 void processSourceSDFT(SkSpan<const SkGlyphPos> masks,
Herb Derby36a54c12019-06-06 10:50:56 -0400530 const SkStrikeSpec& strikeSpec,
Herb Derbyc72594a2019-03-05 17:39:38 -0500531 const SkFont& runFont,
Herb Derbyc72594a2019-03-05 17:39:38 -0500532 SkScalar minScale,
533 SkScalar maxScale,
Herb Derbyac962c92019-03-08 12:31:47 -0500534 bool hasWCoord) override;
Herb Derbyc72594a2019-03-05 17:39:38 -0500535
Herb Derby44b35192019-03-12 16:17:03 -0400536 void processSourceFallback(SkSpan<const SkGlyphPos> masks,
Herb Derby36a54c12019-06-06 10:50:56 -0400537 const SkStrikeSpec& strikeSpec,
Herb Derbyac962c92019-03-08 12:31:47 -0500538 bool hasW) override;
Herb Derbyc72594a2019-03-05 17:39:38 -0500539
Herb Derby44b35192019-03-12 16:17:03 -0400540 void processDeviceFallback(SkSpan<const SkGlyphPos> masks,
Herb Derby36a54c12019-06-06 10:50:56 -0400541 const SkStrikeSpec& strikeSpec) override;
Herb Derbyc72594a2019-03-05 17:39:38 -0500542
joshualitt374b2f72015-07-21 08:05:03 -0700543 struct StrokeInfo {
544 SkScalar fFrameWidth;
545 SkScalar fMiterLimit;
546 SkPaint::Join fJoin;
547 };
548
549 enum TextType {
550 kHasDistanceField_TextType = 0x1,
551 kHasBitmap_TextType = 0x2,
552 };
553
554 // all glyph / vertex offsets are into these pools.
Brian Salomon18923f92017-11-06 16:26:02 -0500555 char* fVertices;
joshualitt374b2f72015-07-21 08:05:03 -0700556 GrGlyph** fGlyphs;
557 Run* fRuns;
Herb Derbya00da612019-03-04 17:10:01 -0500558
559 // Lifetime: The GrStrikeCache is owned by and has the same lifetime as the GrRecordingContext.
560 // The GrRecordingContext also owns the GrTextBlob cache which owns this GrTextBlob.
561 GrStrikeCache* const fStrikeCache;
Mike Reed80747ef2018-01-23 15:29:32 -0500562 SkMaskFilterBase::BlurRec fBlurRec;
joshualitt374b2f72015-07-21 08:05:03 -0700563 StrokeInfo fStrokeInfo;
joshualitt374b2f72015-07-21 08:05:03 -0700564 Key fKey;
joshualitt8e0ef292016-02-19 14:13:03 -0800565 SkMatrix fInitialViewMatrix;
joshualitt7481e752016-01-22 06:08:48 -0800566 SkMatrix fInitialViewMatrixInverse;
joshualitt2f2ee832016-02-10 08:52:24 -0800567 size_t fSize;
Jim Van Verthbc2cdd12017-06-08 11:14:35 -0400568 SkColor fLuminanceColor;
joshualitt7481e752016-01-22 06:08:48 -0800569 SkScalar fInitialX;
570 SkScalar fInitialY;
joshualitt374b2f72015-07-21 08:05:03 -0700571
572 // We can reuse distance field text, but only if the new viewmatrix would not result in
573 // a mip change. Because there can be multiple runs in a blob, we track the overall
574 // maximum minimum scale, and minimum maximum scale, we can support before we need to regen
Herb Derbya00da612019-03-04 17:10:01 -0500575 SkScalar fMaxMinScale{-SK_ScalarMax};
576 SkScalar fMinMaxScale{SK_ScalarMax};
Herb Derby2bb343c2018-11-08 15:03:48 -0500577 int fRunCount{0};
578 int fRunCountLimit;
Herb Derbya00da612019-03-04 17:10:01 -0500579 uint8_t fTextType{0};
joshualitt374b2f72015-07-21 08:05:03 -0700580};
581
Brian Salomon18923f92017-11-06 16:26:02 -0500582/**
583 * Used to produce vertices for a subrun of a blob. The vertices are cached in the blob itself.
584 * This is invoked each time a sub run is drawn. It regenerates the vertex data as required either
585 * because of changes to the atlas or because of different draw parameters (e.g. color change). In
586 * rare cases the draw may have to interrupted and flushed in the middle of the sub run in order to
587 * free up atlas space. Thus, this generator is stateful and should be invoked in a loop until the
588 * entire sub run has been completed.
589 */
Herb Derby86240592018-05-24 16:12:31 -0400590class GrTextBlob::VertexRegenerator {
Brian Salomon18923f92017-11-06 16:26:02 -0500591public:
592 /**
593 * Consecutive VertexRegenerators often use the same SkGlyphCache. If the same instance of
594 * SkAutoGlyphCache is reused then it can save the cost of multiple detach/attach operations of
595 * SkGlyphCache.
596 */
Herb Derby86240592018-05-24 16:12:31 -0400597 VertexRegenerator(GrResourceProvider*, GrTextBlob*, int runIdx, int subRunIdx,
Robert Phillips4bc70112018-03-01 10:24:02 -0500598 const SkMatrix& viewMatrix, SkScalar x, SkScalar y, GrColor color,
Herb Derby081e6f32019-01-16 13:46:02 -0500599 GrDeferredUploadTarget*, GrStrikeCache*, GrAtlasManager*,
Herb Derby7956b592018-03-22 16:10:30 -0400600 SkExclusiveStrikePtr*);
Brian Salomon18923f92017-11-06 16:26:02 -0500601
602 struct Result {
603 /**
604 * Was regenerate() able to draw all the glyphs from the sub run? If not flush all glyph
605 * draws and call regenerate() again.
606 */
607 bool fFinished = true;
608
609 /**
610 * How many glyphs were regenerated. Will be equal to the sub run's glyph count if
611 * fType is kFinished.
612 */
613 int fGlyphsRegenerated = 0;
614
615 /**
616 * Pointer where the caller finds the first regenerated vertex.
617 */
618 const char* fFirstVertex;
619 };
620
Robert Phillipsd2e9f762018-03-07 11:54:37 -0500621 bool regenerate(Result*);
Brian Salomon18923f92017-11-06 16:26:02 -0500622
623private:
Brian Osman5d6be8f2019-01-08 12:02:51 -0500624 bool doRegen(Result*, bool regenPos, bool regenCol, bool regenTexCoords, bool regenGlyphs);
Brian Salomon18923f92017-11-06 16:26:02 -0500625
Robert Phillips4bc70112018-03-01 10:24:02 -0500626 GrResourceProvider* fResourceProvider;
Brian Salomon18923f92017-11-06 16:26:02 -0500627 const SkMatrix& fViewMatrix;
Herb Derby86240592018-05-24 16:12:31 -0400628 GrTextBlob* fBlob;
Brian Salomon18923f92017-11-06 16:26:02 -0500629 GrDeferredUploadTarget* fUploadTarget;
Herb Derby081e6f32019-01-16 13:46:02 -0500630 GrStrikeCache* fGlyphCache;
Robert Phillipsc4039ea2018-03-01 11:36:45 -0500631 GrAtlasManager* fFullAtlasManager;
Herb Derbye7efd082019-05-28 11:30:33 -0400632 SkExclusiveStrikePtr* fLazyStrike;
Herb Derby69ff8952018-11-12 11:39:12 -0500633 SubRun* fSubRun;
Brian Salomon18923f92017-11-06 16:26:02 -0500634 GrColor fColor;
635 SkScalar fTransX;
636 SkScalar fTransY;
637
638 uint32_t fRegenFlags = 0;
639 int fCurrGlyph = 0;
640 bool fBrokenRun = false;
641};
642
Herb Derby86240592018-05-24 16:12:31 -0400643#endif // GrTextBlob_DEFINED