blob: 15e3ef6d9fd4b53a326dc2d98e70e8faca48ddb1 [file] [log] [blame]
Robert Phillipsae7d3f32017-09-21 08:26:08 -04001/*
2 * Copyright 2017 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// This is a GPU-backend specific test.
9
10#include "Test.h"
11
Robert Phillipsae7d3f32017-09-21 08:26:08 -040012#include "GrBackendSurface.h"
13#include "GrContextPriv.h"
14#include "GrResourceCache.h"
Robert Phillips1afd4cd2018-01-08 13:40:32 -050015#include "GrProxyProvider.h"
Robert Phillipsae7d3f32017-09-21 08:26:08 -040016#include "GrResourceProvider.h"
17#include "GrTest.h"
18#include "GrTexture.h"
19#include "GrTextureProxy.h"
20
21#include "SkGr.h"
22#include "SkImage.h"
23
Robert Phillips1afd4cd2018-01-08 13:40:32 -050024int GrProxyProvider::numUniqueKeyProxies_TestOnly() const {
Robert Phillipsae7d3f32017-09-21 08:26:08 -040025 return fUniquelyKeyedProxies.count();
26}
27
Robert Phillipsfe0253f2018-03-16 16:47:25 -040028static GrSurfaceDesc make_desc(GrSurfaceDescFlags descFlags) {
Robert Phillipsae7d3f32017-09-21 08:26:08 -040029 GrSurfaceDesc desc;
Robert Phillipsfe0253f2018-03-16 16:47:25 -040030 desc.fFlags = descFlags;
Robert Phillipsae7d3f32017-09-21 08:26:08 -040031 desc.fWidth = 64;
32 desc.fHeight = 64;
33 desc.fConfig = kRGBA_8888_GrPixelConfig;
Brian Salomonbdecacf2018-02-02 20:32:49 -050034 desc.fSampleCnt = 1;
Robert Phillipsae7d3f32017-09-21 08:26:08 -040035
36 return desc;
37}
38
39///////////////////////////////////////////////////////////////////////////////////////////////////
40// Basic test
41
Greg Danielcd871402017-09-26 12:49:26 -040042static sk_sp<GrTextureProxy> deferred_tex(skiatest::Reporter* reporter,
Robert Phillipsadbe1322018-01-17 13:35:46 -050043 GrProxyProvider* proxyProvider, SkBackingFit fit) {
Robert Phillips0bd24dc2018-01-16 08:06:32 -050044 const GrSurfaceDesc desc = make_desc(kNone_GrSurfaceFlags);
Robert Phillipsae7d3f32017-09-21 08:26:08 -040045
Brian Salomon2a4f9832018-03-03 22:43:43 -050046 sk_sp<GrTextureProxy> proxy =
47 proxyProvider->createProxy(desc, kBottomLeft_GrSurfaceOrigin, fit, SkBudgeted::kYes);
Robert Phillipsae7d3f32017-09-21 08:26:08 -040048 // Only budgeted & wrapped external proxies get to carry uniqueKeys
Greg Danielcd871402017-09-26 12:49:26 -040049 REPORTER_ASSERT(reporter, !proxy->getUniqueKey().isValid());
50 return proxy;
Robert Phillipsae7d3f32017-09-21 08:26:08 -040051}
52
Greg Danielcd871402017-09-26 12:49:26 -040053static sk_sp<GrTextureProxy> deferred_texRT(skiatest::Reporter* reporter,
Robert Phillipsadbe1322018-01-17 13:35:46 -050054 GrProxyProvider* proxyProvider, SkBackingFit fit) {
Robert Phillips0bd24dc2018-01-16 08:06:32 -050055 const GrSurfaceDesc desc = make_desc(kRenderTarget_GrSurfaceFlag);
Robert Phillipsae7d3f32017-09-21 08:26:08 -040056
Brian Salomon2a4f9832018-03-03 22:43:43 -050057 sk_sp<GrTextureProxy> proxy =
58 proxyProvider->createProxy(desc, kBottomLeft_GrSurfaceOrigin, fit, SkBudgeted::kYes);
Robert Phillipsae7d3f32017-09-21 08:26:08 -040059 // Only budgeted & wrapped external proxies get to carry uniqueKeys
Greg Danielcd871402017-09-26 12:49:26 -040060 REPORTER_ASSERT(reporter, !proxy->getUniqueKey().isValid());
61 return proxy;
Robert Phillipsae7d3f32017-09-21 08:26:08 -040062}
63
Greg Danielcd871402017-09-26 12:49:26 -040064static sk_sp<GrTextureProxy> wrapped(skiatest::Reporter* reporter,
Robert Phillipsadbe1322018-01-17 13:35:46 -050065 GrProxyProvider* proxyProvider, SkBackingFit fit) {
Robert Phillips0bd24dc2018-01-16 08:06:32 -050066 const GrSurfaceDesc desc = make_desc(kNone_GrSurfaceFlags);
Robert Phillips1afd4cd2018-01-08 13:40:32 -050067
Brian Salomon2a4f9832018-03-03 22:43:43 -050068 sk_sp<GrTextureProxy> proxy = proxyProvider->createInstantiatedProxy(
69 desc, kBottomLeft_GrSurfaceOrigin, fit, SkBudgeted::kYes);
Robert Phillips0bd24dc2018-01-16 08:06:32 -050070 // Only budgeted & wrapped external proxies get to carry uniqueKeys
Greg Danielcd871402017-09-26 12:49:26 -040071 REPORTER_ASSERT(reporter, !proxy->getUniqueKey().isValid());
72 return proxy;
73}
74
75static sk_sp<GrTextureProxy> wrapped_with_key(skiatest::Reporter* reporter,
Robert Phillipsadbe1322018-01-17 13:35:46 -050076 GrProxyProvider* proxyProvider, SkBackingFit fit) {
Greg Danielcd871402017-09-26 12:49:26 -040077 static GrUniqueKey::Domain d = GrUniqueKey::GenerateDomain();
78 static int kUniqueKeyData = 0;
79
80 GrUniqueKey key;
81
82 GrUniqueKey::Builder builder(&key, d, 1, nullptr);
83 builder[0] = kUniqueKeyData++;
84 builder.finish();
85
Robert Phillips0bd24dc2018-01-16 08:06:32 -050086 const GrSurfaceDesc desc = make_desc(kNone_GrSurfaceFlags);
Greg Danielcd871402017-09-26 12:49:26 -040087
Robert Phillipsadbe1322018-01-17 13:35:46 -050088 // Only budgeted & wrapped external proxies get to carry uniqueKeys
Brian Salomon2a4f9832018-03-03 22:43:43 -050089 sk_sp<GrTextureProxy> proxy = proxyProvider->createInstantiatedProxy(
Robert Phillipsfe0253f2018-03-16 16:47:25 -040090 desc, kBottomLeft_GrSurfaceOrigin, fit, SkBudgeted::kYes);
Robert Phillipsadbe1322018-01-17 13:35:46 -050091 SkAssertResult(proxyProvider->assignUniqueKeyToProxy(key, proxy.get()));
Greg Danielcd871402017-09-26 12:49:26 -040092 REPORTER_ASSERT(reporter, proxy->getUniqueKey().isValid());
93 return proxy;
Robert Phillipsae7d3f32017-09-21 08:26:08 -040094}
95
96static sk_sp<GrTextureProxy> create_wrapped_backend(GrContext* context, SkBackingFit fit,
97 sk_sp<GrTexture>* backingSurface) {
Robert Phillips0bd24dc2018-01-16 08:06:32 -050098 GrProxyProvider* proxyProvider = context->contextPriv().proxyProvider();
Robert Phillips6be756b2018-01-16 15:07:54 -050099 GrResourceProvider* resourceProvider = context->contextPriv().resourceProvider();
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400100
Robert Phillipsb67821d2017-12-13 15:00:45 -0500101 const GrSurfaceDesc desc = make_desc(kNone_GrSurfaceFlags);
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400102
Robert Phillips0bd24dc2018-01-16 08:06:32 -0500103 *backingSurface = resourceProvider->createTexture(desc, SkBudgeted::kNo);
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400104 if (!(*backingSurface)) {
105 return nullptr;
106 }
107
Robert Phillipsb67821d2017-12-13 15:00:45 -0500108 GrBackendTexture backendTex = (*backingSurface)->getBackendTexture();
Greg Daniel108bb232018-07-03 16:18:29 -0400109 backendTex.setPixelConfig(desc.fConfig);
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400110
Brian Salomon7578f3e2018-03-07 14:39:54 -0500111 return proxyProvider->wrapBackendTexture(backendTex, kBottomLeft_GrSurfaceOrigin);
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400112}
113
114
115// This tests the basic capabilities of the uniquely keyed texture proxies. Does assigning
116// and looking them up work, etc.
117static void basic_test(GrContext* context,
118 skiatest::Reporter* reporter,
119 sk_sp<GrTextureProxy> proxy, bool proxyIsCached) {
120 static int id = 1;
121
Robert Phillips6be756b2018-01-16 15:07:54 -0500122 GrResourceProvider* resourceProvider = context->contextPriv().resourceProvider();
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500123 GrProxyProvider* proxyProvider = context->contextPriv().proxyProvider();
Robert Phillips6be756b2018-01-16 15:07:54 -0500124 GrResourceCache* cache = context->contextPriv().getResourceCache();
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400125
126 int startCacheCount = cache->getResourceCount();
127
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400128 GrUniqueKey key;
Greg Danielcd871402017-09-26 12:49:26 -0400129 if (proxy->getUniqueKey().isValid()) {
130 key = proxy->getUniqueKey();
131 } else {
132 GrMakeKeyFromImageID(&key, id, SkIRect::MakeWH(64, 64));
133 ++id;
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400134
Greg Danielcd871402017-09-26 12:49:26 -0400135 // Assigning the uniqueKey adds the proxy to the hash but doesn't force instantiation
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500136 REPORTER_ASSERT(reporter, !proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsadbe1322018-01-17 13:35:46 -0500137 SkAssertResult(proxyProvider->assignUniqueKeyToProxy(key, proxy.get()));
Greg Danielcd871402017-09-26 12:49:26 -0400138 }
139
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500140 REPORTER_ASSERT(reporter, 1 == proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400141 REPORTER_ASSERT(reporter, startCacheCount == cache->getResourceCount());
142
143 // setUniqueKey had better stick
144 REPORTER_ASSERT(reporter, key == proxy->getUniqueKey());
145
146 // We just added it, surely we can find it
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500147 REPORTER_ASSERT(reporter, proxyProvider->findOrCreateProxyByUniqueKey(
148 key, kBottomLeft_GrSurfaceOrigin));
149 REPORTER_ASSERT(reporter, 1 == proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400150
151 // Once instantiated, the backing resource should have the same key
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500152 SkAssertResult(proxy->instantiate(resourceProvider));
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400153 const GrUniqueKey& texKey = proxy->priv().peekSurface()->getUniqueKey();
154 REPORTER_ASSERT(reporter, texKey.isValid());
155 REPORTER_ASSERT(reporter, key == texKey);
156 if (proxyIsCached) {
157 REPORTER_ASSERT(reporter, 1 == cache->getResourceCount());
158 }
159
160 // deleting the proxy should delete it from the hash but not the cache
161 proxy = nullptr;
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500162 REPORTER_ASSERT(reporter, 0 == proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400163 REPORTER_ASSERT(reporter, 1 == cache->getResourceCount());
164
165 // If the proxy was cached refinding it should bring it back to life
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500166 proxy = proxyProvider->findOrCreateProxyByUniqueKey(key, kBottomLeft_GrSurfaceOrigin);
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400167 if (proxyIsCached) {
168 REPORTER_ASSERT(reporter, proxy);
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500169 REPORTER_ASSERT(reporter, 1 == proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400170 } else {
171 REPORTER_ASSERT(reporter, !proxy);
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500172 REPORTER_ASSERT(reporter, 0 == proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400173 }
174 REPORTER_ASSERT(reporter, 1 == cache->getResourceCount());
175
176 // Mega-purging it should remove it from both the hash and the cache
177 proxy = nullptr;
178 cache->purgeAllUnlocked();
179 if (proxyIsCached) {
180 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
181 } else {
182 REPORTER_ASSERT(reporter, 1 == cache->getResourceCount());
183 }
184
185 // We can bring neither the texture nor proxy back from perma-death
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500186 proxy = proxyProvider->findOrCreateProxyByUniqueKey(key, kBottomLeft_GrSurfaceOrigin);
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400187 REPORTER_ASSERT(reporter, !proxy);
188 if (proxyIsCached) {
189 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
190 } else {
191 REPORTER_ASSERT(reporter, 1 == cache->getResourceCount());
192 }
193}
194
195///////////////////////////////////////////////////////////////////////////////////////////////////
196// Invalidation test
197
198// Test if invalidating unique ids operates as expected for texture proxies.
199static void invalidation_test(GrContext* context, skiatest::Reporter* reporter) {
200
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500201 GrProxyProvider* proxyProvider = context->contextPriv().proxyProvider();
Robert Phillips6be756b2018-01-16 15:07:54 -0500202 GrResourceCache* cache = context->contextPriv().getResourceCache();
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400203 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
204
205 sk_sp<SkImage> rasterImg;
206
207 {
208 SkImageInfo ii = SkImageInfo::Make(64, 64, kRGBA_8888_SkColorType, kOpaque_SkAlphaType);
209
210 SkBitmap bm;
211 bm.allocPixels(ii);
212
213 rasterImg = SkImage::MakeFromBitmap(bm);
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500214 REPORTER_ASSERT(reporter, 0 == proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400215 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
216 }
217
218 sk_sp<SkImage> textureImg = rasterImg->makeTextureImage(context, nullptr);
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500219 REPORTER_ASSERT(reporter, 1 == proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400220 REPORTER_ASSERT(reporter, 1 == cache->getResourceCount());
221
222 rasterImg = nullptr; // this invalidates the uniqueKey
223
224 // this forces the cache to respond to the inval msg
225 int maxNum;
226 size_t maxBytes;
227 context->getResourceCacheLimits(&maxNum, &maxBytes);
228 context->setResourceCacheLimits(maxNum-1, maxBytes);
229
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500230 REPORTER_ASSERT(reporter, 0 == proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400231 REPORTER_ASSERT(reporter, 1 == cache->getResourceCount());
232
233 textureImg = nullptr;
Robert Phillips0c4b7b12018-03-06 08:20:37 -0500234 context->contextPriv().purgeAllUnlockedResources_ForTesting();
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400235
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500236 REPORTER_ASSERT(reporter, 0 == proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400237 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
238}
239
Brian Osman28c434b2017-09-27 13:11:16 -0400240// Test if invalidating unique ids prior to instantiating operates as expected
241static void invalidation_and_instantiation_test(GrContext* context, skiatest::Reporter* reporter) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500242 GrProxyProvider* proxyProvider = context->contextPriv().proxyProvider();
Robert Phillips6be756b2018-01-16 15:07:54 -0500243 GrResourceProvider* resourceProvider = context->contextPriv().resourceProvider();
244 GrResourceCache* cache = context->contextPriv().getResourceCache();
Brian Osman28c434b2017-09-27 13:11:16 -0400245 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
246
247 static GrUniqueKey::Domain d = GrUniqueKey::GenerateDomain();
248 GrUniqueKey key;
249 GrUniqueKey::Builder builder(&key, d, 1, nullptr);
250 builder[0] = 0;
251 builder.finish();
252
253 // Create proxy, assign unique key
Robert Phillipsadbe1322018-01-17 13:35:46 -0500254 sk_sp<GrTextureProxy> proxy = deferred_tex(reporter, proxyProvider, SkBackingFit::kExact);
255 SkAssertResult(proxyProvider->assignUniqueKeyToProxy(key, proxy.get()));
Brian Osman28c434b2017-09-27 13:11:16 -0400256
257 // Send an invalidation message, which will be sitting in the cache's inbox
Brian Salomon143cf8e2018-07-11 20:01:19 +0000258 SkMessageBus<GrUniqueKeyInvalidatedMessage>::Post(GrUniqueKeyInvalidatedMessage(key));
Brian Osman28c434b2017-09-27 13:11:16 -0400259
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500260 REPORTER_ASSERT(reporter, 1 == proxyProvider->numUniqueKeyProxies_TestOnly());
Brian Osman28c434b2017-09-27 13:11:16 -0400261 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
262
263 // Instantiate the proxy. This will trigger the message to be processed, so the resulting
264 // texture should *not* have the unique key on it!
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500265 SkAssertResult(proxy->instantiate(resourceProvider));
Brian Osman28c434b2017-09-27 13:11:16 -0400266
267 REPORTER_ASSERT(reporter, !proxy->getUniqueKey().isValid());
268 REPORTER_ASSERT(reporter, !proxy->priv().peekTexture()->getUniqueKey().isValid());
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500269 REPORTER_ASSERT(reporter, 0 == proxyProvider->numUniqueKeyProxies_TestOnly());
Brian Osman28c434b2017-09-27 13:11:16 -0400270 REPORTER_ASSERT(reporter, 1 == cache->getResourceCount());
271
272 proxy = nullptr;
Robert Phillips0c4b7b12018-03-06 08:20:37 -0500273 context->contextPriv().purgeAllUnlockedResources_ForTesting();
Brian Osman28c434b2017-09-27 13:11:16 -0400274
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500275 REPORTER_ASSERT(reporter, 0 == proxyProvider->numUniqueKeyProxies_TestOnly());
Brian Osman28c434b2017-09-27 13:11:16 -0400276 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
277}
278
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400279DEF_GPUTEST_FOR_RENDERING_CONTEXTS(TextureProxyTest, reporter, ctxInfo) {
280 GrContext* context = ctxInfo.grContext();
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500281 GrProxyProvider* proxyProvider = context->contextPriv().proxyProvider();
Robert Phillips6be756b2018-01-16 15:07:54 -0500282 GrResourceCache* cache = context->contextPriv().getResourceCache();
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400283
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500284 REPORTER_ASSERT(reporter, !proxyProvider->numUniqueKeyProxies_TestOnly());
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400285 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
286
287 for (auto fit : { SkBackingFit::kExact, SkBackingFit::kApprox }) {
Greg Danielcd871402017-09-26 12:49:26 -0400288 for (auto create : { deferred_tex, deferred_texRT, wrapped, wrapped_with_key }) {
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400289 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
Robert Phillipsadbe1322018-01-17 13:35:46 -0500290 basic_test(context, reporter, create(reporter, proxyProvider, fit), true);
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400291 }
292
293 REPORTER_ASSERT(reporter, 0 == cache->getResourceCount());
294 sk_sp<GrTexture> backingTex;
295 sk_sp<GrTextureProxy> proxy = create_wrapped_backend(context, fit, &backingTex);
296 basic_test(context, reporter, std::move(proxy), false);
297
298 backingTex = nullptr;
299 cache->purgeAllUnlocked();
300 }
301
302 invalidation_test(context, reporter);
Robert Phillipsfa8c0802017-10-04 08:42:28 -0400303 invalidation_and_instantiation_test(context, reporter);
Robert Phillipsae7d3f32017-09-21 08:26:08 -0400304}