blob: 7dcb0ee9425811395ff54d9a69cde47f2fea2b62 [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
8// Include here to ensure SK_SUPPORT_GPU is set correctly before it is examined.
9#include "SkTypes.h"
10
Robert Phillipsfa8c0802017-10-04 08:42:28 -040011#if SK_SUPPORT_GPU
12#ifndef SK_DISABLE_DEFERRED_PROXIES
Robert Phillips5af44de2017-07-18 14:49:38 -040013#include "Test.h"
14
Robert Phillips57aa3672017-07-21 11:38:13 -040015#include "GrContextPriv.h"
16#include "GrGpu.h"
Robert Phillips0bd24dc2018-01-16 08:06:32 -050017#include "GrProxyProvider.h"
Robert Phillips5af44de2017-07-18 14:49:38 -040018#include "GrResourceAllocator.h"
Robert Phillips57aa3672017-07-21 11:38:13 -040019#include "GrResourceProvider.h"
Robert Phillips5af44de2017-07-18 14:49:38 -040020#include "GrSurfaceProxyPriv.h"
Robert Phillips57aa3672017-07-21 11:38:13 -040021#include "GrTest.h"
22#include "GrTexture.h"
Robert Phillips5af44de2017-07-18 14:49:38 -040023#include "GrTextureProxy.h"
24
Robert Phillips57aa3672017-07-21 11:38:13 -040025struct ProxyParams {
26 int fSize;
27 bool fIsRT;
28 GrPixelConfig fConfig;
29 SkBackingFit fFit;
30 int fSampleCnt;
31 GrSurfaceOrigin fOrigin;
32 // TODO: do we care about mipmapping
33};
34
Robert Phillips1afd4cd2018-01-08 13:40:32 -050035static sk_sp<GrSurfaceProxy> make_deferred(GrProxyProvider* proxyProvider, const ProxyParams& p) {
Robert Phillips57aa3672017-07-21 11:38:13 -040036 GrSurfaceDesc desc;
37 desc.fFlags = p.fIsRT ? kRenderTarget_GrSurfaceFlag : kNone_GrSurfaceFlags;
38 desc.fOrigin = p.fOrigin;
39 desc.fWidth = p.fSize;
40 desc.fHeight = p.fSize;
41 desc.fConfig = p.fConfig;
42 desc.fSampleCnt = p.fSampleCnt;
43
Robert Phillips0bd24dc2018-01-16 08:06:32 -050044 return proxyProvider->createProxy(desc, p.fFit, SkBudgeted::kNo);
Robert Phillips57aa3672017-07-21 11:38:13 -040045}
46
47static sk_sp<GrSurfaceProxy> make_backend(GrContext* context, const ProxyParams& p,
Robert Phillipsd21b2a52017-12-12 13:01:25 -050048 GrBackendTexture* backendTex) {
Robert Phillips0bd24dc2018-01-16 08:06:32 -050049 GrProxyProvider* proxyProvider = context->contextPriv().proxyProvider();
Robert Phillipsf35fd8d2018-01-22 10:48:15 -050050 GrGpu* gpu = context->contextPriv().getGpu();
Robert Phillips0bd24dc2018-01-16 08:06:32 -050051
Robert Phillipsf35fd8d2018-01-22 10:48:15 -050052 *backendTex = gpu->createTestingOnlyBackendTexture(nullptr, p.fSize, p.fSize,
53 p.fConfig, false,
54 GrMipMapped::kNo);
Robert Phillips57aa3672017-07-21 11:38:13 -040055
Robert Phillips0bd24dc2018-01-16 08:06:32 -050056 return proxyProvider->createWrappedTextureProxy(*backendTex, p.fOrigin);
Robert Phillips57aa3672017-07-21 11:38:13 -040057}
58
Robert Phillipsd21b2a52017-12-12 13:01:25 -050059static void cleanup_backend(GrContext* context, GrBackendTexture* backendTex) {
Robert Phillipsf35fd8d2018-01-22 10:48:15 -050060 context->contextPriv().getGpu()->deleteTestingOnlyBackendTexture(backendTex);
Robert Phillips57aa3672017-07-21 11:38:13 -040061}
62
Robert Phillips5af44de2017-07-18 14:49:38 -040063// Basic test that two proxies with overlapping intervals and compatible descriptors are
64// assigned different GrSurfaces.
Robert Phillips57aa3672017-07-21 11:38:13 -040065static void overlap_test(skiatest::Reporter* reporter, GrResourceProvider* resourceProvider,
66 sk_sp<GrSurfaceProxy> p1, sk_sp<GrSurfaceProxy> p2,
67 bool expectedResult) {
Robert Phillips5af44de2017-07-18 14:49:38 -040068 GrResourceAllocator alloc(resourceProvider);
69
70 alloc.addInterval(p1.get(), 0, 4);
71 alloc.addInterval(p2.get(), 1, 2);
Robert Phillipseafd48a2017-11-16 07:52:08 -050072 alloc.markEndOfOpList(0);
Robert Phillips5af44de2017-07-18 14:49:38 -040073
Robert Phillipseafd48a2017-11-16 07:52:08 -050074 int startIndex, stopIndex;
Greg Danielaa3dfbe2018-01-29 10:34:25 -050075 GrResourceAllocator::AssignError error;
76 alloc.assign(&startIndex, &stopIndex, &error);
77 REPORTER_ASSERT(reporter, GrResourceAllocator::AssignError::kNoError == error);
Robert Phillips5af44de2017-07-18 14:49:38 -040078
79 REPORTER_ASSERT(reporter, p1->priv().peekSurface());
80 REPORTER_ASSERT(reporter, p2->priv().peekSurface());
Robert Phillips57aa3672017-07-21 11:38:13 -040081 bool doTheBackingStoresMatch = p1->underlyingUniqueID() == p2->underlyingUniqueID();
82 REPORTER_ASSERT(reporter, expectedResult == doTheBackingStoresMatch);
Robert Phillips5af44de2017-07-18 14:49:38 -040083}
84
Robert Phillips57aa3672017-07-21 11:38:13 -040085// Test various cases when two proxies do not have overlapping intervals.
86// This mainly acts as a test of the ResourceAllocator's free pool.
87static void non_overlap_test(skiatest::Reporter* reporter, GrResourceProvider* resourceProvider,
88 sk_sp<GrSurfaceProxy> p1, sk_sp<GrSurfaceProxy> p2,
89 bool expectedResult) {
90 GrResourceAllocator alloc(resourceProvider);
91
92 alloc.addInterval(p1.get(), 0, 2);
93 alloc.addInterval(p2.get(), 3, 5);
Robert Phillipseafd48a2017-11-16 07:52:08 -050094 alloc.markEndOfOpList(0);
Robert Phillips57aa3672017-07-21 11:38:13 -040095
Robert Phillipseafd48a2017-11-16 07:52:08 -050096 int startIndex, stopIndex;
Greg Danielaa3dfbe2018-01-29 10:34:25 -050097 GrResourceAllocator::AssignError error;
98 alloc.assign(&startIndex, &stopIndex, &error);
99 REPORTER_ASSERT(reporter, GrResourceAllocator::AssignError::kNoError == error);
Robert Phillips57aa3672017-07-21 11:38:13 -0400100
101 REPORTER_ASSERT(reporter, p1->priv().peekSurface());
102 REPORTER_ASSERT(reporter, p2->priv().peekSurface());
103 bool doTheBackingStoresMatch = p1->underlyingUniqueID() == p2->underlyingUniqueID();
104 REPORTER_ASSERT(reporter, expectedResult == doTheBackingStoresMatch);
105}
106
107DEF_GPUTEST_FOR_RENDERING_CONTEXTS(ResourceAllocatorTest, reporter, ctxInfo) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500108 GrProxyProvider* proxyProvider = ctxInfo.grContext()->contextPriv().proxyProvider();
Robert Phillips6be756b2018-01-16 15:07:54 -0500109 GrResourceProvider* resourceProvider = ctxInfo.grContext()->contextPriv().resourceProvider();
Robert Phillips5af44de2017-07-18 14:49:38 -0400110
Robert Phillips57aa3672017-07-21 11:38:13 -0400111 struct TestCase {
112 ProxyParams fP1;
113 ProxyParams fP2;
114 bool fExpectation;
115 };
116
117 constexpr bool kRT = true;
118 constexpr bool kNotRT = false;
119
120 constexpr bool kShare = true;
121 constexpr bool kDontShare = false;
122 // Non-RT GrSurfaces are never recycled on some platforms.
123 bool kConditionallyShare = resourceProvider->caps()->reuseScratchTextures();
124
125 const GrPixelConfig kRGBA = kRGBA_8888_GrPixelConfig;
126 const GrPixelConfig kBGRA = kBGRA_8888_GrPixelConfig;
127
128 const SkBackingFit kE = SkBackingFit::kExact;
129 const SkBackingFit kA = SkBackingFit::kApprox;
130
131 const GrSurfaceOrigin kTL = kTopLeft_GrSurfaceOrigin;
132 const GrSurfaceOrigin kBL = kBottomLeft_GrSurfaceOrigin;
133
134 //--------------------------------------------------------------------------------------------
135 TestCase gOverlappingTests[] = {
136 //----------------------------------------------------------------------------------------
137 // Two proxies with overlapping intervals and compatible descriptors should never share
138 // RT version
139 { { 64, kRT, kRGBA, kA, 0, kTL }, { 64, kRT, kRGBA, kA, 0, kTL }, kDontShare },
140 // non-RT version
141 { { 64, kNotRT, kRGBA, kA, 0, kTL }, { 64, kNotRT, kRGBA, kA, 0, kTL }, kDontShare },
142 };
143
144 for (auto test : gOverlappingTests) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500145 sk_sp<GrSurfaceProxy> p1 = make_deferred(proxyProvider, test.fP1);
146 sk_sp<GrSurfaceProxy> p2 = make_deferred(proxyProvider, test.fP2);
Robert Phillipseafd48a2017-11-16 07:52:08 -0500147 overlap_test(reporter, resourceProvider,
148 std::move(p1), std::move(p2), test.fExpectation);
Robert Phillips57aa3672017-07-21 11:38:13 -0400149 }
150
Brian Salomon18c52a72018-02-02 06:53:26 -0500151 int k2 = ctxInfo.grContext()->caps()->getSampleCount(2, kRGBA);
152 int k4 = ctxInfo.grContext()->caps()->getSampleCount(4, kRGBA);
Robert Phillips57aa3672017-07-21 11:38:13 -0400153
154 //--------------------------------------------------------------------------------------------
155 TestCase gNonOverlappingTests[] = {
156 //----------------------------------------------------------------------------------------
157 // Two non-overlapping intervals w/ compatible proxies should share
158 // both same size & approx
159 { { 64, kRT, kRGBA, kA, 0, kTL }, { 64, kRT, kRGBA, kA, 0, kTL }, kShare },
160 { { 64, kNotRT, kRGBA, kA, 0, kTL }, { 64, kNotRT, kRGBA, kA, 0, kTL }, kConditionallyShare },
161 // diffs sizes but still approx
162 { { 64, kRT, kRGBA, kA, 0, kTL }, { 50, kRT, kRGBA, kA, 0, kTL }, kShare },
163 { { 64, kNotRT, kRGBA, kA, 0, kTL }, { 50, kNotRT, kRGBA, kA, 0, kTL }, kConditionallyShare },
164 // sames sizes but exact
165 { { 64, kRT, kRGBA, kE, 0, kTL }, { 64, kRT, kRGBA, kE, 0, kTL }, kShare },
166 { { 64, kNotRT, kRGBA, kE, 0, kTL }, { 64, kNotRT, kRGBA, kE, 0, kTL }, kConditionallyShare },
167 //----------------------------------------------------------------------------------------
168 // Two non-overlapping intervals w/ different exact sizes should not share
169 { { 56, kRT, kRGBA, kE, 0, kTL }, { 54, kRT, kRGBA, kE, 0, kTL }, kDontShare },
170 // Two non-overlapping intervals w/ _very different_ approx sizes should not share
171 { { 255, kRT, kRGBA, kA, 0, kTL }, { 127, kRT, kRGBA, kA, 0, kTL }, kDontShare },
172 // Two non-overlapping intervals w/ different MSAA sample counts should not share
173 { { 64, kRT, kRGBA, kA, k2, kTL },{ 64, kRT, kRGBA, kA, k4, kTL}, k2 == k4 },
174 // Two non-overlapping intervals w/ different configs should not share
175 { { 64, kRT, kRGBA, kA, 0, kTL }, { 64, kRT, kBGRA, kA, 0, kTL }, kDontShare },
176 // Two non-overlapping intervals w/ different RT classifications should never share
177 { { 64, kRT, kRGBA, kA, 0, kTL }, { 64, kNotRT, kRGBA, kA, 0, kTL }, kDontShare },
178 { { 64, kNotRT, kRGBA, kA, 0, kTL }, { 64, kRT, kRGBA, kA, 0, kTL }, kDontShare },
Robert Phillipsb0e93a22017-08-29 08:26:54 -0400179 // Two non-overlapping intervals w/ different origins should share
180 { { 64, kRT, kRGBA, kA, 0, kTL }, { 64, kRT, kRGBA, kA, 0, kBL }, kShare },
Robert Phillips57aa3672017-07-21 11:38:13 -0400181 };
182
183 for (auto test : gNonOverlappingTests) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500184 sk_sp<GrSurfaceProxy> p1 = make_deferred(proxyProvider, test.fP1);
185 sk_sp<GrSurfaceProxy> p2 = make_deferred(proxyProvider, test.fP2);
Robert Phillips57aa3672017-07-21 11:38:13 -0400186 if (!p1 || !p2) {
187 continue; // creation can fail (i.e., for msaa4 on iOS)
188 }
Robert Phillipseafd48a2017-11-16 07:52:08 -0500189 non_overlap_test(reporter, resourceProvider,
190 std::move(p1), std::move(p2), test.fExpectation);
Robert Phillips57aa3672017-07-21 11:38:13 -0400191 }
192
193 {
194 // Wrapped backend textures should never be reused
195 TestCase t[1] = {
196 { { 64, kNotRT, kRGBA, kE, 0, kTL }, { 64, kNotRT, kRGBA, kE, 0, kTL }, kDontShare }
197 };
198
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500199 GrBackendTexture backEndTex;
200 sk_sp<GrSurfaceProxy> p1 = make_backend(ctxInfo.grContext(), t[0].fP1, &backEndTex);
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500201 sk_sp<GrSurfaceProxy> p2 = make_deferred(proxyProvider, t[0].fP2);
Robert Phillipseafd48a2017-11-16 07:52:08 -0500202 non_overlap_test(reporter, resourceProvider,
203 std::move(p1), std::move(p2), t[0].fExpectation);
Robert Phillipsd21b2a52017-12-12 13:01:25 -0500204 cleanup_backend(ctxInfo.grContext(), &backEndTex);
Robert Phillips57aa3672017-07-21 11:38:13 -0400205 }
Robert Phillips5af44de2017-07-18 14:49:38 -0400206}
207
208#endif
Robert Phillipsfa8c0802017-10-04 08:42:28 -0400209#endif