• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2014 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 
8 #include "SkTextBlob.h"
9 
10 #include "SkReadBuffer.h"
11 #include "SkWriteBuffer.h"
12 
13 //
14 // Textblob data is laid out into externally-managed storage as follows:
15 //
16 //    -----------------------------------------------------------------------------
17 //   | SkTextBlob | RunRecord | Glyphs[] | Pos[] | RunRecord | Glyphs[] | Pos[] | ...
18 //    -----------------------------------------------------------------------------
19 //
20 //  Each run record describes a text blob run, and can be used to determine the (implicit)
21 //  location of the following record.
22 
23 SkDEBUGCODE(static const unsigned kRunRecordMagic = 0xb10bcafe;)
24 
25 class SkTextBlob::RunRecord {
26 public:
RunRecord(uint32_t count,const SkPoint & offset,const SkPaint & font,GlyphPositioning pos)27     RunRecord(uint32_t count, const SkPoint& offset, const SkPaint& font, GlyphPositioning pos)
28         : fCount(count)
29         , fOffset(offset)
30         , fFont(font)
31         , fPositioning(pos) {
32         SkDEBUGCODE(fMagic = kRunRecordMagic);
33     }
34 
glyphCount() const35     uint32_t glyphCount() const {
36         return fCount;
37     }
38 
offset() const39     const SkPoint& offset() const {
40         return fOffset;
41     }
42 
font() const43     const SkPaint& font() const {
44         return fFont;
45     }
46 
positioning() const47     GlyphPositioning positioning() const {
48         return fPositioning;
49     }
50 
glyphBuffer() const51     uint16_t* glyphBuffer() const {
52         // Glyph are stored immediately following the record.
53         return reinterpret_cast<uint16_t*>(const_cast<RunRecord*>(this) + 1);
54     }
55 
posBuffer() const56     SkScalar* posBuffer() const {
57         // Position scalars follow the (aligned) glyph buffer.
58         return reinterpret_cast<SkScalar*>(reinterpret_cast<uint8_t*>(this->glyphBuffer()) +
59                                            SkAlign4(fCount * sizeof(uint16_t)));
60     }
61 
StorageSize(int glyphCount,SkTextBlob::GlyphPositioning positioning)62     static size_t StorageSize(int glyphCount, SkTextBlob::GlyphPositioning positioning) {
63         // RunRecord object + (aligned) glyph buffer + position buffer
64         return SkAlignPtr(sizeof(SkTextBlob::RunRecord)
65                         + SkAlign4(glyphCount* sizeof(uint16_t))
66                         + glyphCount * sizeof(SkScalar) * ScalarsPerGlyph(positioning));
67     }
68 
First(const SkTextBlob * blob)69     static const RunRecord* First(const SkTextBlob* blob) {
70         // The first record (if present) is stored following the blob object.
71         return reinterpret_cast<const RunRecord*>(blob + 1);
72     }
73 
Next(const RunRecord * run)74     static const RunRecord* Next(const RunRecord* run) {
75         return reinterpret_cast<const RunRecord*>(reinterpret_cast<const uint8_t*>(run)
76             + StorageSize(run->glyphCount(), run->positioning()));
77     }
78 
validate(uint8_t * storageTop) const79     void validate(uint8_t* storageTop) const {
80         SkASSERT(kRunRecordMagic == fMagic);
81         SkASSERT((uint8_t*)Next(this) <= storageTop);
82         SkASSERT(glyphBuffer() + fCount <= (uint16_t*)posBuffer());
83         SkASSERT(posBuffer() + fCount * ScalarsPerGlyph(fPositioning) <= (SkScalar*)Next(this));
84     }
85 
86 private:
87     friend class SkTextBlobBuilder;
88 
grow(uint32_t count)89     void grow(uint32_t count) {
90         SkScalar* initialPosBuffer = posBuffer();
91         uint32_t initialCount = fCount;
92         fCount += count;
93 
94         // Move the initial pos scalars to their new location.
95         size_t copySize = initialCount * sizeof(SkScalar) * ScalarsPerGlyph(fPositioning);
96         SkASSERT((uint8_t*)posBuffer() + copySize <= (uint8_t*)Next(this));
97 
98         // memmove, as the buffers may overlap
99         memmove(posBuffer(), initialPosBuffer, copySize);
100     }
101 
102     uint32_t         fCount;
103     SkPoint          fOffset;
104     SkPaint          fFont;
105     GlyphPositioning fPositioning;
106 
107     SkDEBUGCODE(unsigned fMagic;)
108 };
109 
SkTextBlob(int runCount,const SkRect & bounds)110 SkTextBlob::SkTextBlob(int runCount, const SkRect& bounds)
111     : fRunCount(runCount)
112     , fBounds(bounds) {
113 }
114 
~SkTextBlob()115 SkTextBlob::~SkTextBlob() {
116     const RunRecord* run = RunRecord::First(this);
117     for (int i = 0; i < fRunCount; ++i) {
118         const RunRecord* nextRun = RunRecord::Next(run);
119         SkDEBUGCODE(run->validate((uint8_t*)this + fStorageSize);)
120         run->~RunRecord();
121         run = nextRun;
122     }
123 }
124 
internal_dispose() const125 void SkTextBlob::internal_dispose() const {
126     // SkTextBlobs use externally-managed storage.
127     this->internal_dispose_restore_refcnt_to_1();
128     this->~SkTextBlob();
129     sk_free(const_cast<SkTextBlob*>(this));
130 }
131 
uniqueID() const132 uint32_t SkTextBlob::uniqueID() const {
133     static int32_t  gTextBlobGenerationID; // = 0;
134 
135     // loop in case our global wraps around, as we never want to return SK_InvalidGenID
136     while (SK_InvalidGenID == fUniqueID) {
137         fUniqueID = sk_atomic_inc(&gTextBlobGenerationID) + 1;
138     }
139 
140     return fUniqueID;
141 }
142 
flatten(SkWriteBuffer & buffer) const143 void SkTextBlob::flatten(SkWriteBuffer& buffer) const {
144     int runCount = fRunCount;
145 
146     buffer.write32(runCount);
147     buffer.writeRect(fBounds);
148 
149     SkPaint runPaint;
150     RunIterator it(this);
151     while (!it.done()) {
152         SkASSERT(it.glyphCount() > 0);
153 
154         buffer.write32(it.glyphCount());
155         buffer.write32(it.positioning());
156         buffer.writePoint(it.offset());
157         // This should go away when switching to SkFont
158         it.applyFontToPaint(&runPaint);
159         buffer.writePaint(runPaint);
160 
161         buffer.writeByteArray(it.glyphs(), it.glyphCount() * sizeof(uint16_t));
162         buffer.writeByteArray(it.pos(),
163             it.glyphCount() * sizeof(SkScalar) * ScalarsPerGlyph(it.positioning()));
164 
165         it.next();
166         SkDEBUGCODE(runCount--);
167     }
168     SkASSERT(0 == runCount);
169 }
170 
CreateFromBuffer(SkReadBuffer & reader)171 const SkTextBlob* SkTextBlob::CreateFromBuffer(SkReadBuffer& reader) {
172     int runCount = reader.read32();
173     if (runCount < 0) {
174         return NULL;
175     }
176 
177     SkRect bounds;
178     reader.readRect(&bounds);
179 
180     SkTextBlobBuilder blobBuilder;
181     for (int i = 0; i < runCount; ++i) {
182         int glyphCount = reader.read32();
183         GlyphPositioning pos = static_cast<GlyphPositioning>(reader.read32());
184         if (glyphCount <= 0 || pos > kFull_Positioning) {
185             return NULL;
186         }
187 
188         SkPoint offset;
189         reader.readPoint(&offset);
190         SkPaint font;
191         reader.readPaint(&font);
192 
193         const SkTextBlobBuilder::RunBuffer* buf = NULL;
194         switch (pos) {
195         case kDefault_Positioning:
196             buf = &blobBuilder.allocRun(font, glyphCount, offset.x(), offset.y(), &bounds);
197             break;
198         case kHorizontal_Positioning:
199             buf = &blobBuilder.allocRunPosH(font, glyphCount, offset.y(), &bounds);
200             break;
201         case kFull_Positioning:
202             buf = &blobBuilder.allocRunPos(font, glyphCount, &bounds);
203             break;
204         default:
205             return NULL;
206         }
207 
208         if (!reader.readByteArray(buf->glyphs, glyphCount * sizeof(uint16_t)) ||
209             !reader.readByteArray(buf->pos,
210                                   glyphCount * sizeof(SkScalar) * ScalarsPerGlyph(pos))) {
211             return NULL;
212         }
213     }
214 
215     return blobBuilder.build();
216 }
217 
ScalarsPerGlyph(GlyphPositioning pos)218 unsigned SkTextBlob::ScalarsPerGlyph(GlyphPositioning pos) {
219     // GlyphPositioning values are directly mapped to scalars-per-glyph.
220     SkASSERT(pos <= 2);
221     return pos;
222 }
223 
RunIterator(const SkTextBlob * blob)224 SkTextBlob::RunIterator::RunIterator(const SkTextBlob* blob)
225     : fCurrentRun(RunRecord::First(blob))
226     , fRemainingRuns(blob->fRunCount) {
227     SkDEBUGCODE(fStorageTop = (uint8_t*)blob + blob->fStorageSize;)
228 }
229 
done() const230 bool SkTextBlob::RunIterator::done() const {
231     return fRemainingRuns <= 0;
232 }
233 
next()234 void SkTextBlob::RunIterator::next() {
235     SkASSERT(!this->done());
236 
237     if (!this->done()) {
238         SkDEBUGCODE(fCurrentRun->validate(fStorageTop);)
239         fCurrentRun = RunRecord::Next(fCurrentRun);
240         fRemainingRuns--;
241     }
242 }
243 
glyphCount() const244 uint32_t SkTextBlob::RunIterator::glyphCount() const {
245     SkASSERT(!this->done());
246     return fCurrentRun->glyphCount();
247 }
248 
glyphs() const249 const uint16_t* SkTextBlob::RunIterator::glyphs() const {
250     SkASSERT(!this->done());
251     return fCurrentRun->glyphBuffer();
252 }
253 
pos() const254 const SkScalar* SkTextBlob::RunIterator::pos() const {
255     SkASSERT(!this->done());
256     return fCurrentRun->posBuffer();
257 }
258 
offset() const259 const SkPoint& SkTextBlob::RunIterator::offset() const {
260     SkASSERT(!this->done());
261     return fCurrentRun->offset();
262 }
263 
positioning() const264 SkTextBlob::GlyphPositioning SkTextBlob::RunIterator::positioning() const {
265     SkASSERT(!this->done());
266     return fCurrentRun->positioning();
267 }
268 
applyFontToPaint(SkPaint * paint) const269 void SkTextBlob::RunIterator::applyFontToPaint(SkPaint* paint) const {
270     SkASSERT(!this->done());
271 
272     const SkPaint& font = fCurrentRun->font();
273 
274     paint->setTypeface(font.getTypeface());
275     paint->setTextEncoding(font.getTextEncoding());
276     paint->setTextSize(font.getTextSize());
277     paint->setTextScaleX(font.getTextScaleX());
278     paint->setTextSkewX(font.getTextSkewX());
279     paint->setHinting(font.getHinting());
280 
281     uint32_t flagsMask = SkPaint::kAntiAlias_Flag
282                        | SkPaint::kUnderlineText_Flag
283                        | SkPaint::kStrikeThruText_Flag
284                        | SkPaint::kFakeBoldText_Flag
285                        | SkPaint::kLinearText_Flag
286                        | SkPaint::kSubpixelText_Flag
287                        | SkPaint::kDevKernText_Flag
288                        | SkPaint::kLCDRenderText_Flag
289                        | SkPaint::kEmbeddedBitmapText_Flag
290                        | SkPaint::kAutoHinting_Flag
291                        | SkPaint::kVerticalText_Flag
292                        | SkPaint::kGenA8FromLCD_Flag
293                        | SkPaint::kDistanceFieldTextTEMP_Flag;
294     paint->setFlags((paint->getFlags() & ~flagsMask) | (font.getFlags() & flagsMask));
295 }
296 
SkTextBlobBuilder()297 SkTextBlobBuilder::SkTextBlobBuilder()
298     : fStorageSize(0)
299     , fStorageUsed(0)
300     , fRunCount(0)
301     , fDeferredBounds(false)
302     , fLastRun(0) {
303     fBounds.setEmpty();
304 }
305 
~SkTextBlobBuilder()306 SkTextBlobBuilder::~SkTextBlobBuilder() {
307     if (NULL != fStorage.get()) {
308         // We are abandoning runs and must destruct the associated font data.
309         // The easiest way to accomplish that is to use the blob destructor.
310         build()->unref();
311     }
312 }
313 
updateDeferredBounds()314 void SkTextBlobBuilder::updateDeferredBounds() {
315     SkASSERT(!fDeferredBounds || fRunCount > 0);
316 
317     if (!fDeferredBounds) {
318         return;
319     }
320 
321     // FIXME: measure the current run & union bounds
322     fDeferredBounds = false;
323 }
324 
reserve(size_t size)325 void SkTextBlobBuilder::reserve(size_t size) {
326     // We don't currently pre-allocate, but maybe someday...
327     if (fStorageUsed + size <= fStorageSize) {
328         return;
329     }
330 
331     if (0 == fRunCount) {
332         SkASSERT(NULL == fStorage.get());
333         SkASSERT(0 == fStorageSize);
334         SkASSERT(0 == fStorageUsed);
335 
336         // the first allocation also includes blob storage
337         fStorageUsed += sizeof(SkTextBlob);
338     }
339 
340     fStorageSize = fStorageUsed + size;
341     // FYI: This relies on everything we store being relocatable, particularly SkPaint.
342     fStorage.realloc(fStorageSize);
343 }
344 
mergeRun(const SkPaint & font,SkTextBlob::GlyphPositioning positioning,int count,SkPoint offset)345 bool SkTextBlobBuilder::mergeRun(const SkPaint &font, SkTextBlob::GlyphPositioning positioning,
346                                  int count, SkPoint offset) {
347     if (0 == fLastRun) {
348         SkASSERT(0 == fRunCount);
349         return false;
350     }
351 
352     SkASSERT(fLastRun >= sizeof(SkTextBlob));
353     SkTextBlob::RunRecord* run = reinterpret_cast<SkTextBlob::RunRecord*>(fStorage.get() +
354                                                                           fLastRun);
355     SkASSERT(run->glyphCount() > 0);
356 
357     if (run->positioning() != positioning
358         || run->font() != font
359         || (run->glyphCount() + count < run->glyphCount())) {
360         return false;
361     }
362 
363     // we can merge same-font/same-positioning runs in the following cases:
364     //   * fully positioned run following another fully positioned run
365     //   * horizontally postioned run following another horizontally positioned run with the same
366     //     y-offset
367     if (SkTextBlob::kFull_Positioning != positioning
368         && (SkTextBlob::kHorizontal_Positioning != positioning
369             || run->offset().y() != offset.y())) {
370         return false;
371     }
372 
373     size_t sizeDelta = SkTextBlob::RunRecord::StorageSize(run->glyphCount() + count, positioning) -
374                        SkTextBlob::RunRecord::StorageSize(run->glyphCount(), positioning);
375     this->reserve(sizeDelta);
376 
377     // reserve may have realloced
378     run = reinterpret_cast<SkTextBlob::RunRecord*>(fStorage.get() + fLastRun);
379     uint32_t preMergeCount = run->glyphCount();
380     run->grow(count);
381 
382     // Callers expect the buffers to point at the newly added slice, ant not at the beginning.
383     fCurrentRunBuffer.glyphs = run->glyphBuffer() + preMergeCount;
384     fCurrentRunBuffer.pos = run->posBuffer()
385                           + preMergeCount * SkTextBlob::ScalarsPerGlyph(positioning);
386 
387     fStorageUsed += sizeDelta;
388 
389     SkASSERT(fStorageUsed <= fStorageSize);
390     run->validate(fStorage.get() + fStorageUsed);
391 
392     return true;
393 }
394 
allocInternal(const SkPaint & font,SkTextBlob::GlyphPositioning positioning,int count,SkPoint offset,const SkRect * bounds)395 void SkTextBlobBuilder::allocInternal(const SkPaint &font,
396                                       SkTextBlob::GlyphPositioning positioning,
397                                       int count, SkPoint offset, const SkRect* bounds) {
398     SkASSERT(count > 0);
399     SkASSERT(SkPaint::kGlyphID_TextEncoding == font.getTextEncoding());
400 
401     if (!this->mergeRun(font, positioning, count, offset)) {
402         updateDeferredBounds();
403 
404         size_t runSize = SkTextBlob::RunRecord::StorageSize(count, positioning);
405         this->reserve(runSize);
406 
407         SkASSERT(fStorageUsed >= sizeof(SkTextBlob));
408         SkASSERT(fStorageUsed + runSize <= fStorageSize);
409 
410         SkTextBlob::RunRecord* run = new (fStorage.get() + fStorageUsed)
411                                          SkTextBlob::RunRecord(count, offset, font, positioning);
412 
413         fCurrentRunBuffer.glyphs = run->glyphBuffer();
414         fCurrentRunBuffer.pos = run->posBuffer();
415 
416         fLastRun = fStorageUsed;
417         fStorageUsed += runSize;
418         fRunCount++;
419 
420         SkASSERT(fStorageUsed <= fStorageSize);
421         run->validate(fStorage.get() + fStorageUsed);
422     }
423 
424     if (!fDeferredBounds) {
425         if (bounds) {
426             fBounds.join(*bounds);
427         } else {
428             fDeferredBounds = true;
429         }
430     }
431 }
432 
allocRun(const SkPaint & font,int count,SkScalar x,SkScalar y,const SkRect * bounds)433 const SkTextBlobBuilder::RunBuffer& SkTextBlobBuilder::allocRun(const SkPaint& font, int count,
434                                                                 SkScalar x, SkScalar y,
435                                                                 const SkRect* bounds) {
436     this->allocInternal(font, SkTextBlob::kDefault_Positioning, count, SkPoint::Make(x, y), bounds);
437 
438     return fCurrentRunBuffer;
439 }
440 
allocRunPosH(const SkPaint & font,int count,SkScalar y,const SkRect * bounds)441 const SkTextBlobBuilder::RunBuffer& SkTextBlobBuilder::allocRunPosH(const SkPaint& font, int count,
442                                                                     SkScalar y,
443                                                                     const SkRect* bounds) {
444     this->allocInternal(font, SkTextBlob::kHorizontal_Positioning, count, SkPoint::Make(0, y),
445                         bounds);
446 
447     return fCurrentRunBuffer;
448 }
449 
allocRunPos(const SkPaint & font,int count,const SkRect * bounds)450 const SkTextBlobBuilder::RunBuffer& SkTextBlobBuilder::allocRunPos(const SkPaint& font, int count,
451                                                                    const SkRect *bounds) {
452     this->allocInternal(font, SkTextBlob::kFull_Positioning, count, SkPoint::Make(0, 0), bounds);
453 
454     return fCurrentRunBuffer;
455 }
456 
build()457 const SkTextBlob* SkTextBlobBuilder::build() {
458     SkASSERT((fRunCount > 0) == (NULL != fStorage.get()));
459 
460     this->updateDeferredBounds();
461 
462     if (0 == fRunCount) {
463         SkASSERT(NULL == fStorage.get());
464         fStorageUsed = sizeof(SkTextBlob);
465         fStorage.realloc(fStorageUsed);
466     }
467 
468     SkDEBUGCODE(
469         size_t validateSize = sizeof(SkTextBlob);
470         const SkTextBlob::RunRecord* run =
471             SkTextBlob::RunRecord::First(reinterpret_cast<const SkTextBlob*>(fStorage.get()));
472         for (int i = 0; i < fRunCount; ++i) {
473             validateSize += SkTextBlob::RunRecord::StorageSize(run->fCount, run->fPositioning);
474             run->validate(fStorage.get() + fStorageUsed);
475             run = SkTextBlob::RunRecord::Next(run);
476         }
477         SkASSERT(validateSize == fStorageUsed);
478     )
479 
480     const SkTextBlob* blob = new (fStorage.detach()) SkTextBlob(fRunCount, fBounds);
481     SkDEBUGCODE(const_cast<SkTextBlob*>(blob)->fStorageSize = fStorageSize;)
482 
483     fStorageUsed = 0;
484     fStorageSize = 0;
485     fRunCount = 0;
486     fLastRun = 0;
487     fBounds.setEmpty();
488 
489     return blob;
490 }
491 
492