blob: 01c881c90b6ef6c231cd2521e8a5b1265a6bbe4b [file] [log] [blame]
robertphillips76948d42016-05-04 12:47:41 -07001/*
2 * Copyright 2016 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
12#if SK_SUPPORT_GPU
Robert Phillips8bf1f9f2017-05-31 15:01:20 -040013
Greg Danielbcf612b2017-05-01 13:50:58 +000014#include "GrBackendSurface.h"
Robert Phillips1afd4cd2018-01-08 13:40:32 -050015#include "GrContextPriv.h"
Robert Phillipsc7635fa2016-10-28 13:25:24 -040016#include "GrRenderTargetPriv.h"
robertphillips76948d42016-05-04 12:47:41 -070017#include "GrRenderTargetProxy.h"
Brian Osman32342f02017-03-04 08:12:46 -050018#include "GrResourceProvider.h"
Robert Phillipsf95b1752017-08-31 08:56:07 -040019#include "GrSurfaceProxyPriv.h"
Robert Phillips646e4292017-06-13 12:44:56 -040020#include "GrTexture.h"
Brian Osman32342f02017-03-04 08:12:46 -050021#include "GrTextureProxy.h"
robertphillips76948d42016-05-04 12:47:41 -070022
robertphillips8abb3702016-08-31 14:04:06 -070023// Check that the surface proxy's member vars are set as expected
robertphillips76948d42016-05-04 12:47:41 -070024static void check_surface(skiatest::Reporter* reporter,
25 GrSurfaceProxy* proxy,
26 GrSurfaceOrigin origin,
Greg Danielbcf612b2017-05-01 13:50:58 +000027 int width, int height,
robertphillips8abb3702016-08-31 14:04:06 -070028 GrPixelConfig config,
Robert Phillips294870f2016-11-11 12:38:40 -050029 const GrGpuResource::UniqueID& uniqueID,
Robert Phillipsabacf092016-11-02 10:23:32 -040030 SkBudgeted budgeted) {
robertphillips76948d42016-05-04 12:47:41 -070031 REPORTER_ASSERT(reporter, proxy->origin() == origin);
32 REPORTER_ASSERT(reporter, proxy->width() == width);
33 REPORTER_ASSERT(reporter, proxy->height() == height);
34 REPORTER_ASSERT(reporter, proxy->config() == config);
Robert Phillips294870f2016-11-11 12:38:40 -050035 if (!uniqueID.isInvalid()) {
36 REPORTER_ASSERT(reporter, proxy->uniqueID().asUInt() == uniqueID.asUInt());
37 } else {
38 REPORTER_ASSERT(reporter, !proxy->uniqueID().isInvalid());
robertphillips8abb3702016-08-31 14:04:06 -070039 }
Robert Phillipsabacf092016-11-02 10:23:32 -040040 REPORTER_ASSERT(reporter, proxy->isBudgeted() == budgeted);
robertphillips76948d42016-05-04 12:47:41 -070041}
42
43static void check_rendertarget(skiatest::Reporter* reporter,
Robert Phillipsec2249f2016-11-09 08:54:35 -050044 const GrCaps& caps,
Brian Osman32342f02017-03-04 08:12:46 -050045 GrResourceProvider* provider,
robertphillips76948d42016-05-04 12:47:41 -070046 GrRenderTargetProxy* rtProxy,
Robert Phillipsabacf092016-11-02 10:23:32 -040047 int numSamples,
Robert Phillipsec2249f2016-11-09 08:54:35 -050048 SkBackingFit fit,
Robert Phillips294870f2016-11-11 12:38:40 -050049 int expectedMaxWindowRects,
50 bool wasWrapped) {
Robert Phillipsec2249f2016-11-09 08:54:35 -050051 REPORTER_ASSERT(reporter, rtProxy->maxWindowRectangles(caps) == expectedMaxWindowRects);
Robert Phillipsabacf092016-11-02 10:23:32 -040052 REPORTER_ASSERT(reporter, rtProxy->numStencilSamples() == numSamples);
53
Robert Phillips294870f2016-11-11 12:38:40 -050054 GrSurfaceProxy::UniqueID idBefore = rtProxy->uniqueID();
Robert Phillipseee4d6e2017-06-05 09:26:07 -040055 REPORTER_ASSERT(reporter, rtProxy->instantiate(provider));
56 GrRenderTarget* rt = rtProxy->priv().peekRenderTarget();
robertphillips76948d42016-05-04 12:47:41 -070057
Robert Phillips294870f2016-11-11 12:38:40 -050058 REPORTER_ASSERT(reporter, rtProxy->uniqueID() == idBefore);
59 if (wasWrapped) {
60 // Wrapped resources share their uniqueID with the wrapping RenderTargetProxy
61 REPORTER_ASSERT(reporter, rtProxy->uniqueID().asUInt() == rt->uniqueID().asUInt());
62 } else {
63 // Deferred resources should always have a different ID from their instantiated rendertarget
64 REPORTER_ASSERT(reporter, rtProxy->uniqueID().asUInt() != rt->uniqueID().asUInt());
65 }
66
robertphillips76948d42016-05-04 12:47:41 -070067 if (SkBackingFit::kExact == fit) {
68 REPORTER_ASSERT(reporter, rt->width() == rtProxy->width());
69 REPORTER_ASSERT(reporter, rt->height() == rtProxy->height());
70 } else {
71 REPORTER_ASSERT(reporter, rt->width() >= rtProxy->width());
Robert Phillips294870f2016-11-11 12:38:40 -050072 REPORTER_ASSERT(reporter, rt->height() >= rtProxy->height());
robertphillips76948d42016-05-04 12:47:41 -070073 }
74 REPORTER_ASSERT(reporter, rt->config() == rtProxy->config());
75
Brian Salomon7c8460e2017-05-12 11:36:10 -040076 REPORTER_ASSERT(reporter, rt->fsaaType() == rtProxy->fsaaType());
robertphillips76948d42016-05-04 12:47:41 -070077 REPORTER_ASSERT(reporter, rt->numColorSamples() == rtProxy->numColorSamples());
78 REPORTER_ASSERT(reporter, rt->numStencilSamples() == rtProxy->numStencilSamples());
csmartdaltonf9635992016-08-10 11:09:07 -070079 REPORTER_ASSERT(reporter, rt->renderTargetPriv().flags() == rtProxy->testingOnly_getFlags());
robertphillips76948d42016-05-04 12:47:41 -070080}
81
82static void check_texture(skiatest::Reporter* reporter,
Brian Osman32342f02017-03-04 08:12:46 -050083 GrResourceProvider* provider,
robertphillips76948d42016-05-04 12:47:41 -070084 GrTextureProxy* texProxy,
Robert Phillips294870f2016-11-11 12:38:40 -050085 SkBackingFit fit,
86 bool wasWrapped) {
87 GrSurfaceProxy::UniqueID idBefore = texProxy->uniqueID();
Robert Phillipseee4d6e2017-06-05 09:26:07 -040088
89 REPORTER_ASSERT(reporter, texProxy->instantiate(provider));
90 GrTexture* tex = texProxy->priv().peekTexture();
robertphillips76948d42016-05-04 12:47:41 -070091
Robert Phillips294870f2016-11-11 12:38:40 -050092 REPORTER_ASSERT(reporter, texProxy->uniqueID() == idBefore);
93 if (wasWrapped) {
94 // Wrapped resources share their uniqueID with the wrapping TextureProxy
95 REPORTER_ASSERT(reporter, texProxy->uniqueID().asUInt() == tex->uniqueID().asUInt());
96 } else {
97 // Deferred resources should always have a different ID from their instantiated texture
98 REPORTER_ASSERT(reporter, texProxy->uniqueID().asUInt() != tex->uniqueID().asUInt());
99 }
100
robertphillips76948d42016-05-04 12:47:41 -0700101 if (SkBackingFit::kExact == fit) {
102 REPORTER_ASSERT(reporter, tex->width() == texProxy->width());
103 REPORTER_ASSERT(reporter, tex->height() == texProxy->height());
104 } else {
105 REPORTER_ASSERT(reporter, tex->width() >= texProxy->width());
106 REPORTER_ASSERT(reporter, tex->height() >= texProxy->height());
107 }
108 REPORTER_ASSERT(reporter, tex->config() == texProxy->config());
109}
110
111
robertphillips8abb3702016-08-31 14:04:06 -0700112DEF_GPUTEST_FOR_RENDERING_CONTEXTS(DeferredProxyTest, reporter, ctxInfo) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500113 GrProxyProvider* proxyProvider = ctxInfo.grContext()->contextPriv().proxyProvider();
114 GrResourceProvider* resourceProvider = ctxInfo.grContext()->resourceProvider();
Robert Phillipsabacf092016-11-02 10:23:32 -0400115 const GrCaps& caps = *ctxInfo.grContext()->caps();
robertphillips76948d42016-05-04 12:47:41 -0700116
Robert Phillips294870f2016-11-11 12:38:40 -0500117 const GrGpuResource::UniqueID kInvalidResourceID = GrGpuResource::UniqueID::InvalidID();
118
Robert Phillips6520a692017-02-01 09:20:00 -0500119 int attempt = 0; // useful for debugging
120
robertphillips76948d42016-05-04 12:47:41 -0700121 for (auto origin : { kBottomLeft_GrSurfaceOrigin, kTopLeft_GrSurfaceOrigin }) {
Robert Phillips40d94952017-01-30 14:15:59 -0500122 for (auto widthHeight : { 100, 128, 1048576 }) {
Robert Phillips6520a692017-02-01 09:20:00 -0500123 for (auto config : { kAlpha_8_GrPixelConfig, kRGB_565_GrPixelConfig,
Robert Phillips92de6312017-05-23 07:43:48 -0400124 kRGBA_8888_GrPixelConfig }) {
robertphillips76948d42016-05-04 12:47:41 -0700125 for (auto fit : { SkBackingFit::kExact, SkBackingFit::kApprox }) {
126 for (auto budgeted : { SkBudgeted::kYes, SkBudgeted::kNo }) {
Robert Phillips6520a692017-02-01 09:20:00 -0500127 for (auto numSamples : { 0, 4, 16, 128 }) {
robertphillips76948d42016-05-04 12:47:41 -0700128 GrSurfaceDesc desc;
Robert Phillips84a81202016-11-04 11:59:10 -0400129 desc.fFlags = kRenderTarget_GrSurfaceFlag;
robertphillips76948d42016-05-04 12:47:41 -0700130 desc.fOrigin = origin;
131 desc.fWidth = widthHeight;
132 desc.fHeight = widthHeight;
133 desc.fConfig = config;
134 desc.fSampleCnt = numSamples;
135
Robert Phillips6520a692017-02-01 09:20:00 -0500136 {
137 sk_sp<GrTexture> tex;
138 if (SkBackingFit::kApprox == fit) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500139 tex = resourceProvider->createApproxTexture(desc, 0);
Robert Phillips6520a692017-02-01 09:20:00 -0500140 } else {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500141 tex = resourceProvider->createTexture(desc, budgeted);
Robert Phillips6520a692017-02-01 09:20:00 -0500142 }
143
Robert Phillips2f493142017-03-02 18:18:38 -0500144 sk_sp<GrTextureProxy> proxy(GrSurfaceProxy::MakeDeferred(
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500145 proxyProvider, desc,
Robert Phillipsabacf092016-11-02 10:23:32 -0400146 fit, budgeted));
Robert Phillips2f493142017-03-02 18:18:38 -0500147 REPORTER_ASSERT(reporter, SkToBool(tex) == SkToBool(proxy));
148 if (proxy) {
149 REPORTER_ASSERT(reporter, proxy->asRenderTargetProxy());
Robert Phillips40d94952017-01-30 14:15:59 -0500150 // This forces the proxy to compute and cache its
151 // pre-instantiation size guess. Later, when it is actually
152 // instantiated, it checks that the instantiated size is <= to
153 // the pre-computation. If the proxy never computed its
154 // pre-instantiation size then the check is skipped.
Robert Phillips2f493142017-03-02 18:18:38 -0500155 proxy->gpuMemorySize();
Robert Phillipsb4460882016-11-17 14:43:51 -0500156
Robert Phillips2f493142017-03-02 18:18:38 -0500157 check_surface(reporter, proxy.get(), origin,
Robert Phillips40d94952017-01-30 14:15:59 -0500158 widthHeight, widthHeight, config,
159 kInvalidResourceID, budgeted);
Greg Daniel81e7bf82017-07-19 14:47:42 -0400160 int supportedSamples = caps.getSampleCount(numSamples, config);
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500161 check_rendertarget(reporter, caps, resourceProvider,
Robert Phillips2f493142017-03-02 18:18:38 -0500162 proxy->asRenderTargetProxy(),
Greg Daniel81e7bf82017-07-19 14:47:42 -0400163 supportedSamples,
Robert Phillips40d94952017-01-30 14:15:59 -0500164 fit, caps.maxWindowRectangles(), false);
165 }
robertphillips76948d42016-05-04 12:47:41 -0700166 }
167
Robert Phillips84a81202016-11-04 11:59:10 -0400168 desc.fFlags = kNone_GrSurfaceFlags;
robertphillips76948d42016-05-04 12:47:41 -0700169
Robert Phillips6520a692017-02-01 09:20:00 -0500170 {
171 sk_sp<GrTexture> tex;
172 if (SkBackingFit::kApprox == fit) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500173 tex = resourceProvider->createApproxTexture(desc, 0);
Robert Phillips6520a692017-02-01 09:20:00 -0500174 } else {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500175 tex = resourceProvider->createTexture(desc, budgeted);
Robert Phillips6520a692017-02-01 09:20:00 -0500176 }
Robert Phillipsb4460882016-11-17 14:43:51 -0500177
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500178 sk_sp<GrTextureProxy> proxy(GrSurfaceProxy::MakeDeferred(
179 proxyProvider,
180 desc,
181 fit,
182 budgeted));
Robert Phillips2f493142017-03-02 18:18:38 -0500183 REPORTER_ASSERT(reporter, SkToBool(tex) == SkToBool(proxy));
184 if (proxy) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500185 // This forces the proxy to compute and cache its
186 // pre-instantiation size guess. Later, when it is actually
187 // instantiated, it checks that the instantiated size is <= to
188 // the pre-computation. If the proxy never computed its
189 // pre-instantiation size then the check is skipped.
Robert Phillips2f493142017-03-02 18:18:38 -0500190 proxy->gpuMemorySize();
Robert Phillips6520a692017-02-01 09:20:00 -0500191
Robert Phillips2f493142017-03-02 18:18:38 -0500192 check_surface(reporter, proxy.get(), origin,
Robert Phillips6520a692017-02-01 09:20:00 -0500193 widthHeight, widthHeight, config,
194 kInvalidResourceID, budgeted);
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500195 check_texture(reporter, resourceProvider,
196 proxy->asTextureProxy(), fit, false);
Robert Phillips6520a692017-02-01 09:20:00 -0500197 }
Robert Phillips40d94952017-01-30 14:15:59 -0500198 }
Robert Phillips6520a692017-02-01 09:20:00 -0500199
200 attempt++;
robertphillips76948d42016-05-04 12:47:41 -0700201 }
202 }
203 }
204 }
205 }
206 }
207}
208
egdanielab527a52016-06-28 08:07:26 -0700209DEF_GPUTEST_FOR_RENDERING_CONTEXTS(WrappedProxyTest, reporter, ctxInfo) {
Brian Osman32342f02017-03-04 08:12:46 -0500210 GrResourceProvider* provider = ctxInfo.grContext()->resourceProvider();
csmartdaltonf9635992016-08-10 11:09:07 -0700211 const GrCaps& caps = *ctxInfo.grContext()->caps();
robertphillips76948d42016-05-04 12:47:41 -0700212
213 static const int kWidthHeight = 100;
214
215 for (auto origin : { kBottomLeft_GrSurfaceOrigin, kTopLeft_GrSurfaceOrigin }) {
216 for (auto config : { kAlpha_8_GrPixelConfig, kRGBA_8888_GrPixelConfig }) {
217 for (auto budgeted : { SkBudgeted::kYes, SkBudgeted::kNo }) {
218 for (auto numSamples: { 0, 4}) {
Greg Daniel81e7bf82017-07-19 14:47:42 -0400219 int supportedNumSamples = caps.getSampleCount(numSamples, config);
Robert Phillips78de2122017-04-26 07:44:26 -0400220
csmartdaltonf9635992016-08-10 11:09:07 -0700221 bool renderable = caps.isConfigRenderable(config, numSamples > 0);
robertphillips76948d42016-05-04 12:47:41 -0700222
223 GrSurfaceDesc desc;
224 desc.fOrigin = origin;
225 desc.fWidth = kWidthHeight;
226 desc.fHeight = kWidthHeight;
227 desc.fConfig = config;
Greg Daniel81e7bf82017-07-19 14:47:42 -0400228 desc.fSampleCnt = supportedNumSamples;
robertphillips76948d42016-05-04 12:47:41 -0700229
csmartdaltonf9635992016-08-10 11:09:07 -0700230 // External on-screen render target.
231 if (renderable && kOpenGL_GrBackend == ctxInfo.backend()) {
Greg Danielbcf612b2017-05-01 13:50:58 +0000232 GrGLFramebufferInfo fboInfo;
233 fboInfo.fFBOID = 0;
234 GrBackendRenderTarget backendRT(kWidthHeight, kWidthHeight, numSamples, 8,
235 config, fboInfo);
csmartdaltonf9635992016-08-10 11:09:07 -0700236
csmartdaltonf9635992016-08-10 11:09:07 -0700237 sk_sp<GrRenderTarget> defaultFBO(
Robert Phillipsb0e93a22017-08-29 08:26:54 -0400238 provider->wrapBackendRenderTarget(backendRT));
csmartdaltonf9635992016-08-10 11:09:07 -0700239
Robert Phillips066f0202017-07-25 10:16:35 -0400240 sk_sp<GrSurfaceProxy> sProxy(GrSurfaceProxy::MakeWrapped(defaultFBO,
241 origin));
Robert Phillips37430132016-11-09 06:50:43 -0500242 check_surface(reporter, sProxy.get(), origin,
Robert Phillipsabacf092016-11-02 10:23:32 -0400243 kWidthHeight, kWidthHeight, config,
244 defaultFBO->uniqueID(), SkBudgeted::kNo);
Robert Phillipsec2249f2016-11-09 08:54:35 -0500245 check_rendertarget(reporter, caps, provider, sProxy->asRenderTargetProxy(),
Greg Danield62f5542017-07-20 09:11:41 -0400246 supportedNumSamples, SkBackingFit::kExact, 0, true);
csmartdaltonf9635992016-08-10 11:09:07 -0700247 }
248
robertphillips76948d42016-05-04 12:47:41 -0700249 sk_sp<GrTexture> tex;
250
csmartdaltonf9635992016-08-10 11:09:07 -0700251 // Internal offscreen render target.
robertphillips76948d42016-05-04 12:47:41 -0700252 if (renderable) {
253 desc.fFlags = kRenderTarget_GrSurfaceFlag;
Robert Phillipse78b7252017-04-06 07:59:41 -0400254 tex = provider->createTexture(desc, budgeted);
Robert Phillipsd9971c02017-08-02 11:51:43 -0400255 if (!tex) {
256 continue; // This can fail on Mesa
257 }
robertphillips76948d42016-05-04 12:47:41 -0700258 sk_sp<GrRenderTarget> rt(sk_ref_sp(tex->asRenderTarget()));
259
Robert Phillips066f0202017-07-25 10:16:35 -0400260 sk_sp<GrSurfaceProxy> sProxy(GrSurfaceProxy::MakeWrapped(rt, origin));
Robert Phillips37430132016-11-09 06:50:43 -0500261 check_surface(reporter, sProxy.get(), origin,
Robert Phillipsabacf092016-11-02 10:23:32 -0400262 kWidthHeight, kWidthHeight, config,
263 rt->uniqueID(), budgeted);
Robert Phillipsec2249f2016-11-09 08:54:35 -0500264 check_rendertarget(reporter, caps, provider, sProxy->asRenderTargetProxy(),
Greg Danield62f5542017-07-20 09:11:41 -0400265 supportedNumSamples, SkBackingFit::kExact,
Robert Phillips294870f2016-11-11 12:38:40 -0500266 caps.maxWindowRectangles(), true);
robertphillips76948d42016-05-04 12:47:41 -0700267 }
268
269 if (!tex) {
270 SkASSERT(kNone_GrSurfaceFlags == desc.fFlags );
271 desc.fSampleCnt = 0;
Robert Phillipse78b7252017-04-06 07:59:41 -0400272 tex = provider->createTexture(desc, budgeted);
robertphillips76948d42016-05-04 12:47:41 -0700273 }
274
Robert Phillips066f0202017-07-25 10:16:35 -0400275 sk_sp<GrSurfaceProxy> sProxy(GrSurfaceProxy::MakeWrapped(tex, origin));
Robert Phillips37430132016-11-09 06:50:43 -0500276 check_surface(reporter, sProxy.get(), origin,
Robert Phillipsabacf092016-11-02 10:23:32 -0400277 kWidthHeight, kWidthHeight, config, tex->uniqueID(), budgeted);
Robert Phillips37430132016-11-09 06:50:43 -0500278 check_texture(reporter, provider, sProxy->asTextureProxy(),
Robert Phillips294870f2016-11-11 12:38:40 -0500279 SkBackingFit::kExact, true);
robertphillips76948d42016-05-04 12:47:41 -0700280 }
281 }
282 }
283 }
284}
285
Robert Phillips78de2122017-04-26 07:44:26 -0400286DEF_GPUTEST_FOR_RENDERING_CONTEXTS(ZeroSizedProxyTest, reporter, ctxInfo) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500287 GrProxyProvider* provider = ctxInfo.grContext()->contextPriv().proxyProvider();
Robert Phillips78de2122017-04-26 07:44:26 -0400288
289 for (auto flags : { kRenderTarget_GrSurfaceFlag, kNone_GrSurfaceFlags }) {
290 for (auto fit : { SkBackingFit::kExact, SkBackingFit::kApprox }) {
291 for (int width : { 0, 100 }) {
292 for (int height : { 0, 100}) {
293 if (width && height) {
294 continue; // not zero-sized
295 }
296
297 GrSurfaceDesc desc;
298 desc.fFlags = flags;
299 desc.fOrigin = kBottomLeft_GrSurfaceOrigin;
300 desc.fWidth = width;
301 desc.fHeight = height;
302 desc.fConfig = kRGBA_8888_GrPixelConfig;
303 desc.fSampleCnt = 0;
304
305 sk_sp<GrTextureProxy> proxy(GrSurfaceProxy::MakeDeferred(provider,
306 desc,
307 fit,
308 SkBudgeted::kNo));
309 REPORTER_ASSERT(reporter, !proxy);
310 }
311 }
312 }
313 }
314}
315
robertphillips76948d42016-05-04 12:47:41 -0700316#endif