blob: 39523febf18177fd4b19fee56246f1f8deca5b1e [file] [log] [blame]
Robert Phillips5af44de2017-07-18 14:49:38 -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
Robert Phillipse19babf2020-04-06 13:57:30 -04008#include "include/core/SkCanvas.h"
9#include "include/core/SkSurface.h"
Robert Phillips6d344c32020-07-06 10:56:46 -040010#include "include/gpu/GrDirectContext.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "src/gpu/GrContextPriv.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "src/gpu/GrGpu.h"
13#include "src/gpu/GrProxyProvider.h"
14#include "src/gpu/GrResourceAllocator.h"
15#include "src/gpu/GrResourceProvider.h"
16#include "src/gpu/GrSurfaceProxyPriv.h"
Greg Daniel456f9b52020-03-05 19:14:18 +000017#include "src/gpu/GrTexture.h"
Greg Danielf91aeb22019-06-18 09:58:02 -040018#include "src/gpu/GrTextureProxy.h"
Robert Phillipse19babf2020-04-06 13:57:30 -040019#include "tests/Test.h"
Greg Danielc1ad77c2020-05-06 11:40:03 -040020#include "tests/TestUtils.h"
Robert Phillips1734dd32018-08-21 13:52:09 -040021
Robert Phillips57aa3672017-07-21 11:38:13 -040022struct ProxyParams {
23 int fSize;
Brian Salomonf2c2ba92019-07-17 09:59:59 -040024 GrRenderable fRenderable;
Robert Phillipsc80b0e92019-07-23 10:27:09 -040025 GrColorType fColorType;
Robert Phillips57aa3672017-07-21 11:38:13 -040026 SkBackingFit fFit;
27 int fSampleCnt;
Robert Phillipsc476e5d2019-03-26 14:50:08 -040028 SkBudgeted fBudgeted;
Robert Phillips57aa3672017-07-21 11:38:13 -040029 // TODO: do we care about mipmapping
30};
31
Robert Phillips3d4cac52019-06-11 08:08:08 -040032static sk_sp<GrSurfaceProxy> make_deferred(GrProxyProvider* proxyProvider, const GrCaps* caps,
33 const ProxyParams& p) {
Robert Phillips0a15cc62019-07-30 12:49:10 -040034 const GrBackendFormat format = caps->getDefaultBackendFormat(p.fColorType, p.fRenderable);
Brian Salomondf1bd6d2020-03-26 20:37:01 -040035 return proxyProvider->createProxy(format, {p.fSize, p.fSize}, p.fRenderable, p.fSampleCnt,
Brian Salomon7e67dca2020-07-21 09:27:25 -040036 GrMipmapped::kNo, p.fFit, p.fBudgeted, GrProtected::kNo);
Robert Phillips57aa3672017-07-21 11:38:13 -040037}
38
Robert Phillipseffd13f2020-07-20 15:00:36 -040039static sk_sp<GrSurfaceProxy> make_backend(GrDirectContext* dContext,
40 const ProxyParams& p,
Robert Phillips3d4cac52019-06-11 08:08:08 -040041 GrBackendTexture* backendTex) {
Robert Phillipseffd13f2020-07-20 15:00:36 -040042 GrProxyProvider* proxyProvider = dContext->priv().proxyProvider();
Robert Phillips0bd24dc2018-01-16 08:06:32 -050043
Robert Phillipsc80b0e92019-07-23 10:27:09 -040044 SkColorType skColorType = GrColorTypeToSkColorType(p.fColorType);
45 SkASSERT(SkColorType::kUnknown_SkColorType != skColorType);
46
Robert Phillipseffd13f2020-07-20 15:00:36 -040047 CreateBackendTexture(dContext, backendTex, p.fSize, p.fSize, skColorType,
Brian Salomon7e67dca2020-07-21 09:27:25 -040048 SkColors::kTransparent, GrMipmapped::kNo, GrRenderable::kNo);
Greg Danielc1ad77c2020-05-06 11:40:03 -040049
Robert Phillips646f6372018-09-25 09:31:10 -040050 if (!backendTex->isValid()) {
51 return nullptr;
52 }
Robert Phillips57aa3672017-07-21 11:38:13 -040053
Brian Salomon8a78e9c2020-03-27 10:42:15 -040054 return proxyProvider->wrapBackendTexture(*backendTex, kBorrow_GrWrapOwnership,
Greg Daniel3a365112020-02-14 10:47:18 -050055 GrWrapCacheable::kNo, kRead_GrIOType);
Robert Phillips57aa3672017-07-21 11:38:13 -040056}
57
Robert Phillipseffd13f2020-07-20 15:00:36 -040058static void cleanup_backend(GrDirectContext* dContext, const GrBackendTexture& backendTex) {
59 dContext->deleteBackendTexture(backendTex);
Robert Phillips57aa3672017-07-21 11:38:13 -040060}
61
Robert Phillips5af44de2017-07-18 14:49:38 -040062// Basic test that two proxies with overlapping intervals and compatible descriptors are
63// assigned different GrSurfaces.
Robert Phillips57aa3672017-07-21 11:38:13 -040064static void overlap_test(skiatest::Reporter* reporter, GrResourceProvider* resourceProvider,
Robert Phillips3d4cac52019-06-11 08:08:08 -040065 sk_sp<GrSurfaceProxy> p1, sk_sp<GrSurfaceProxy> p2,
Brian Salomon2c791fc2019-04-02 11:52:03 -040066 bool expectedResult) {
Brian Salomonbeb7f522019-08-30 16:19:42 -040067 GrResourceAllocator alloc(resourceProvider SkDEBUGCODE(, 1));
Robert Phillips5af44de2017-07-18 14:49:38 -040068
Robert Phillips3d4cac52019-06-11 08:08:08 -040069 alloc.addInterval(p1.get(), 0, 4, GrResourceAllocator::ActualUse::kYes);
Robert Phillipsc476e5d2019-03-26 14:50:08 -040070 alloc.incOps();
Robert Phillips3d4cac52019-06-11 08:08:08 -040071 alloc.addInterval(p2.get(), 1, 2, GrResourceAllocator::ActualUse::kYes);
Robert Phillipsc476e5d2019-03-26 14:50:08 -040072 alloc.incOps();
Greg Danielf41b2bd2019-08-22 16:19:24 -040073 alloc.markEndOfOpsTask(0);
Robert Phillips5af44de2017-07-18 14:49:38 -040074
Robert Phillipsc73666f2019-04-24 08:49:48 -040075 alloc.determineRecyclability();
76
Robert Phillipseafd48a2017-11-16 07:52:08 -050077 int startIndex, stopIndex;
Greg Danielaa3dfbe2018-01-29 10:34:25 -050078 GrResourceAllocator::AssignError error;
Brian Salomon577aa0f2018-11-30 13:32:23 -050079 alloc.assign(&startIndex, &stopIndex, &error);
Greg Danielaa3dfbe2018-01-29 10:34:25 -050080 REPORTER_ASSERT(reporter, GrResourceAllocator::AssignError::kNoError == error);
Robert Phillips5af44de2017-07-18 14:49:38 -040081
Brian Salomonfd98c2c2018-07-31 17:25:29 -040082 REPORTER_ASSERT(reporter, p1->peekSurface());
83 REPORTER_ASSERT(reporter, p2->peekSurface());
Robert Phillips57aa3672017-07-21 11:38:13 -040084 bool doTheBackingStoresMatch = p1->underlyingUniqueID() == p2->underlyingUniqueID();
85 REPORTER_ASSERT(reporter, expectedResult == doTheBackingStoresMatch);
Robert Phillips5af44de2017-07-18 14:49:38 -040086}
87
Robert Phillips57aa3672017-07-21 11:38:13 -040088// Test various cases when two proxies do not have overlapping intervals.
89// This mainly acts as a test of the ResourceAllocator's free pool.
90static void non_overlap_test(skiatest::Reporter* reporter, GrResourceProvider* resourceProvider,
Robert Phillips3d4cac52019-06-11 08:08:08 -040091 sk_sp<GrSurfaceProxy> p1, sk_sp<GrSurfaceProxy> p2,
Robert Phillips57aa3672017-07-21 11:38:13 -040092 bool expectedResult) {
Brian Salomonbeb7f522019-08-30 16:19:42 -040093 GrResourceAllocator alloc(resourceProvider SkDEBUGCODE(, 1));
Robert Phillipsc476e5d2019-03-26 14:50:08 -040094
95 alloc.incOps();
96 alloc.incOps();
97 alloc.incOps();
98 alloc.incOps();
99 alloc.incOps();
100 alloc.incOps();
Robert Phillips57aa3672017-07-21 11:38:13 -0400101
Robert Phillips3d4cac52019-06-11 08:08:08 -0400102 alloc.addInterval(p1.get(), 0, 2, GrResourceAllocator::ActualUse::kYes);
103 alloc.addInterval(p2.get(), 3, 5, GrResourceAllocator::ActualUse::kYes);
Greg Danielf41b2bd2019-08-22 16:19:24 -0400104 alloc.markEndOfOpsTask(0);
Robert Phillips57aa3672017-07-21 11:38:13 -0400105
Robert Phillipsc73666f2019-04-24 08:49:48 -0400106 alloc.determineRecyclability();
107
Robert Phillipseafd48a2017-11-16 07:52:08 -0500108 int startIndex, stopIndex;
Greg Danielaa3dfbe2018-01-29 10:34:25 -0500109 GrResourceAllocator::AssignError error;
Brian Salomon577aa0f2018-11-30 13:32:23 -0500110 alloc.assign(&startIndex, &stopIndex, &error);
Greg Danielaa3dfbe2018-01-29 10:34:25 -0500111 REPORTER_ASSERT(reporter, GrResourceAllocator::AssignError::kNoError == error);
Robert Phillips57aa3672017-07-21 11:38:13 -0400112
Brian Salomonfd98c2c2018-07-31 17:25:29 -0400113 REPORTER_ASSERT(reporter, p1->peekSurface());
114 REPORTER_ASSERT(reporter, p2->peekSurface());
Robert Phillips57aa3672017-07-21 11:38:13 -0400115 bool doTheBackingStoresMatch = p1->underlyingUniqueID() == p2->underlyingUniqueID();
116 REPORTER_ASSERT(reporter, expectedResult == doTheBackingStoresMatch);
117}
118
119DEF_GPUTEST_FOR_RENDERING_CONTEXTS(ResourceAllocatorTest, reporter, ctxInfo) {
Robert Phillips6d344c32020-07-06 10:56:46 -0400120 auto direct = ctxInfo.directContext();
121 const GrCaps* caps = direct->priv().caps();
122 GrProxyProvider* proxyProvider = direct->priv().proxyProvider();
123 GrResourceProvider* resourceProvider = direct->priv().resourceProvider();
Robert Phillips5af44de2017-07-18 14:49:38 -0400124
Robert Phillips57aa3672017-07-21 11:38:13 -0400125 struct TestCase {
126 ProxyParams fP1;
127 ProxyParams fP2;
128 bool fExpectation;
129 };
130
Brian Salomonf2c2ba92019-07-17 09:59:59 -0400131 constexpr GrRenderable kRT = GrRenderable::kYes;
132 constexpr GrRenderable kNotRT = GrRenderable::kNo;
Robert Phillips57aa3672017-07-21 11:38:13 -0400133
134 constexpr bool kShare = true;
135 constexpr bool kDontShare = false;
136 // Non-RT GrSurfaces are never recycled on some platforms.
137 bool kConditionallyShare = resourceProvider->caps()->reuseScratchTextures();
138
Robert Phillipsc80b0e92019-07-23 10:27:09 -0400139 const GrColorType kRGBA = GrColorType::kRGBA_8888;
Greg Danield51fa2f2020-01-22 16:53:38 -0500140 const GrColorType kAlpha = GrColorType::kAlpha_8;
Robert Phillips57aa3672017-07-21 11:38:13 -0400141
142 const SkBackingFit kE = SkBackingFit::kExact;
143 const SkBackingFit kA = SkBackingFit::kApprox;
144
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400145 const SkBudgeted kNotB = SkBudgeted::kNo;
146
Robert Phillips57aa3672017-07-21 11:38:13 -0400147 //--------------------------------------------------------------------------------------------
148 TestCase gOverlappingTests[] = {
Greg Daniel3a365112020-02-14 10:47:18 -0500149 //----------------------------------------------------------------------------------------
150 // Two proxies with overlapping intervals and compatible descriptors should never share
151 // RT version
152 {{64, kRT, kRGBA, kA, 1, kNotB}, {64, kRT, kRGBA, kA, 1, kNotB}, kDontShare},
153 // non-RT version
154 {{64, kNotRT, kRGBA, kA, 1, kNotB}, {64, kNotRT, kRGBA, kA, 1, kNotB}, kDontShare},
Robert Phillips57aa3672017-07-21 11:38:13 -0400155 };
156
157 for (auto test : gOverlappingTests) {
Robert Phillips3d4cac52019-06-11 08:08:08 -0400158 sk_sp<GrSurfaceProxy> p1 = make_deferred(proxyProvider, caps, test.fP1);
159 sk_sp<GrSurfaceProxy> p2 = make_deferred(proxyProvider, caps, test.fP2);
Robert Phillipse5f73282019-06-18 17:15:04 -0400160 overlap_test(reporter, resourceProvider, std::move(p1), std::move(p2), test.fExpectation);
Robert Phillips57aa3672017-07-21 11:38:13 -0400161 }
162
Greg Daniel6fa62e22019-08-07 15:52:37 -0400163 auto beFormat = caps->getDefaultBackendFormat(GrColorType::kRGBA_8888, GrRenderable::kYes);
Robert Phillips6d344c32020-07-06 10:56:46 -0400164 int k2 = direct->priv().caps()->getRenderTargetSampleCount(2, beFormat);
165 int k4 = direct->priv().caps()->getRenderTargetSampleCount(4, beFormat);
Robert Phillips57aa3672017-07-21 11:38:13 -0400166
167 //--------------------------------------------------------------------------------------------
168 TestCase gNonOverlappingTests[] = {
Greg Daniel3a365112020-02-14 10:47:18 -0500169 //----------------------------------------------------------------------------------------
170 // Two non-overlapping intervals w/ compatible proxies should share
171 // both same size & approx
172 {{64, kRT, kRGBA, kA, 1, kNotB}, {64, kRT, kRGBA, kA, 1, kNotB}, kShare},
173 {{64, kNotRT, kRGBA, kA, 1, kNotB},
174 {64, kNotRT, kRGBA, kA, 1, kNotB},
175 kConditionallyShare},
176 // diffs sizes but still approx
177 {{64, kRT, kRGBA, kA, 1, kNotB}, {50, kRT, kRGBA, kA, 1, kNotB}, kShare},
178 {{64, kNotRT, kRGBA, kA, 1, kNotB},
179 {50, kNotRT, kRGBA, kA, 1, kNotB},
180 kConditionallyShare},
181 // sames sizes but exact
182 {{64, kRT, kRGBA, kE, 1, kNotB}, {64, kRT, kRGBA, kE, 1, kNotB}, kShare},
183 {{64, kNotRT, kRGBA, kE, 1, kNotB},
184 {64, kNotRT, kRGBA, kE, 1, kNotB},
185 kConditionallyShare},
186 //----------------------------------------------------------------------------------------
187 // Two non-overlapping intervals w/ different exact sizes should not share
188 {{56, kRT, kRGBA, kE, 1, kNotB}, {54, kRT, kRGBA, kE, 1, kNotB}, kDontShare},
189 // Two non-overlapping intervals w/ _very different_ approx sizes should not share
190 {{255, kRT, kRGBA, kA, 1, kNotB}, {127, kRT, kRGBA, kA, 1, kNotB}, kDontShare},
191 // Two non-overlapping intervals w/ different MSAA sample counts should not share
192 {{64, kRT, kRGBA, kA, k2, kNotB}, {64, kRT, kRGBA, kA, k4, kNotB}, k2 == k4},
193 // Two non-overlapping intervals w/ different configs should not share
194 {{64, kRT, kRGBA, kA, 1, kNotB}, {64, kRT, kAlpha, kA, 1, kNotB}, kDontShare},
195 // Two non-overlapping intervals w/ different RT classifications should never share
196 {{64, kRT, kRGBA, kA, 1, kNotB}, {64, kNotRT, kRGBA, kA, 1, kNotB}, kDontShare},
197 {{64, kNotRT, kRGBA, kA, 1, kNotB}, {64, kRT, kRGBA, kA, 1, kNotB}, kDontShare},
198 // Two non-overlapping intervals w/ different origins should share
199 {{64, kRT, kRGBA, kA, 1, kNotB}, {64, kRT, kRGBA, kA, 1, kNotB}, kShare},
Robert Phillips57aa3672017-07-21 11:38:13 -0400200 };
201
202 for (auto test : gNonOverlappingTests) {
Robert Phillips3d4cac52019-06-11 08:08:08 -0400203 sk_sp<GrSurfaceProxy> p1 = make_deferred(proxyProvider, caps, test.fP1);
204 sk_sp<GrSurfaceProxy> p2 = make_deferred(proxyProvider, caps, test.fP2);
Robert Phillips715d08c2018-07-18 13:56:48 -0400205
Robert Phillips57aa3672017-07-21 11:38:13 -0400206 if (!p1 || !p2) {
207 continue; // creation can fail (i.e., for msaa4 on iOS)
208 }
Robert Phillips715d08c2018-07-18 13:56:48 -0400209
Robert Phillipse5f73282019-06-18 17:15:04 -0400210 non_overlap_test(reporter, resourceProvider, std::move(p1), std::move(p2),
211 test.fExpectation);
Robert Phillips57aa3672017-07-21 11:38:13 -0400212 }
213
214 {
215 // Wrapped backend textures should never be reused
216 TestCase t[1] = {
Greg Daniel3a365112020-02-14 10:47:18 -0500217 {{64, kNotRT, kRGBA, kE, 1, kNotB}, {64, kNotRT, kRGBA, kE, 1, kNotB}, kDontShare}};
Robert Phillips57aa3672017-07-21 11:38:13 -0400218
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500219 GrBackendTexture backEndTex;
Robert Phillips6d344c32020-07-06 10:56:46 -0400220 sk_sp<GrSurfaceProxy> p1 = make_backend(direct, t[0].fP1, &backEndTex);
Robert Phillips3d4cac52019-06-11 08:08:08 -0400221 sk_sp<GrSurfaceProxy> p2 = make_deferred(proxyProvider, caps, t[0].fP2);
Robert Phillips715d08c2018-07-18 13:56:48 -0400222
Robert Phillipse5f73282019-06-18 17:15:04 -0400223 non_overlap_test(reporter, resourceProvider, std::move(p1), std::move(p2),
224 t[0].fExpectation);
Robert Phillips715d08c2018-07-18 13:56:48 -0400225
Robert Phillips6d344c32020-07-06 10:56:46 -0400226 cleanup_backend(direct, backEndTex);
Robert Phillips57aa3672017-07-21 11:38:13 -0400227 }
Robert Phillips5af44de2017-07-18 14:49:38 -0400228}
Robert Phillips1734dd32018-08-21 13:52:09 -0400229
Robert Phillipseffd13f2020-07-20 15:00:36 -0400230static void draw(GrRecordingContext* rContext) {
Robert Phillips1734dd32018-08-21 13:52:09 -0400231 SkImageInfo ii = SkImageInfo::Make(1024, 1024, kRGBA_8888_SkColorType, kPremul_SkAlphaType);
232
Robert Phillipseffd13f2020-07-20 15:00:36 -0400233 sk_sp<SkSurface> s = SkSurface::MakeRenderTarget(rContext, SkBudgeted::kYes,
Robert Phillips1734dd32018-08-21 13:52:09 -0400234 ii, 1, kTopLeft_GrSurfaceOrigin, nullptr);
235
236 SkCanvas* c = s->getCanvas();
237
238 c->clear(SK_ColorBLACK);
239}
240
241
242DEF_GPUTEST_FOR_RENDERING_CONTEXTS(ResourceAllocatorStressTest, reporter, ctxInfo) {
Robert Phillips6d344c32020-07-06 10:56:46 -0400243 auto context = ctxInfo.directContext();
Robert Phillips1734dd32018-08-21 13:52:09 -0400244
Robert Phillipscf39f372019-09-03 10:29:20 -0400245 size_t maxBytes = context->getResourceCacheLimit();
Robert Phillips1734dd32018-08-21 13:52:09 -0400246
Robert Phillipscf39f372019-09-03 10:29:20 -0400247 context->setResourceCacheLimit(0); // We'll always be overbudget
Robert Phillips1734dd32018-08-21 13:52:09 -0400248
249 draw(context);
250 draw(context);
251 draw(context);
252 draw(context);
Greg Daniel0a2464f2020-05-14 15:45:44 -0400253 context->flushAndSubmit();
Robert Phillips1734dd32018-08-21 13:52:09 -0400254
Robert Phillipscf39f372019-09-03 10:29:20 -0400255 context->setResourceCacheLimit(maxBytes);
Robert Phillips1734dd32018-08-21 13:52:09 -0400256}
Brian Salomon577aa0f2018-11-30 13:32:23 -0500257
258sk_sp<GrSurfaceProxy> make_lazy(GrProxyProvider* proxyProvider, const GrCaps* caps,
Brian Salomonbeb7f522019-08-30 16:19:42 -0400259 const ProxyParams& p) {
Brian Salomon4eb38b72019-08-05 12:58:39 -0400260 const auto format = caps->getDefaultBackendFormat(p.fColorType, p.fRenderable);
Brian Salomon577aa0f2018-11-30 13:32:23 -0500261
Brian Salomon63410e92020-03-23 18:32:50 -0400262 auto callback = [](GrResourceProvider* resourceProvider,
263 const GrSurfaceProxy::LazySurfaceDesc& desc) {
Brian Salomonb6a3a3b2019-04-01 12:29:34 -0400264 sk_sp<GrTexture> texture;
Brian Salomon63410e92020-03-23 18:32:50 -0400265 if (desc.fFit == SkBackingFit::kApprox) {
266 texture = resourceProvider->createApproxTexture(desc.fDimensions, desc.fFormat,
267 desc.fRenderable, desc.fSampleCnt,
268 desc.fProtected);
Brian Salomon577aa0f2018-11-30 13:32:23 -0500269 } else {
Brian Salomon63410e92020-03-23 18:32:50 -0400270 texture = resourceProvider->createTexture(
271 desc.fDimensions, desc.fFormat, desc.fRenderable, desc.fSampleCnt,
272 desc.fMipMapped, desc.fBudgeted, desc.fProtected);
Brian Salomon577aa0f2018-11-30 13:32:23 -0500273 }
Brian Salomonbeb7f522019-08-30 16:19:42 -0400274 return GrSurfaceProxy::LazyCallbackResult(std::move(texture));
Brian Salomon577aa0f2018-11-30 13:32:23 -0500275 };
Brian Salomon577aa0f2018-11-30 13:32:23 -0500276 GrInternalSurfaceFlags flags = GrInternalSurfaceFlags::kNone;
Brian Salomon63410e92020-03-23 18:32:50 -0400277 SkISize dims = {p.fSize, p.fSize};
Brian Salomondf1bd6d2020-03-26 20:37:01 -0400278 return proxyProvider->createLazyProxy(callback, format, dims, p.fRenderable, p.fSampleCnt,
Brian Salomon7e67dca2020-07-21 09:27:25 -0400279 GrMipmapped::kNo, GrMipMapsStatus::kNotAllocated, flags,
Brian Salomondf1bd6d2020-03-26 20:37:01 -0400280 p.fFit, p.fBudgeted, GrProtected::kNo,
281 GrSurfaceProxy::UseAllocator::kYes);
Brian Salomon577aa0f2018-11-30 13:32:23 -0500282}
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400283
Greg Danielf41b2bd2019-08-22 16:19:24 -0400284// Set up so there are two opsTasks that need to be flushed but the resource allocator thinks
285// it is over budget. The two opsTasks should be flushed separately and the opsTask indices
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400286// returned from assign should be correct.
287DEF_GPUTEST_FOR_RENDERING_CONTEXTS(ResourceAllocatorOverBudgetTest, reporter, ctxInfo) {
Robert Phillips6d344c32020-07-06 10:56:46 -0400288 auto context = ctxInfo.directContext();
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400289 const GrCaps* caps = context->priv().caps();
290 GrProxyProvider* proxyProvider = context->priv().proxyProvider();
291 GrResourceProvider* resourceProvider = context->priv().resourceProvider();
292
Robert Phillipscf39f372019-09-03 10:29:20 -0400293 size_t origMaxBytes = context->getResourceCacheLimit();
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400294
295 // Force the resource allocator to always believe it is over budget
Robert Phillipscf39f372019-09-03 10:29:20 -0400296 context->setResourceCacheLimit(0);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400297
Greg Daniel3a365112020-02-14 10:47:18 -0500298 const ProxyParams params = {
299 64, GrRenderable::kNo, GrColorType::kRGBA_8888, SkBackingFit::kExact,
300 1, SkBudgeted::kYes};
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400301
302 {
Robert Phillips3d4cac52019-06-11 08:08:08 -0400303 sk_sp<GrSurfaceProxy> p1 = make_deferred(proxyProvider, caps, params);
304 sk_sp<GrSurfaceProxy> p2 = make_deferred(proxyProvider, caps, params);
305 sk_sp<GrSurfaceProxy> p3 = make_deferred(proxyProvider, caps, params);
306 sk_sp<GrSurfaceProxy> p4 = make_deferred(proxyProvider, caps, params);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400307
Brian Salomonbeb7f522019-08-30 16:19:42 -0400308 GrResourceAllocator alloc(resourceProvider SkDEBUGCODE(, 2));
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400309
Robert Phillips3d4cac52019-06-11 08:08:08 -0400310 alloc.addInterval(p1.get(), 0, 0, GrResourceAllocator::ActualUse::kYes);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400311 alloc.incOps();
Robert Phillips3d4cac52019-06-11 08:08:08 -0400312 alloc.addInterval(p2.get(), 1, 1, GrResourceAllocator::ActualUse::kYes);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400313 alloc.incOps();
Greg Danielf41b2bd2019-08-22 16:19:24 -0400314 alloc.markEndOfOpsTask(0);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400315
Robert Phillips3d4cac52019-06-11 08:08:08 -0400316 alloc.addInterval(p3.get(), 2, 2, GrResourceAllocator::ActualUse::kYes);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400317 alloc.incOps();
Robert Phillips3d4cac52019-06-11 08:08:08 -0400318 alloc.addInterval(p4.get(), 3, 3, GrResourceAllocator::ActualUse::kYes);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400319 alloc.incOps();
Greg Danielf41b2bd2019-08-22 16:19:24 -0400320 alloc.markEndOfOpsTask(1);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400321
322 int startIndex, stopIndex;
323 GrResourceAllocator::AssignError error;
324
Robert Phillipsc73666f2019-04-24 08:49:48 -0400325 alloc.determineRecyclability();
326
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400327 alloc.assign(&startIndex, &stopIndex, &error);
328 REPORTER_ASSERT(reporter, GrResourceAllocator::AssignError::kNoError == error);
329 REPORTER_ASSERT(reporter, 0 == startIndex && 1 == stopIndex);
330
331 alloc.assign(&startIndex, &stopIndex, &error);
332 REPORTER_ASSERT(reporter, GrResourceAllocator::AssignError::kNoError == error);
333 REPORTER_ASSERT(reporter, 1 == startIndex && 2 == stopIndex);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400334 }
335
Robert Phillipscf39f372019-09-03 10:29:20 -0400336 context->setResourceCacheLimit(origMaxBytes);
Robert Phillipsc476e5d2019-03-26 14:50:08 -0400337}
Greg Danield72dd4d2019-08-29 14:37:46 -0400338
339// This test is used to make sure we are tracking the current task index during the assign call in
340// the GrResourceAllocator. Specifically we can fall behind if we have intervals that don't
341// use the allocator. In this case we need to possibly increment the fCurOpsTaskIndex multiple times
342// to get in back in sync. We had a bug where we'd only every increment the index by one,
343// http://crbug.com/996610.
344DEF_GPUTEST_FOR_RENDERING_CONTEXTS(ResourceAllocatorCurOpsTaskIndexTest,
345 reporter, ctxInfo) {
Robert Phillips6d344c32020-07-06 10:56:46 -0400346 auto context = ctxInfo.directContext();
Greg Danield72dd4d2019-08-29 14:37:46 -0400347 const GrCaps* caps = context->priv().caps();
348 GrProxyProvider* proxyProvider = context->priv().proxyProvider();
349 GrResourceProvider* resourceProvider = context->priv().resourceProvider();
350
Robert Phillipscf39f372019-09-03 10:29:20 -0400351 size_t origMaxBytes = context->getResourceCacheLimit();
Greg Danield72dd4d2019-08-29 14:37:46 -0400352
353 // Force the resource allocator to always believe it is over budget
Robert Phillipscf39f372019-09-03 10:29:20 -0400354 context->setResourceCacheLimit(0);
Greg Danield72dd4d2019-08-29 14:37:46 -0400355
356 ProxyParams params;
357 params.fFit = SkBackingFit::kExact;
Greg Danield72dd4d2019-08-29 14:37:46 -0400358 params.fColorType = GrColorType::kRGBA_8888;
359 params.fRenderable = GrRenderable::kYes;
360 params.fSampleCnt = 1;
361 params.fSize = 100;
362 params.fBudgeted = SkBudgeted::kYes;
363
364 sk_sp<GrSurfaceProxy> proxy1 = make_deferred(proxyProvider, caps, params);
365 if (!proxy1) {
366 return;
367 }
368 sk_sp<GrSurfaceProxy> proxy2 = make_deferred(proxyProvider, caps, params);
369 if (!proxy2) {
370 return;
371 }
372
373 // Wrapped proxy that will be ignored by the resourceAllocator. We use this to try and get the
374 // resource allocator fCurOpsTaskIndex to fall behind what it really should be.
375 GrBackendTexture backEndTex;
Robert Phillips6d344c32020-07-06 10:56:46 -0400376 sk_sp<GrSurfaceProxy> proxyWrapped = make_backend(context, params, &backEndTex);
Greg Danield72dd4d2019-08-29 14:37:46 -0400377 if (!proxyWrapped) {
378 return;
379 }
Greg Danield72dd4d2019-08-29 14:37:46 -0400380
381 // Same as above, but we actually need to have at least two intervals that don't go through the
382 // resource allocator to expose the index bug.
383 GrBackendTexture backEndTex2;
Robert Phillips6d344c32020-07-06 10:56:46 -0400384 sk_sp<GrSurfaceProxy> proxyWrapped2 = make_backend(context, params, &backEndTex2);
Greg Danield72dd4d2019-08-29 14:37:46 -0400385 if (!proxyWrapped2) {
Robert Phillips6d344c32020-07-06 10:56:46 -0400386 cleanup_backend(context, backEndTex);
Greg Danield72dd4d2019-08-29 14:37:46 -0400387 return;
388 }
Greg Danield72dd4d2019-08-29 14:37:46 -0400389
Brian Salomonbeb7f522019-08-30 16:19:42 -0400390 GrResourceAllocator alloc(resourceProvider SkDEBUGCODE(, 4));
Greg Danield72dd4d2019-08-29 14:37:46 -0400391
392 alloc.addInterval(proxyWrapped.get(), 0, 0, GrResourceAllocator::ActualUse::kYes);
393 alloc.incOps();
394 alloc.markEndOfOpsTask(0);
395
396 alloc.addInterval(proxyWrapped2.get(), 1, 1, GrResourceAllocator::ActualUse::kYes);
397 alloc.incOps();
398 alloc.markEndOfOpsTask(1);
399
400 alloc.addInterval(proxy1.get(), 2, 2, GrResourceAllocator::ActualUse::kYes);
401 alloc.incOps();
402 alloc.markEndOfOpsTask(2);
403
404 // We want to force the resource allocator to do a intermediateFlush for the previous interval.
405 // But if it is the resource allocator is at the of its list of intervals it skips the
406 // intermediate flush call, so we add another interval here so it is not skipped.
407 alloc.addInterval(proxy2.get(), 3, 3, GrResourceAllocator::ActualUse::kYes);
408 alloc.incOps();
409 alloc.markEndOfOpsTask(3);
410
411 int startIndex, stopIndex;
412 GrResourceAllocator::AssignError error;
413
414 alloc.determineRecyclability();
415
416 alloc.assign(&startIndex, &stopIndex, &error);
417 REPORTER_ASSERT(reporter, GrResourceAllocator::AssignError::kNoError == error);
418 // The original bug in the allocator here would return a stopIndex of 2 since it would have only
419 // incremented its fCurOpsTaskIndex once instead of the needed two times to skip the first two
420 // unused intervals.
421 REPORTER_ASSERT(reporter, 0 == startIndex && 3 == stopIndex);
422
423 alloc.assign(&startIndex, &stopIndex, &error);
424 REPORTER_ASSERT(reporter, GrResourceAllocator::AssignError::kNoError == error);
425 REPORTER_ASSERT(reporter, 3 == startIndex && 4 == stopIndex);
426
Robert Phillips6d344c32020-07-06 10:56:46 -0400427 cleanup_backend(context, backEndTex);
428 cleanup_backend(context, backEndTex2);
Greg Danield72dd4d2019-08-29 14:37:46 -0400429
Robert Phillipscf39f372019-09-03 10:29:20 -0400430 context->setResourceCacheLimit(origMaxBytes);
Greg Danield72dd4d2019-08-29 14:37:46 -0400431}
432