1 /*------------------------------------------------------------------------
2 * Vulkan Conformance Tests
3 * ------------------------
4 *
5 * Copyright (c) 2020 Google LLC
6 * Copyright (c) 2020 The Khronos Group Inc.
7 *
8 * Licensed under the Apache License, Version 2.0 (the "License");
9 * you may not use this file except in compliance with the License.
10 * You may obtain a copy of the License at
11 *
12 * http://www.apache.org/licenses/LICENSE-2.0
13 *
14 * Unless required by applicable law or agreed to in writing, software
15 * distributed under the License is distributed on an "AS IS" BASIS,
16 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17 * See the License for the specific language governing permissions and
18 * limitations under the License.
19 *
20 *//*!
21 * \file
22 * \brief Test new features in VK_KHR_shader_terminate_invocation
23 *//*--------------------------------------------------------------------*/
24
25 #include <string>
26 #include <vector>
27 #include <amber/amber.h>
28
29 #include "tcuDefs.hpp"
30
31 #include "vkDefs.hpp"
32 #include "vktTestGroupUtil.hpp"
33 #include "vktAmberTestCase.hpp"
34 #include "vktSpvAsmTerminateInvocationTests.hpp"
35 #include "vktTestGroupUtil.hpp"
36
37 namespace vkt
38 {
39 namespace SpirVAssembly
40 {
41 namespace
42 {
43
44 struct Case
45 {
Casevkt::SpirVAssembly::__anon2b6660520111::Case46 Case(const char* b, const char* d, bool v) : basename(b), description(d), spv1p3(v), requirements() { }
Casevkt::SpirVAssembly::__anon2b6660520111::Case47 Case(const char* b, const char* d, bool v, const std::vector<std::string>& e) : basename(b), description(d), spv1p3(v), requirements(e) { }
48 const char *basename;
49 const char *description;
50 bool spv1p3;
51 // Additional Vulkan requirements, if any.
52 std::vector<std::string> requirements;
53 };
54 struct CaseGroup
55 {
CaseGroupvkt::SpirVAssembly::__anon2b6660520111::CaseGroup56 CaseGroup(const char* the_data_dir) : data_dir(the_data_dir) { }
addvkt::SpirVAssembly::__anon2b6660520111::CaseGroup57 void add(const char* basename, const char* description, bool spv1p3)
58 {
59 cases.push_back(Case(basename, description, spv1p3));
60 }
addvkt::SpirVAssembly::__anon2b6660520111::CaseGroup61 void add(const char* basename, const char* description, bool spv1p3, const std::vector<std::string>& requirements)
62 {
63 cases.push_back(Case(basename, description, spv1p3, requirements));
64 }
65
66 const char* data_dir;
67 std::vector<Case> cases;
68 };
69
70 #ifndef CTS_USES_VULKANSC
71
addTestsForAmberFiles(tcu::TestCaseGroup * tests,CaseGroup group)72 void addTestsForAmberFiles (tcu::TestCaseGroup* tests, CaseGroup group)
73 {
74 tcu::TestContext& testCtx = tests->getTestContext();
75 const std::string data_dir(group.data_dir);
76 const std::string category = data_dir;
77 std::vector<Case> cases(group.cases);
78
79 for (unsigned i = 0; i < cases.size() ; ++i)
80 {
81 deUint32 vulkan_version = cases[i].spv1p3 ? VK_MAKE_API_VERSION(0, 1, 1, 0) : VK_MAKE_API_VERSION(0, 1, 0, 0);
82 vk::SpirvVersion spirv_version = cases[i].spv1p3 ? vk::SPIRV_VERSION_1_3 : vk::SPIRV_VERSION_1_0;
83 vk::SpirVAsmBuildOptions asm_options(vulkan_version, spirv_version);
84
85 const std::string file = std::string(cases[i].basename) + ".amber";
86 cts_amber::AmberTestCase *testCase = cts_amber::createAmberTestCase(testCtx,
87 cases[i].basename,
88 cases[i].description,
89 category.c_str(),
90 file);
91 DE_ASSERT(testCase != DE_NULL);
92 testCase->addRequirement("VK_KHR_shader_terminate_invocation");
93 const std::vector<std::string>& reqmts = cases[i].requirements;
94 for (size_t r = 0; r < reqmts.size() ; ++r)
95 {
96 testCase->addRequirement(reqmts[r]);
97 }
98
99 testCase->setSpirVAsmBuildOptions(asm_options);
100 tests->addChild(testCase);
101 }
102 }
103
104 #endif // CTS_USES_VULKANSC
105
106 } // anonymous
107
createTerminateInvocationGroup(tcu::TestContext & testCtx)108 tcu::TestCaseGroup* createTerminateInvocationGroup(tcu::TestContext& testCtx)
109 {
110 de::MovePtr<tcu::TestCaseGroup> terminateTests(new tcu::TestCaseGroup(testCtx, "terminate_invocation", "VK_KHR_shader_terminate_invocation tests"));
111
112 const char* data_data = "spirv_assembly/instruction/terminate_invocation";
113
114 std::vector<std::string> Stores;
115 Stores.push_back("Features.fragmentStoresAndAtomics");
116
117 std::vector<std::string> VarPtr;
118 VarPtr.push_back("VariablePointerFeatures.variablePointersStorageBuffer");
119 VarPtr.push_back("Features.fragmentStoresAndAtomics");
120
121 std::vector<std::string> Vote;
122 Vote.push_back("SubgroupSupportedOperations.vote");
123 Vote.push_back("SubgroupSupportedStages.fragment");
124
125 std::vector<std::string> Ballot;
126 Ballot.push_back("SubgroupSupportedOperations.ballot");
127 Ballot.push_back("SubgroupSupportedStages.fragment");
128
129 CaseGroup group(data_data);
130 group.add("no_output_write", "no write to after calling terminate invocation", false);
131 group.add("no_output_write_before_terminate", "no write to output despite occurring before terminate invocation", false);
132 group.add("no_ssbo_store", "no store to SSBO when it occurs after terminate invocation", false, Stores);
133 group.add("no_ssbo_atomic", "no atomic update to SSBO when it occurs after terminate invocation", false, Stores);
134 group.add("ssbo_store_before_terminate", "ssbo store commits when it occurs before terminate invocation", false, Stores);
135 group.add("no_image_store", "no image write when it occurs after terminate invocation", false, Stores);
136 group.add("no_image_atomic", "no image atomic updates when it occurs after terminate invocation", false, Stores);
137 group.add("no_null_pointer_load", "null pointer should not be accessed by a load in a terminated invocation", false, VarPtr);
138 group.add("no_null_pointer_store", "null pointer should not be accessed by a store in a terminated invocation", false, VarPtr);
139 group.add("no_out_of_bounds_load", "out of bounds pointer should not be accessed by a load in a terminated invocation", false, VarPtr);
140 group.add("no_out_of_bounds_store", "out of bounds pointer should not be accessed by a store in a terminated invocation", false, VarPtr);
141 group.add("no_out_of_bounds_atomic", "out of bounds pointer should not be accessed by an atomic in a terminated invocation", false, VarPtr);
142 group.add("terminate_loop", "\"inifinite\" loop that calls terminate invocation", false);
143 group.add("subgroup_ballot", "checks that terminated invocations don't participate in the ballot", true, Ballot);
144 group.add("subgroup_vote", "checks that a subgroup all does not include any terminated invocations", true, Vote);
145 #ifndef CTS_USES_VULKANSC
146 terminateTests->addChild(createTestGroup(testCtx, "terminate", "Terminate Invocation", addTestsForAmberFiles, group));
147 #endif // CTS_USES_VULKANSC
148
149 return terminateTests.release();
150 }
151
152 } // SpirVAssembly
153 } // vkt
154