1 /*
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 
8 #ifndef GrTextBlobCache_DEFINED
9 #define GrTextBlobCache_DEFINED
10 
11 #include "GrAtlasTextContext.h"
12 #include "SkMessageBus.h"
13 #include "SkRefCnt.h"
14 #include "SkTArray.h"
15 #include "SkTextBlobRunIterator.h"
16 #include "SkTHash.h"
17 
18 class GrTextBlobCache {
19 public:
20     /**
21      * The callback function used by the cache when it is still over budget after a purge. The
22      * passed in 'data' is the same 'data' handed to setOverbudgetCallback.
23      */
24     typedef void (*PFOverBudgetCB)(void* data);
25 
GrTextBlobCache(PFOverBudgetCB cb,void * data)26     GrTextBlobCache(PFOverBudgetCB cb, void* data)
27         : fPool(kPreAllocSize, kMinGrowthSize)
28         , fCallback(cb)
29         , fData(data)
30         , fBudget(kDefaultBudget) {
31         SkASSERT(cb && data);
32     }
33     ~GrTextBlobCache();
34 
35     // creates an uncached blob
makeBlob(int glyphCount,int runCount)36     sk_sp<GrAtlasTextBlob> makeBlob(int glyphCount, int runCount) {
37         return GrAtlasTextBlob::Make(&fPool, glyphCount, runCount);
38     }
39 
makeBlob(const SkTextBlob * blob)40     sk_sp<GrAtlasTextBlob> makeBlob(const SkTextBlob* blob) {
41         int glyphCount = 0;
42         int runCount = 0;
43         BlobGlyphCount(&glyphCount, &runCount, blob);
44         return GrAtlasTextBlob::Make(&fPool, glyphCount, runCount);
45     }
46 
makeCachedBlob(const SkTextBlob * blob,const GrAtlasTextBlob::Key & key,const SkMaskFilter::BlurRec & blurRec,const SkPaint & paint)47     sk_sp<GrAtlasTextBlob> makeCachedBlob(const SkTextBlob* blob,
48                                           const GrAtlasTextBlob::Key& key,
49                                           const SkMaskFilter::BlurRec& blurRec,
50                                           const SkPaint& paint) {
51         sk_sp<GrAtlasTextBlob> cacheBlob(this->makeBlob(blob));
52         cacheBlob->setupKey(key, blurRec, paint);
53         this->add(cacheBlob);
54         blob->notifyAddedToCache();
55         return cacheBlob;
56     }
57 
find(const GrAtlasTextBlob::Key & key)58     sk_sp<GrAtlasTextBlob> find(const GrAtlasTextBlob::Key& key) const {
59         const auto* idEntry = fBlobIDCache.find(key.fUniqueID);
60         return idEntry ? idEntry->find(key) : nullptr;
61     }
62 
remove(GrAtlasTextBlob * blob)63     void remove(GrAtlasTextBlob* blob) {
64         auto  id      = GrAtlasTextBlob::GetKey(*blob).fUniqueID;
65         auto* idEntry = fBlobIDCache.find(id);
66         SkASSERT(idEntry);
67 
68         fBlobList.remove(blob);
69         idEntry->removeBlob(blob);
70         if (idEntry->fBlobs.empty()) {
71             fBlobIDCache.remove(id);
72         }
73     }
74 
makeMRU(GrAtlasTextBlob * blob)75     void makeMRU(GrAtlasTextBlob* blob) {
76         if (fBlobList.head() == blob) {
77             return;
78         }
79 
80         fBlobList.remove(blob);
81         fBlobList.addToHead(blob);
82     }
83 
84     void freeAll();
85 
86     // TODO move to SkTextBlob
BlobGlyphCount(int * glyphCount,int * runCount,const SkTextBlob * blob)87     static void BlobGlyphCount(int* glyphCount, int* runCount, const SkTextBlob* blob) {
88         SkTextBlobRunIterator itCounter(blob);
89         for (; !itCounter.done(); itCounter.next(), (*runCount)++) {
90             *glyphCount += itCounter.glyphCount();
91         }
92     }
93 
setBudget(size_t budget)94     void setBudget(size_t budget) {
95         fBudget = budget;
96         this->checkPurge();
97     }
98 
99     struct PurgeBlobMessage {
100         uint32_t fID;
101     };
102 
103     static void PostPurgeBlobMessage(uint32_t);
104 
105 private:
106     using BitmapBlobList = SkTInternalLList<GrAtlasTextBlob>;
107 
108     struct BlobIDCacheEntry {
BlobIDCacheEntryBlobIDCacheEntry109         BlobIDCacheEntry() : fID(SK_InvalidGenID) {}
BlobIDCacheEntryBlobIDCacheEntry110         explicit BlobIDCacheEntry(uint32_t id) : fID(id) {}
111 
GetKeyBlobIDCacheEntry112         static uint32_t GetKey(const BlobIDCacheEntry& entry) {
113             return entry.fID;
114         }
115 
addBlobBlobIDCacheEntry116         void addBlob(sk_sp<GrAtlasTextBlob> blob) {
117             SkASSERT(blob);
118             SkASSERT(GrAtlasTextBlob::GetKey(*blob).fUniqueID == fID);
119             SkASSERT(!this->find(GrAtlasTextBlob::GetKey(*blob)));
120 
121             fBlobs.emplace_back(std::move(blob));
122         }
123 
removeBlobBlobIDCacheEntry124         void removeBlob(GrAtlasTextBlob* blob) {
125             SkASSERT(blob);
126             SkASSERT(GrAtlasTextBlob::GetKey(*blob).fUniqueID == fID);
127 
128             auto index = this->findBlobIndex(GrAtlasTextBlob::GetKey(*blob));
129             SkASSERT(index >= 0);
130 
131             fBlobs.removeShuffle(index);
132         }
133 
findBlobIDCacheEntry134         sk_sp<GrAtlasTextBlob> find(const GrAtlasTextBlob::Key& key) const {
135             auto index = this->findBlobIndex(key);
136             return index < 0 ? nullptr : fBlobs[index];
137         }
138 
findBlobIndexBlobIDCacheEntry139         int findBlobIndex(const GrAtlasTextBlob::Key& key) const{
140             for (int i = 0; i < fBlobs.count(); ++i) {
141                 if (GrAtlasTextBlob::GetKey(*fBlobs[i]) == key) {
142                     return i;
143                 }
144             }
145             return -1;
146         }
147 
148         uint32_t                             fID;
149         // Current clients don't generate multiple GrAtlasTextBlobs per SkTextBlob, so an array w/
150         // linear search is acceptable.  If usage changes, we should re-evaluate this structure.
151         SkSTArray<1, sk_sp<GrAtlasTextBlob>, true> fBlobs;
152     };
153 
add(sk_sp<GrAtlasTextBlob> blob)154     void add(sk_sp<GrAtlasTextBlob> blob) {
155         auto  id      = GrAtlasTextBlob::GetKey(*blob).fUniqueID;
156         auto* idEntry = fBlobIDCache.find(id);
157         if (!idEntry) {
158             idEntry = fBlobIDCache.set(id, BlobIDCacheEntry(id));
159         }
160 
161         // Safe to retain a raw ptr temporarily here, because the cache will hold a ref.
162         GrAtlasTextBlob* rawBlobPtr = blob.get();
163         fBlobList.addToHead(rawBlobPtr);
164         idEntry->addBlob(std::move(blob));
165 
166         this->checkPurge(rawBlobPtr);
167     }
168 
169     void checkPurge(GrAtlasTextBlob* blob = nullptr) {
170         // First, purge all stale blob IDs.
171         {
172             SkTArray<PurgeBlobMessage> msgs;
173             fPurgeBlobInbox.poll(&msgs);
174 
175             for (const auto& msg : msgs) {
176                 auto* idEntry = fBlobIDCache.find(msg.fID);
177                 if (!idEntry) {
178                     // no cache entries for id
179                     continue;
180                 }
181 
182                 // remove all blob entries from the LRU list
183                 for (const auto& blob : idEntry->fBlobs) {
184                     fBlobList.remove(blob.get());
185                 }
186 
187                 // drop the idEntry itself (unrefs all blobs)
188                 fBlobIDCache.remove(msg.fID);
189             }
190         }
191 
192         // If we are still overbudget, then unref until we are below budget again
193         if (fPool.size() > fBudget) {
194             BitmapBlobList::Iter iter;
195             iter.init(fBlobList, BitmapBlobList::Iter::kTail_IterStart);
196             GrAtlasTextBlob* lruBlob = nullptr;
197             while (fPool.size() > fBudget && (lruBlob = iter.get()) && lruBlob != blob) {
198                 // Backup the iterator before removing and unrefing the blob
199                 iter.prev();
200 
201                 this->remove(lruBlob);
202             }
203 
204             // If we break out of the loop with lruBlob == blob, then we haven't purged enough
205             // use the call back and try to free some more.  If we are still overbudget after this,
206             // then this single textblob is over our budget
207             if (blob && lruBlob == blob) {
208                 (*fCallback)(fData);
209             }
210 
211 #ifdef SPEW_BUDGET_MESSAGE
212             if (fPool.size() > fBudget) {
213                 SkDebugf("Single textblob is larger than our whole budget");
214             }
215 #endif
216         }
217     }
218 
219     // Budget was chosen to be ~4 megabytes.  The min alloc and pre alloc sizes in the pool are
220     // based off of the largest cached textblob I have seen in the skps(a couple of kilobytes).
221     static const int kPreAllocSize = 1 << 17;
222     static const int kMinGrowthSize = 1 << 17;
223     static const int kDefaultBudget = 1 << 22;
224     GrMemoryPool fPool;
225     BitmapBlobList fBlobList;
226     SkTHashMap<uint32_t, BlobIDCacheEntry> fBlobIDCache;
227     PFOverBudgetCB fCallback;
228     void* fData;
229     size_t fBudget;
230     SkMessageBus<PurgeBlobMessage>::Inbox fPurgeBlobInbox;
231 };
232 
233 #endif
234