• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2018 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. It relies on static intializers to work
9 
10 #include "include/core/SkTypes.h"
11 
12 #if defined(SK_VULKAN)
13 
14 #include "include/gpu/vk/GrVkVulkan.h"
15 
16 #include "tests/Test.h"
17 
18 #include "include/core/SkImage.h"
19 #include "include/gpu/GrBackendSurface.h"
20 #include "include/gpu/GrTexture.h"
21 #include "include/gpu/vk/GrVkTypes.h"
22 #include "src/gpu/GrContextPriv.h"
23 #include "src/gpu/GrRenderTargetContext.h"
24 #include "src/gpu/GrTextureProxy.h"
25 #include "src/gpu/SkGpuDevice.h"
26 #include "src/gpu/vk/GrVkGpu.h"
27 #include "src/gpu/vk/GrVkImageLayout.h"
28 #include "src/gpu/vk/GrVkTexture.h"
29 #include "src/image/SkImage_Base.h"
30 #include "src/image/SkSurface_Gpu.h"
31 
DEF_GPUTEST_FOR_VULKAN_CONTEXT(VkImageLayoutTest,reporter,ctxInfo)32 DEF_GPUTEST_FOR_VULKAN_CONTEXT(VkImageLayoutTest, reporter, ctxInfo) {
33     GrContext* context = ctxInfo.grContext();
34 
35     GrBackendTexture backendTex = context->createBackendTexture(1, 1,
36                                                                 kRGBA_8888_SkColorType,
37                                                                 SkColors::kTransparent,
38                                                                 GrMipMapped::kNo,
39                                                                 GrRenderable::kNo,
40                                                                 GrProtected::kNo);
41     REPORTER_ASSERT(reporter, backendTex.isValid());
42 
43     GrVkImageInfo info;
44     REPORTER_ASSERT(reporter, backendTex.getVkImageInfo(&info));
45     VkImageLayout initLayout = info.fImageLayout;
46 
47     // Verify that setting that layout via a copy of a backendTexture is reflected in all the
48     // backendTextures.
49     GrBackendTexture backendTexCopy = backendTex;
50     REPORTER_ASSERT(reporter, backendTexCopy.getVkImageInfo(&info));
51     REPORTER_ASSERT(reporter, initLayout == info.fImageLayout);
52 
53     backendTexCopy.setVkImageLayout(VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL);
54 
55     REPORTER_ASSERT(reporter, backendTex.getVkImageInfo(&info));
56     REPORTER_ASSERT(reporter, VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL == info.fImageLayout);
57 
58     REPORTER_ASSERT(reporter, backendTexCopy.getVkImageInfo(&info));
59     REPORTER_ASSERT(reporter, VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL == info.fImageLayout);
60 
61     // Setting back the layout since we didn't actually change it
62     backendTex.setVkImageLayout(initLayout);
63 
64     sk_sp<SkImage> wrappedImage = SkImage::MakeFromTexture(context, backendTex,
65                                                            kTopLeft_GrSurfaceOrigin,
66                                                            kRGBA_8888_SkColorType,
67                                                            kPremul_SkAlphaType, nullptr);
68     REPORTER_ASSERT(reporter, wrappedImage.get());
69 
70     sk_sp<GrTextureProxy> texProxy = as_IB(wrappedImage)->asTextureProxyRef(context);
71     REPORTER_ASSERT(reporter, texProxy.get());
72     REPORTER_ASSERT(reporter, texProxy->isInstantiated());
73     GrTexture* texture = texProxy->peekTexture();
74     REPORTER_ASSERT(reporter, texture);
75 
76     // Verify that modifying the layout via the GrVkTexture is reflected in the GrBackendTexture
77     GrVkTexture* vkTexture = static_cast<GrVkTexture*>(texture);
78     REPORTER_ASSERT(reporter, initLayout == vkTexture->currentLayout());
79     vkTexture->updateImageLayout(VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL);
80 
81     REPORTER_ASSERT(reporter, backendTex.getVkImageInfo(&info));
82     REPORTER_ASSERT(reporter, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL == info.fImageLayout);
83 
84     GrBackendTexture backendTexImage = wrappedImage->getBackendTexture(false);
85     REPORTER_ASSERT(reporter, backendTexImage.getVkImageInfo(&info));
86     REPORTER_ASSERT(reporter, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL == info.fImageLayout);
87 
88     // Verify that modifying the layout via the GrBackendTexutre is reflected in the GrVkTexture
89     backendTexImage.setVkImageLayout(VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL);
90     REPORTER_ASSERT(reporter, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL == vkTexture->currentLayout());
91 
92     vkTexture->updateImageLayout(initLayout);
93 
94     REPORTER_ASSERT(reporter, backendTex.getVkImageInfo(&info));
95     REPORTER_ASSERT(reporter, initLayout == info.fImageLayout);
96 
97     REPORTER_ASSERT(reporter, backendTexCopy.getVkImageInfo(&info));
98     REPORTER_ASSERT(reporter, initLayout == info.fImageLayout);
99 
100     REPORTER_ASSERT(reporter, backendTexImage.getVkImageInfo(&info));
101     REPORTER_ASSERT(reporter, initLayout == info.fImageLayout);
102 
103     // Check that we can do things like assigning the backend texture to invalid one, assign an
104     // invalid one, assin a backend texture to inself etc. Success here is that we don't hit any of
105     // our ref counting asserts.
106     REPORTER_ASSERT(reporter, GrBackendTexture::TestingOnly_Equals(backendTex, backendTexCopy));
107 
108     GrBackendTexture invalidTexture;
109     REPORTER_ASSERT(reporter, !invalidTexture.isValid());
110     REPORTER_ASSERT(reporter, !GrBackendTexture::TestingOnly_Equals(invalidTexture, backendTexCopy));
111 
112     backendTexCopy = invalidTexture;
113     REPORTER_ASSERT(reporter, !backendTexCopy.isValid());
114     REPORTER_ASSERT(reporter, !GrBackendTexture::TestingOnly_Equals(invalidTexture, backendTexCopy));
115 
116     invalidTexture = backendTex;
117     REPORTER_ASSERT(reporter, invalidTexture.isValid());
118     REPORTER_ASSERT(reporter, GrBackendTexture::TestingOnly_Equals(invalidTexture, backendTex));
119 
120     invalidTexture = static_cast<decltype(invalidTexture)&>(invalidTexture);
121     REPORTER_ASSERT(reporter, invalidTexture.isValid());
122     REPORTER_ASSERT(reporter, GrBackendTexture::TestingOnly_Equals(invalidTexture, invalidTexture));
123 
124     context->deleteBackendTexture(backendTex);
125 }
126 
testing_release_proc(void * ctx)127 static void testing_release_proc(void* ctx) {
128     int* count = (int*)ctx;
129     *count += 1;
130 }
131 
132 // Test to make sure we don't call our release proc on an image until we've transferred it back to
133 // its original queue family.
DEF_GPUTEST_FOR_VULKAN_CONTEXT(VkReleaseExternalQueueTest,reporter,ctxInfo)134 DEF_GPUTEST_FOR_VULKAN_CONTEXT(VkReleaseExternalQueueTest, reporter, ctxInfo) {
135     GrContext* context = ctxInfo.grContext();
136     GrGpu* gpu = context->priv().getGpu();
137     GrVkGpu* vkGpu = static_cast<GrVkGpu*>(gpu);
138     if (!vkGpu->vkCaps().supportsExternalMemory()) {
139         return;
140     }
141 
142     for (bool useExternal : {false, true}) {
143         GrBackendTexture backendTex = context->createBackendTexture(1, 1,
144                                                                     kRGBA_8888_SkColorType,
145                                                                     SkColors::kTransparent,
146                                                                     GrMipMapped::kNo,
147                                                                     GrRenderable::kNo,
148                                                                     GrProtected::kNo);
149         sk_sp<SkImage> image;
150         int count = 0;
151         if (useExternal) {
152             // Make a backend texture with an external queue family;
153             GrVkImageInfo vkInfo;
154             if (!backendTex.getVkImageInfo(&vkInfo)) {
155                 return;
156             }
157             vkInfo.fCurrentQueueFamily = VK_QUEUE_FAMILY_EXTERNAL;
158 
159             GrBackendTexture vkExtTex(1, 1, vkInfo);
160             REPORTER_ASSERT(reporter, vkExtTex.isValid());
161             image = SkImage::MakeFromTexture(context, vkExtTex,
162                                              kTopLeft_GrSurfaceOrigin,
163                                              kRGBA_8888_SkColorType,
164                                              kPremul_SkAlphaType,
165                                              nullptr, testing_release_proc,
166                                              (void*)&count);
167 
168         } else {
169             image = SkImage::MakeFromTexture(context, backendTex,
170                                              kTopLeft_GrSurfaceOrigin,
171                                              kRGBA_8888_SkColorType,
172                                              kPremul_SkAlphaType,
173                                              nullptr, testing_release_proc,
174                                              (void*)&count);
175         }
176 
177         if (!image) {
178             continue;
179         }
180 
181         REPORTER_ASSERT(reporter, !count);
182 
183         GrTexture* texture = image->getTexture();
184         REPORTER_ASSERT(reporter, texture);
185         GrVkTexture* vkTex = static_cast<GrVkTexture*>(texture);
186 
187         if (useExternal) {
188             // Testing helper so we claim that we don't need to transition from our fake external
189             // queue first.
190             vkTex->setCurrentQueueFamilyToGraphicsQueue(vkGpu);
191         }
192 
193         image.reset();
194 
195         // Resetting the image should only trigger the release proc if we are not using an external
196         // queue. When using an external queue when we free the SkImage and the underlying
197         // GrTexture, we submit a queue transition on the command buffer.
198         if (useExternal) {
199             REPORTER_ASSERT(reporter, !count);
200         } else {
201             REPORTER_ASSERT(reporter, count == 1);
202         }
203 
204         gpu->testingOnly_flushGpuAndSync();
205 
206         // Now that we flushed and waited the release proc should have be triggered.
207         REPORTER_ASSERT(reporter, count == 1);
208 
209         context->deleteBackendTexture(backendTex);
210     }
211 }
212 
213 // Test to make sure we transition to the original queue when requests for prepareforexternalio are
214 // in flush calls
DEF_GPUTEST_FOR_VULKAN_CONTEXT(VkPrepareForExternalIOQueueTransitionTest,reporter,ctxInfo)215 DEF_GPUTEST_FOR_VULKAN_CONTEXT(VkPrepareForExternalIOQueueTransitionTest, reporter, ctxInfo) {
216     GrContext* context = ctxInfo.grContext();
217 
218     GrVkGpu* vkGpu = static_cast<GrVkGpu*>(context->priv().getGpu());
219     if (!vkGpu->vkCaps().supportsExternalMemory()) {
220         return;
221     }
222 
223     for (bool useSurface : {false, true}) {
224         for (bool preparePresent : {false, true}) {
225             if (!useSurface && preparePresent) {
226                 // We don't set textures to present
227                 continue;
228             }
229             GrBackendTexture backendTex = context->createBackendTexture(
230                     4, 4, kRGBA_8888_SkColorType,
231                     SkColors::kTransparent, GrMipMapped::kNo,
232                     useSurface ? GrRenderable::kYes : GrRenderable::kNo,
233                     GrProtected::kNo);
234 
235             // Make a backend texture with an external queue family and general layout.
236             GrVkImageInfo vkInfo;
237             if (!backendTex.getVkImageInfo(&vkInfo)) {
238                 return;
239             }
240 
241             // We can't actually make an external texture in our test. However, we lie and say it is
242             // and then will manually go and swap the queue to the graphics queue once we wrap it.
243             if (preparePresent) {
244                 // We don't transition to present to things that are going to external for foreign
245                 // queues.
246                 vkInfo.fCurrentQueueFamily = vkGpu->queueIndex();
247             } else {
248                 vkInfo.fCurrentQueueFamily = VK_QUEUE_FAMILY_EXTERNAL;
249             }
250 
251             GrBackendTexture vkExtTex(1, 1, vkInfo);
252 
253             sk_sp<SkImage> image;
254             sk_sp<SkSurface> surface;
255             GrTexture* texture;
256             if (useSurface) {
257                 surface = SkSurface::MakeFromBackendTexture(context, vkExtTex,
258                         kTopLeft_GrSurfaceOrigin, 0, kRGBA_8888_SkColorType, nullptr, nullptr);
259                 REPORTER_ASSERT(reporter, surface.get());
260                 if (!surface) {
261                     continue;
262                 }
263                 SkSurface_Gpu* gpuSurface = static_cast<SkSurface_Gpu*>(surface.get());
264                 auto* rtc = gpuSurface->getDevice()->accessRenderTargetContext();
265                 texture = rtc->asTextureProxy()->peekTexture();
266             } else {
267                 image = SkImage::MakeFromTexture(context, vkExtTex, kTopLeft_GrSurfaceOrigin,
268                         kRGBA_8888_SkColorType, kPremul_SkAlphaType, nullptr, nullptr, nullptr);
269 
270                 REPORTER_ASSERT(reporter, image.get());
271                 if (!image) {
272                     continue;
273                 }
274 
275                 texture = image->getTexture();
276             }
277 
278             REPORTER_ASSERT(reporter, texture);
279             GrVkTexture* vkTex = static_cast<GrVkTexture*>(texture);
280 
281             // Testing helper so we claim that we don't need to transition from our fake external
282             // queue first.
283             vkTex->setCurrentQueueFamilyToGraphicsQueue(vkGpu);
284 
285             GrBackendTexture newBackendTexture;
286             if (useSurface) {
287                 newBackendTexture = surface->getBackendTexture(
288                         SkSurface::kFlushRead_TextureHandleAccess);
289             } else {
290                 newBackendTexture = image->getBackendTexture(false);
291             }
292             GrVkImageInfo newVkInfo;
293             REPORTER_ASSERT(reporter, newBackendTexture.getVkImageInfo(&newVkInfo));
294             REPORTER_ASSERT(reporter, newVkInfo.fCurrentQueueFamily == vkGpu->queueIndex());
295             VkImageLayout oldLayout = newVkInfo.fImageLayout;
296 
297             GrPrepareForExternalIORequests externalRequests;
298             SkImage* imagePtr;
299             SkSurface* surfacePtr;
300             if (useSurface) {
301                 externalRequests.fNumSurfaces = 1;
302                 surfacePtr = surface.get();
303                 externalRequests.fSurfaces = &surfacePtr;
304                 externalRequests.fPrepareSurfaceForPresent = &preparePresent;
305             } else {
306                 externalRequests.fNumImages = 1;
307                 imagePtr = image.get();
308                 externalRequests.fImages = &imagePtr;
309 
310             }
311             context->flush(GrFlushInfo(), externalRequests);
312 
313             if (useSurface) {
314                 newBackendTexture = surface->getBackendTexture(
315                         SkSurface::kFlushRead_TextureHandleAccess);
316             } else {
317                 newBackendTexture = image->getBackendTexture(false);
318             }
319             REPORTER_ASSERT(reporter, newBackendTexture.getVkImageInfo(&newVkInfo));
320             if (preparePresent) {
321                 REPORTER_ASSERT(reporter, newVkInfo.fCurrentQueueFamily == vkGpu->queueIndex());
322                 REPORTER_ASSERT(reporter,
323                                 newVkInfo.fImageLayout == VK_IMAGE_LAYOUT_PRESENT_SRC_KHR);
324             } else {
325                 REPORTER_ASSERT(reporter, newVkInfo.fCurrentQueueFamily == VK_QUEUE_FAMILY_EXTERNAL);
326                 REPORTER_ASSERT(reporter, newVkInfo.fImageLayout == oldLayout);
327             }
328 
329             GrFlushInfo flushInfo;
330             flushInfo.fFlags = kSyncCpu_GrFlushFlag;
331             context->flush(flushInfo);
332             context->deleteBackendTexture(backendTex);
333         }
334     }
335 }
336 
337 // This test is disabled because it executes illegal vulkan calls which cause the validations layers
338 // to fail and makes us assert. Once fixed to use a valid vulkan call sequence it should be
339 // renenabled, see skbug.com/8936.
340 #if 0
341 // Test to make sure we transition from the EXTERNAL queue even when no layout transition is needed.
342 DEF_GPUTEST_FOR_VULKAN_CONTEXT(VkTransitionExternalQueueTest, reporter, ctxInfo) {
343     GrContext* context = ctxInfo.grContext();
344     GrGpu* gpu = context->priv().getGpu();
345     GrVkGpu* vkGpu = static_cast<GrVkGpu*>(gpu);
346     if (!vkGpu->vkCaps().supportsExternalMemory()) {
347         return;
348     }
349 
350     GrBackendTexture backendTex = context->createBackendTexture(
351             1, 1, kRGBA_8888_SkColorType,
352             SkColors::kTransparent, GrMipMapped::kNo, GrRenderable::kNo);
353     sk_sp<SkImage> image;
354     // Make a backend texture with an external queue family and general layout.
355     GrVkImageInfo vkInfo;
356     if (!backendTex.getVkImageInfo(&vkInfo)) {
357         return;
358     }
359     vkInfo.fCurrentQueueFamily = VK_QUEUE_FAMILY_EXTERNAL;
360     // Use a read-only layout as these are the ones where we can otherwise skip a transition.
361     vkInfo.fImageLayout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL;
362 
363     GrBackendTexture vkExtTex(1, 1, vkInfo);
364     REPORTER_ASSERT(reporter, vkExtTex.isValid());
365     image = SkImage::MakeFromTexture(context, vkExtTex, kTopLeft_GrSurfaceOrigin,
366                                      kRGBA_8888_SkColorType, kPremul_SkAlphaType, nullptr, nullptr,
367                                      nullptr);
368 
369     if (!image) {
370         return;
371     }
372 
373     GrTexture* texture = image->getTexture();
374     REPORTER_ASSERT(reporter, texture);
375     GrVkTexture* vkTex = static_cast<GrVkTexture*>(texture);
376 
377     // Change our backend texture to the internal queue, with the same layout. This should force a
378     // queue transition even though the layouts match.
379     vkTex->setImageLayout(vkGpu, VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL, 0,
380                           VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT, false, false);
381 
382     // Get our image info again and make sure we transitioned queues.
383     GrBackendTexture newBackendTexture = image->getBackendTexture(true);
384     GrVkImageInfo newVkInfo;
385     REPORTER_ASSERT(reporter, newBackendTexture.getVkImageInfo(&newVkInfo));
386     REPORTER_ASSERT(reporter, newVkInfo.fCurrentQueueFamily == vkGpu->queueIndex());
387 
388     image.reset();
389     gpu->testingOnly_flushGpuAndSync();
390     context->deleteBackendTexture(backendTex);
391 }
392 #endif
393 
394 #endif
395