blob: 5c665d13af068e4609ada141ef5780ab121a8bad [file] [log] [blame]
bsalomon@google.com686bcb82013-04-09 15:04:12 +00001/*
2 * Copyright 2013 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
bsalomona2c23232014-11-25 07:41:12 -08008#include "SkTypes.h"
9
bsalomon@google.com686bcb82013-04-09 15:04:12 +000010#if SK_SUPPORT_GPU
11
bsalomon@google.com686bcb82013-04-09 15:04:12 +000012#include "GrContext.h"
Greg Daniel7ef28f32017-04-20 16:41:55 +000013#include "GrContextPriv.h"
bsalomon091f60c2015-11-10 11:54:56 -080014#include "GrGpu.h"
Robert Phillips2890fbf2017-07-26 15:48:41 -040015#include "GrRenderTarget.h"
Brian Osman32342f02017-03-04 08:12:46 -050016#include "GrResourceProvider.h"
Greg Daniel7ef28f32017-04-20 16:41:55 +000017#include "GrTest.h"
bsalomon@google.com686bcb82013-04-09 15:04:12 +000018#include "GrTexture.h"
bsalomonafbf2d62014-09-30 12:18:44 -070019#include "GrSurfacePriv.h"
Brian Osman48c99192017-06-02 08:45:06 -040020#include "SkMipMap.h"
tfarina@chromium.org4ee16bf2014-01-10 22:08:27 +000021#include "Test.h"
bsalomon@google.com686bcb82013-04-09 15:04:12 +000022
bsalomona2c23232014-11-25 07:41:12 -080023// Tests that GrSurface::asTexture(), GrSurface::asRenderTarget(), and static upcasting of texture
24// and render targets to GrSurface all work as expected.
bsalomon758586c2016-04-06 14:02:39 -070025DEF_GPUTEST_FOR_NULLGL_CONTEXT(GrSurface, reporter, ctxInfo) {
bsalomon8b7451a2016-05-11 06:33:06 -070026 GrContext* context = ctxInfo.grContext();
kkinnunen15302832015-12-01 04:35:26 -080027 GrSurfaceDesc desc;
kkinnunen15302832015-12-01 04:35:26 -080028 desc.fFlags = kRenderTarget_GrSurfaceFlag;
Robert Phillips16d8ec62017-07-27 16:16:25 -040029 desc.fOrigin = kBottomLeft_GrSurfaceOrigin;
kkinnunen15302832015-12-01 04:35:26 -080030 desc.fWidth = 256;
31 desc.fHeight = 256;
Robert Phillips16d8ec62017-07-27 16:16:25 -040032 desc.fConfig = kRGBA_8888_GrPixelConfig;
kkinnunen15302832015-12-01 04:35:26 -080033 desc.fSampleCnt = 0;
Robert Phillipse78b7252017-04-06 07:59:41 -040034 sk_sp<GrSurface> texRT1 = context->resourceProvider()->createTexture(desc, SkBudgeted::kNo);
bsalomona2c23232014-11-25 07:41:12 -080035
Robert Phillipse78b7252017-04-06 07:59:41 -040036 REPORTER_ASSERT(reporter, texRT1.get() == texRT1->asRenderTarget());
37 REPORTER_ASSERT(reporter, texRT1.get() == texRT1->asTexture());
kkinnunen15302832015-12-01 04:35:26 -080038 REPORTER_ASSERT(reporter, static_cast<GrSurface*>(texRT1->asRenderTarget()) ==
39 texRT1->asTexture());
40 REPORTER_ASSERT(reporter, texRT1->asRenderTarget() ==
41 static_cast<GrSurface*>(texRT1->asTexture()));
42 REPORTER_ASSERT(reporter, static_cast<GrSurface*>(texRT1->asRenderTarget()) ==
43 static_cast<GrSurface*>(texRT1->asTexture()));
bsalomona2c23232014-11-25 07:41:12 -080044
kkinnunen15302832015-12-01 04:35:26 -080045 desc.fFlags = kNone_GrSurfaceFlags;
Robert Phillips16d8ec62017-07-27 16:16:25 -040046 desc.fOrigin = kTopLeft_GrSurfaceOrigin;
Robert Phillipse78b7252017-04-06 07:59:41 -040047 sk_sp<GrTexture> tex1 = context->resourceProvider()->createTexture(desc, SkBudgeted::kNo);
kkinnunen15302832015-12-01 04:35:26 -080048 REPORTER_ASSERT(reporter, nullptr == tex1->asRenderTarget());
Robert Phillipse78b7252017-04-06 07:59:41 -040049 REPORTER_ASSERT(reporter, tex1.get() == tex1->asTexture());
50 REPORTER_ASSERT(reporter, static_cast<GrSurface*>(tex1.get()) == tex1->asTexture());
bsalomon@google.com686bcb82013-04-09 15:04:12 +000051
Greg Daniel7ef28f32017-04-20 16:41:55 +000052 GrBackendObject backendTexHandle = context->getGpu()->createTestingOnlyBackendTexture(
Brian Osman777b5632016-10-14 09:16:21 -040053 nullptr, 256, 256, kRGBA_8888_GrPixelConfig);
Greg Daniel7ef28f32017-04-20 16:41:55 +000054 GrBackendTexture backendTex = GrTest::CreateBackendTexture(context->contextPriv().getBackend(),
55 256,
56 256,
57 kRGBA_8888_GrPixelConfig,
58 backendTexHandle);
bsalomon091f60c2015-11-10 11:54:56 -080059
Brian Salomond17f6582017-07-19 18:28:58 -040060 sk_sp<GrSurface> texRT2 = context->resourceProvider()->wrapRenderableBackendTexture(
61 backendTex, kTopLeft_GrSurfaceOrigin, 0, kBorrow_GrWrapOwnership);
Greg Daniel7ef28f32017-04-20 16:41:55 +000062
bungeman6bd52842016-10-27 09:30:08 -070063 REPORTER_ASSERT(reporter, texRT2.get() == texRT2->asRenderTarget());
64 REPORTER_ASSERT(reporter, texRT2.get() == texRT2->asTexture());
kkinnunen15302832015-12-01 04:35:26 -080065 REPORTER_ASSERT(reporter, static_cast<GrSurface*>(texRT2->asRenderTarget()) ==
66 texRT2->asTexture());
67 REPORTER_ASSERT(reporter, texRT2->asRenderTarget() ==
68 static_cast<GrSurface*>(texRT2->asTexture()));
69 REPORTER_ASSERT(reporter, static_cast<GrSurface*>(texRT2->asRenderTarget()) ==
70 static_cast<GrSurface*>(texRT2->asTexture()));
bsalomon@google.com686bcb82013-04-09 15:04:12 +000071
Greg Daniel7ef28f32017-04-20 16:41:55 +000072 context->getGpu()->deleteTestingOnlyBackendTexture(backendTexHandle);
bsalomon@google.com686bcb82013-04-09 15:04:12 +000073}
74
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -040075// This test checks that the isConfigTexturable and isConfigRenderable are
76// consistent with createTexture's result.
77DEF_GPUTEST_FOR_ALL_CONTEXTS(GrSurfaceRenderability, reporter, ctxInfo) {
78 GrContext* context = ctxInfo.grContext();
Robert Phillips3b3307f2017-05-24 07:44:02 -040079 GrResourceProvider* resourceProvider = context->resourceProvider();
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -040080 const GrCaps* caps = context->caps();
81
82 GrPixelConfig configs[] = {
83 kUnknown_GrPixelConfig,
84 kAlpha_8_GrPixelConfig,
85 kGray_8_GrPixelConfig,
86 kRGB_565_GrPixelConfig,
87 kRGBA_4444_GrPixelConfig,
88 kRGBA_8888_GrPixelConfig,
89 kBGRA_8888_GrPixelConfig,
90 kSRGBA_8888_GrPixelConfig,
91 kSBGRA_8888_GrPixelConfig,
92 kRGBA_8888_sint_GrPixelConfig,
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -040093 kRGBA_float_GrPixelConfig,
94 kRG_float_GrPixelConfig,
95 kAlpha_half_GrPixelConfig,
96 kRGBA_half_GrPixelConfig,
97 };
98 SkASSERT(kGrPixelConfigCnt == SK_ARRAY_COUNT(configs));
99
100 GrSurfaceDesc desc;
101 desc.fWidth = 64;
102 desc.fHeight = 64;
103
Brian Osman48c99192017-06-02 08:45:06 -0400104 // Enough space for the first mip of our largest pixel config
105 const size_t pixelBufferSize = desc.fWidth * desc.fHeight *
106 GrBytesPerPixel(kRGBA_float_GrPixelConfig);
107 std::unique_ptr<char[]> pixelData(new char[pixelBufferSize]);
108 memset(pixelData.get(), 0, pixelBufferSize);
109
110 // We re-use the same mip level objects (with updated pointers and rowBytes) for each config
111 const int levelCount = SkMipMap::ComputeLevelCount(desc.fWidth, desc.fHeight) + 1;
112 std::unique_ptr<GrMipLevel[]> texels(new GrMipLevel[levelCount]);
113
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -0400114 for (GrPixelConfig config : configs) {
115 for (GrSurfaceOrigin origin : { kTopLeft_GrSurfaceOrigin, kBottomLeft_GrSurfaceOrigin }) {
116 desc.fFlags = kNone_GrSurfaceFlags;
117 desc.fOrigin = origin;
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -0400118 desc.fConfig = config;
Robert Phillips16d8ec62017-07-27 16:16:25 -0400119 desc.fSampleCnt = 0;
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -0400120
Robert Phillips3b3307f2017-05-24 07:44:02 -0400121 sk_sp<GrSurface> tex = resourceProvider->createTexture(desc, SkBudgeted::kNo);
122 REPORTER_ASSERT(reporter, SkToBool(tex.get()) == caps->isConfigTexturable(desc.fConfig));
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -0400123
Brian Osman48c99192017-06-02 08:45:06 -0400124 size_t rowBytes = desc.fWidth * GrBytesPerPixel(desc.fConfig);
125 for (int i = 0; i < levelCount; ++i) {
126 texels[i].fPixels = pixelData.get();
127 texels[i].fRowBytes = rowBytes >> i;
128 }
Robert Phillips8e8c7552017-07-10 12:06:05 -0400129 sk_sp<GrTextureProxy> proxy = GrSurfaceProxy::MakeDeferredMipMap(resourceProvider,
130 desc, SkBudgeted::kNo,
131 texels.get(),
132 levelCount);
Brian Osman48c99192017-06-02 08:45:06 -0400133 REPORTER_ASSERT(reporter, SkToBool(proxy.get()) ==
134 (caps->isConfigTexturable(desc.fConfig) &&
135 caps->mipMapSupport() &&
136 !GrPixelConfigIsSint(desc.fConfig)));
137
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -0400138 desc.fFlags = kRenderTarget_GrSurfaceFlag;
Robert Phillips3b3307f2017-05-24 07:44:02 -0400139 tex = resourceProvider->createTexture(desc, SkBudgeted::kNo);
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -0400140 REPORTER_ASSERT(reporter, SkToBool(tex.get()) == caps->isConfigRenderable(config, false));
141
142 desc.fSampleCnt = 4;
Robert Phillips3b3307f2017-05-24 07:44:02 -0400143 tex = resourceProvider->createTexture(desc, SkBudgeted::kNo);
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -0400144 REPORTER_ASSERT(reporter, SkToBool(tex.get()) == caps->isConfigRenderable(config, true));
145 }
146 }
147}
Robert Phillipsb7b7e5f2017-05-22 13:23:19 -0400148
Brian Salomond17b4a62017-05-23 16:53:47 -0400149#include "GrDrawingManager.h"
150#include "GrSurfaceProxy.h"
151#include "GrTextureContext.h"
152
153DEF_GPUTEST_FOR_RENDERING_CONTEXTS(InitialTextureClear, reporter, context_info) {
154 static constexpr int kSize = 100;
155 GrSurfaceDesc desc;
156 desc.fWidth = desc.fHeight = kSize;
157 std::unique_ptr<uint32_t[]> data(new uint32_t[kSize * kSize]);
158 GrContext* context = context_info.grContext();
159 for (int c = 0; c <= kLast_GrPixelConfig; ++c) {
160 desc.fConfig = static_cast<GrPixelConfig>(c);
161 if (!context_info.grContext()->caps()->isConfigTexturable(desc.fConfig)) {
162 continue;
163 }
164 desc.fFlags = kPerformInitialClear_GrSurfaceFlag;
165 for (bool rt : {false, true}) {
166 if (rt && !context->caps()->isConfigRenderable(desc.fConfig, false)) {
167 continue;
168 }
169 desc.fFlags |= rt ? kRenderTarget_GrSurfaceFlag : kNone_GrSurfaceFlags;
170 for (bool mipped : {false, true}) {
171 desc.fIsMipMapped = mipped;
172 for (GrSurfaceOrigin origin :
173 {kTopLeft_GrSurfaceOrigin, kBottomLeft_GrSurfaceOrigin}) {
174 desc.fOrigin = origin;
175 for (bool approx : {false, true}) {
176 auto resourceProvider = context->resourceProvider();
177 // Try directly creating the texture.
178 // Do this twice in an attempt to hit the cache on the second time through.
179 for (int i = 0; i < 2; ++i) {
180 sk_sp<GrTexture> tex;
181 if (approx) {
182 tex = sk_sp<GrTexture>(
183 resourceProvider->createApproxTexture(desc, 0));
184 } else {
185 tex = resourceProvider->createTexture(desc, SkBudgeted::kYes);
186 }
187 if (!tex) {
188 continue;
189 }
Robert Phillips066f0202017-07-25 10:16:35 -0400190 auto proxy = GrSurfaceProxy::MakeWrapped(std::move(tex), desc.fOrigin);
Brian Salomond17b4a62017-05-23 16:53:47 -0400191 auto texCtx = context->contextPriv().makeWrappedSurfaceContext(
192 std::move(proxy), nullptr);
193 SkImageInfo info = SkImageInfo::Make(
194 kSize, kSize, kRGBA_8888_SkColorType, kPremul_SkAlphaType);
195 memset(data.get(), 0xAB, kSize * kSize * sizeof(uint32_t));
196 if (texCtx->readPixels(info, data.get(), 0, 0, 0)) {
197 uint32_t cmp = GrPixelConfigIsOpaque(desc.fConfig) ? 0xFF000000 : 0;
198 for (int i = 0; i < kSize * kSize; ++i) {
199 if (cmp != data.get()[i]) {
200 ERRORF(reporter, "Failed on config %d", desc.fConfig);
201 break;
202 }
203 }
204 }
205 memset(data.get(), 0xBC, kSize * kSize * sizeof(uint32_t));
206 // Here we overwrite the texture so that the second time through we
207 // test against recycling without reclearing.
208 if (0 == i) {
209 texCtx->writePixels(info, data.get(), 0, 0, 0);
210 }
211 }
212 context->purgeAllUnlockedResources();
213
214 // Try creating the texture as a deferred proxy.
215 for (int i = 0; i < 2; ++i) {
216 auto surfCtx = context->contextPriv().makeDeferredSurfaceContext(
217 desc, approx ? SkBackingFit::kApprox : SkBackingFit::kExact,
218 SkBudgeted::kYes);
219 if (!surfCtx) {
220 continue;
221 }
222 SkImageInfo info = SkImageInfo::Make(
223 kSize, kSize, kRGBA_8888_SkColorType, kPremul_SkAlphaType);
224 memset(data.get(), 0xAB, kSize * kSize * sizeof(uint32_t));
225 if (surfCtx->readPixels(info, data.get(), 0, 0, 0)) {
226 uint32_t cmp = GrPixelConfigIsOpaque(desc.fConfig) ? 0xFF000000 : 0;
227 for (int i = 0; i < kSize * kSize; ++i) {
228 if (cmp != data.get()[i]) {
229 ERRORF(reporter, "Failed on config %d", desc.fConfig);
230 break;
231 }
232 }
233 }
234 // Here we overwrite the texture so that the second time through we
235 // test against recycling without reclearing.
236 if (0 == i) {
237 surfCtx->writePixels(info, data.get(), 0, 0, 0);
238 }
239 }
240 context->purgeAllUnlockedResources();
241 }
242 }
243 }
244 }
245 }
246}
bsalomon@google.com686bcb82013-04-09 15:04:12 +0000247#endif