forked from KhronosGroup/Vulkan-ValidationLayers
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathvkrenderframework.h
798 lines (694 loc) · 37.6 KB
/
vkrenderframework.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
/*
* Copyright (c) 2015-2023 The Khronos Group Inc.
* Copyright (c) 2015-2023 Valve Corporation
* Copyright (c) 2015-2023 LunarG, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#pragma once
#include "lvt_function_pointers.h"
#include "vkerrormonitor.h"
#ifdef ANDROID
#include "vktestframeworkandroid.h"
class VkImageObj;
#else
#include "vktestframework.h"
#endif
#if defined(ANDROID)
#include <android/log.h>
#if defined(VALIDATION_APK)
#include <android_native_app_glue.h>
#endif
#endif
#include <algorithm>
#include <array>
#include <memory>
#include <vector>
#include <unordered_map>
#include <unordered_set>
using vk_testing::MakeVkHandles;
static constexpr uint64_t kWaitTimeout{10000000000}; // 10 seconds in ns
template <class Dst, class Src>
std::vector<Dst *> MakeTestbindingHandles(const std::vector<Src *> &v) {
std::vector<Dst *> handles;
handles.reserve(v.size());
std::transform(v.begin(), v.end(), std::back_inserter(handles), [](const Src *o) { return static_cast<Dst *>(o); });
return handles;
}
typedef vk_testing::Queue VkQueueObj;
class VkDeviceObj : public vk_testing::Device {
public:
VkDeviceObj(uint32_t id, VkPhysicalDevice obj);
VkDeviceObj(uint32_t id, VkPhysicalDevice obj, std::vector<const char *> &extension_names,
VkPhysicalDeviceFeatures *features = nullptr, void *create_device_pnext = nullptr);
// Find a queue family with and without desired capabilities
std::optional<uint32_t> QueueFamilyMatching(VkQueueFlags with, VkQueueFlags without, bool all_bits = true);
std::optional<uint32_t> QueueFamilyWithoutCapabilities(VkQueueFlags capabilities) {
// an all_bits match with 0 matches all
return QueueFamilyMatching(VkQueueFlags(0), capabilities, true /* all_bits with */);
}
VkDevice device() { return handle(); }
void SetDeviceQueue();
VkQueueObj *GetDefaultQueue();
VkQueueObj *GetDefaultComputeQueue();
uint32_t id;
VkPhysicalDeviceProperties props;
std::vector<VkQueueFamilyProperties> queue_props;
VkQueueObj *m_queue_obj = nullptr;
VkQueue m_queue;
};
struct DebugReporter {
void Create(VkInstance instance) noexcept;
void Destroy(VkInstance instance) noexcept;
ErrorMonitor error_monitor_;
#ifdef VK_USE_PLATFORM_ANDROID_KHR
static VKAPI_ATTR VkBool32 VKAPI_CALL DebugCallback(VkDebugReportFlagsEXT message_flags, VkDebugReportObjectTypeEXT, uint64_t,
size_t, int32_t, const char *, const char *msg, void *user_data);
const char *debug_extension_name = VK_EXT_DEBUG_REPORT_EXTENSION_NAME;
VkDebugReportCallbackCreateInfoEXT debug_create_info_ = {VK_STRUCTURE_TYPE_DEBUG_REPORT_CALLBACK_CREATE_INFO_EXT, nullptr,
VK_DEBUG_REPORT_ERROR_BIT_EXT | VK_DEBUG_REPORT_WARNING_BIT_EXT |
VK_DEBUG_REPORT_PERFORMANCE_WARNING_BIT_EXT |
VK_DEBUG_REPORT_INFORMATION_BIT_EXT,
&DebugCallback, &error_monitor_};
using DebugCreateFnType = PFN_vkCreateDebugReportCallbackEXT;
const char *debug_create_fn_name_ = "vkCreateDebugReportCallbackEXT";
using DebugDestroyFnType = PFN_vkDestroyDebugReportCallbackEXT;
const char *debug_destroy_fn_name_ = "vkDestroyDebugReportCallbackEXT";
VkDebugReportCallbackEXT debug_obj_ = VK_NULL_HANDLE;
#else
static VKAPI_ATTR VkBool32 VKAPI_CALL DebugCallback(VkDebugUtilsMessageSeverityFlagBitsEXT message_severity,
VkDebugUtilsMessageTypeFlagsEXT message_types,
const VkDebugUtilsMessengerCallbackDataEXT *callback_data, void *user_data);
const char *debug_extension_name = VK_EXT_DEBUG_UTILS_EXTENSION_NAME;
VkDebugUtilsMessengerCreateInfoEXT debug_create_info_ = {
VK_STRUCTURE_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT,
nullptr,
0,
VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT | VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT |
VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT,
VK_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT | VK_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT,
&DebugCallback,
&error_monitor_};
using DebugCreateFnType = PFN_vkCreateDebugUtilsMessengerEXT;
const char *debug_create_fn_name_ = "vkCreateDebugUtilsMessengerEXT";
using DebugDestroyFnType = PFN_vkDestroyDebugUtilsMessengerEXT;
const char *debug_destroy_fn_name_ = "vkDestroyDebugUtilsMessengerEXT";
VkDebugUtilsMessengerEXT debug_obj_ = VK_NULL_HANDLE;
#endif
};
class VkCommandPoolObj;
class VkCommandBufferObj;
class VkDepthStencilObj;
typedef enum {
kGalaxyS10,
kPixel3,
kPixelC,
kNexusPlayer,
kShieldTV,
kShieldTVb,
kPixel3aXL,
kPixel2XL,
kMockICD,
} PlatformType;
const std::unordered_map<PlatformType, std::string, std::hash<int>> vk_gpu_table = {
{kGalaxyS10, "Mali-G76"},
{kPixel3, "Adreno (TM) 630"},
{kPixelC, "NVIDIA Tegra X1"},
{kNexusPlayer, "PowerVR Rogue G6430"},
{kShieldTV, "NVIDIA Tegra X1 (nvgpu)"},
{kShieldTVb, "NVIDIA Tegra X1 (rev B) (nvgpu)"},
{kPixel3aXL, "Adreno (TM) 615"},
{kPixel2XL, "Adreno (TM) 540"},
{kMockICD, "Vulkan Mock Device"},
};
class VkRenderFramework : public VkTestFramework {
public:
VkInstance instance() const { return instance_; }
VkDevice device() const { return m_device->device(); }
VkDeviceObj *DeviceObj() const { return m_device; }
VkPhysicalDevice gpu() const;
VkRenderPass renderPass() const { return m_renderPass; }
const VkRenderPassCreateInfo &RenderPassInfo() const { return m_renderPass_info; };
VkFramebuffer framebuffer() const { return m_framebuffer; }
ErrorMonitor &Monitor();
const VkPhysicalDeviceProperties &physDevProps() const;
bool InstanceLayerSupported(const char *layer_name, uint32_t spec_version = 0, uint32_t impl_version = 0);
bool InstanceExtensionSupported(const char *extension_name, uint32_t spec_version = 0);
VkInstanceCreateInfo GetInstanceCreateInfo() const;
void InitFramework(void * /*unused compatibility parameter*/ = NULL, void *instance_pnext = NULL);
void ShutdownFramework();
void InitViewport(float width, float height);
void InitViewport();
bool InitSurface();
bool InitSurface(VkSurfaceKHR &surface);
void InitSwapchainInfo();
bool InitSwapchain(VkSurfaceKHR &surface, VkImageUsageFlags imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT,
VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR);
bool InitSwapchain(VkImageUsageFlags imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT,
VkSurfaceTransformFlagBitsKHR preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR);
bool InitSwapchain(VkSurfaceKHR &surface, VkImageUsageFlags imageUsage, VkSurfaceTransformFlagBitsKHR preTransform, VkSwapchainKHR &swapchain, VkSwapchainKHR oldSwapchain = 0);
void DestroySwapchain();
void InitRenderTarget();
void InitRenderTarget(uint32_t targets);
void InitRenderTarget(VkImageView *dsBinding);
void InitRenderTarget(uint32_t targets, VkImageView *dsBinding);
void DestroyRenderTarget();
static bool IgnoreDisableChecks();
bool IsPlatform(PlatformType platform);
void GetPhysicalDeviceFeatures(VkPhysicalDeviceFeatures *features);
void GetPhysicalDeviceProperties(VkPhysicalDeviceProperties *props);
VkFormat GetRenderTargetFormat();
void InitState(VkPhysicalDeviceFeatures *features = nullptr, void *create_device_pnext = nullptr,
const VkCommandPoolCreateFlags flags = 0);
const VkRenderPassBeginInfo &renderPassBeginInfo() const { return m_renderPassBeginInfo; }
bool InstanceExtensionEnabled(const char *name);
bool DeviceExtensionSupported(const char *extension_name, uint32_t spec_version = 0) const;
bool DeviceExtensionSupported(VkPhysicalDevice, const char *, const char *name,
uint32_t spec_version = 0) const { // deprecated
return DeviceExtensionSupported(name, spec_version);
}
bool DeviceExtensionEnabled(const char *name);
// Tracks ext_name to be enabled at device creation time and attempts to enable any required instance extensions.
// Does not return anything as the caller should use AreRequiredExtensionsEnabled or AddOptionalExtensions then
// `ext_name` can refer to a device or instance extension.
void AddRequiredExtensions(const char *ext_name);
// Same as AddRequiredExtensions but won't fail a check to AreRequiredExtensionsEnabled
void AddOptionalExtensions(const char *ext_name);
// After instance and physical device creation (e.g., after InitFramework), returns true if all required extensions are
// available, false otherwise
bool AreRequiredExtensionsEnabled() const;
// After instance and physical device creation (e.g., after InitFramework), returns if extension was enabled
bool IsExtensionsEnabled(const char *ext_name) const;
// if requested extensions are not supported, helper function to get string to print out
std::string RequiredExtensionsNotSupported() const;
void *SetupValidationSettings(void *first_pnext);
template <typename GLSLContainer>
std::vector<uint32_t> GLSLToSPV(VkShaderStageFlagBits stage, const GLSLContainer &code, const char *entry_point = "main",
const VkSpecializationInfo *spec_info = nullptr, const spv_target_env env = SPV_ENV_VULKAN_1_0,
bool debug = false) {
std::vector<uint32_t> spv;
GLSLtoSPV(&m_device->props.limits, stage, code, spv, debug, env);
return spv;
}
protected:
VkRenderFramework();
virtual ~VkRenderFramework() = 0;
std::vector<VkLayerProperties> available_layers_; // allow caching of available layers
std::vector<VkExtensionProperties> available_extensions_; // allow caching of available instance extensions
DebugReporter debug_reporter_;
ErrorMonitor *m_errorMonitor = &debug_reporter_.error_monitor_; // compatibility alias name
VkApplicationInfo app_info_;
std::vector<const char *> instance_layers_;
std::vector<const char *> instance_extensions_;
std::vector<const char *> &m_instance_extension_names = instance_extensions_; // compatibility alias name
VkInstance instance_;
VkPhysicalDevice gpu_ = VK_NULL_HANDLE;
VkPhysicalDeviceProperties physDevProps_;
uint32_t m_gpu_index;
VkDeviceObj *m_device;
VkCommandPoolObj *m_commandPool;
VkCommandBufferObj *m_commandBuffer;
VkRenderPass m_renderPass = VK_NULL_HANDLE;
VkRenderPassCreateInfo m_renderPass_info = {};
std::vector<VkAttachmentDescription> m_renderPass_attachments;
std::vector<VkSubpassDescription> m_renderPass_subpasses;
std::vector<VkSubpassDependency> m_renderPass_dependencies;
VkFramebuffer m_framebuffer;
VkFramebufferCreateInfo m_framebuffer_info;
std::vector<VkImageView> m_framebuffer_attachments;
// WSI items
VkSurfaceKHR m_surface = VK_NULL_HANDLE;
#if defined(VK_USE_PLATFORM_WIN32_KHR)
HWND m_win32Window;
#endif
#if defined(VK_USE_PLATFORM_XLIB_KHR)
Display *m_surface_dpy;
Window m_surface_window;
#endif
#if defined(VK_USE_PLATFORM_XCB_KHR)
xcb_connection_t *m_surface_xcb_conn;
#endif
VkSwapchainKHR m_swapchain;
VkSurfaceCapabilitiesKHR m_surface_capabilities;
std::vector<VkSurfaceFormatKHR> m_surface_formats;
std::vector<VkPresentModeKHR> m_surface_present_modes;
VkPresentModeKHR m_surface_non_shared_present_mode;
VkCompositeAlphaFlagBitsKHR m_surface_composite_alpha;
std::vector<VkViewport> m_viewports;
std::vector<VkRect2D> m_scissors;
float m_lineWidth;
float m_depthBiasConstantFactor;
float m_depthBiasClamp;
float m_depthBiasSlopeFactor;
float m_blendConstants[4];
float m_minDepthBounds;
float m_maxDepthBounds;
uint32_t m_compareMask;
uint32_t m_writeMask;
uint32_t m_reference;
bool m_addRenderPassSelfDependency;
std::vector<VkSubpassDependency> m_additionalSubpassDependencies;
std::vector<VkClearValue> m_renderPassClearValues;
VkRenderPassBeginInfo m_renderPassBeginInfo;
std::vector<std::unique_ptr<VkImageObj>> m_renderTargets;
float m_width, m_height;
VkFormat m_render_target_fmt;
VkFormat m_depth_stencil_fmt;
VkClearColorValue m_clear_color;
bool m_clear_via_load_op;
float m_depth_clear_color;
uint32_t m_stencil_clear_color;
VkDepthStencilObj *m_depthStencil;
// Requested extensions to enable at device creation time
std::vector<const char *> m_required_extensions;
// Optional extensions to try and enable at device creation time
std::vector<const char *> m_optional_extensions;
// Device extensions to enable
std::vector<const char *> m_device_extension_names;
VkValidationFeaturesEXT validation_features;
VkValidationFeatureEnableEXT validation_enable_all[4] = {VK_VALIDATION_FEATURE_ENABLE_GPU_ASSISTED_EXT,
VK_VALIDATION_FEATURE_ENABLE_BEST_PRACTICES_EXT,
VK_VALIDATION_FEATURE_ENABLE_GPU_ASSISTED_RESERVE_BINDING_SLOT_EXT,
VK_VALIDATION_FEATURE_ENABLE_SYNCHRONIZATION_VALIDATION_EXT};
VkValidationFeatureDisableEXT validation_disable_all = VK_VALIDATION_FEATURE_DISABLE_ALL_EXT;
private:
// Add ext_name, the names of all instance extensions required by ext_name, and return true if ext_name is supported. If the
// extension is not supported, no extension names are added for instance creation. `ext_name` can refer to a device or instance
// extension.
bool AddRequestedInstanceExtensions(const char *ext_name);
// Returns true if the instance extension inst_ext_name is enabled. This call is only valid _after_ previous
// `AddRequired*Extensions` calls and InitFramework has been called. `inst_ext_name` must be an instance extension name; false
// is returned for all device extension names.
bool CanEnableInstanceExtension(const std::string &inst_ext_name) const;
// Add dev_ext_name, then names of _device_ extensions required by dev_ext_name, and return true if dev_ext_name is supported.
// If the extension is not supported, no extension names are added for device creation. This function has no effect if
// dev_ext_name refers to an instance extension.
bool AddRequestedDeviceExtensions(const char *dev_ext_name);
// Returns true if the device extension is enabled. This call is only valid _after_ previous `AddRequired*Extensions` calls and
// InitFramework has been called.
// `dev_ext_name` must be an instance extension name; false is returned for all instance extension names.
bool CanEnableDeviceExtension(const std::string &dev_ext_name) const;
};
class VkDescriptorSetObj;
class VkConstantBufferObj;
class VkPipelineObj;
typedef vk_testing::Event VkEventObj;
typedef vk_testing::Fence VkFenceObj;
typedef vk_testing::Semaphore VkSemaphoreObj;
typedef vk_testing::Buffer VkBufferObj;
typedef vk_testing::AccelerationStructure VkAccelerationStructureObj;
typedef vk_testing::AccelerationStructureKHR VkAccelerationStructurekhrObj;
class VkCommandPoolObj : public vk_testing::CommandPool {
public:
VkCommandPoolObj() : vk_testing::CommandPool(){};
void Init(VkDeviceObj *device, uint32_t queue_family_index, VkCommandPoolCreateFlags flags = 0);
VkCommandPoolObj(VkDeviceObj *device, uint32_t queue_family_index, VkCommandPoolCreateFlags flags = 0);
};
class VkCommandBufferObj : public vk_testing::CommandBuffer {
public:
VkCommandBufferObj() : vk_testing::CommandBuffer() {}
VkCommandBufferObj(VkDeviceObj *device, VkCommandPoolObj *pool, VkCommandBufferLevel level = VK_COMMAND_BUFFER_LEVEL_PRIMARY,
VkQueueObj *queue = nullptr);
void Init(VkDeviceObj *device, VkCommandPoolObj *pool, VkCommandBufferLevel level = VK_COMMAND_BUFFER_LEVEL_PRIMARY,
VkQueueObj *queue = nullptr);
void PipelineBarrier(VkPipelineStageFlags src_stages, VkPipelineStageFlags dest_stages, VkDependencyFlags dependencyFlags,
uint32_t memoryBarrierCount, const VkMemoryBarrier *pMemoryBarriers, uint32_t bufferMemoryBarrierCount,
const VkBufferMemoryBarrier *pBufferMemoryBarriers, uint32_t imageMemoryBarrierCount,
const VkImageMemoryBarrier *pImageMemoryBarriers);
void PipelineBarrier2KHR(const VkDependencyInfoKHR *pDependencyInfo);
void ClearAllBuffers(const std::vector<std::unique_ptr<VkImageObj>> &color_objs, VkClearColorValue clear_color,
VkDepthStencilObj *depth_stencil_obj, float depth_clear_value, uint32_t stencil_clear_value);
void PrepareAttachments(const std::vector<std::unique_ptr<VkImageObj>> &color_atts, VkDepthStencilObj *depth_stencil_att);
void BindDescriptorSet(VkDescriptorSetObj &descriptorSet);
void BindIndexBuffer(VkBufferObj *indexBuffer, VkDeviceSize offset, VkIndexType indexType);
void BindVertexBuffer(VkConstantBufferObj *vertexBuffer, VkDeviceSize offset, uint32_t binding);
void BeginRenderPass(const VkRenderPassBeginInfo &info, VkSubpassContents contents = VK_SUBPASS_CONTENTS_INLINE);
void NextSubpass(VkSubpassContents contents = VK_SUBPASS_CONTENTS_INLINE);
void EndRenderPass();
void BeginRendering(const VkRenderingInfoKHR &renderingInfo);
void EndRendering();
void BeginVideoCoding(const VkVideoBeginCodingInfoKHR &beginInfo);
void ControlVideoCoding(const VkVideoCodingControlInfoKHR &controlInfo);
void DecodeVideo(const VkVideoDecodeInfoKHR &decodeInfo);
void EndVideoCoding(const VkVideoEndCodingInfoKHR &endInfo);
void FillBuffer(VkBuffer buffer, VkDeviceSize offset, VkDeviceSize fill_size, uint32_t data);
void Draw(uint32_t vertexCount, uint32_t instanceCount, uint32_t firstVertex, uint32_t firstInstance);
void DrawIndexed(uint32_t indexCount, uint32_t instanceCount, uint32_t firstIndex, int32_t vertexOffset,
uint32_t firstInstance);
void QueueCommandBuffer(bool check_success = true);
void QueueCommandBuffer(const VkFenceObj &fence, bool check_success = true, bool submit_2 = false);
void SetViewport(uint32_t firstViewport, uint32_t viewportCount, const VkViewport *pViewports);
void SetStencilReference(VkStencilFaceFlags faceMask, uint32_t reference);
void UpdateBuffer(VkBuffer buffer, VkDeviceSize dstOffset, VkDeviceSize dataSize, const void *pData);
void CopyImage(VkImage srcImage, VkImageLayout srcImageLayout, VkImage dstImage, VkImageLayout dstImageLayout,
uint32_t regionCount, const VkImageCopy *pRegions);
void ResolveImage(VkImage srcImage, VkImageLayout srcImageLayout, VkImage dstImage, VkImageLayout dstImageLayout,
uint32_t regionCount, const VkImageResolve *pRegions);
void ClearColorImage(VkImage image, VkImageLayout imageLayout, const VkClearColorValue *pColor, uint32_t rangeCount,
const VkImageSubresourceRange *pRanges);
void ClearDepthStencilImage(VkImage image, VkImageLayout imageLayout, const VkClearDepthStencilValue *pColor,
uint32_t rangeCount, const VkImageSubresourceRange *pRanges);
void BuildAccelerationStructure(VkAccelerationStructureObj *as, VkBuffer scratchBuffer);
void BuildAccelerationStructure(VkAccelerationStructureObj *as, VkBuffer scratchBuffer, VkBuffer instanceData);
void SetEvent(VkEventObj &event, VkPipelineStageFlags stageMask) { event.cmd_set(*this, stageMask); }
void ResetEvent(VkEventObj &event, VkPipelineStageFlags stageMask) { event.cmd_reset(*this, stageMask); }
void WaitEvents(uint32_t eventCount, const VkEvent *pEvents, VkPipelineStageFlags srcStageMask,
VkPipelineStageFlags dstStageMask, uint32_t memoryBarrierCount, const VkMemoryBarrier *pMemoryBarriers,
uint32_t bufferMemoryBarrierCount, const VkBufferMemoryBarrier *pBufferMemoryBarriers,
uint32_t imageMemoryBarrierCount, const VkImageMemoryBarrier *pImageMemoryBarriers) {
vk::CmdWaitEvents(handle(), eventCount, pEvents, srcStageMask, dstStageMask, memoryBarrierCount, pMemoryBarriers,
bufferMemoryBarrierCount, pBufferMemoryBarriers, imageMemoryBarrierCount, pImageMemoryBarriers);
}
protected:
VkDeviceObj *m_device;
VkQueueObj *m_queue;
};
class VkConstantBufferObj : public VkBufferObj {
public:
VkConstantBufferObj(VkDeviceObj *device,
VkBufferUsageFlags usage = VK_BUFFER_USAGE_TRANSFER_SRC_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT);
VkConstantBufferObj(VkDeviceObj *device, VkDeviceSize size, const void *data,
VkBufferUsageFlags usage = VK_BUFFER_USAGE_TRANSFER_SRC_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT);
VkDescriptorBufferInfo m_descriptorBufferInfo;
protected:
VkDeviceObj *m_device;
};
class VkRenderpassObj : public vk_testing::RenderPass {
public:
VkRenderpassObj(VkDeviceObj *device, VkFormat format = VK_FORMAT_B8G8R8A8_UNORM);
VkRenderpassObj(VkDeviceObj *device, VkFormat format, bool depthStencil);
};
class VkImageObj : public vk_testing::Image {
public:
VkImageObj(VkDeviceObj *dev);
bool IsCompatible(VkImageUsageFlags usages, VkFormatFeatureFlags2 features);
public:
static VkImageCreateInfo ImageCreateInfo2D(uint32_t const width, uint32_t const height, uint32_t const mipLevels,
uint32_t const layers, VkFormat const format, VkFlags const usage,
VkImageTiling const requested_tiling = VK_IMAGE_TILING_LINEAR,
const std::vector<uint32_t> *queue_families = nullptr);
void Init(uint32_t const width, uint32_t const height, uint32_t const mipLevels, VkFormat const format, VkFlags const usage,
VkImageTiling const tiling = VK_IMAGE_TILING_LINEAR, VkMemoryPropertyFlags const reqs = 0,
const std::vector<uint32_t> *queue_families = nullptr, bool memory = true);
void Init(const VkImageCreateInfo &create_info, VkMemoryPropertyFlags const reqs = 0, bool memory = true);
void init(const VkImageCreateInfo *create_info);
void init_no_mem(const vk_testing::Device &dev, const VkImageCreateInfo &info);
void InitNoLayout(uint32_t const width, uint32_t const height, uint32_t const mipLevels, VkFormat const format,
VkFlags const usage, VkImageTiling tiling = VK_IMAGE_TILING_LINEAR, VkMemoryPropertyFlags reqs = 0,
const std::vector<uint32_t> *queue_families = nullptr, bool memory = true);
void InitNoLayout(const VkImageCreateInfo &create_info, VkMemoryPropertyFlags reqs = 0, bool memory = true);
// void clear( CommandBuffer*, uint32_t[4] );
void Layout(VkImageLayout const layout) { m_descriptorImageInfo.imageLayout = layout; }
VkDeviceMemory memory() const { return Image::memory().handle(); }
void *MapMemory() { return Image::memory().map(); }
void UnmapMemory() { Image::memory().unmap(); }
void ImageMemoryBarrier(VkCommandBufferObj *cmd, VkImageAspectFlags aspect, VkFlags output_mask, VkFlags input_mask,
VkImageLayout image_layout, VkPipelineStageFlags src_stages = VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
VkPipelineStageFlags dest_stages = VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
uint32_t srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
uint32_t dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED);
VkResult CopyImage(VkImageObj &src_image);
VkResult CopyImageOut(VkImageObj &dst_image);
std::array<std::array<uint32_t, 16>, 16> Read();
VkImage image() const { return handle(); }
VkImageViewCreateInfo TargetViewCI(VkFormat format) const {
auto ci = LvlInitStruct<VkImageViewCreateInfo>();
ci.format = format;
ci.viewType = VK_IMAGE_VIEW_TYPE_2D;
ci.components.r = VK_COMPONENT_SWIZZLE_R;
ci.components.g = VK_COMPONENT_SWIZZLE_G;
ci.components.b = VK_COMPONENT_SWIZZLE_B;
ci.components.a = VK_COMPONENT_SWIZZLE_A;
ci.subresourceRange = {
VK_IMAGE_ASPECT_COLOR_BIT,
0, // base mip level
VK_REMAINING_MIP_LEVELS, // level count
0, // base array layer
VK_REMAINING_ARRAY_LAYERS, // layer count
};
ci.flags = 0;
return ci;
}
VkImageView targetView(VkImageViewCreateInfo ci) {
if (!m_targetView.initialized()) {
ci.image = handle();
m_targetView.init(*m_device, ci);
}
return m_targetView.handle();
}
VkImageView targetView(VkFormat format, VkImageAspectFlags aspect = VK_IMAGE_ASPECT_COLOR_BIT, uint32_t baseMipLevel = 0,
uint32_t levelCount = VK_REMAINING_MIP_LEVELS, uint32_t baseArrayLayer = 0,
uint32_t layerCount = VK_REMAINING_ARRAY_LAYERS, VkImageViewType type = VK_IMAGE_VIEW_TYPE_2D) {
if (!m_targetView.initialized()) {
VkImageViewCreateInfo createView = LvlInitStruct<VkImageViewCreateInfo>();
createView.image = handle();
createView.viewType = type;
createView.format = format;
createView.components.r = VK_COMPONENT_SWIZZLE_R;
createView.components.g = VK_COMPONENT_SWIZZLE_G;
createView.components.b = VK_COMPONENT_SWIZZLE_B;
createView.components.a = VK_COMPONENT_SWIZZLE_A;
createView.subresourceRange = {aspect, baseMipLevel, levelCount, baseArrayLayer, layerCount};
createView.flags = 0;
m_targetView.init(*m_device, createView);
}
return m_targetView.handle();
}
void SetLayout(VkCommandBufferObj *cmd_buf, VkImageAspectFlags aspect, VkImageLayout image_layout);
void SetLayout(VkImageAspectFlags aspect, VkImageLayout image_layout);
void SetLayout(VkImageLayout image_layout) { SetLayout(aspect_mask(), image_layout); };
VkImageLayout Layout() const { return m_descriptorImageInfo.imageLayout; }
uint32_t width() const { return extent().width; }
uint32_t height() const { return extent().height; }
VkDeviceObj *device() const { return m_device; }
protected:
VkDeviceObj *m_device;
vk_testing::ImageView m_targetView;
VkDescriptorImageInfo m_descriptorImageInfo;
uint32_t m_mipLevels;
uint32_t m_arrayLayers;
};
class VkTextureObj : public VkImageObj {
public:
VkTextureObj(VkDeviceObj *device, uint32_t *colors = NULL);
const VkDescriptorImageInfo &DescriptorImageInfo() const { return m_descriptorImageInfo; }
protected:
VkDeviceObj *m_device;
vk_testing::ImageView m_textureView;
};
class VkDepthStencilObj : public VkImageObj {
public:
VkDepthStencilObj(VkDeviceObj *device);
void Init(VkDeviceObj *device, int32_t width, int32_t height, VkFormat format,
VkImageUsageFlags usage = VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT, VkImageAspectFlags aspect = 0);
bool Initialized();
VkImageView *BindInfo();
VkFormat Format() const;
protected:
VkDeviceObj *m_device;
bool m_initialized;
vk_testing::ImageView m_imageView;
VkFormat m_depth_stencil_fmt;
VkImageView m_attachmentBindInfo;
};
class VkSamplerObj : public vk_testing::Sampler {
public:
VkSamplerObj(VkDeviceObj *device);
protected:
VkDeviceObj *m_device;
};
class VkDescriptorSetLayoutObj : public vk_testing::DescriptorSetLayout {
public:
VkDescriptorSetLayoutObj() = default;
VkDescriptorSetLayoutObj(const VkDeviceObj *device,
const std::vector<VkDescriptorSetLayoutBinding> &descriptor_set_bindings = {},
VkDescriptorSetLayoutCreateFlags flags = 0, void *pNext = NULL);
// Move constructor and move assignment operator for Visual Studio 2013
VkDescriptorSetLayoutObj(VkDescriptorSetLayoutObj &&src) noexcept : DescriptorSetLayout(std::move(src)){};
VkDescriptorSetLayoutObj &operator=(VkDescriptorSetLayoutObj &&src) noexcept {
DescriptorSetLayout::operator=(std::move(src));
return *this;
}
};
class VkDescriptorSetObj : public vk_testing::DescriptorPool {
public:
VkDescriptorSetObj(VkDeviceObj *device);
~VkDescriptorSetObj() noexcept;
int AppendDummy();
int AppendBuffer(VkDescriptorType type, VkConstantBufferObj &constantBuffer);
int AppendSamplerTexture(VkSamplerObj *sampler, VkTextureObj *texture);
void CreateVKDescriptorSet(VkCommandBufferObj *commandBuffer);
VkDescriptorSet GetDescriptorSetHandle() const;
VkPipelineLayout GetPipelineLayout() const;
VkDescriptorSetLayout GetDescriptorSetLayout() const;
protected:
VkDeviceObj *m_device;
std::vector<VkDescriptorSetLayoutBinding> m_layout_bindings;
std::map<VkDescriptorType, int> m_type_counts;
int m_nextSlot;
std::vector<VkDescriptorImageInfo> m_imageSamplerDescriptors;
std::vector<VkWriteDescriptorSet> m_writes;
vk_testing::DescriptorSetLayout m_layout;
vk_testing::PipelineLayout m_pipeline_layout;
vk_testing::DescriptorSet *m_set = NULL;
};
// What is the incoming source to be turned into VkShaderModuleCreateInfo::pCode
typedef enum {
SPV_SOURCE_GLSL,
SPV_SOURCE_ASM,
// TRY == Won't try in contructor as need to be called as function that can return the VkResult
SPV_SOURCE_GLSL_TRY,
SPV_SOURCE_ASM_TRY,
} SpvSourceType;
class VkShaderObj : public vk_testing::ShaderModule {
public:
// optional arguments listed order of most likely to be changed manually by a test
VkShaderObj(VkRenderFramework *framework, const char *source, VkShaderStageFlagBits stage,
const spv_target_env env = SPV_ENV_VULKAN_1_0, SpvSourceType source_type = SPV_SOURCE_GLSL,
const VkSpecializationInfo *spec_info = nullptr, char const *name = "main", bool debug = false);
VkPipelineShaderStageCreateInfo const &GetStageCreateInfo() const;
bool InitFromGLSL(bool debug = false);
VkResult InitFromGLSLTry(bool debug = false, const VkDeviceObj *custom_device = nullptr);
bool InitFromASM();
VkResult InitFromASMTry();
// These functions return a pointer to a newly created _and initialized_ VkShaderObj if initialization was successful.
// Otherwise, {} is returned.
static std::unique_ptr<VkShaderObj> CreateFromGLSL(VkRenderFramework &framework, VkShaderStageFlagBits stage,
const std::string &code, const char *entry_point = "main",
const VkSpecializationInfo *spec_info = nullptr,
const spv_target_env = SPV_ENV_VULKAN_1_0, bool debug = false);
static std::unique_ptr<VkShaderObj> CreateFromASM(VkRenderFramework &framework, VkShaderStageFlagBits stage,
const std::string &code, const char *entry_point = "main",
const VkSpecializationInfo *spec_info = nullptr,
const spv_target_env spv_env = SPV_ENV_VULKAN_1_0);
// TODO (ncesario) remove ifndef once android build consolidation changes go in
#ifndef __ANDROID__
struct GlslangTargetEnv {
GlslangTargetEnv(const spv_target_env env) {
switch (env) {
case SPV_ENV_UNIVERSAL_1_0:
language_version = glslang::EShTargetSpv_1_0;
break;
case SPV_ENV_UNIVERSAL_1_1:
language_version = glslang::EShTargetSpv_1_1;
break;
case SPV_ENV_UNIVERSAL_1_2:
language_version = glslang::EShTargetSpv_1_2;
break;
case SPV_ENV_UNIVERSAL_1_3:
language_version = glslang::EShTargetSpv_1_3;
break;
case SPV_ENV_UNIVERSAL_1_4:
language_version = glslang::EShTargetSpv_1_4;
break;
case SPV_ENV_UNIVERSAL_1_5:
language_version = glslang::EShTargetSpv_1_5;
break;
case SPV_ENV_UNIVERSAL_1_6:
language_version = glslang::EShTargetSpv_1_6;
break;
case SPV_ENV_VULKAN_1_0:
client_version = glslang::EShTargetVulkan_1_0;
break;
case SPV_ENV_VULKAN_1_1:
client_version = glslang::EShTargetVulkan_1_1;
language_version = glslang::EShTargetSpv_1_3;
break;
case SPV_ENV_VULKAN_1_2:
client_version = glslang::EShTargetVulkan_1_2;
language_version = glslang::EShTargetSpv_1_5;
break;
case SPV_ENV_VULKAN_1_3:
client_version = glslang::EShTargetVulkan_1_3;
language_version = glslang::EShTargetSpv_1_6;
break;
default:
assert(false && "Invalid SPIR-V environment");
break;
}
}
operator glslang::EShTargetLanguageVersion() const {
return language_version;
}
operator glslang::EShTargetClientVersion() const {
return client_version;
}
private:
glslang::EShTargetLanguageVersion language_version = glslang::EShTargetSpv_1_0;
glslang::EShTargetClientVersion client_version = glslang::EShTargetVulkan_1_0;
};
#endif
protected:
VkPipelineShaderStageCreateInfo m_stage_info;
VkRenderFramework &m_framework;
VkDeviceObj &m_device;
const char *m_source;
spv_target_env m_spv_env;
};
class VkPipelineLayoutObj : public vk_testing::PipelineLayout {
public:
VkPipelineLayoutObj() = default;
VkPipelineLayoutObj(VkDeviceObj *device, const std::vector<const VkDescriptorSetLayoutObj *> &descriptor_layouts = {},
const std::vector<VkPushConstantRange> &push_constant_ranges = {},
VkPipelineLayoutCreateFlags flags = static_cast<VkPipelineLayoutCreateFlags>(0));
// Move constructor and move assignment operator for Visual Studio 2013
VkPipelineLayoutObj(VkPipelineLayoutObj &&src) noexcept : PipelineLayout(std::move(src)) {}
VkPipelineLayoutObj &operator=(VkPipelineLayoutObj &&src) noexcept {
PipelineLayout::operator=(std::move(src));
return *this;
}
void Reset();
};
class VkPipelineObj : public vk_testing::Pipeline {
public:
VkPipelineObj(VkDeviceObj *device);
void AddShader(VkShaderObj *shaderObj);
void AddShader(VkPipelineShaderStageCreateInfo const &createInfo);
void AddVertexInputAttribs(VkVertexInputAttributeDescription *vi_attrib, uint32_t count);
void AddVertexInputBindings(VkVertexInputBindingDescription *vi_binding, uint32_t count);
void AddColorAttachment(uint32_t binding, const VkPipelineColorBlendAttachmentState &att);
void MakeDynamic(VkDynamicState state);
void AddDefaultColorAttachment(VkColorComponentFlags writeMask = 0xf /*=R|G|B|A*/) {
VkPipelineColorBlendAttachmentState att = {};
att.blendEnable = VK_FALSE;
att.colorWriteMask = writeMask;
AddColorAttachment(0, att);
}
void SetDepthStencil(const VkPipelineDepthStencilStateCreateInfo *);
void SetMSAA(const VkPipelineMultisampleStateCreateInfo *ms_state);
void SetInputAssembly(const VkPipelineInputAssemblyStateCreateInfo *ia_state);
void SetRasterization(const VkPipelineRasterizationStateCreateInfo *rs_state);
void SetTessellation(const VkPipelineTessellationStateCreateInfo *te_state);
void SetViewport(const std::vector<VkViewport> &viewports);
void SetScissor(const std::vector<VkRect2D> &scissors);
void SetLineState(const VkPipelineRasterizationLineStateCreateInfoEXT *line_state);
void InitGraphicsPipelineCreateInfo(VkGraphicsPipelineCreateInfo *gp_ci);
VkResult CreateVKPipeline(VkPipelineLayout layout, VkRenderPass render_pass, VkGraphicsPipelineCreateInfo *gp_ci = nullptr);
protected:
VkPipelineVertexInputStateCreateInfo m_vi_state;
VkPipelineInputAssemblyStateCreateInfo m_ia_state;
VkPipelineRasterizationStateCreateInfo m_rs_state;
VkPipelineColorBlendStateCreateInfo m_cb_state;
VkPipelineDepthStencilStateCreateInfo const *m_ds_state;
VkPipelineViewportStateCreateInfo m_vp_state;
VkPipelineMultisampleStateCreateInfo m_ms_state;
VkPipelineTessellationStateCreateInfo const *m_te_state;
VkPipelineDynamicStateCreateInfo m_pd_state;
VkPipelineRasterizationLineStateCreateInfoEXT m_line_state;
std::vector<VkDynamicState> m_dynamic_state_enables;
std::vector<VkViewport> m_viewports;
std::vector<VkRect2D> m_scissors;
VkDeviceObj *m_device;
std::vector<VkPipelineShaderStageCreateInfo> m_shaderStages;
std::vector<VkPipelineColorBlendAttachmentState> m_colorAttachments;
};