1 /*-------------------------------------------------------------------------
2 * Vulkan Conformance Tests
3 * ------------------------
4 *
5 * Copyright (c) 2017 Google Inc.
6 *
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 *
19 *//*!
20 * \file
21 * \brief Tests reading of samples from a previous subpass.
22 *//*--------------------------------------------------------------------*/
23
24 #include "vktRenderPassSampleReadTests.hpp"
25 #include "vktRenderPassTestsUtil.hpp"
26
27 #include "vktTestCaseUtil.hpp"
28 #include "vktTestGroupUtil.hpp"
29
30 #include "vkDefs.hpp"
31 #include "vkImageUtil.hpp"
32 #include "vkMemUtil.hpp"
33 #include "vkPrograms.hpp"
34 #include "vkQueryUtil.hpp"
35 #include "vkRef.hpp"
36 #include "vkRefUtil.hpp"
37 #include "vkTypeUtil.hpp"
38 #include "vkCmdUtil.hpp"
39 #include "vkObjUtil.hpp"
40
41 #include "tcuImageCompare.hpp"
42 #include "tcuResultCollector.hpp"
43
44 #include "deUniquePtr.hpp"
45
46 using namespace vk;
47
48 using tcu::UVec4;
49 using tcu::Vec4;
50
51 using tcu::ConstPixelBufferAccess;
52 using tcu::PixelBufferAccess;
53
54 using tcu::TestLog;
55
56 using std::string;
57 using std::vector;
58
59 namespace vkt
60 {
61 namespace
62 {
63 using namespace renderpass;
64
createBufferMemory(const DeviceInterface & vk,VkDevice device,Allocator & allocator,VkBuffer buffer)65 de::MovePtr<Allocation> createBufferMemory (const DeviceInterface& vk,
66 VkDevice device,
67 Allocator& allocator,
68 VkBuffer buffer)
69 {
70 de::MovePtr<Allocation> allocation (allocator.allocate(getBufferMemoryRequirements(vk, device, buffer), MemoryRequirement::HostVisible));
71 VK_CHECK(vk.bindBufferMemory(device, buffer, allocation->getMemory(), allocation->getOffset()));
72 return allocation;
73 }
74
createImageMemory(const DeviceInterface & vk,VkDevice device,Allocator & allocator,VkImage image)75 de::MovePtr<Allocation> createImageMemory (const DeviceInterface& vk,
76 VkDevice device,
77 Allocator& allocator,
78 VkImage image)
79 {
80 de::MovePtr<Allocation> allocation (allocator.allocate(getImageMemoryRequirements(vk, device, image), MemoryRequirement::Any));
81 VK_CHECK(vk.bindImageMemory(device, image, allocation->getMemory(), allocation->getOffset()));
82 return allocation;
83 }
84
createImage(const DeviceInterface & vk,VkDevice device,VkImageCreateFlags flags,VkImageType imageType,VkFormat format,VkExtent3D extent,deUint32 mipLevels,deUint32 arrayLayers,VkSampleCountFlagBits samples,VkImageTiling tiling,VkImageUsageFlags usage,VkSharingMode sharingMode,deUint32 queueFamilyCount,const deUint32 * pQueueFamilyIndices,VkImageLayout initialLayout)85 Move<VkImage> createImage (const DeviceInterface& vk,
86 VkDevice device,
87 VkImageCreateFlags flags,
88 VkImageType imageType,
89 VkFormat format,
90 VkExtent3D extent,
91 deUint32 mipLevels,
92 deUint32 arrayLayers,
93 VkSampleCountFlagBits samples,
94 VkImageTiling tiling,
95 VkImageUsageFlags usage,
96 VkSharingMode sharingMode,
97 deUint32 queueFamilyCount,
98 const deUint32* pQueueFamilyIndices,
99 VkImageLayout initialLayout)
100 {
101 const VkImageCreateInfo createInfo =
102 {
103 VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
104 DE_NULL,
105 flags,
106 imageType,
107 format,
108 extent,
109 mipLevels,
110 arrayLayers,
111 samples,
112 tiling,
113 usage,
114 sharingMode,
115 queueFamilyCount,
116 pQueueFamilyIndices,
117 initialLayout
118 };
119 return createImage(vk, device, &createInfo);
120 }
121
createImageView(const DeviceInterface & vk,VkDevice device,VkImageViewCreateFlags flags,VkImage image,VkImageViewType viewType,VkFormat format,VkComponentMapping components,VkImageSubresourceRange subresourceRange)122 Move<VkImageView> createImageView (const DeviceInterface& vk,
123 VkDevice device,
124 VkImageViewCreateFlags flags,
125 VkImage image,
126 VkImageViewType viewType,
127 VkFormat format,
128 VkComponentMapping components,
129 VkImageSubresourceRange subresourceRange)
130 {
131 const VkImageViewCreateInfo pCreateInfo =
132 {
133 VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
134 DE_NULL,
135 flags,
136 image,
137 viewType,
138 format,
139 components,
140 subresourceRange,
141 };
142 return createImageView(vk, device, &pCreateInfo);
143 }
144
createImage(const InstanceInterface & vki,VkPhysicalDevice physicalDevice,const DeviceInterface & vkd,VkDevice device,VkFormat vkFormat,VkSampleCountFlagBits sampleCountBit,VkImageUsageFlags usage,deUint32 width,deUint32 height)145 Move<VkImage> createImage (const InstanceInterface& vki,
146 VkPhysicalDevice physicalDevice,
147 const DeviceInterface& vkd,
148 VkDevice device,
149 VkFormat vkFormat,
150 VkSampleCountFlagBits sampleCountBit,
151 VkImageUsageFlags usage,
152 deUint32 width,
153 deUint32 height)
154 {
155 try
156 {
157 const VkImageType imageType (VK_IMAGE_TYPE_2D);
158 const VkImageTiling imageTiling (VK_IMAGE_TILING_OPTIMAL);
159 const VkImageFormatProperties imageFormatProperties (getPhysicalDeviceImageFormatProperties(vki, physicalDevice, vkFormat, imageType, imageTiling, usage, 0u));
160 const VkExtent3D imageExtent =
161 {
162 width,
163 height,
164 1u
165 };
166
167 if (imageFormatProperties.maxExtent.width < imageExtent.width
168 || imageFormatProperties.maxExtent.height < imageExtent.height
169 || ((imageFormatProperties.sampleCounts & sampleCountBit) == 0))
170 {
171 TCU_THROW(NotSupportedError, "Image type not supported");
172 }
173
174 return createImage(vkd, device, 0u, imageType, vkFormat, imageExtent, 1u, 1u, sampleCountBit, imageTiling, usage, VK_SHARING_MODE_EXCLUSIVE, 0u, DE_NULL, VK_IMAGE_LAYOUT_UNDEFINED);
175 }
176 catch (const vk::Error& error)
177 {
178 if (error.getError() == VK_ERROR_FORMAT_NOT_SUPPORTED)
179 TCU_THROW(NotSupportedError, "Image format not supported");
180
181 throw;
182 }
183 }
184
createImageView(const DeviceInterface & vkd,VkDevice device,VkImage image,VkFormat format,VkImageAspectFlags aspect)185 Move<VkImageView> createImageView (const DeviceInterface& vkd,
186 VkDevice device,
187 VkImage image,
188 VkFormat format,
189 VkImageAspectFlags aspect)
190 {
191 const VkImageSubresourceRange range =
192 {
193 aspect,
194 0u,
195 1u,
196 0u,
197 1u
198 };
199
200 return createImageView(vkd, device, 0u, image, VK_IMAGE_VIEW_TYPE_2D, format, makeComponentMappingRGBA(), range);
201 }
202
getPixelSize(VkFormat vkFormat)203 VkDeviceSize getPixelSize (VkFormat vkFormat)
204 {
205 const tcu::TextureFormat format (mapVkFormat(vkFormat));
206
207 return format.getPixelSize();
208 }
209
createBuffer(const DeviceInterface & vkd,VkDevice device,VkFormat format,deUint32 width,deUint32 height)210 Move<VkBuffer> createBuffer (const DeviceInterface& vkd,
211 VkDevice device,
212 VkFormat format,
213 deUint32 width,
214 deUint32 height)
215 {
216 const VkBufferUsageFlags bufferUsage (VK_BUFFER_USAGE_TRANSFER_SRC_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT);
217 const VkDeviceSize pixelSize (getPixelSize(format));
218 const VkBufferCreateInfo createInfo =
219 {
220 VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO,
221 DE_NULL,
222 0u,
223
224 width * height * pixelSize,
225 bufferUsage,
226
227 VK_SHARING_MODE_EXCLUSIVE,
228 0u,
229 DE_NULL
230 };
231 return createBuffer(vkd, device, &createInfo);
232 }
233
sampleCountBitFromSampleCount(deUint32 count)234 VkSampleCountFlagBits sampleCountBitFromSampleCount (deUint32 count)
235 {
236 switch (count)
237 {
238 case 1: return VK_SAMPLE_COUNT_1_BIT;
239 case 2: return VK_SAMPLE_COUNT_2_BIT;
240 case 4: return VK_SAMPLE_COUNT_4_BIT;
241 case 8: return VK_SAMPLE_COUNT_8_BIT;
242 case 16: return VK_SAMPLE_COUNT_16_BIT;
243 case 32: return VK_SAMPLE_COUNT_32_BIT;
244 case 64: return VK_SAMPLE_COUNT_64_BIT;
245
246 default:
247 DE_FATAL("Invalid sample count");
248 return (VkSampleCountFlagBits)(0x1u << count);
249 }
250 }
251
252 template<typename AttachmentDesc, typename AttachmentRef, typename SubpassDesc, typename SubpassDep, typename RenderPassCreateInfo>
createRenderPass(const DeviceInterface & vkd,VkDevice device,VkFormat srcFormat,VkFormat dstFormat,deUint32 sampleCount,RenderPassType renderPassType)253 Move<VkRenderPass> createRenderPass (const DeviceInterface& vkd,
254 VkDevice device,
255 VkFormat srcFormat,
256 VkFormat dstFormat,
257 deUint32 sampleCount,
258 RenderPassType renderPassType)
259 {
260 const VkSampleCountFlagBits samples (sampleCountBitFromSampleCount(sampleCount));
261 const VkImageAspectFlagBits aspectFlag ((renderPassType == RENDERPASS_TYPE_RENDERPASS2) ? VK_IMAGE_ASPECT_COLOR_BIT :
262 static_cast<VkImageAspectFlagBits>(0u));
263 const AttachmentRef srcAttachmentRef // VkAttachmentReference || VkAttachmentReference2KHR
264 (
265 // || VkStructureType sType;
266 DE_NULL, // || const void* pNext;
267 0u, // deUint32 attachment; || deUint32 attachment;
268 VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, // VkImageLayout layout; || VkImageLayout layout;
269 0u // || VkImageAspectFlags aspectMask;
270 );
271 const AttachmentRef srcAttachmentInputRef // VkAttachmentReference || VkAttachmentReference2KHR
272 (
273 // || VkStructureType sType;
274 DE_NULL, // || const void* pNext;
275 0u, // deUint32 attachment; || deUint32 attachment;
276 VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL, // VkImageLayout layout; || VkImageLayout layout;
277 aspectFlag // || VkImageAspectFlags aspectMask;
278 );
279 const AttachmentRef dstAttachmentRef // VkAttachmentReference || VkAttachmentReference2KHR
280 (
281 // || VkStructureType sType;
282 DE_NULL, // || const void* pNext;
283 1u, // deUint32 attachment; || deUint32 attachment;
284 VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, // VkImageLayout layout; || VkImageLayout layout;
285 0u // || VkImageAspectFlags aspectMask;
286 );
287 const AttachmentRef dstResolveAttachmentRef // VkAttachmentReference || VkAttachmentReference2KHR
288 (
289 // || VkStructureType sType;
290 DE_NULL, // || const void* pNext;
291 2u, // deUint32 attachment; || deUint32 attachment;
292 VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, // VkImageLayout layout; || VkImageLayout layout;
293 0u // || VkImageAspectFlags aspectMask;
294 );
295 const SubpassDep dependency // VkSubpassDependency || VkSubpassDependency2KHR
296 (
297 // || VkStructureType sType;
298 DE_NULL, // || const void* pNext;
299 0u, // deUint32 srcSubpass; || deUint32 srcSubpass;
300 1u, // deUint32 dstSubpass; || deUint32 dstSubpass;
301 VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT, // VkPipelineStageFlags srcStageMask; || VkPipelineStageFlags srcStageMask;
302 VK_PIPELINE_STAGE_FRAGMENT_SHADER_BIT, // VkPipelineStageFlags dstStageMask; || VkPipelineStageFlags dstStageMask;
303 VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT, // VkAccessFlags srcAccessMask; || VkAccessFlags srcAccessMask;
304 VK_ACCESS_INPUT_ATTACHMENT_READ_BIT, // VkAccessFlags dstAccessMask; || VkAccessFlags dstAccessMask;
305 VK_DEPENDENCY_BY_REGION_BIT, // VkDependencyFlags dependencyFlags; || VkDependencyFlags dependencyFlags;
306 0u // || deInt32 viewOffset;
307 );
308 const AttachmentDesc srcAttachment // VkAttachmentDescription || VkAttachmentDescription2KHR
309 (
310 // || VkStructureType sType;
311 DE_NULL, // || const void* pNext;
312 0u, // VkAttachmentDescriptionFlags flags; || VkAttachmentDescriptionFlags flags;
313 srcFormat, // VkFormat format; || VkFormat format;
314 samples, // VkSampleCountFlagBits samples; || VkSampleCountFlagBits samples;
315 VK_ATTACHMENT_LOAD_OP_DONT_CARE, // VkAttachmentLoadOp loadOp; || VkAttachmentLoadOp loadOp;
316 VK_ATTACHMENT_STORE_OP_DONT_CARE, // VkAttachmentStoreOp storeOp; || VkAttachmentStoreOp storeOp;
317 VK_ATTACHMENT_LOAD_OP_DONT_CARE, // VkAttachmentLoadOp stencilLoadOp; || VkAttachmentLoadOp stencilLoadOp;
318 VK_ATTACHMENT_STORE_OP_DONT_CARE, // VkAttachmentStoreOp stencilStoreOp; || VkAttachmentStoreOp stencilStoreOp;
319 VK_IMAGE_LAYOUT_UNDEFINED, // VkImageLayout initialLayout; || VkImageLayout initialLayout;
320 VK_IMAGE_LAYOUT_GENERAL // VkImageLayout finalLayout; || VkImageLayout finalLayout;
321 );
322 const AttachmentDesc dstMultisampleAttachment // VkAttachmentDescription || VkAttachmentDescription2KHR
323 (
324 // || VkStructureType sType;
325 DE_NULL, // || const void* pNext;
326 0u, // VkAttachmentDescriptionFlags flags; || VkAttachmentDescriptionFlags flags;
327 dstFormat, // VkFormat format; || VkFormat format;
328 samples, // VkSampleCountFlagBits samples; || VkSampleCountFlagBits samples;
329 VK_ATTACHMENT_LOAD_OP_DONT_CARE, // VkAttachmentLoadOp loadOp; || VkAttachmentLoadOp loadOp;
330 VK_ATTACHMENT_STORE_OP_DONT_CARE, // VkAttachmentStoreOp storeOp; || VkAttachmentStoreOp storeOp;
331 VK_ATTACHMENT_LOAD_OP_DONT_CARE, // VkAttachmentLoadOp stencilLoadOp; || VkAttachmentLoadOp stencilLoadOp;
332 VK_ATTACHMENT_STORE_OP_DONT_CARE, // VkAttachmentStoreOp stencilStoreOp; || VkAttachmentStoreOp stencilStoreOp;
333 VK_IMAGE_LAYOUT_UNDEFINED, // VkImageLayout initialLayout; || VkImageLayout initialLayout;
334 VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL // VkImageLayout finalLayout; || VkImageLayout finalLayout;
335 );
336 const AttachmentDesc dstResolveAttachment // VkAttachmentDescription || VkAttachmentDescription2KHR
337 (
338 // || VkStructureType sType;
339 DE_NULL, // || const void* pNext;
340 0u, // VkAttachmentDescriptionFlags flags; || VkAttachmentDescriptionFlags flags;
341 dstFormat, // VkFormat format; || VkFormat format;
342 VK_SAMPLE_COUNT_1_BIT, // VkSampleCountFlagBits samples; || VkSampleCountFlagBits samples;
343 VK_ATTACHMENT_LOAD_OP_DONT_CARE, // VkAttachmentLoadOp loadOp; || VkAttachmentLoadOp loadOp;
344 VK_ATTACHMENT_STORE_OP_STORE, // VkAttachmentStoreOp storeOp; || VkAttachmentStoreOp storeOp;
345 VK_ATTACHMENT_LOAD_OP_DONT_CARE, // VkAttachmentLoadOp stencilLoadOp; || VkAttachmentLoadOp stencilLoadOp;
346 VK_ATTACHMENT_STORE_OP_STORE, // VkAttachmentStoreOp stencilStoreOp; || VkAttachmentStoreOp stencilStoreOp;
347 VK_IMAGE_LAYOUT_UNDEFINED, // VkImageLayout initialLayout; || VkImageLayout initialLayout;
348 VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL // VkImageLayout finalLayout; || VkImageLayout finalLayout;
349 );
350 const AttachmentDesc attachments[] =
351 {
352 srcAttachment,
353 dstMultisampleAttachment,
354 dstResolveAttachment
355 };
356 const SubpassDesc subpass1 // VkSubpassDescription || VkSubpassDescription2KHR
357 (
358 // || VkStructureType sType;
359 DE_NULL, // || const void* pNext;
360 (VkSubpassDescriptionFlags)0, // VkSubpassDescriptionFlags flags; || VkSubpassDescriptionFlags flags;
361 VK_PIPELINE_BIND_POINT_GRAPHICS, // VkPipelineBindPoint pipelineBindPoint; || VkPipelineBindPoint pipelineBindPoint;
362 0u, // || deUint32 viewMask;
363 0u, // deUint32 inputAttachmentCount; || deUint32 inputAttachmentCount;
364 DE_NULL, // const VkAttachmentReference* pInputAttachments; || const VkAttachmentReference2KHR* pInputAttachments;
365 1u, // deUint32 colorAttachmentCount; || deUint32 colorAttachmentCount;
366 &srcAttachmentRef, // const VkAttachmentReference* pColorAttachments; || const VkAttachmentReference2KHR* pColorAttachments;
367 DE_NULL, // const VkAttachmentReference* pResolveAttachments; || const VkAttachmentReference2KHR* pResolveAttachments;
368 DE_NULL, // const VkAttachmentReference* pDepthStencilAttachment; || const VkAttachmentReference2KHR* pDepthStencilAttachment;
369 0u, // deUint32 preserveAttachmentCount; || deUint32 preserveAttachmentCount;
370 DE_NULL // const deUint32* pPreserveAttachments; || const deUint32* pPreserveAttachments;
371 );
372 const SubpassDesc subpass2 // VkSubpassDescription || VkSubpassDescription2KHR
373 (
374 // || VkStructureType sType;
375 DE_NULL, // || const void* pNext;
376 (VkSubpassDescriptionFlags)0, // VkSubpassDescriptionFlags flags; || VkSubpassDescriptionFlags flags;
377 VK_PIPELINE_BIND_POINT_GRAPHICS, // VkPipelineBindPoint pipelineBindPoint; || VkPipelineBindPoint pipelineBindPoint;
378 0u, // || deUint32 viewMask;
379 1u, // deUint32 inputAttachmentCount; || deUint32 inputAttachmentCount;
380 &srcAttachmentInputRef, // const VkAttachmentReference* pInputAttachments; || const VkAttachmentReference2KHR* pInputAttachments;
381 1u, // deUint32 colorAttachmentCount; || deUint32 colorAttachmentCount;
382 &dstAttachmentRef, // const VkAttachmentReference* pColorAttachments; || const VkAttachmentReference2KHR* pColorAttachments;
383 &dstResolveAttachmentRef, // const VkAttachmentReference* pResolveAttachments; || const VkAttachmentReference2KHR* pResolveAttachments;
384 DE_NULL, // const VkAttachmentReference* pDepthStencilAttachment; || const VkAttachmentReference2KHR* pDepthStencilAttachment;
385 0u, // deUint32 preserveAttachmentCount; || deUint32 preserveAttachmentCount;
386 DE_NULL // const deUint32* pPreserveAttachments; || const deUint32* pPreserveAttachments;
387 );
388 const SubpassDesc subpasses[] =
389 {
390 subpass1,
391 subpass2
392 };
393 const RenderPassCreateInfo renderPassCreator // VkRenderPassCreateInfo || VkRenderPassCreateInfo2KHR
394 (
395 // VkStructureType sType; || VkStructureType sType;
396 DE_NULL, // const void* pNext; || const void* pNext;
397 (VkRenderPassCreateFlags)0u, // VkRenderPassCreateFlags flags; || VkRenderPassCreateFlags flags;
398 3u, // deUint32 attachmentCount; || deUint32 attachmentCount;
399 attachments, // const VkAttachmentDescription* pAttachments; || const VkAttachmentDescription2KHR* pAttachments;
400 2u, // deUint32 subpassCount; || deUint32 subpassCount;
401 subpasses, // const VkSubpassDescription* pSubpasses; || const VkSubpassDescription2KHR* pSubpasses;
402 1u, // deUint32 dependencyCount; || deUint32 dependencyCount;
403 &dependency, // const VkSubpassDependency* pDependencies; || const VkSubpassDependency2KHR* pDependencies;
404 0u, // || deUint32 correlatedViewMaskCount;
405 DE_NULL // || const deUint32* pCorrelatedViewMasks;
406 );
407
408 return renderPassCreator.createRenderPass(vkd, device);
409 }
410
createRenderPass(const DeviceInterface & vkd,VkDevice device,VkFormat srcFormat,VkFormat dstFormat,deUint32 sampleCount,RenderPassType renderPassType)411 Move<VkRenderPass> createRenderPass (const DeviceInterface& vkd,
412 VkDevice device,
413 VkFormat srcFormat,
414 VkFormat dstFormat,
415 deUint32 sampleCount,
416 RenderPassType renderPassType)
417 {
418 switch (renderPassType)
419 {
420 case RENDERPASS_TYPE_LEGACY:
421 return createRenderPass<AttachmentDescription1, AttachmentReference1, SubpassDescription1, SubpassDependency1, RenderPassCreateInfo1>(vkd, device, srcFormat, dstFormat, sampleCount, renderPassType);
422 case RENDERPASS_TYPE_RENDERPASS2:
423 return createRenderPass<AttachmentDescription2, AttachmentReference2, SubpassDescription2, SubpassDependency2, RenderPassCreateInfo2>(vkd, device, srcFormat, dstFormat, sampleCount, renderPassType);
424 default:
425 TCU_THROW(InternalError, "Impossible");
426 }
427 }
428
createFramebuffer(const DeviceInterface & vkd,VkDevice device,VkRenderPass renderPass,VkImageView srcImageView,VkImageView dstMultisampleImageView,VkImageView dstSinglesampleImageView,deUint32 width,deUint32 height)429 Move<VkFramebuffer> createFramebuffer (const DeviceInterface& vkd,
430 VkDevice device,
431 VkRenderPass renderPass,
432 VkImageView srcImageView,
433 VkImageView dstMultisampleImageView,
434 VkImageView dstSinglesampleImageView,
435 deUint32 width,
436 deUint32 height)
437 {
438 VkImageView attachments[] =
439 {
440 srcImageView,
441 dstMultisampleImageView,
442 dstSinglesampleImageView
443 };
444
445 const VkFramebufferCreateInfo createInfo =
446 {
447 VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO,
448 DE_NULL,
449 0u,
450
451 renderPass,
452 3u,
453 attachments,
454
455 width,
456 height,
457 1u
458 };
459
460 return createFramebuffer(vkd, device, &createInfo);
461 }
462
createRenderPipelineLayout(const DeviceInterface & vkd,VkDevice device)463 Move<VkPipelineLayout> createRenderPipelineLayout (const DeviceInterface& vkd,
464 VkDevice device)
465 {
466 const VkPipelineLayoutCreateInfo createInfo =
467 {
468 VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO,
469 DE_NULL,
470 (vk::VkPipelineLayoutCreateFlags)0,
471
472 0u,
473 DE_NULL,
474
475 0u,
476 DE_NULL
477 };
478
479 return createPipelineLayout(vkd, device, &createInfo);
480 }
481
createRenderPipeline(const DeviceInterface & vkd,VkDevice device,VkRenderPass renderPass,VkPipelineLayout pipelineLayout,const vk::BinaryCollection & binaryCollection,deUint32 width,deUint32 height,deUint32 sampleCount)482 Move<VkPipeline> createRenderPipeline (const DeviceInterface& vkd,
483 VkDevice device,
484 VkRenderPass renderPass,
485 VkPipelineLayout pipelineLayout,
486 const vk::BinaryCollection& binaryCollection,
487 deUint32 width,
488 deUint32 height,
489 deUint32 sampleCount)
490 {
491 const Unique<VkShaderModule> vertexShaderModule (createShaderModule(vkd, device, binaryCollection.get("quad-vert"), 0u));
492 const Unique<VkShaderModule> fragmentShaderModule (createShaderModule(vkd, device, binaryCollection.get("quad-frag"), 0u));
493 const VkPipelineVertexInputStateCreateInfo vertexInputState =
494 {
495 VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO,
496 DE_NULL,
497 (VkPipelineVertexInputStateCreateFlags)0u,
498
499 0u,
500 DE_NULL,
501
502 0u,
503 DE_NULL
504 };
505 const std::vector<VkViewport> viewports (1, makeViewport(tcu::UVec2(width, height)));
506 const std::vector<VkRect2D> scissors (1, makeRect2D(tcu::UVec2(width, height)));
507
508 const VkPipelineMultisampleStateCreateInfo multisampleState =
509 {
510 VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO,
511 DE_NULL,
512 (VkPipelineMultisampleStateCreateFlags)0u,
513
514 sampleCountBitFromSampleCount(sampleCount),
515 VK_TRUE,
516 1.0f,
517 DE_NULL,
518 VK_FALSE,
519 VK_FALSE,
520 };
521
522 return makeGraphicsPipeline(vkd, // const DeviceInterface& vk
523 device, // const VkDevice device
524 pipelineLayout, // const VkPipelineLayout pipelineLayout
525 *vertexShaderModule, // const VkShaderModule vertexShaderModule
526 DE_NULL, // const VkShaderModule tessellationControlShaderModule
527 DE_NULL, // const VkShaderModule tessellationEvalShaderModule
528 DE_NULL, // const VkShaderModule geometryShaderModule
529 *fragmentShaderModule, // const VkShaderModule fragmentShaderModule
530 renderPass, // const VkRenderPass renderPass
531 viewports, // const std::vector<VkViewport>& viewports
532 scissors, // const std::vector<VkRect2D>& scissors
533 VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST, // const VkPrimitiveTopology topology
534 0u, // const deUint32 subpass
535 0u, // const deUint32 patchControlPoints
536 &vertexInputState, // const VkPipelineVertexInputStateCreateInfo* vertexInputStateCreateInfo
537 DE_NULL, // const VkPipelineRasterizationStateCreateInfo* rasterizationStateCreateInfo
538 &multisampleState); // const VkPipelineMultisampleStateCreateInfo* multisampleStateCreateInfo
539 }
540
createSubpassDescriptorSetLayout(const DeviceInterface & vkd,VkDevice device)541 Move<VkDescriptorSetLayout> createSubpassDescriptorSetLayout (const DeviceInterface& vkd,
542 VkDevice device)
543 {
544 const VkDescriptorSetLayoutBinding bindings[] =
545 {
546 {
547 0u,
548 VK_DESCRIPTOR_TYPE_INPUT_ATTACHMENT,
549 1u,
550 VK_SHADER_STAGE_FRAGMENT_BIT,
551 DE_NULL
552 },
553 {
554 1u,
555 VK_DESCRIPTOR_TYPE_INPUT_ATTACHMENT,
556 1u,
557 VK_SHADER_STAGE_FRAGMENT_BIT,
558 DE_NULL
559 }
560 };
561 const VkDescriptorSetLayoutCreateInfo createInfo =
562 {
563 VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
564 DE_NULL,
565 0u,
566
567 1u,
568 bindings
569 };
570
571 return createDescriptorSetLayout(vkd, device, &createInfo);
572 }
573
createSubpassPipelineLayout(const DeviceInterface & vkd,VkDevice device,VkDescriptorSetLayout descriptorSetLayout)574 Move<VkPipelineLayout> createSubpassPipelineLayout (const DeviceInterface& vkd,
575 VkDevice device,
576 VkDescriptorSetLayout descriptorSetLayout)
577 {
578 const VkPipelineLayoutCreateInfo createInfo =
579 {
580 VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO,
581 DE_NULL,
582 (vk::VkPipelineLayoutCreateFlags)0,
583
584 1u,
585 &descriptorSetLayout,
586
587 0u,
588 DE_NULL
589 };
590
591 return createPipelineLayout(vkd, device, &createInfo);
592 }
593
createSubpassPipeline(const DeviceInterface & vkd,VkDevice device,VkRenderPass renderPass,VkPipelineLayout pipelineLayout,const vk::BinaryCollection & binaryCollection,deUint32 width,deUint32 height,deUint32 sampleCount)594 Move<VkPipeline> createSubpassPipeline (const DeviceInterface& vkd,
595 VkDevice device,
596 VkRenderPass renderPass,
597 VkPipelineLayout pipelineLayout,
598 const vk::BinaryCollection& binaryCollection,
599 deUint32 width,
600 deUint32 height,
601 deUint32 sampleCount)
602 {
603 const Unique<VkShaderModule> vertexShaderModule (createShaderModule(vkd, device, binaryCollection.get("quad-vert"), 0u));
604 const Unique<VkShaderModule> fragmentShaderModule (createShaderModule(vkd, device, binaryCollection.get("quad-subpass-frag"), 0u));
605
606 const VkPipelineVertexInputStateCreateInfo vertexInputState =
607 {
608 VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO,
609 DE_NULL,
610 (VkPipelineVertexInputStateCreateFlags)0u,
611
612 0u,
613 DE_NULL,
614
615 0u,
616 DE_NULL
617 };
618
619 const std::vector<VkViewport> viewports (1, makeViewport(tcu::UVec2(width, height)));
620 const std::vector<VkRect2D> scissors (1, makeRect2D(tcu::UVec2(width, height)));
621
622 const VkPipelineMultisampleStateCreateInfo multisampleState =
623 {
624 VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO,
625 DE_NULL,
626 (VkPipelineMultisampleStateCreateFlags)0u,
627
628 sampleCountBitFromSampleCount(sampleCount),
629 VK_FALSE,
630 0.0f,
631 DE_NULL,
632 VK_FALSE,
633 VK_FALSE,
634 };
635
636 return makeGraphicsPipeline(vkd, // const DeviceInterface& vk
637 device, // const VkDevice device
638 pipelineLayout, // const VkPipelineLayout pipelineLayout
639 *vertexShaderModule, // const VkShaderModule vertexShaderModule
640 DE_NULL, // const VkShaderModule tessellationControlShaderModule
641 DE_NULL, // const VkShaderModule tessellationEvalShaderModule
642 DE_NULL, // const VkShaderModule geometryShaderModule
643 *fragmentShaderModule, // const VkShaderModule fragmentShaderModule
644 renderPass, // const VkRenderPass renderPass
645 viewports, // const std::vector<VkViewport>& viewports
646 scissors, // const std::vector<VkRect2D>& scissors
647 VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST, // const VkPrimitiveTopology topology
648 1u, // const deUint32 subpass
649 0u, // const deUint32 patchControlPoints
650 &vertexInputState, // const VkPipelineVertexInputStateCreateInfo* vertexInputStateCreateInfo
651 DE_NULL, // const VkPipelineRasterizationStateCreateInfo* rasterizationStateCreateInfo
652 &multisampleState); // const VkPipelineMultisampleStateCreateInfo* multisampleStateCreateInfo
653 }
654
createSubpassDescriptorPool(const DeviceInterface & vkd,VkDevice device)655 Move<VkDescriptorPool> createSubpassDescriptorPool (const DeviceInterface& vkd,
656 VkDevice device)
657 {
658 const VkDescriptorPoolSize size =
659 {
660 VK_DESCRIPTOR_TYPE_INPUT_ATTACHMENT, 2u
661 };
662 const VkDescriptorPoolCreateInfo createInfo =
663 {
664 VK_STRUCTURE_TYPE_DESCRIPTOR_POOL_CREATE_INFO,
665 DE_NULL,
666 VK_DESCRIPTOR_POOL_CREATE_FREE_DESCRIPTOR_SET_BIT,
667
668 2u,
669 1u,
670 &size
671 };
672
673 return createDescriptorPool(vkd, device, &createInfo);
674 }
675
createSubpassDescriptorSet(const DeviceInterface & vkd,VkDevice device,VkDescriptorPool pool,VkDescriptorSetLayout layout,VkImageView imageView)676 Move<VkDescriptorSet> createSubpassDescriptorSet (const DeviceInterface& vkd,
677 VkDevice device,
678 VkDescriptorPool pool,
679 VkDescriptorSetLayout layout,
680 VkImageView imageView)
681 {
682 const VkDescriptorSetAllocateInfo allocateInfo =
683 {
684 VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO,
685 DE_NULL,
686
687 pool,
688 1u,
689 &layout
690 };
691 Move<VkDescriptorSet> set (allocateDescriptorSet(vkd, device, &allocateInfo));
692
693 {
694 const VkDescriptorImageInfo imageInfo =
695 {
696 (VkSampler)0u,
697 imageView,
698 VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL
699 };
700 const VkWriteDescriptorSet write =
701 {
702 VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
703 DE_NULL,
704
705 *set,
706 0u,
707 0u,
708 1u,
709 VK_DESCRIPTOR_TYPE_INPUT_ATTACHMENT,
710 &imageInfo,
711 DE_NULL,
712 DE_NULL
713 };
714
715 vkd.updateDescriptorSets(device, 1u, &write, 0u, DE_NULL);
716 }
717 return set;
718 }
719
720 enum TestMode
721 {
722 TESTMODE_ADD = 0,
723 TESTMODE_SELECT,
724
725 TESTMODE_LAST
726 };
727
728 struct TestConfig
729 {
TestConfigvkt::__anon4d5745420111::TestConfig730 TestConfig (deUint32 sampleCount_, TestMode testMode_, deUint32 selectedSample_, RenderPassType renderPassType_)
731 : sampleCount (sampleCount_)
732 , testMode (testMode_)
733 , selectedSample (selectedSample_)
734 , renderPassType (renderPassType_)
735 {
736 }
737
738 deUint32 sampleCount;
739 TestMode testMode;
740 deUint32 selectedSample;
741 RenderPassType renderPassType;
742 };
743
744 class SampleReadTestInstance : public TestInstance
745 {
746 public:
747 SampleReadTestInstance (Context& context, TestConfig config);
748 ~SampleReadTestInstance (void);
749
750 tcu::TestStatus iterate (void);
751
752 template<typename RenderpassSubpass>
753 tcu::TestStatus iterateInternal (void);
754
755 private:
756 const bool m_extensionSupported;
757 const RenderPassType m_renderPassType;
758
759 const deUint32 m_sampleCount;
760 const deUint32 m_width;
761 const deUint32 m_height;
762 const TestMode m_testMode;
763 const deUint32 m_selectedSample;
764
765 const Unique<VkImage> m_srcImage;
766 const de::UniquePtr<Allocation> m_srcImageMemory;
767 const Unique<VkImageView> m_srcImageView;
768 const Unique<VkImageView> m_srcInputImageView;
769
770 const Unique<VkImage> m_dstMultisampleImage;
771 const de::UniquePtr<Allocation> m_dstMultisampleImageMemory;
772 const Unique<VkImageView> m_dstMultisampleImageView;
773
774 const Unique<VkImage> m_dstSinglesampleImage;
775 const de::UniquePtr<Allocation> m_dstSinglesampleImageMemory;
776 const Unique<VkImageView> m_dstSinglesampleImageView;
777
778 const Unique<VkBuffer> m_dstBuffer;
779 const de::UniquePtr<Allocation> m_dstBufferMemory;
780
781 const Unique<VkRenderPass> m_renderPass;
782 const Unique<VkFramebuffer> m_framebuffer;
783
784 const Unique<VkPipelineLayout> m_renderPipelineLayout;
785 const Unique<VkPipeline> m_renderPipeline;
786
787 const Unique<VkDescriptorSetLayout> m_subpassDescriptorSetLayout;
788 const Unique<VkPipelineLayout> m_subpassPipelineLayout;
789 const Unique<VkPipeline> m_subpassPipeline;
790 const Unique<VkDescriptorPool> m_subpassDescriptorPool;
791 const Unique<VkDescriptorSet> m_subpassDescriptorSet;
792
793 const Unique<VkCommandPool> m_commandPool;
794 tcu::ResultCollector m_resultCollector;
795 };
796
SampleReadTestInstance(Context & context,TestConfig config)797 SampleReadTestInstance::SampleReadTestInstance (Context& context, TestConfig config)
798 : TestInstance (context)
799 , m_extensionSupported (context.requireDeviceCoreFeature(DEVICE_CORE_FEATURE_SAMPLE_RATE_SHADING) &&
800 ((config.renderPassType != RENDERPASS_TYPE_RENDERPASS2) || context.requireDeviceFunctionality("VK_KHR_create_renderpass2")))
801 , m_renderPassType (config.renderPassType)
802 , m_sampleCount (config.sampleCount)
803 , m_width (32u)
804 , m_height (32u)
805 , m_testMode (config.testMode)
806 , m_selectedSample (config.selectedSample)
807 , m_srcImage (createImage(context.getInstanceInterface(), context.getPhysicalDevice(), context.getDeviceInterface(), context.getDevice(), VK_FORMAT_R32_UINT, sampleCountBitFromSampleCount(m_sampleCount), VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT, m_width, m_height))
808 , m_srcImageMemory (createImageMemory(context.getDeviceInterface(), context.getDevice(), context.getDefaultAllocator(), *m_srcImage))
809 , m_srcImageView (createImageView(context.getDeviceInterface(), context.getDevice(), *m_srcImage, VK_FORMAT_R32_UINT, VK_IMAGE_ASPECT_COLOR_BIT))
810 , m_srcInputImageView (createImageView(context.getDeviceInterface(), context.getDevice(), *m_srcImage, VK_FORMAT_R32_UINT, VK_IMAGE_ASPECT_COLOR_BIT))
811 , m_dstMultisampleImage (createImage(context.getInstanceInterface(), context.getPhysicalDevice(), context.getDeviceInterface(), context.getDevice(), VK_FORMAT_R32_UINT, sampleCountBitFromSampleCount(m_sampleCount), VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT, m_width, m_height))
812 , m_dstMultisampleImageMemory (createImageMemory(context.getDeviceInterface(), context.getDevice(), context.getDefaultAllocator(), *m_dstMultisampleImage))
813 , m_dstMultisampleImageView (createImageView(context.getDeviceInterface(), context.getDevice(), *m_dstMultisampleImage, VK_FORMAT_R32_UINT, VK_IMAGE_ASPECT_COLOR_BIT))
814 , m_dstSinglesampleImage (createImage(context.getInstanceInterface(), context.getPhysicalDevice(), context.getDeviceInterface(), context.getDevice(), VK_FORMAT_R32_UINT, VK_SAMPLE_COUNT_1_BIT, VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_SRC_BIT, m_width, m_height))
815 , m_dstSinglesampleImageMemory (createImageMemory(context.getDeviceInterface(), context.getDevice(), context.getDefaultAllocator(), *m_dstSinglesampleImage))
816 , m_dstSinglesampleImageView (createImageView(context.getDeviceInterface(), context.getDevice(), *m_dstSinglesampleImage, VK_FORMAT_R32_UINT, VK_IMAGE_ASPECT_COLOR_BIT))
817 , m_dstBuffer (createBuffer(context.getDeviceInterface(), context.getDevice(), VK_FORMAT_R32_UINT, m_width, m_height))
818 , m_dstBufferMemory (createBufferMemory(context.getDeviceInterface(), context.getDevice(), context.getDefaultAllocator(), *m_dstBuffer))
819 , m_renderPass (createRenderPass(context.getDeviceInterface(), context.getDevice(), VK_FORMAT_R32_UINT, VK_FORMAT_R32_UINT, m_sampleCount, config.renderPassType))
820 , m_framebuffer (createFramebuffer(context.getDeviceInterface(), context.getDevice(), *m_renderPass, *m_srcImageView, *m_dstMultisampleImageView, *m_dstSinglesampleImageView, m_width, m_height))
821 , m_renderPipelineLayout (createRenderPipelineLayout(context.getDeviceInterface(), context.getDevice()))
822 , m_renderPipeline (createRenderPipeline(context.getDeviceInterface(), context.getDevice(), *m_renderPass, *m_renderPipelineLayout, context.getBinaryCollection(), m_width, m_height, m_sampleCount))
823 , m_subpassDescriptorSetLayout (createSubpassDescriptorSetLayout(context.getDeviceInterface(), context.getDevice()))
824 , m_subpassPipelineLayout (createSubpassPipelineLayout(context.getDeviceInterface(), context.getDevice(), *m_subpassDescriptorSetLayout))
825 , m_subpassPipeline (createSubpassPipeline(context.getDeviceInterface(), context.getDevice(), *m_renderPass, *m_subpassPipelineLayout, context.getBinaryCollection(), m_width, m_height, m_sampleCount))
826 , m_subpassDescriptorPool (createSubpassDescriptorPool(context.getDeviceInterface(), context.getDevice()))
827 , m_subpassDescriptorSet (createSubpassDescriptorSet(context.getDeviceInterface(), context.getDevice(), *m_subpassDescriptorPool, *m_subpassDescriptorSetLayout, *m_srcInputImageView))
828 , m_commandPool (createCommandPool(context.getDeviceInterface(), context.getDevice(), VK_COMMAND_POOL_CREATE_TRANSIENT_BIT, context.getUniversalQueueFamilyIndex()))
829 {
830 }
831
~SampleReadTestInstance(void)832 SampleReadTestInstance::~SampleReadTestInstance (void)
833 {
834 }
835
iterate(void)836 tcu::TestStatus SampleReadTestInstance::iterate (void)
837 {
838 switch (m_renderPassType)
839 {
840 case RENDERPASS_TYPE_LEGACY:
841 return iterateInternal<RenderpassSubpass1>();
842 case RENDERPASS_TYPE_RENDERPASS2:
843 return iterateInternal<RenderpassSubpass2>();
844 default:
845 TCU_THROW(InternalError, "Impossible");
846 }
847 }
848
849 template<typename RenderpassSubpass>
iterateInternal(void)850 tcu::TestStatus SampleReadTestInstance::iterateInternal (void)
851 {
852 const DeviceInterface& vkd (m_context.getDeviceInterface());
853 const VkDevice device (m_context.getDevice());
854 const Unique<VkCommandBuffer> commandBuffer (allocateCommandBuffer(vkd, device, *m_commandPool, VK_COMMAND_BUFFER_LEVEL_PRIMARY));
855 const typename RenderpassSubpass::SubpassBeginInfo subpassBeginInfo (DE_NULL, VK_SUBPASS_CONTENTS_INLINE);
856 const typename RenderpassSubpass::SubpassEndInfo subpassEndInfo (DE_NULL);
857
858 beginCommandBuffer(vkd, *commandBuffer);
859
860 {
861 const VkRenderPassBeginInfo beginInfo =
862 {
863 VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO,
864 DE_NULL,
865
866 *m_renderPass,
867 *m_framebuffer,
868
869 {
870 { 0u, 0u },
871 { m_width, m_height }
872 },
873
874 0u,
875 DE_NULL
876 };
877 RenderpassSubpass::cmdBeginRenderPass(vkd, *commandBuffer, &beginInfo, &subpassBeginInfo);
878 }
879
880 vkd.cmdBindPipeline(*commandBuffer, VK_PIPELINE_BIND_POINT_GRAPHICS, *m_renderPipeline);
881
882 vkd.cmdDraw(*commandBuffer, 6u, 1u, 0u, 0u);
883
884 RenderpassSubpass::cmdNextSubpass(vkd, *commandBuffer, &subpassBeginInfo, &subpassEndInfo);
885
886 vkd.cmdBindPipeline(*commandBuffer, VK_PIPELINE_BIND_POINT_GRAPHICS, *m_subpassPipeline);
887 vkd.cmdBindDescriptorSets(*commandBuffer, VK_PIPELINE_BIND_POINT_GRAPHICS, *m_subpassPipelineLayout, 0u, 1u, &*m_subpassDescriptorSet, 0u, DE_NULL);
888 vkd.cmdDraw(*commandBuffer, 6u, 1u, 0u, 0u);
889
890 RenderpassSubpass::cmdEndRenderPass(vkd, *commandBuffer, &subpassEndInfo);
891
892 copyImageToBuffer(vkd, *commandBuffer, *m_dstSinglesampleImage, *m_dstBuffer, tcu::IVec2(m_width, m_height), VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL);
893
894 endCommandBuffer(vkd, *commandBuffer);
895
896 submitCommandsAndWait(vkd, device, m_context.getUniversalQueue(), *commandBuffer);
897
898 {
899 invalidateAlloc(vkd, device, *m_dstBufferMemory);
900
901 const tcu::TextureFormat format (mapVkFormat(VK_FORMAT_R32_UINT));
902 const void* const ptr (m_dstBufferMemory->getHostPtr());
903 const tcu::ConstPixelBufferAccess access (format, m_width, m_height, 1, ptr);
904 tcu::TextureLevel reference (format, m_width, m_height);
905
906 for (deUint32 y = 0; y < m_height; y++)
907 for (deUint32 x = 0; x < m_width; x++)
908 {
909 deUint32 bits;
910
911 if (m_testMode == TESTMODE_ADD)
912 bits = m_sampleCount == 32 ? 0xffffffff : (1u << m_sampleCount) - 1;
913 else
914 bits = 1u << m_selectedSample;
915
916 const UVec4 color (bits, 0, 0, 0xffffffff);
917
918 reference.getAccess().setPixel(color, x, y);
919 }
920
921 if (!tcu::intThresholdCompare(m_context.getTestContext().getLog(), "", "", reference.getAccess(), access, UVec4(0u), tcu::COMPARE_LOG_ON_ERROR))
922 m_resultCollector.fail("Compare failed.");
923 }
924
925 return tcu::TestStatus(m_resultCollector.getResult(), m_resultCollector.getMessage());
926 }
927
928 struct Programs
929 {
initvkt::__anon4d5745420111::Programs930 void init (vk::SourceCollections& dst, TestConfig config) const
931 {
932 std::ostringstream fragmentShader;
933 std::ostringstream subpassShader;
934
935 dst.glslSources.add("quad-vert") << glu::VertexSource(
936 "#version 450\n"
937 "out gl_PerVertex {\n"
938 "\tvec4 gl_Position;\n"
939 "};\n"
940 "highp float;\n"
941 "void main (void)\n"
942 "{\n"
943 " gl_Position = vec4(((gl_VertexIndex + 2) / 3) % 2 == 0 ? -1.0 : 1.0,\n"
944 " ((gl_VertexIndex + 1) / 3) % 2 == 0 ? -1.0 : 1.0, 0.0, 1.0);\n"
945 "}\n");
946
947 fragmentShader <<
948 "#version 450\n"
949 "layout(location = 0) out highp uvec4 o_color;\n"
950 "void main (void)\n"
951 "{\n"
952 " o_color = uvec4(1u << gl_SampleID, 0, 0, 0);\n"
953 "}\n";
954
955 dst.glslSources.add("quad-frag") << glu::FragmentSource(fragmentShader.str());
956
957 subpassShader <<
958 "#version 450\n"
959 "layout(input_attachment_index = 0, set = 0, binding = 0) uniform highp usubpassInputMS i_color;\n"
960 "layout(location = 0) out highp uvec4 o_color;\n"
961 "void main (void)\n"
962 "{\n"
963 " o_color = uvec4(0);\n";
964
965 if (config.testMode == TESTMODE_ADD)
966 {
967 subpassShader <<
968 " for (int i = 0; i < " << config.sampleCount << "; i++)\n" <<
969 " o_color.r += subpassLoad(i_color, i).r;\n";
970 }
971 else
972 {
973 subpassShader <<
974 " o_color.r = subpassLoad(i_color, " << de::toString(config.selectedSample) << ").r;\n";
975 }
976
977 subpassShader << "}\n";
978
979 dst.glslSources.add("quad-subpass-frag") << glu::FragmentSource(subpassShader.str());
980 }
981 };
982
initTests(tcu::TestCaseGroup * group,RenderPassType renderPassType)983 void initTests (tcu::TestCaseGroup* group, RenderPassType renderPassType)
984 {
985 const deUint32 sampleCounts[] = { 2u, 4u, 8u, 16u, 32u };
986 tcu::TestContext& testCtx (group->getTestContext());
987
988 for (deUint32 sampleCountNdx = 0; sampleCountNdx < DE_LENGTH_OF_ARRAY(sampleCounts); sampleCountNdx++)
989 {
990 const deUint32 sampleCount (sampleCounts[sampleCountNdx]);
991 {
992 const TestConfig testConfig (sampleCount, TESTMODE_ADD, 0, renderPassType);
993 const std::string testName ("numsamples_" + de::toString(sampleCount) + "_add");
994
995 group->addChild(new InstanceFactory1<SampleReadTestInstance, TestConfig, Programs>(testCtx, tcu::NODETYPE_SELF_VALIDATE, testName.c_str(), testName.c_str(), testConfig));
996 }
997
998 for (deUint32 sample = 0; sample < sampleCount; sample++)
999 {
1000 const TestConfig testConfig (sampleCount, TESTMODE_SELECT, sample, renderPassType);
1001 const std::string testName ("numsamples_" + de::toString(sampleCount) + "_selected_sample_" + de::toString(sample));
1002
1003 group->addChild(new InstanceFactory1<SampleReadTestInstance, TestConfig, Programs>(testCtx, tcu::NODETYPE_SELF_VALIDATE, testName.c_str(), testName.c_str(), testConfig));
1004 }
1005 }
1006 }
1007
1008 } // anonymous
1009
createRenderPassSampleReadTests(tcu::TestContext & testCtx)1010 tcu::TestCaseGroup* createRenderPassSampleReadTests (tcu::TestContext& testCtx)
1011 {
1012 return createTestGroup(testCtx, "sampleread", "Sample reading tests", initTests, RENDERPASS_TYPE_LEGACY);
1013 }
1014
createRenderPass2SampleReadTests(tcu::TestContext & testCtx)1015 tcu::TestCaseGroup* createRenderPass2SampleReadTests (tcu::TestContext& testCtx)
1016 {
1017 return createTestGroup(testCtx, "sampleread", "Sample reading tests", initTests, RENDERPASS_TYPE_RENDERPASS2);
1018 }
1019
1020 } // vkt
1021