Flutter Engine
The Flutter Engine
SkResourceCache.h
Go to the documentation of this file.
1/*
2 * Copyright 2013 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 SkResourceCache_DEFINED
9#define SkResourceCache_DEFINED
10
13
14#include <cstddef>
15#include <cstdint>
16
17class SkCachedData;
20
21/**
22 * Cache object for bitmaps (with possible scale in X Y as part of the key).
23 *
24 * Multiple caches can be instantiated, but each instance is not implicitly
25 * thread-safe, so if a given instance is to be shared across threads, the
26 * caller must manage the access itself (e.g. via a mutex).
27 *
28 * As a convenience, a global instance is also defined, which can be safely
29 * access across threads via the static methods (e.g. FindAndLock, etc.).
30 */
32public:
33 struct Key {
34 /** Key subclasses must call this after their own fields and data are initialized.
35 * All fields and data must be tightly packed.
36 * @param nameSpace must be unique per Key subclass.
37 * @param sharedID == 0 means ignore this field, does not support group purging.
38 * @param dataSize is size of fields and data of the subclass, must be a multiple of 4.
39 */
40 void init(void* nameSpace, uint64_t sharedID, size_t dataSize);
41
42 /** Returns the size of this key. */
43 size_t size() const {
44 return fCount32 << 2;
45 }
46
47 void* getNamespace() const { return fNamespace; }
48 uint64_t getSharedID() const { return ((uint64_t)fSharedID_hi << 32) | fSharedID_lo; }
49
50 // This is only valid after having called init().
51 uint32_t hash() const { return fHash; }
52
53 bool operator==(const Key& other) const {
54 const uint32_t* a = this->as32();
55 const uint32_t* b = other.as32();
56 for (int i = 0; i < fCount32; ++i) { // (This checks fCount == other.fCount first.)
57 if (a[i] != b[i]) {
58 return false;
59 }
60 }
61 return true;
62 }
63
64 private:
65 int32_t fCount32; // local + user contents count32
66 uint32_t fHash;
67 // split uint64_t into hi and lo so we don't force ourselves to pad on 32bit machines.
68 uint32_t fSharedID_lo;
69 uint32_t fSharedID_hi;
70 void* fNamespace; // A unique namespace tag. This is hashed.
71 /* uint32_t fContents32[] */
72
73 const uint32_t* as32() const { return (const uint32_t*)this; }
74 };
75
76 struct Rec {
78
79 Rec() {}
80 virtual ~Rec() {}
81
82 uint32_t getHash() const { return this->getKey().hash(); }
83
84 virtual const Key& getKey() const = 0;
85 virtual size_t bytesUsed() const = 0;
86
87 // Called if the cache needs to purge/remove/delete the Rec. Default returns true.
88 // Subclass may return false if there are outstanding references to it (e.g. bitmaps).
89 // Will only be deleted/removed-from-the-cache when this returns true.
90 virtual bool canBePurged() { return true; }
91
92 // A rec is first created/initialized, and then added to the cache. As part of the add(),
93 // the cache will callback into the rec with postAddInstall, passing in whatever payload
94 // was passed to add/Add.
95 //
96 // This late-install callback exists because the process of add-ing might end up deleting
97 // the new rec (if an existing rec in the cache has the same key and cannot be purged).
98 // If the new rec will be deleted during add, the pre-existing one (with the same key)
99 // will have postAddInstall() called on it instead, so that either way an "install" will
100 // happen during the add.
101 virtual void postAddInstall(void*) {}
102
103 // for memory usage diagnostics
104 virtual const char* getCategory() const = 0;
105 virtual SkDiscardableMemory* diagnostic_only_getDiscardable() const { return nullptr; }
106
107 private:
108 Rec* fNext;
109 Rec* fPrev;
110
111 friend class SkResourceCache;
112 };
113
114 // Used with SkMessageBus
116 PurgeSharedIDMessage(uint64_t sharedID) : fSharedID(sharedID) {}
117 uint64_t fSharedID;
118 };
119
120 typedef const Rec* ID;
121
122 /**
123 * Callback function for find(). If called, the cache will have found a match for the
124 * specified Key, and will pass in the corresponding Rec, along with a caller-specified
125 * context. The function can read the data in Rec, and copy whatever it likes into context
126 * (casting context to whatever it really is).
127 *
128 * The return value determines what the cache will do with the Rec. If the function returns
129 * true, then the Rec is considered "valid". If false is returned, the Rec will be considered
130 * "stale" and will be purged from the cache.
131 */
132 typedef bool (*FindVisitor)(const Rec&, void* context);
133
134 /**
135 * Returns a locked/pinned SkDiscardableMemory instance for the specified
136 * number of bytes, or nullptr on failure.
137 */
138 typedef SkDiscardableMemory* (*DiscardableFactory)(size_t bytes);
139
140 /*
141 * The following static methods are thread-safe wrappers around a global
142 * instance of this cache.
143 */
144
145 /**
146 * Returns true if the visitor was called on a matching Key, and the visitor returned true.
147 *
148 * Find() will search the cache for the specified Key. If no match is found, return false and
149 * do not call the FindVisitor. If a match is found, return whatever the visitor returns.
150 * Its return value is interpreted to mean:
151 * true : Rec is valid
152 * false : Rec is "stale" -- the cache will purge it.
153 */
154 static bool Find(const Key& key, FindVisitor, void* context);
155 static void Add(Rec*, void* payload = nullptr);
156
157 typedef void (*Visitor)(const Rec&, void* context);
158 // Call the visitor for every Rec in the cache.
159 static void VisitAll(Visitor, void* context);
160
161 static size_t GetTotalBytesUsed();
162 static size_t GetTotalByteLimit();
163 static size_t SetTotalByteLimit(size_t newLimit);
164
165 static size_t SetSingleAllocationByteLimit(size_t);
166 static size_t GetSingleAllocationByteLimit();
168
169 static void PurgeAll();
170 static void CheckMessages();
171
172 static void TestDumpMemoryStatistics();
173
174 /** Dump memory usage statistics of every Rec in the cache using the
175 SkTraceMemoryDump interface.
176 */
178
179 /**
180 * Returns the DiscardableFactory used by the global cache, or nullptr.
181 */
183
184 static SkCachedData* NewCachedData(size_t bytes);
185
186 static void PostPurgeSharedID(uint64_t sharedID);
187
188 /**
189 * Call SkDebugf() with diagnostic information about the state of the cache
190 */
191 static void Dump();
192
193 ///////////////////////////////////////////////////////////////////////////
194
195 /**
196 * Construct the cache to call DiscardableFactory when it
197 * allocates memory for the pixels. In this mode, the cache has
198 * not explicit budget, and so methods like getTotalBytesUsed()
199 * and getTotalByteLimit() will return 0, and setTotalByteLimit
200 * will ignore its argument and return 0.
201 */
203
204 /**
205 * Construct the cache, allocating memory with malloc, and respect the
206 * byteLimit, purging automatically when a new image is added to the cache
207 * that pushes the total bytesUsed over the limit. Note: The limit can be
208 * changed at runtime with setTotalByteLimit.
209 */
210 explicit SkResourceCache(size_t byteLimit);
212
213 /**
214 * Returns true if the visitor was called on a matching Key, and the visitor returned true.
215 *
216 * find() will search the cache for the specified Key. If no match is found, return false and
217 * do not call the FindVisitor. If a match is found, return whatever the visitor returns.
218 * Its return value is interpreted to mean:
219 * true : Rec is valid
220 * false : Rec is "stale" -- the cache will purge it.
221 */
222 bool find(const Key&, FindVisitor, void* context);
223 void add(Rec*, void* payload = nullptr);
224 void visitAll(Visitor, void* context);
225
226 size_t getTotalBytesUsed() const { return fTotalBytesUsed; }
227 size_t getTotalByteLimit() const { return fTotalByteLimit; }
228
229 /**
230 * This is respected by SkBitmapProcState::possiblyScaleImage.
231 * 0 is no maximum at all; this is the default.
232 * setSingleAllocationByteLimit() returns the previous value.
233 */
234 size_t setSingleAllocationByteLimit(size_t maximumAllocationSize);
235 size_t getSingleAllocationByteLimit() const;
236 // returns the logical single allocation size (pinning against the budget when the cache
237 // is not backed by discardable memory.
239
240 /**
241 * Set the maximum number of bytes available to this cache. If the current
242 * cache exceeds this new value, it will be purged to try to fit within
243 * this new limit.
244 */
245 size_t setTotalByteLimit(size_t newLimit);
246
247 void purgeSharedID(uint64_t sharedID);
248
249 void purgeAll() {
250 this->purgeAsNeeded(true);
251 }
252
253 DiscardableFactory discardableFactory() const { return fDiscardableFactory; }
254
255 SkCachedData* newCachedData(size_t bytes);
256
257 /**
258 * Call SkDebugf() with diagnostic information about the state of the cache
259 */
260 void dump() const;
261
262private:
263 Rec* fHead;
264 Rec* fTail;
265
266 class Hash;
267 Hash* fHash;
268
269 DiscardableFactory fDiscardableFactory;
270
271 size_t fTotalBytesUsed;
272 size_t fTotalByteLimit;
273 size_t fSingleAllocationByteLimit;
274 int fCount;
275
277
278 void checkMessages();
279 void purgeAsNeeded(bool forcePurge = false);
280
281 // linklist management
282 void moveToHead(Rec*);
283 void addToHead(Rec*);
284 void release(Rec*);
285 void remove(Rec*);
286
287 void init(); // called by constructors
288
289#ifdef SK_DEBUG
290 void validate() const;
291#else
292 void validate() const {}
293#endif
294};
295#endif
SkResourceCache(DiscardableFactory)
static size_t SetSingleAllocationByteLimit(size_t)
static SkCachedData * NewCachedData(size_t bytes)
static void Add(Rec *, void *payload=nullptr)
size_t getSingleAllocationByteLimit() const
size_t getTotalByteLimit() const
static size_t GetEffectiveSingleAllocationByteLimit()
DiscardableFactory discardableFactory() const
size_t getTotalBytesUsed() const
void visitAll(Visitor, void *context)
static void PurgeAll()
static void PostPurgeSharedID(uint64_t sharedID)
static size_t GetTotalBytesUsed()
void purgeSharedID(uint64_t sharedID)
static void CheckMessages()
static DiscardableFactory GetDiscardableFactory()
bool find(const Key &, FindVisitor, void *context)
static void DumpMemoryStatistics(SkTraceMemoryDump *dump)
static size_t GetSingleAllocationByteLimit()
static void VisitAll(Visitor, void *context)
static size_t GetTotalByteLimit()
static bool Find(const Key &key, FindVisitor, void *context)
static void TestDumpMemoryStatistics()
size_t setTotalByteLimit(size_t newLimit)
SkCachedData * newCachedData(size_t bytes)
SkDiscardableMemory *(* DiscardableFactory)(size_t bytes)
bool(* FindVisitor)(const Rec &, void *context)
size_t setSingleAllocationByteLimit(size_t maximumAllocationSize)
size_t getEffectiveSingleAllocationByteLimit() const
void(* Visitor)(const Rec &, void *context)
static void Dump()
static size_t SetTotalByteLimit(size_t newLimit)
void add(Rec *, void *payload=nullptr)
static bool b
struct MyStruct a[10]
static bool init()
static uint32_t Hash(uint32_t key)
Definition: hashmap_test.cc:65
def remove(*paths)
uint64_t getSharedID() const
uint32_t hash() const
void * getNamespace() const
size_t size() const
bool operator==(const Key &other) const
void init(void *nameSpace, uint64_t sharedID, size_t dataSize)
virtual void postAddInstall(void *)
virtual const char * getCategory() const =0
virtual SkDiscardableMemory * diagnostic_only_getDiscardable() const
virtual size_t bytesUsed() const =0
uint32_t getHash() const
virtual const Key & getKey() const =0
SkResourceCache::Key Key
virtual bool canBePurged()