• 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 #ifndef GrResourceCache_DEFINED
9 #define GrResourceCache_DEFINED
10 
11 #include <set>
12 #include <stack>
13 
14 #include "include/core/SkRefCnt.h"
15 #include "include/gpu/GrDirectContext.h"
16 #include "include/private/GrResourceKey.h"
17 #include "include/private/SkTArray.h"
18 #include "include/private/SkTHash.h"
19 #include "src/core/SkMessageBus.h"
20 #include "src/core/SkTDPQueue.h"
21 #include "src/core/SkTInternalLList.h"
22 #include "src/core/SkTMultiMap.h"
23 #include "src/gpu/GrGpuResource.h"
24 #include "src/gpu/GrGpuResourceCacheAccess.h"
25 #include "src/gpu/GrGpuResourcePriv.h"
26 
27 class GrCaps;
28 class GrProxyProvider;
29 class SkString;
30 class SkTraceMemoryDump;
31 class GrSingleOwner;
32 class GrTexture;
33 class GrThreadSafeCache;
34 
35 struct GrTextureFreedMessage {
36     GrTexture* fTexture;
37     GrDirectContext::DirectContextID fIntendedRecipient;
38 };
39 
SkShouldPostMessageToBus(const GrTextureFreedMessage & msg,GrDirectContext::DirectContextID potentialRecipient)40 static inline bool SkShouldPostMessageToBus(
41         const GrTextureFreedMessage& msg, GrDirectContext::DirectContextID potentialRecipient) {
42     return potentialRecipient == msg.fIntendedRecipient;
43 }
44 
45 /**
46  * Manages the lifetime of all GrGpuResource instances.
47  *
48  * Resources may have optionally have two types of keys:
49  *      1) A scratch key. This is for resources whose allocations are cached but not their contents.
50  *         Multiple resources can share the same scratch key. This is so a caller can have two
51  *         resource instances with the same properties (e.g. multipass rendering that ping-pongs
52  *         between two temporary surfaces). The scratch key is set at resource creation time and
53  *         should never change. Resources need not have a scratch key.
54  *      2) A unique key. This key's meaning is specific to the domain that created the key. Only one
55  *         resource may have a given unique key. The unique key can be set, cleared, or changed
56  *         anytime after resource creation.
57  *
58  * A unique key always takes precedence over a scratch key when a resource has both types of keys.
59  * If a resource has neither key type then it will be deleted as soon as the last reference to it
60  * is dropped.
61  */
62 class GrResourceCache {
63 public:
64     GrResourceCache(GrSingleOwner* owner,
65                     GrDirectContext::DirectContextID owningContextID,
66                     uint32_t familyID);
67     ~GrResourceCache();
68 
69     // Default maximum number of bytes of gpu memory of budgeted resources in the cache.
70     static const size_t kDefaultMaxSize             = 256 * (1 << 20);
71 
72     /** Used to access functionality needed by GrGpuResource for lifetime management. */
73     class ResourceAccess;
74     ResourceAccess resourceAccess();
75 
76     /**
77      * Get current resource tag for gpu cache recycle.
78      */
79     GrGpuResourceTag getCurrentGrResourceTag() const;
80 
81     /**
82      * Set current resourcetag for gpu cache recycle.
83      */
84     void setCurrentGrResourceTag(const GrGpuResourceTag tag);
85 
86     /** Unique ID of the owning GrContext. */
contextUniqueID()87     uint32_t contextUniqueID() const { return fContextUniqueID; }
88 
89     /** Sets the max gpu memory byte size of the cache. */
90     void setLimit(size_t bytes);
91 
92     /**
93      * Returns the number of resources.
94      */
getResourceCount()95     int getResourceCount() const {
96         return fPurgeableQueue.count() + fNonpurgeableResources.count();
97     }
98 
99     /**
100      * Returns the number of resources that count against the budget.
101      */
getBudgetedResourceCount()102     int getBudgetedResourceCount() const { return fBudgetedCount; }
103 
104     /**
105      * Returns the number of bytes consumed by resources.
106      */
getResourceBytes()107     size_t getResourceBytes() const { return fBytes; }
108 
109     /**
110      * Returns the number of bytes held by unlocked resources which are available for purging.
111      */
getPurgeableBytes()112     size_t getPurgeableBytes() const { return fPurgeableBytes; }
113 
114     /**
115      * Returns the number of bytes consumed by budgeted resources.
116      */
getBudgetedResourceBytes()117     size_t getBudgetedResourceBytes() const { return fBudgetedBytes; }
118 
119     /**
120      * Returns the number of bytes consumed by cached resources.
121      */
getMaxResourceBytes()122     size_t getMaxResourceBytes() const { return fMaxBytes; }
123 
124     /**
125      * Abandons the backend API resources owned by all GrGpuResource objects and removes them from
126      * the cache.
127      */
128     void abandonAll();
129 
130     /**
131      * Releases the backend API resources owned by all GrGpuResource objects and removes them from
132      * the cache.
133      */
134     void releaseAll();
135 
136     /**
137      * Release GrGpuResource objects and removes them from the cache by tag.
138      */
139     void releaseByTag(const GrGpuResourceTag tag);
140     /**
141      * Get all GrGpuResource tags.
142     */
143     std::set<GrGpuResourceTag> getAllGrGpuResourceTag() const;
144 
145     /**
146      * Find a resource that matches a scratch key.
147      */
148     GrGpuResource* findAndRefScratchResource(const GrScratchKey& scratchKey);
149 
150 #ifdef SK_DEBUG
151     // This is not particularly fast and only used for validation, so debug only.
countScratchEntriesForKey(const GrScratchKey & scratchKey)152     int countScratchEntriesForKey(const GrScratchKey& scratchKey) const {
153         return fScratchMap.countForKey(scratchKey);
154     }
155 #endif
156 
157     /**
158      * Find a resource that matches a unique key.
159      */
findAndRefUniqueResource(const GrUniqueKey & key)160     GrGpuResource* findAndRefUniqueResource(const GrUniqueKey& key) {
161         GrGpuResource* resource = fUniqueHash.find(key);
162         if (resource && this->isInCache(resource)) {
163             this->refAndMakeResourceMRU(resource);
164             return resource;
165         }
166         SkDebugf("OHOS resource is not in cache, return nullptr!");
167         return nullptr;
168     }
169 
170     /**
171      * Query whether a unique key exists in the cache.
172      */
hasUniqueKey(const GrUniqueKey & key)173     bool hasUniqueKey(const GrUniqueKey& key) const {
174         return SkToBool(fUniqueHash.find(key));
175     }
176 
177     /** Purges resources to become under budget and processes resources with invalidated unique
178         keys. */
179     void purgeAsNeeded();
180 
181     // Purge unlocked resources. If 'scratchResourcesOnly' is true the purgeable resources
182     // containing persistent data are spared. If it is false then all purgeable resources will
183     // be deleted.
184     void purgeUnlockedResources(bool scratchResourcesOnly=false) {
185         this->purgeUnlockedResources(/*purgeTime=*/nullptr, scratchResourcesOnly);
186     }
187 
188     void purgeUnlockedResourcesByTag(bool scratchResourceOnly, const GrGpuResourceTag tag);
189     void purgeUnlockAndSafeCacheGpuResources();
190 
191     // Purge unlocked resources not used since the passed point in time. If 'scratchResourcesOnly'
192     // is true the purgeable resources containing persistent data are spared. If it is false then
193     // all purgeable resources older than 'purgeTime' will be deleted.
194     void purgeResourcesNotUsedSince(GrStdSteadyClock::time_point purgeTime,
195                                     bool scratchResourcesOnly=false) {
196         this->purgeUnlockedResources(&purgeTime, scratchResourcesOnly);
197     }
198 
199     /** If it's possible to purge enough resources to get the provided amount of budget
200         headroom, do so and return true. If it's not possible, do nothing and return false.
201      */
202     bool purgeToMakeHeadroom(size_t desiredHeadroomBytes);
203 
overBudget()204     bool overBudget() const { return fBudgetedBytes > fMaxBytes; }
205 
206     /**
207      * Purge unlocked resources from the cache until the the provided byte count has been reached
208      * or we have purged all unlocked resources. The default policy is to purge in LRU order, but
209      * can be overridden to prefer purging scratch resources (in LRU order) prior to purging other
210      * resource types.
211      *
212      * @param maxBytesToPurge the desired number of bytes to be purged.
213      * @param preferScratchResources If true scratch resources will be purged prior to other
214      *                               resource types.
215      */
216     void purgeUnlockedResources(size_t bytesToPurge, bool preferScratchResources);
217 
218     /** Returns true if the cache would like a flush to occur in order to make more resources
219         purgeable. */
220     bool requestsFlush() const;
221 
222     /** Maintain a ref to this texture until we receive a GrTextureFreedMessage. */
223     void insertDelayedTextureUnref(GrTexture*);
224 
225 #if GR_CACHE_STATS
226     struct Stats {
227         int fTotal;
228         int fNumPurgeable;
229         int fNumNonPurgeable;
230 
231         int fScratch;
232         int fWrapped;
233         size_t fUnbudgetedSize;
234 
StatsStats235         Stats() { this->reset(); }
236 
resetStats237         void reset() {
238             fTotal = 0;
239             fNumPurgeable = 0;
240             fNumNonPurgeable = 0;
241             fScratch = 0;
242             fWrapped = 0;
243             fUnbudgetedSize = 0;
244         }
245 
updateStats246         void update(GrGpuResource* resource) {
247             if (resource->cacheAccess().isScratch()) {
248                 ++fScratch;
249             }
250             if (resource->resourcePriv().refsWrappedObjects()) {
251                 ++fWrapped;
252             }
253             if (GrBudgetedType::kBudgeted != resource->resourcePriv().budgetedType()) {
254                 fUnbudgetedSize += resource->gpuMemorySize();
255             }
256         }
257     };
258 
259     void getStats(Stats*) const;
260 
261 #if GR_TEST_UTILS
262     void dumpStats(SkString*) const;
263 
264     void dumpStatsKeyValuePairs(SkTArray<SkString>* keys, SkTArray<double>* value) const;
265 #endif
266 
267 #endif // GR_CACHE_STATS
268 
269 #if GR_TEST_UTILS
270     int countUniqueKeysWithTag(const char* tag) const;
271 
272     void changeTimestamp(uint32_t newTimestamp);
273 #endif
274 
275     // Enumerates all cached resources and dumps their details to traceMemoryDump.
276     void dumpMemoryStatistics(SkTraceMemoryDump* traceMemoryDump) const;
277     void dumpMemoryStatistics(SkTraceMemoryDump* traceMemoryDump, GrGpuResourceTag tag) const;
278 
setProxyProvider(GrProxyProvider * proxyProvider)279     void setProxyProvider(GrProxyProvider* proxyProvider) { fProxyProvider = proxyProvider; }
setThreadSafeCache(GrThreadSafeCache * threadSafeCache)280     void setThreadSafeCache(GrThreadSafeCache* threadSafeCache) {
281         fThreadSafeCache = threadSafeCache;
282     }
283 
284     std::set<GrGpuResourceTag> getAllGrGpuResourceTags() const; // Get the tag of all GPU resources
285 
286 private:
287     ///////////////////////////////////////////////////////////////////////////
288     /// @name Methods accessible via ResourceAccess
289     ////
290     void insertResource(GrGpuResource*);
291     void removeResource(GrGpuResource*);
292     void notifyARefCntReachedZero(GrGpuResource*, GrGpuResource::LastRemovedRef);
293     void changeUniqueKey(GrGpuResource*, const GrUniqueKey&);
294     void removeUniqueKey(GrGpuResource*);
295     void willRemoveScratchKey(const GrGpuResource*);
296     void didChangeBudgetStatus(GrGpuResource*);
297     void refResource(GrGpuResource* resource);
298     /// @}
299 
300     void refAndMakeResourceMRU(GrGpuResource*);
301     void processFreedGpuResources();
302     void addToNonpurgeableArray(GrGpuResource*);
303     void removeFromNonpurgeableArray(GrGpuResource*);
304 
wouldFit(size_t bytes)305     bool wouldFit(size_t bytes) const { return fBudgetedBytes+bytes <= fMaxBytes; }
306 
307     uint32_t getNextTimestamp();
308 
309     void purgeUnlockedResources(const GrStdSteadyClock::time_point* purgeTime,
310                                 bool scratchResourcesOnly);
311     bool isInCache(const GrGpuResource* r) const;
312     bool isInPurgeableCache(const GrGpuResource* r) const;
313     bool isInNonpurgeableCache(const GrGpuResource* r) const;
314 #ifdef SK_DEBUG
315     void validate() const;
316 #else
validate()317     void validate() const {}
318 #endif
319 
320     class AutoValidate;
321 
322     class AvailableForScratchUse;
323 
324     struct ScratchMapTraits {
GetKeyScratchMapTraits325         static const GrScratchKey& GetKey(const GrGpuResource& r) {
326             return r.resourcePriv().getScratchKey();
327         }
328 
HashScratchMapTraits329         static uint32_t Hash(const GrScratchKey& key) { return key.hash(); }
OnFreeScratchMapTraits330         static void OnFree(GrGpuResource*) { }
331     };
332     typedef SkTMultiMap<GrGpuResource, GrScratchKey, ScratchMapTraits> ScratchMap;
333 
334     struct UniqueHashTraits {
GetKeyUniqueHashTraits335         static const GrUniqueKey& GetKey(const GrGpuResource& r) { return r.getUniqueKey(); }
336 
HashUniqueHashTraits337         static uint32_t Hash(const GrUniqueKey& key) { return key.hash(); }
338     };
339     typedef SkTDynamicHash<GrGpuResource, GrUniqueKey, UniqueHashTraits> UniqueHash;
340 
341     class TextureAwaitingUnref {
342     public:
343         TextureAwaitingUnref();
344         TextureAwaitingUnref(GrTexture* texture);
345         TextureAwaitingUnref(const TextureAwaitingUnref&) = delete;
346         TextureAwaitingUnref& operator=(const TextureAwaitingUnref&) = delete;
347         TextureAwaitingUnref(TextureAwaitingUnref&&);
348         TextureAwaitingUnref& operator=(TextureAwaitingUnref&&);
349         ~TextureAwaitingUnref();
350         void addRef();
351         void unref();
352         bool finished();
353 
354     private:
355         GrTexture* fTexture = nullptr;
356         int fNumUnrefs = 0;
357     };
358     using TexturesAwaitingUnref = SkTHashMap<uint32_t, TextureAwaitingUnref>;
359 
CompareTimestamp(GrGpuResource * const & a,GrGpuResource * const & b)360     static bool CompareTimestamp(GrGpuResource* const& a, GrGpuResource* const& b) {
361         return a->cacheAccess().timestamp() < b->cacheAccess().timestamp();
362     }
363 
AccessResourceIndex(GrGpuResource * const & res)364     static int* AccessResourceIndex(GrGpuResource* const& res) {
365         return res->cacheAccess().accessCacheIndex();
366     }
367 
368     using TextureFreedMessageBus = SkMessageBus<GrTextureFreedMessage,
369                                                 GrDirectContext::DirectContextID>;
370 
371     typedef SkMessageBus<GrUniqueKeyInvalidatedMessage, uint32_t>::Inbox InvalidUniqueKeyInbox;
372     typedef SkTDPQueue<GrGpuResource*, CompareTimestamp, AccessResourceIndex> PurgeableQueue;
373     typedef SkTDArray<GrGpuResource*> ResourceArray;
374 
375     GrProxyProvider*                    fProxyProvider = nullptr;
376     GrThreadSafeCache*                  fThreadSafeCache = nullptr;
377 
378     // Whenever a resource is added to the cache or the result of a cache lookup, fTimestamp is
379     // assigned as the resource's timestamp and then incremented. fPurgeableQueue orders the
380     // purgeable resources by this value, and thus is used to purge resources in LRU order.
381     uint32_t                            fTimestamp = 0;
382     PurgeableQueue                      fPurgeableQueue;
383     ResourceArray                       fNonpurgeableResources;
384 
385     // This map holds all resources that can be used as scratch resources.
386     ScratchMap                          fScratchMap;
387     // This holds all resources that have unique keys.
388     UniqueHash                          fUniqueHash;
389 
390     // our budget, used in purgeAsNeeded()
391     size_t                              fMaxBytes = kDefaultMaxSize;
392 
393 #if GR_CACHE_STATS
394     int                                 fHighWaterCount = 0;
395     size_t                              fHighWaterBytes = 0;
396     int                                 fBudgetedHighWaterCount = 0;
397     size_t                              fBudgetedHighWaterBytes = 0;
398 #endif
399 
400     // our current stats for all resources
401     SkDEBUGCODE(int                     fCount = 0;)
402     size_t                              fBytes = 0;
403 
404     // our current stats for resources that count against the budget
405     int                                 fBudgetedCount = 0;
406     size_t                              fBudgetedBytes = 0;
407     size_t                              fPurgeableBytes = 0;
408     int                                 fNumBudgetedResourcesFlushWillMakePurgeable = 0;
409 
410     InvalidUniqueKeyInbox               fInvalidUniqueKeyInbox;
411     TextureFreedMessageBus::Inbox       fFreedTextureInbox;
412     TexturesAwaitingUnref               fTexturesAwaitingUnref;
413 
414     GrDirectContext::DirectContextID    fOwningContextID;
415     uint32_t                            fContextUniqueID = SK_InvalidUniqueID;
416     GrSingleOwner*                      fSingleOwner = nullptr;
417 
418     // This resource is allowed to be in the nonpurgeable array for the sake of validate() because
419     // we're in the midst of converting it to purgeable status.
420     SkDEBUGCODE(GrGpuResource*          fNewlyPurgeableResourceForValidation = nullptr;)
421 
422     //Indicates the cached resource tags.
423     std::stack<GrGpuResourceTag> grResourceTagCacheStack;
424 };
425 
426 class GrResourceCache::ResourceAccess {
427 private:
ResourceAccess(GrResourceCache * cache)428     ResourceAccess(GrResourceCache* cache) : fCache(cache) { }
ResourceAccess(const ResourceAccess & that)429     ResourceAccess(const ResourceAccess& that) : fCache(that.fCache) { }
430     ResourceAccess& operator=(const ResourceAccess&) = delete;
431 
432     /**
433      * Insert a resource into the cache.
434      */
insertResource(GrGpuResource * resource)435     void insertResource(GrGpuResource* resource) { fCache->insertResource(resource); }
436 
437     /**
438      * Removes a resource from the cache.
439      */
removeResource(GrGpuResource * resource)440     void removeResource(GrGpuResource* resource) { fCache->removeResource(resource); }
441 
442     /**
443      * Adds a ref to a resource with proper tracking if the resource has 0 refs prior to
444      * adding the ref.
445      */
refResource(GrGpuResource * resource)446     void refResource(GrGpuResource* resource) { fCache->refResource(resource); }
447 
448     /**
449      * Get current resource tag for gpu cache recycle.
450      */
getCurrentGrResourceTag()451     GrGpuResourceTag getCurrentGrResourceTag() const { return fCache->getCurrentGrResourceTag(); }
452 
453     /**
454      * Notifications that should be sent to the cache when the ref/io cnt status of resources
455      * changes.
456      */
457     enum RefNotificationFlags {
458         /** All types of refs on the resource have reached zero. */
459         kAllCntsReachedZero_RefNotificationFlag = 0x1,
460         /** The normal (not pending IO type) ref cnt has reached zero. */
461         kRefCntReachedZero_RefNotificationFlag  = 0x2,
462     };
463     /**
464      * Called by GrGpuResources when they detect one of their ref cnts have reached zero. This may
465      * either be the main ref or the command buffer usage ref.
466      */
notifyARefCntReachedZero(GrGpuResource * resource,GrGpuResource::LastRemovedRef removedRef)467     void notifyARefCntReachedZero(GrGpuResource* resource,
468                                   GrGpuResource::LastRemovedRef removedRef) {
469         fCache->notifyARefCntReachedZero(resource, removedRef);
470     }
471 
472     /**
473      * Called by GrGpuResources to change their unique keys.
474      */
changeUniqueKey(GrGpuResource * resource,const GrUniqueKey & newKey)475     void changeUniqueKey(GrGpuResource* resource, const GrUniqueKey& newKey) {
476          fCache->changeUniqueKey(resource, newKey);
477     }
478 
479     /**
480      * Called by a GrGpuResource to remove its unique key.
481      */
removeUniqueKey(GrGpuResource * resource)482     void removeUniqueKey(GrGpuResource* resource) { fCache->removeUniqueKey(resource); }
483 
484     /**
485      * Called by a GrGpuResource when it removes its scratch key.
486      */
willRemoveScratchKey(const GrGpuResource * resource)487     void willRemoveScratchKey(const GrGpuResource* resource) {
488         fCache->willRemoveScratchKey(resource);
489     }
490 
491     /**
492      * Called by GrGpuResources when they change from budgeted to unbudgeted or vice versa.
493      */
didChangeBudgetStatus(GrGpuResource * resource)494     void didChangeBudgetStatus(GrGpuResource* resource) { fCache->didChangeBudgetStatus(resource); }
495 
496     // No taking addresses of this type.
497     const ResourceAccess* operator&() const;
498     ResourceAccess* operator&();
499 
500     GrResourceCache* fCache;
501 
502     friend class GrGpuResource; // To access all the proxy inline methods.
503     friend class GrResourceCache; // To create this type.
504 };
505 
resourceAccess()506 inline GrResourceCache::ResourceAccess GrResourceCache::resourceAccess() {
507     return ResourceAccess(this);
508 }
509 
510 #endif
511