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 "SkBitmapCache.h"
9 #include "SkImage.h"
10 #include "SkResourceCache.h"
11 #include "SkMipMap.h"
12 #include "SkPixelRef.h"
13 #include "SkRect.h"
14
15 /**
16 * Use this for bitmapcache and mipmapcache entries.
17 */
SkMakeResourceCacheSharedIDForBitmap(uint32_t bitmapGenID)18 uint64_t SkMakeResourceCacheSharedIDForBitmap(uint32_t bitmapGenID) {
19 uint64_t sharedID = SkSetFourByteTag('b', 'm', 'a', 'p');
20 return (sharedID << 32) | bitmapGenID;
21 }
22
SkNotifyBitmapGenIDIsStale(uint32_t bitmapGenID)23 void SkNotifyBitmapGenIDIsStale(uint32_t bitmapGenID) {
24 SkResourceCache::PostPurgeSharedID(SkMakeResourceCacheSharedIDForBitmap(bitmapGenID));
25 }
26
27 ///////////////////////////////////////////////////////////////////////////////////////////////////
28
GetAllocator()29 SkBitmap::Allocator* SkBitmapCache::GetAllocator() {
30 return SkResourceCache::GetAllocator();
31 }
32
33 /**
34 This function finds the bounds of the bitmap *within its pixelRef*.
35 If the bitmap lacks a pixelRef, it will return an empty rect, since
36 that doesn't make sense. This may be a useful enough function that
37 it should be somewhere else (in SkBitmap?).
38 */
get_bounds_from_bitmap(const SkBitmap & bm)39 static SkIRect get_bounds_from_bitmap(const SkBitmap& bm) {
40 if (!(bm.pixelRef())) {
41 return SkIRect::MakeEmpty();
42 }
43 SkIPoint origin = bm.pixelRefOrigin();
44 return SkIRect::MakeXYWH(origin.fX, origin.fY, bm.width(), bm.height());
45 }
46
47 /**
48 * This function finds the bounds of the image. Today this is just the entire bounds,
49 * but in the future we may support subsets within an image, in which case this should
50 * return that subset (see get_bounds_from_bitmap).
51 */
get_bounds_from_image(const SkImage * image)52 static SkIRect get_bounds_from_image(const SkImage* image) {
53 return SkIRect::MakeWH(image->width(), image->height());
54 }
55
Make(const SkBitmap & bm,int width,int height)56 SkBitmapCacheDesc SkBitmapCacheDesc::Make(const SkBitmap& bm, int width, int height) {
57 SkBitmapCacheDesc desc;
58 desc.fImageID = bm.getGenerationID();
59 desc.fWidth = width;
60 desc.fHeight = height;
61 desc.fBounds = get_bounds_from_bitmap(bm);
62 return desc;
63 }
64
Make(const SkBitmap & bm)65 SkBitmapCacheDesc SkBitmapCacheDesc::Make(const SkBitmap& bm) {
66 return Make(bm, bm.width(), bm.height());
67 }
68
Make(const SkImage * image,int width,int height)69 SkBitmapCacheDesc SkBitmapCacheDesc::Make(const SkImage* image, int width, int height) {
70 SkBitmapCacheDesc desc;
71 desc.fImageID = image->uniqueID();
72 desc.fWidth = width;
73 desc.fHeight = height;
74 desc.fBounds = get_bounds_from_image(image);
75 return desc;
76 }
77
Make(const SkImage * image)78 SkBitmapCacheDesc SkBitmapCacheDesc::Make(const SkImage* image) {
79 return Make(image, image->width(), image->height());
80 }
81
82 namespace {
83 static unsigned gBitmapKeyNamespaceLabel;
84
85 struct BitmapKey : public SkResourceCache::Key {
86 public:
BitmapKey__anon2a2c9cf70111::BitmapKey87 BitmapKey(uint32_t genID, int width, int height, const SkIRect& bounds)
88 : fGenID(genID)
89 , fWidth(width)
90 , fHeight(height)
91 , fBounds(bounds)
92 {
93 this->init(&gBitmapKeyNamespaceLabel, SkMakeResourceCacheSharedIDForBitmap(fGenID),
94 sizeof(fGenID) + sizeof(fWidth) + sizeof(fHeight) + sizeof(fBounds));
95 }
96
BitmapKey__anon2a2c9cf70111::BitmapKey97 BitmapKey(const SkBitmapCacheDesc& desc)
98 : fGenID(desc.fImageID)
99 , fWidth(desc.fWidth)
100 , fHeight(desc.fHeight)
101 , fBounds(desc.fBounds)
102 {
103 this->init(&gBitmapKeyNamespaceLabel, SkMakeResourceCacheSharedIDForBitmap(fGenID),
104 sizeof(fGenID) + sizeof(fWidth) + sizeof(fHeight) + sizeof(fBounds));
105 }
106
dump__anon2a2c9cf70111::BitmapKey107 void dump() const {
108 SkDebugf("-- add [%d %d] %d [%d %d %d %d]\n", fWidth, fHeight, fGenID,
109 fBounds.x(), fBounds.y(), fBounds.width(), fBounds.height());
110 }
111
112 const uint32_t fGenID;
113 const int fWidth;
114 const int fHeight;
115 const SkIRect fBounds;
116 };
117
118 struct BitmapRec : public SkResourceCache::Rec {
BitmapRec__anon2a2c9cf70111::BitmapRec119 BitmapRec(uint32_t genID, int width, int height, const SkIRect& bounds,
120 const SkBitmap& result)
121 : fKey(genID, width, height, bounds)
122 , fBitmap(result)
123 {
124 #ifdef TRACE_NEW_BITMAP_CACHE_RECS
125 fKey.dump();
126 #endif
127 }
128
BitmapRec__anon2a2c9cf70111::BitmapRec129 BitmapRec(const SkBitmapCacheDesc& desc, const SkBitmap& result)
130 : fKey(desc)
131 , fBitmap(result)
132 {
133 #ifdef TRACE_NEW_BITMAP_CACHE_RECS
134 fKey.dump();
135 #endif
136 }
137
getKey__anon2a2c9cf70111::BitmapRec138 const Key& getKey() const override { return fKey; }
bytesUsed__anon2a2c9cf70111::BitmapRec139 size_t bytesUsed() const override { return sizeof(fKey) + fBitmap.getSize(); }
140
getCategory__anon2a2c9cf70111::BitmapRec141 const char* getCategory() const override { return "bitmap"; }
diagnostic_only_getDiscardable__anon2a2c9cf70111::BitmapRec142 SkDiscardableMemory* diagnostic_only_getDiscardable() const override {
143 return fBitmap.pixelRef()->diagnostic_only_getDiscardable();
144 }
145
Finder__anon2a2c9cf70111::BitmapRec146 static bool Finder(const SkResourceCache::Rec& baseRec, void* contextBitmap) {
147 const BitmapRec& rec = static_cast<const BitmapRec&>(baseRec);
148 SkBitmap* result = (SkBitmap*)contextBitmap;
149
150 *result = rec.fBitmap;
151 result->lockPixels();
152 return SkToBool(result->getPixels());
153 }
154
155 private:
156 BitmapKey fKey;
157 SkBitmap fBitmap;
158 };
159 } // namespace
160
161 #define CHECK_LOCAL(localCache, localName, globalName, ...) \
162 ((localCache) ? localCache->localName(__VA_ARGS__) : SkResourceCache::globalName(__VA_ARGS__))
163
FindWH(const SkBitmapCacheDesc & desc,SkBitmap * result,SkResourceCache * localCache)164 bool SkBitmapCache::FindWH(const SkBitmapCacheDesc& desc, SkBitmap* result,
165 SkResourceCache* localCache) {
166 if (0 == desc.fWidth || 0 == desc.fHeight) {
167 // degenerate
168 return false;
169 }
170 return CHECK_LOCAL(localCache, find, Find, BitmapKey(desc), BitmapRec::Finder, result);
171 }
172
AddWH(const SkBitmapCacheDesc & desc,const SkBitmap & result,SkResourceCache * localCache)173 bool SkBitmapCache::AddWH(const SkBitmapCacheDesc& desc, const SkBitmap& result,
174 SkResourceCache* localCache) {
175 if (0 == desc.fWidth || 0 == desc.fHeight) {
176 // degenerate, and the key we use for mipmaps
177 return false;
178 }
179 SkASSERT(result.isImmutable());
180 BitmapRec* rec = new BitmapRec(desc, result);
181 CHECK_LOCAL(localCache, add, Add, rec);
182 return true;
183 }
184
Find(uint32_t genID,const SkIRect & subset,SkBitmap * result,SkResourceCache * localCache)185 bool SkBitmapCache::Find(uint32_t genID, const SkIRect& subset, SkBitmap* result,
186 SkResourceCache* localCache) {
187 BitmapKey key(genID, SK_Scalar1, SK_Scalar1, subset);
188
189 return CHECK_LOCAL(localCache, find, Find, key, BitmapRec::Finder, result);
190 }
191
Add(SkPixelRef * pr,const SkIRect & subset,const SkBitmap & result,SkResourceCache * localCache)192 bool SkBitmapCache::Add(SkPixelRef* pr, const SkIRect& subset, const SkBitmap& result,
193 SkResourceCache* localCache) {
194 SkASSERT(result.isImmutable());
195
196 if (subset.isEmpty()
197 || subset.top() < 0
198 || subset.left() < 0
199 || result.width() != subset.width()
200 || result.height() != subset.height()) {
201 return false;
202 } else {
203 BitmapRec* rec = new BitmapRec(pr->getGenerationID(), 1, 1, subset, result);
204
205 CHECK_LOCAL(localCache, add, Add, rec);
206 pr->notifyAddedToCache();
207 return true;
208 }
209 }
210
Find(uint32_t genID,SkBitmap * result,SkResourceCache * localCache)211 bool SkBitmapCache::Find(uint32_t genID, SkBitmap* result, SkResourceCache* localCache) {
212 BitmapKey key(genID, SK_Scalar1, SK_Scalar1, SkIRect::MakeEmpty());
213
214 return CHECK_LOCAL(localCache, find, Find, key, BitmapRec::Finder, result);
215 }
216
Add(uint32_t genID,const SkBitmap & result,SkResourceCache * localCache)217 void SkBitmapCache::Add(uint32_t genID, const SkBitmap& result, SkResourceCache* localCache) {
218 SkASSERT(result.isImmutable());
219
220 BitmapRec* rec = new BitmapRec(genID, 1, 1, SkIRect::MakeEmpty(), result);
221
222 CHECK_LOCAL(localCache, add, Add, rec);
223 }
224
225 //////////////////////////////////////////////////////////////////////////////////////////
226 //////////////////////////////////////////////////////////////////////////////////////////
227
228 namespace {
229 static unsigned gMipMapKeyNamespaceLabel;
230
231 struct MipMapKey : public SkResourceCache::Key {
232 public:
MipMapKey__anon2a2c9cf70211::MipMapKey233 MipMapKey(uint32_t genID, const SkIRect& bounds) : fGenID(genID), fBounds(bounds) {
234 this->init(&gMipMapKeyNamespaceLabel, SkMakeResourceCacheSharedIDForBitmap(genID),
235 sizeof(fGenID) + sizeof(fBounds));
236 }
237
238 uint32_t fGenID;
239 SkIRect fBounds;
240 };
241
242 struct MipMapRec : public SkResourceCache::Rec {
MipMapRec__anon2a2c9cf70211::MipMapRec243 MipMapRec(const SkBitmap& src, const SkMipMap* result)
244 : fKey(src.getGenerationID(), get_bounds_from_bitmap(src))
245 , fMipMap(result)
246 {
247 fMipMap->attachToCacheAndRef();
248 }
249
~MipMapRec__anon2a2c9cf70211::MipMapRec250 virtual ~MipMapRec() {
251 fMipMap->detachFromCacheAndUnref();
252 }
253
getKey__anon2a2c9cf70211::MipMapRec254 const Key& getKey() const override { return fKey; }
bytesUsed__anon2a2c9cf70211::MipMapRec255 size_t bytesUsed() const override { return sizeof(fKey) + fMipMap->size(); }
getCategory__anon2a2c9cf70211::MipMapRec256 const char* getCategory() const override { return "mipmap"; }
diagnostic_only_getDiscardable__anon2a2c9cf70211::MipMapRec257 SkDiscardableMemory* diagnostic_only_getDiscardable() const override {
258 return fMipMap->diagnostic_only_getDiscardable();
259 }
260
Finder__anon2a2c9cf70211::MipMapRec261 static bool Finder(const SkResourceCache::Rec& baseRec, void* contextMip) {
262 const MipMapRec& rec = static_cast<const MipMapRec&>(baseRec);
263 const SkMipMap* mm = SkRef(rec.fMipMap);
264 // the call to ref() above triggers a "lock" in the case of discardable memory,
265 // which means we can now check for null (in case the lock failed).
266 if (nullptr == mm->data()) {
267 mm->unref(); // balance our call to ref()
268 return false;
269 }
270 // the call must call unref() when they are done.
271 *(const SkMipMap**)contextMip = mm;
272 return true;
273 }
274
275 private:
276 MipMapKey fKey;
277 const SkMipMap* fMipMap;
278 };
279 }
280
FindAndRef(const SkBitmapCacheDesc & desc,SkResourceCache * localCache)281 const SkMipMap* SkMipMapCache::FindAndRef(const SkBitmapCacheDesc& desc,
282 SkResourceCache* localCache) {
283 // Note: we ignore width/height from desc, just need id and bounds
284 MipMapKey key(desc.fImageID, desc.fBounds);
285 const SkMipMap* result;
286
287 if (!CHECK_LOCAL(localCache, find, Find, key, MipMapRec::Finder, &result)) {
288 result = nullptr;
289 }
290 return result;
291 }
292
get_fact(SkResourceCache * localCache)293 static SkResourceCache::DiscardableFactory get_fact(SkResourceCache* localCache) {
294 return localCache ? localCache->GetDiscardableFactory()
295 : SkResourceCache::GetDiscardableFactory();
296 }
297
AddAndRef(const SkBitmap & src,SkResourceCache * localCache)298 const SkMipMap* SkMipMapCache::AddAndRef(const SkBitmap& src, SkResourceCache* localCache) {
299 SkMipMap* mipmap = SkMipMap::Build(src, get_fact(localCache));
300 if (mipmap) {
301 MipMapRec* rec = new MipMapRec(src, mipmap);
302 CHECK_LOCAL(localCache, add, Add, rec);
303 src.pixelRef()->notifyAddedToCache();
304 }
305 return mipmap;
306 }
307