blob: 507c192f43e929b132aeba819243a7ec45191117 [file] [log] [blame]
bsalomonc8dc1f72014-08-21 13:02:13 -07001/*
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
bsalomon0ea80f42015-02-11 10:49:59 -08008#ifndef GrResourceCache_DEFINED
9#define GrResourceCache_DEFINED
bsalomonc8dc1f72014-08-21 13:02:13 -070010
bsalomon744998e2014-08-28 09:54:34 -070011#include "GrGpuResource.h"
bsalomon9f2d1572015-02-17 11:47:40 -080012#include "GrGpuResourceCacheAccess.h"
bsalomon3582d3e2015-02-13 14:20:05 -080013#include "GrGpuResourcePriv.h"
bsalomon744998e2014-08-28 09:54:34 -070014#include "GrResourceKey.h"
bsalomon23e619c2015-02-06 11:54:28 -080015#include "SkMessageBus.h"
bsalomon8b79d232014-11-10 10:19:06 -080016#include "SkRefCnt.h"
bsalomon23e619c2015-02-06 11:54:28 -080017#include "SkTArray.h"
bsalomon9f2d1572015-02-17 11:47:40 -080018#include "SkTDPQueue.h"
bsalomonc8dc1f72014-08-21 13:02:13 -070019#include "SkTInternalLList.h"
bsalomon744998e2014-08-28 09:54:34 -070020#include "SkTMultiMap.h"
bsalomonc8dc1f72014-08-21 13:02:13 -070021
robertphillips63926682015-08-20 09:39:02 -070022class GrCaps;
Robert Phillips1afd4cd2018-01-08 13:40:32 -050023class GrProxyProvider;
mtkleinb9eb4ac2015-02-02 18:26:03 -080024class SkString;
ericrk0a5fa482015-09-15 14:16:10 -070025class SkTraceMemoryDump;
mtkleinb9eb4ac2015-02-02 18:26:03 -080026
Brian Osman13dddce2017-05-09 13:19:50 -040027struct GrGpuResourceFreedMessage {
28 GrGpuResource* fResource;
29 uint32_t fOwningUniqueID;
Brian Salomon238069b2018-07-11 15:58:57 -040030 bool shouldSend(uint32_t inboxID) const {
31 // The inbox's ID is the unique ID of the owning GrContext.
32 return inboxID == fOwningUniqueID;
33 }
Brian Osman13dddce2017-05-09 13:19:50 -040034};
35
bsalomonc8dc1f72014-08-21 13:02:13 -070036/**
bsalomon71cb0c22014-11-14 12:10:14 -080037 * Manages the lifetime of all GrGpuResource instances.
38 *
39 * Resources may have optionally have two types of keys:
40 * 1) A scratch key. This is for resources whose allocations are cached but not their contents.
41 * Multiple resources can share the same scratch key. This is so a caller can have two
bsalomon84c8e622014-11-17 09:33:27 -080042 * resource instances with the same properties (e.g. multipass rendering that ping-pongs
bsalomon8718aaf2015-02-19 07:24:21 -080043 * between two temporary surfaces). The scratch key is set at resource creation time and
bsalomon71cb0c22014-11-14 12:10:14 -080044 * should never change. Resources need not have a scratch key.
bsalomon8718aaf2015-02-19 07:24:21 -080045 * 2) A unique key. This key's meaning is specific to the domain that created the key. Only one
bsalomonf99e9612015-02-19 08:24:16 -080046 * resource may have a given unique key. The unique key can be set, cleared, or changed
47 * anytime after resource creation.
48 *
bsalomon8718aaf2015-02-19 07:24:21 -080049 * A unique key always takes precedence over a scratch key when a resource has both types of keys.
bsalomon71cb0c22014-11-14 12:10:14 -080050 * If a resource has neither key type then it will be deleted as soon as the last reference to it
bsalomon8718aaf2015-02-19 07:24:21 -080051 * is dropped.
bsalomonc8dc1f72014-08-21 13:02:13 -070052 */
bsalomon0ea80f42015-02-11 10:49:59 -080053class GrResourceCache {
bsalomonc8dc1f72014-08-21 13:02:13 -070054public:
Robert Phillips1afd4cd2018-01-08 13:40:32 -050055 GrResourceCache(const GrCaps*, uint32_t contextUniqueID);
bsalomon0ea80f42015-02-11 10:49:59 -080056 ~GrResourceCache();
bsalomonc8dc1f72014-08-21 13:02:13 -070057
bsalomon3f324322015-04-08 11:01:54 -070058 // Default maximum number of budgeted resources in the cache.
59 static const int kDefaultMaxCount = 2 * (1 << 12);
60 // Default maximum number of bytes of gpu memory of budgeted resources in the cache.
61 static const size_t kDefaultMaxSize = 96 * (1 << 20);
Brian Salomon5e150852017-03-22 14:53:13 -040062 // Default number of external flushes a budgeted resources can go unused in the cache before it
63 // is purged. Using a value <= 0 disables this feature. This will be removed once Chrome
64 // starts using time-based purging.
bsalomone2e87f32016-09-22 12:42:11 -070065 static const int kDefaultMaxUnusedFlushes =
66 1 * /* flushes per frame */
67 60 * /* fps */
68 30; /* seconds */
bsalomon3f324322015-04-08 11:01:54 -070069
bsalomon71cb0c22014-11-14 12:10:14 -080070 /** Used to access functionality needed by GrGpuResource for lifetime management. */
71 class ResourceAccess;
72 ResourceAccess resourceAccess();
bsalomonc8dc1f72014-08-21 13:02:13 -070073
Brian Salomon238069b2018-07-11 15:58:57 -040074 /** Unique ID of the owning GrContext. */
75 uint32_t contextUniqueID() const { return fContextUniqueID; }
76
bsalomon71cb0c22014-11-14 12:10:14 -080077 /**
bsalomon3f324322015-04-08 11:01:54 -070078 * Sets the cache limits in terms of number of resources, max gpu memory byte size, and number
bsalomone2e87f32016-09-22 12:42:11 -070079 * of external GrContext flushes that a resource can be unused before it is evicted. The latter
80 * value is a suggestion and there is no promise that a resource will be purged immediately
81 * after it hasn't been used in maxUnusedFlushes flushes.
bsalomon71cb0c22014-11-14 12:10:14 -080082 */
bsalomon3f324322015-04-08 11:01:54 -070083 void setLimits(int count, size_t bytes, int maxUnusedFlushes = kDefaultMaxUnusedFlushes);
bsalomonc8dc1f72014-08-21 13:02:13 -070084
bsalomon71cb0c22014-11-14 12:10:14 -080085 /**
bsalomondace19e2014-11-17 07:34:06 -080086 * Returns the number of resources.
bsalomon71cb0c22014-11-14 12:10:14 -080087 */
bsalomonf320e042015-02-17 15:09:34 -080088 int getResourceCount() const {
89 return fPurgeableQueue.count() + fNonpurgeableResources.count();
90 }
bsalomon8b79d232014-11-10 10:19:06 -080091
bsalomon71cb0c22014-11-14 12:10:14 -080092 /**
bsalomondace19e2014-11-17 07:34:06 -080093 * Returns the number of resources that count against the budget.
94 */
95 int getBudgetedResourceCount() const { return fBudgetedCount; }
96
97 /**
98 * Returns the number of bytes consumed by resources.
bsalomon71cb0c22014-11-14 12:10:14 -080099 */
100 size_t getResourceBytes() const { return fBytes; }
101
102 /**
Derek Sollenbergeree479142017-05-24 11:41:33 -0400103 * Returns the number of bytes held by unlocked reosources which are available for purging.
104 */
105 size_t getPurgeableBytes() const { return fPurgeableBytes; }
106
107 /**
bsalomondace19e2014-11-17 07:34:06 -0800108 * Returns the number of bytes consumed by budgeted resources.
109 */
110 size_t getBudgetedResourceBytes() const { return fBudgetedBytes; }
111
112 /**
bsalomon71cb0c22014-11-14 12:10:14 -0800113 * Returns the cached resources count budget.
114 */
115 int getMaxResourceCount() const { return fMaxCount; }
116
117 /**
118 * Returns the number of bytes consumed by cached resources.
119 */
120 size_t getMaxResourceBytes() const { return fMaxBytes; }
121
122 /**
123 * Abandons the backend API resources owned by all GrGpuResource objects and removes them from
124 * the cache.
125 */
bsalomonc8dc1f72014-08-21 13:02:13 -0700126 void abandonAll();
127
bsalomon71cb0c22014-11-14 12:10:14 -0800128 /**
129 * Releases the backend API resources owned by all GrGpuResource objects and removes them from
130 * the cache.
131 */
bsalomonc8dc1f72014-08-21 13:02:13 -0700132 void releaseAll();
133
bsalomon000f8292014-10-15 19:04:14 -0700134 enum {
135 /** Preferentially returns scratch resources with no pending IO. */
136 kPreferNoPendingIO_ScratchFlag = 0x1,
137 /** Will not return any resources that match but have pending IO. */
138 kRequireNoPendingIO_ScratchFlag = 0x2,
139 };
bsalomon71cb0c22014-11-14 12:10:14 -0800140
141 /**
142 * Find a resource that matches a scratch key.
143 */
robertphillips6e83ac72015-08-13 05:19:14 -0700144 GrGpuResource* findAndRefScratchResource(const GrScratchKey& scratchKey,
145 size_t resourceSize,
146 uint32_t flags);
halcanary9d524f22016-03-29 09:03:52 -0700147
bsalomon8b79d232014-11-10 10:19:06 -0800148#ifdef SK_DEBUG
149 // This is not particularly fast and only used for validation, so debug only.
bsalomon7775c852014-12-30 12:50:52 -0800150 int countScratchEntriesForKey(const GrScratchKey& scratchKey) const {
bsalomon8b79d232014-11-10 10:19:06 -0800151 return fScratchMap.countForKey(scratchKey);
152 }
153#endif
154
bsalomon71cb0c22014-11-14 12:10:14 -0800155 /**
bsalomon8718aaf2015-02-19 07:24:21 -0800156 * Find a resource that matches a unique key.
bsalomon71cb0c22014-11-14 12:10:14 -0800157 */
robertphillipsee843b22016-10-04 05:30:20 -0700158 GrGpuResource* findAndRefUniqueResource(const GrUniqueKey& key) {
159 GrGpuResource* resource = fUniqueHash.find(key);
160 if (resource) {
161 this->refAndMakeResourceMRU(resource);
162 }
163 return resource;
164 }
bsalomon8b79d232014-11-10 10:19:06 -0800165
Greg Danielcd871402017-09-26 12:49:26 -0400166 /**
bsalomon8718aaf2015-02-19 07:24:21 -0800167 * Query whether a unique key exists in the cache.
bsalomon71cb0c22014-11-14 12:10:14 -0800168 */
bsalomon8718aaf2015-02-19 07:24:21 -0800169 bool hasUniqueKey(const GrUniqueKey& key) const {
170 return SkToBool(fUniqueHash.find(key));
bsalomon8b79d232014-11-10 10:19:06 -0800171 }
bsalomonbcf0a522014-10-08 08:40:09 -0700172
bsalomon8718aaf2015-02-19 07:24:21 -0800173 /** Purges resources to become under budget and processes resources with invalidated unique
bsalomon23e619c2015-02-06 11:54:28 -0800174 keys. */
robertphillipsee843b22016-10-04 05:30:20 -0700175 void purgeAsNeeded();
bsalomon23e619c2015-02-06 11:54:28 -0800176
bsalomon71cb0c22014-11-14 12:10:14 -0800177 /** Purges all resources that don't have external owners. */
Robert Phillips6eba0632018-03-28 12:25:42 -0400178 void purgeAllUnlocked() { this->purgeUnlockedResources(false); }
179
180 // Purge unlocked resources. If 'scratchResourcesOnly' is true the purgeable resources
181 // containing persistent data are spared. If it is false then all purgeable resources will
182 // be deleted.
183 void purgeUnlockedResources(bool scratchResourcesOnly);
bsalomon71cb0c22014-11-14 12:10:14 -0800184
Brian Salomon5e150852017-03-22 14:53:13 -0400185 /** Purge all resources not used since the passed in time. */
186 void purgeResourcesNotUsedSince(GrStdSteadyClock::time_point);
187
Robert Phillipseafd48a2017-11-16 07:52:08 -0500188 bool overBudget() const { return fBudgetedBytes > fMaxBytes || fBudgetedCount > fMaxCount; }
189
Derek Sollenberger5480a182017-05-25 16:43:59 -0400190 /**
191 * Purge unlocked resources from the cache until the the provided byte count has been reached
192 * or we have purged all unlocked resources. The default policy is to purge in LRU order, but
193 * can be overridden to prefer purging scratch resources (in LRU order) prior to purging other
194 * resource types.
195 *
196 * @param maxBytesToPurge the desired number of bytes to be purged.
197 * @param preferScratchResources If true scratch resources will be purged prior to other
198 * resource types.
199 */
200 void purgeUnlockedResources(size_t bytesToPurge, bool preferScratchResources);
201
bsalomonb77a9072016-09-07 10:02:04 -0700202 /** Returns true if the cache would like a flush to occur in order to make more resources
203 purgeable. */
robertphillipsee843b22016-10-04 05:30:20 -0700204 bool requestsFlush() const { return fRequestFlush; }
bsalomon71cb0c22014-11-14 12:10:14 -0800205
bsalomonb77a9072016-09-07 10:02:04 -0700206 enum FlushType {
207 kExternal,
bsalomonb77a9072016-09-07 10:02:04 -0700208 kCacheRequested,
209 };
210 void notifyFlushOccurred(FlushType);
bsalomon71cb0c22014-11-14 12:10:14 -0800211
Brian Osman13dddce2017-05-09 13:19:50 -0400212 /** Maintain a ref to this resource until we receive a GrGpuResourceFreedMessage. */
213 void insertCrossContextGpuResource(GrGpuResource* resource);
214
robertphillips60029a52015-11-09 13:51:06 -0800215#if GR_CACHE_STATS
216 struct Stats {
217 int fTotal;
218 int fNumPurgeable;
219 int fNumNonPurgeable;
220
221 int fScratch;
kkinnunen2e6055b2016-04-22 01:48:29 -0700222 int fWrapped;
robertphillips60029a52015-11-09 13:51:06 -0800223 size_t fUnbudgetedSize;
224
225 Stats() { this->reset(); }
226
227 void reset() {
228 fTotal = 0;
229 fNumPurgeable = 0;
230 fNumNonPurgeable = 0;
231 fScratch = 0;
kkinnunen2e6055b2016-04-22 01:48:29 -0700232 fWrapped = 0;
robertphillips60029a52015-11-09 13:51:06 -0800233 fUnbudgetedSize = 0;
234 }
235
236 void update(GrGpuResource* resource) {
237 if (resource->cacheAccess().isScratch()) {
238 ++fScratch;
239 }
kkinnunen2e6055b2016-04-22 01:48:29 -0700240 if (resource->resourcePriv().refsWrappedObjects()) {
241 ++fWrapped;
robertphillips60029a52015-11-09 13:51:06 -0800242 }
bsalomon5ec26ae2016-02-25 08:33:02 -0800243 if (SkBudgeted::kNo == resource->resourcePriv().isBudgeted()) {
robertphillips60029a52015-11-09 13:51:06 -0800244 fUnbudgetedSize += resource->gpuMemorySize();
245 }
246 }
247 };
248
249 void getStats(Stats*) const;
250
mtkleinb9eb4ac2015-02-02 18:26:03 -0800251 void dumpStats(SkString*) const;
joshualittdc5685a2015-12-02 14:08:25 -0800252
253 void dumpStatsKeyValuePairs(SkTArray<SkString>* keys, SkTArray<double>* value) const;
bsalomon71cb0c22014-11-14 12:10:14 -0800254#endif
255
Brian Salomon1090da62017-01-06 12:04:19 -0500256#ifdef SK_DEBUG
257 int countUniqueKeysWithTag(const char* tag) const;
258#endif
259
bsalomonddf30e62015-02-19 11:38:44 -0800260 // This function is for unit testing and is only defined in test tools.
261 void changeTimestamp(uint32_t newTimestamp);
262
ericrk0a5fa482015-09-15 14:16:10 -0700263 // Enumerates all cached resources and dumps their details to traceMemoryDump.
264 void dumpMemoryStatistics(SkTraceMemoryDump* traceMemoryDump) const;
265
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500266 void setProxyProvider(GrProxyProvider* proxyProvider) { fProxyProvider = proxyProvider; }
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400267
bsalomonc8dc1f72014-08-21 13:02:13 -0700268private:
bsalomon71cb0c22014-11-14 12:10:14 -0800269 ///////////////////////////////////////////////////////////////////////////
270 /// @name Methods accessible via ResourceAccess
271 ////
272 void insertResource(GrGpuResource*);
273 void removeResource(GrGpuResource*);
bsalomon3f324322015-04-08 11:01:54 -0700274 void notifyCntReachedZero(GrGpuResource*, uint32_t flags);
bsalomonf99e9612015-02-19 08:24:16 -0800275 void changeUniqueKey(GrGpuResource*, const GrUniqueKey&);
276 void removeUniqueKey(GrGpuResource*);
bsalomon10e23ca2014-11-25 05:52:06 -0800277 void willRemoveScratchKey(const GrGpuResource*);
bsalomon84c8e622014-11-17 09:33:27 -0800278 void didChangeBudgetStatus(GrGpuResource*);
bsalomon9f2d1572015-02-17 11:47:40 -0800279 void refAndMakeResourceMRU(GrGpuResource*);
bsalomon71cb0c22014-11-14 12:10:14 -0800280 /// @}
281
bsalomon8718aaf2015-02-19 07:24:21 -0800282 void processInvalidUniqueKeys(const SkTArray<GrUniqueKeyInvalidatedMessage>&);
Brian Osman13dddce2017-05-09 13:19:50 -0400283 void processFreedGpuResources();
bsalomonf320e042015-02-17 15:09:34 -0800284 void addToNonpurgeableArray(GrGpuResource*);
285 void removeFromNonpurgeableArray(GrGpuResource*);
bsalomon71cb0c22014-11-14 12:10:14 -0800286
robertphillips6e83ac72015-08-13 05:19:14 -0700287 bool wouldFit(size_t bytes) {
halcanary9d524f22016-03-29 09:03:52 -0700288 return fBudgetedBytes+bytes <= fMaxBytes && fBudgetedCount+1 <= fMaxCount;
robertphillips6e83ac72015-08-13 05:19:14 -0700289 }
290
bsalomonddf30e62015-02-19 11:38:44 -0800291 uint32_t getNextTimestamp();
292
bsalomon16961262014-08-26 14:01:07 -0700293#ifdef SK_DEBUG
bsalomonf320e042015-02-17 15:09:34 -0800294 bool isInCache(const GrGpuResource* r) const;
bsalomon71cb0c22014-11-14 12:10:14 -0800295 void validate() const;
296#else
297 void validate() const {}
bsalomon16961262014-08-26 14:01:07 -0700298#endif
299
bsalomon71cb0c22014-11-14 12:10:14 -0800300 class AutoValidate;
301
bsalomonbcf0a522014-10-08 08:40:09 -0700302 class AvailableForScratchUse;
bsalomon744998e2014-08-28 09:54:34 -0700303
robertphillipsee843b22016-10-04 05:30:20 -0700304 struct ScratchMapTraits {
bsalomon7775c852014-12-30 12:50:52 -0800305 static const GrScratchKey& GetKey(const GrGpuResource& r) {
bsalomon3582d3e2015-02-13 14:20:05 -0800306 return r.resourcePriv().getScratchKey();
bsalomon744998e2014-08-28 09:54:34 -0700307 }
robertphillipsee843b22016-10-04 05:30:20 -0700308
309 static uint32_t Hash(const GrScratchKey& key) { return key.hash(); }
Robert Phillipsf8e25022017-11-08 15:24:31 -0500310 static void OnFree(GrGpuResource*) { }
bsalomon744998e2014-08-28 09:54:34 -0700311 };
bsalomon7775c852014-12-30 12:50:52 -0800312 typedef SkTMultiMap<GrGpuResource, GrScratchKey, ScratchMapTraits> ScratchMap;
bsalomon744998e2014-08-28 09:54:34 -0700313
robertphillipsee843b22016-10-04 05:30:20 -0700314 struct UniqueHashTraits {
bsalomon8718aaf2015-02-19 07:24:21 -0800315 static const GrUniqueKey& GetKey(const GrGpuResource& r) { return r.getUniqueKey(); }
robertphillipsee843b22016-10-04 05:30:20 -0700316
317 static uint32_t Hash(const GrUniqueKey& key) { return key.hash(); }
bsalomon8b79d232014-11-10 10:19:06 -0800318 };
bsalomon8718aaf2015-02-19 07:24:21 -0800319 typedef SkTDynamicHash<GrGpuResource, GrUniqueKey, UniqueHashTraits> UniqueHash;
bsalomon8b79d232014-11-10 10:19:06 -0800320
bsalomon9f2d1572015-02-17 11:47:40 -0800321 static bool CompareTimestamp(GrGpuResource* const& a, GrGpuResource* const& b) {
322 return a->cacheAccess().timestamp() < b->cacheAccess().timestamp();
323 }
bsalomon23e619c2015-02-06 11:54:28 -0800324
bsalomon9f2d1572015-02-17 11:47:40 -0800325 static int* AccessResourceIndex(GrGpuResource* const& res) {
326 return res->cacheAccess().accessCacheIndex();
327 }
328
bsalomon8718aaf2015-02-19 07:24:21 -0800329 typedef SkMessageBus<GrUniqueKeyInvalidatedMessage>::Inbox InvalidUniqueKeyInbox;
Brian Osman13dddce2017-05-09 13:19:50 -0400330 typedef SkMessageBus<GrGpuResourceFreedMessage>::Inbox FreedGpuResourceInbox;
bsalomon9f2d1572015-02-17 11:47:40 -0800331 typedef SkTDPQueue<GrGpuResource*, CompareTimestamp, AccessResourceIndex> PurgeableQueue;
bsalomonf320e042015-02-17 15:09:34 -0800332 typedef SkTDArray<GrGpuResource*> ResourceArray;
bsalomon9f2d1572015-02-17 11:47:40 -0800333
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500334 GrProxyProvider* fProxyProvider;
bsalomon9f2d1572015-02-17 11:47:40 -0800335 // Whenever a resource is added to the cache or the result of a cache lookup, fTimestamp is
336 // assigned as the resource's timestamp and then incremented. fPurgeableQueue orders the
337 // purgeable resources by this value, and thus is used to purge resources in LRU order.
338 uint32_t fTimestamp;
339 PurgeableQueue fPurgeableQueue;
bsalomonf320e042015-02-17 15:09:34 -0800340 ResourceArray fNonpurgeableResources;
bsalomon9f2d1572015-02-17 11:47:40 -0800341
bsalomon744998e2014-08-28 09:54:34 -0700342 // This map holds all resources that can be used as scratch resources.
bsalomon8b79d232014-11-10 10:19:06 -0800343 ScratchMap fScratchMap;
bsalomon8718aaf2015-02-19 07:24:21 -0800344 // This holds all resources that have unique keys.
345 UniqueHash fUniqueHash;
bsalomon71cb0c22014-11-14 12:10:14 -0800346
347 // our budget, used in purgeAsNeeded()
348 int fMaxCount;
349 size_t fMaxBytes;
bsalomon3f324322015-04-08 11:01:54 -0700350 int fMaxUnusedFlushes;
bsalomon71cb0c22014-11-14 12:10:14 -0800351
352#if GR_CACHE_STATS
353 int fHighWaterCount;
354 size_t fHighWaterBytes;
bsalomondace19e2014-11-17 07:34:06 -0800355 int fBudgetedHighWaterCount;
356 size_t fBudgetedHighWaterBytes;
bsalomon71cb0c22014-11-14 12:10:14 -0800357#endif
358
bsalomondace19e2014-11-17 07:34:06 -0800359 // our current stats for all resources
bsalomonf320e042015-02-17 15:09:34 -0800360 SkDEBUGCODE(int fCount;)
bsalomon71cb0c22014-11-14 12:10:14 -0800361 size_t fBytes;
362
bsalomondace19e2014-11-17 07:34:06 -0800363 // our current stats for resources that count against the budget
364 int fBudgetedCount;
365 size_t fBudgetedBytes;
Derek Sollenbergeree479142017-05-24 11:41:33 -0400366 size_t fPurgeableBytes;
bsalomondace19e2014-11-17 07:34:06 -0800367
robertphillipsee843b22016-10-04 05:30:20 -0700368 bool fRequestFlush;
bsalomone2e87f32016-09-22 12:42:11 -0700369 uint32_t fExternalFlushCnt;
bsalomon3f324322015-04-08 11:01:54 -0700370
bsalomon8718aaf2015-02-19 07:24:21 -0800371 InvalidUniqueKeyInbox fInvalidUniqueKeyInbox;
Brian Osman13dddce2017-05-09 13:19:50 -0400372 FreedGpuResourceInbox fFreedGpuResourceInbox;
373
374 uint32_t fContextUniqueID;
bsalomon3f324322015-04-08 11:01:54 -0700375
376 // This resource is allowed to be in the nonpurgeable array for the sake of validate() because
377 // we're in the midst of converting it to purgeable status.
378 SkDEBUGCODE(GrGpuResource* fNewlyPurgeableResourceForValidation;)
robertphillips63926682015-08-20 09:39:02 -0700379
380 bool fPreferVRAMUseOverFlushes;
bsalomonc8dc1f72014-08-21 13:02:13 -0700381};
382
bsalomon0ea80f42015-02-11 10:49:59 -0800383class GrResourceCache::ResourceAccess {
bsalomon71cb0c22014-11-14 12:10:14 -0800384private:
bsalomon0ea80f42015-02-11 10:49:59 -0800385 ResourceAccess(GrResourceCache* cache) : fCache(cache) { }
bsalomon71cb0c22014-11-14 12:10:14 -0800386 ResourceAccess(const ResourceAccess& that) : fCache(that.fCache) { }
387 ResourceAccess& operator=(const ResourceAccess&); // unimpl
388
389 /**
390 * Insert a resource into the cache.
391 */
392 void insertResource(GrGpuResource* resource) { fCache->insertResource(resource); }
393
394 /**
395 * Removes a resource from the cache.
396 */
397 void removeResource(GrGpuResource* resource) { fCache->removeResource(resource); }
398
399 /**
bsalomon3f324322015-04-08 11:01:54 -0700400 * Notifications that should be sent to the cache when the ref/io cnt status of resources
401 * changes.
bsalomon71cb0c22014-11-14 12:10:14 -0800402 */
bsalomon3f324322015-04-08 11:01:54 -0700403 enum RefNotificationFlags {
404 /** All types of refs on the resource have reached zero. */
405 kAllCntsReachedZero_RefNotificationFlag = 0x1,
406 /** The normal (not pending IO type) ref cnt has reached zero. */
407 kRefCntReachedZero_RefNotificationFlag = 0x2,
408 };
409 /**
410 * Called by GrGpuResources when they detect that their ref/io cnts have reached zero. When the
411 * normal ref cnt reaches zero the flags that are set should be:
412 * a) kRefCntReachedZero if a pending IO cnt is still non-zero.
413 * b) (kRefCntReachedZero | kAllCntsReachedZero) when all pending IO cnts are also zero.
414 * kAllCntsReachedZero is set by itself if a pending IO cnt is decremented to zero and all the
415 * the other cnts are already zero.
416 */
417 void notifyCntReachedZero(GrGpuResource* resource, uint32_t flags) {
418 fCache->notifyCntReachedZero(resource, flags);
419 }
bsalomon71cb0c22014-11-14 12:10:14 -0800420
421 /**
bsalomonf99e9612015-02-19 08:24:16 -0800422 * Called by GrGpuResources to change their unique keys.
bsalomon71cb0c22014-11-14 12:10:14 -0800423 */
bsalomonf99e9612015-02-19 08:24:16 -0800424 void changeUniqueKey(GrGpuResource* resource, const GrUniqueKey& newKey) {
425 fCache->changeUniqueKey(resource, newKey);
426 }
bsalomon71cb0c22014-11-14 12:10:14 -0800427
bsalomon10e23ca2014-11-25 05:52:06 -0800428 /**
bsalomonf99e9612015-02-19 08:24:16 -0800429 * Called by a GrGpuResource to remove its unique key.
bsalomon23e619c2015-02-06 11:54:28 -0800430 */
bsalomonf99e9612015-02-19 08:24:16 -0800431 void removeUniqueKey(GrGpuResource* resource) { fCache->removeUniqueKey(resource); }
bsalomon23e619c2015-02-06 11:54:28 -0800432
433 /**
434 * Called by a GrGpuResource when it removes its scratch key.
bsalomon10e23ca2014-11-25 05:52:06 -0800435 */
436 void willRemoveScratchKey(const GrGpuResource* resource) {
437 fCache->willRemoveScratchKey(resource);
438 }
bsalomon84c8e622014-11-17 09:33:27 -0800439
440 /**
441 * Called by GrGpuResources when they change from budgeted to unbudgeted or vice versa.
442 */
443 void didChangeBudgetStatus(GrGpuResource* resource) { fCache->didChangeBudgetStatus(resource); }
444
bsalomon71cb0c22014-11-14 12:10:14 -0800445 // No taking addresses of this type.
446 const ResourceAccess* operator&() const;
447 ResourceAccess* operator&();
448
bsalomon0ea80f42015-02-11 10:49:59 -0800449 GrResourceCache* fCache;
bsalomon71cb0c22014-11-14 12:10:14 -0800450
451 friend class GrGpuResource; // To access all the proxy inline methods.
bsalomon0ea80f42015-02-11 10:49:59 -0800452 friend class GrResourceCache; // To create this type.
bsalomon71cb0c22014-11-14 12:10:14 -0800453};
454
bsalomon0ea80f42015-02-11 10:49:59 -0800455inline GrResourceCache::ResourceAccess GrResourceCache::resourceAccess() {
bsalomon71cb0c22014-11-14 12:10:14 -0800456 return ResourceAccess(this);
457}
458
bsalomonc8dc1f72014-08-21 13:02:13 -0700459#endif