1 /*-------------------------------------------------------------------------
2 * Vulkan Conformance Tests
3 * ------------------------
4 *
5 * Copyright (c) 2019 The Khronos Group 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 SPIR-V Float Control SPIR-V tokens test
22 *//*--------------------------------------------------------------------*/
23
24 #include "vkApiVersion.hpp"
25
26 #include "vktSpvAsmFloatControlsExtensionlessTests.hpp"
27 #include "vktTestCase.hpp"
28 #include "vktSpvAsmComputeShaderCase.hpp"
29
30 #include "deRandom.hpp"
31 #include "deStringUtil.hpp"
32 #include "tcuCommandLine.hpp"
33 #include "vkQueryUtil.hpp"
34
35 namespace vkt
36 {
37 namespace SpirVAssembly
38 {
39
40 static const char* TEST_FEATURE_DENORM_PRESERVE = "DenormPreserve";
41 static const char* TEST_FEATURE_DENORM_FLUSH_TO_ZERO = "DenormFlushToZero";
42 static const char* TEST_FEATURE_SIGNED_ZERO_INF_NAN_PRESERVE = "SignedZeroInfNanPreserve";
43 static const char* TEST_FEATURE_ROUNDING_MODE_RTE = "RoundingModeRTE";
44 static const char* TEST_FEATURE_ROUNDING_MODE_RTZ = "RoundingModeRTZ";
45
46 using namespace vk;
47 using std::map;
48 using std::string;
49 using std::vector;
50
getComputeSourceCode(std::string & computeSourceCode,const std::string & featureName,const int fpWideness)51 static void getComputeSourceCode (std::string& computeSourceCode, const std::string& featureName, const int fpWideness)
52 {
53 const std::string capability = "OpCapability " + featureName + "\n";
54 const std::string exeModes = "OpExecutionMode %main " + featureName + " " + de::toString(fpWideness) + "\n";
55
56 computeSourceCode =
57 string(getComputeAsmShaderPreamble(capability, "", exeModes, "", "%indata %outdata")) +
58
59 "OpSource GLSL 430\n"
60 "OpName %main \"main\"\n"
61 "OpName %id \"gl_GlobalInvocationID\"\n"
62
63 "OpDecorate %id BuiltIn GlobalInvocationId\n"
64
65 + getComputeAsmInputOutputBufferTraits("Block") + getComputeAsmCommonTypes("StorageBuffer") + getComputeAsmInputOutputBuffer("StorageBuffer") +
66
67 "%id = OpVariable %uvec3ptr Input\n"
68 "%zero = OpConstant %i32 0\n"
69
70 "%main = OpFunction %void None %voidf\n"
71 "%label = OpLabel\n"
72 "%idval = OpLoad %uvec3 %id\n"
73 "%x = OpCompositeExtract %u32 %idval 0\n"
74
75 " OpNop\n" // Inside a function body
76
77 "%inloc = OpAccessChain %f32ptr %indata %zero %x\n"
78 "%inval = OpLoad %f32 %inloc\n"
79 "%neg = OpFNegate %f32 %inval\n"
80 "%outloc = OpAccessChain %f32ptr %outdata %zero %x\n"
81 " OpStore %outloc %neg\n"
82 " OpReturn\n"
83 " OpFunctionEnd\n";
84 }
85
getComputeShaderSpec(Context & ctx,const std::string & testCaseName)86 static ComputeShaderSpec getComputeShaderSpec (Context& ctx, const std::string& testCaseName)
87 {
88 const deUint32 baseSeed = deStringHash(testCaseName.c_str()) + static_cast<deUint32>(ctx.getTestContext().getCommandLine().getBaseSeed());
89 de::Random rnd (baseSeed);
90 const int numElements = 64;
91 vector<float> inputFloats (numElements, 0);
92 vector<float> outputFloats (numElements, 0);
93 ComputeShaderSpec spec;
94
95 for (size_t ndx = 0; ndx < numElements; ++ndx)
96 inputFloats[ndx] = rnd.getFloat(1.0f, 100.0f);
97
98 for (size_t ndx = 0; ndx < numElements; ++ndx)
99 outputFloats[ndx] = -inputFloats[ndx];
100
101 // Shader source code can be retrieved to complete definition of ComputeShaderSpec, though it is not required at this stage
102 // getComputeSourceCode (spec.assembly);
103
104 spec.inputs.push_back(BufferSp(new Float32Buffer(inputFloats)));
105 spec.outputs.push_back(BufferSp(new Float32Buffer(outputFloats)));
106
107 spec.numWorkGroups = tcu::IVec3(numElements, 1, 1);
108 spec.verifyIO = &verifyOutput;
109
110 return spec;
111 }
112
getFloatControlsProperty(Context & context,const int fpWideness,const std::string & featureName)113 VkBool32 getFloatControlsProperty(Context& context, const int fpWideness, const std::string& featureName)
114 {
115 VkPhysicalDeviceFloatControlsProperties floatControlsProperties;
116 deMemset(&floatControlsProperties, 0, sizeof(floatControlsProperties));
117 floatControlsProperties.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FLOAT_CONTROLS_PROPERTIES;
118
119 VkPhysicalDeviceProperties2 properties;
120 deMemset(&properties, 0, sizeof(properties));
121 properties.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2;
122 properties.pNext = &floatControlsProperties;
123
124 context.getInstanceInterface().getPhysicalDeviceProperties2(context.getPhysicalDevice(), &properties);
125
126 if (fpWideness == 16)
127 {
128 if (featureName == TEST_FEATURE_DENORM_PRESERVE ) return floatControlsProperties.shaderDenormPreserveFloat16;
129 if (featureName == TEST_FEATURE_DENORM_FLUSH_TO_ZERO ) return floatControlsProperties.shaderDenormFlushToZeroFloat16;
130 if (featureName == TEST_FEATURE_SIGNED_ZERO_INF_NAN_PRESERVE) return floatControlsProperties.shaderSignedZeroInfNanPreserveFloat16;
131 if (featureName == TEST_FEATURE_ROUNDING_MODE_RTE ) return floatControlsProperties.shaderRoundingModeRTEFloat16;
132 if (featureName == TEST_FEATURE_ROUNDING_MODE_RTZ ) return floatControlsProperties.shaderRoundingModeRTZFloat16;
133 }
134
135 if (fpWideness == 32)
136 {
137 if (featureName == TEST_FEATURE_DENORM_PRESERVE ) return floatControlsProperties.shaderDenormPreserveFloat32;
138 if (featureName == TEST_FEATURE_DENORM_FLUSH_TO_ZERO ) return floatControlsProperties.shaderDenormFlushToZeroFloat32;
139 if (featureName == TEST_FEATURE_SIGNED_ZERO_INF_NAN_PRESERVE) return floatControlsProperties.shaderSignedZeroInfNanPreserveFloat32;
140 if (featureName == TEST_FEATURE_ROUNDING_MODE_RTE ) return floatControlsProperties.shaderRoundingModeRTEFloat32;
141 if (featureName == TEST_FEATURE_ROUNDING_MODE_RTZ ) return floatControlsProperties.shaderRoundingModeRTZFloat32;
142 }
143
144 if (fpWideness == 64)
145 {
146 if (featureName == TEST_FEATURE_DENORM_PRESERVE ) return floatControlsProperties.shaderDenormPreserveFloat64;
147 if (featureName == TEST_FEATURE_DENORM_FLUSH_TO_ZERO ) return floatControlsProperties.shaderDenormFlushToZeroFloat64;
148 if (featureName == TEST_FEATURE_SIGNED_ZERO_INF_NAN_PRESERVE) return floatControlsProperties.shaderSignedZeroInfNanPreserveFloat64;
149 if (featureName == TEST_FEATURE_ROUNDING_MODE_RTE ) return floatControlsProperties.shaderRoundingModeRTEFloat64;
150 if (featureName == TEST_FEATURE_ROUNDING_MODE_RTZ ) return floatControlsProperties.shaderRoundingModeRTZFloat64;
151 }
152
153 TCU_THROW(InternalError, "Unknown property requested");
154 }
155
156 class SpvAsmFloatControlsExtensionlessInstance : public ComputeShaderSpec, public SpvAsmComputeShaderInstance
157 {
158 public:
159 SpvAsmFloatControlsExtensionlessInstance (Context& ctx, const std::string& testCaseName);
160 };
161
SpvAsmFloatControlsExtensionlessInstance(Context & ctx,const std::string & testCaseName)162 SpvAsmFloatControlsExtensionlessInstance::SpvAsmFloatControlsExtensionlessInstance (Context& ctx, const std::string& testCaseName)
163 : ComputeShaderSpec(getComputeShaderSpec(ctx, testCaseName))
164 , SpvAsmComputeShaderInstance(ctx, *this)
165 {
166 }
167
SpvAsmFloatControlsExtensionlessCase(tcu::TestContext & testCtx,const char * name,const char * description,const char * featureName,const int fpWideness,const bool spirv14)168 SpvAsmFloatControlsExtensionlessCase::SpvAsmFloatControlsExtensionlessCase (tcu::TestContext& testCtx, const char* name, const char* description, const char* featureName, const int fpWideness, const bool spirv14)
169 : TestCase (testCtx, name, description)
170 , m_featureName (featureName)
171 , m_fpWideness (fpWideness)
172 , m_spirv14 (spirv14)
173 {
174 }
175
initPrograms(SourceCollections & programCollection) const176 void SpvAsmFloatControlsExtensionlessCase::initPrograms (SourceCollections& programCollection) const
177 {
178 const bool allowSpirv14 = true;
179 std::string comp;
180
181 getComputeSourceCode(comp, m_featureName, m_fpWideness);
182
183 programCollection.spirvAsmSources.add("compute") << SpirVAsmBuildOptions(programCollection.usedVulkanVersion, SPIRV_VERSION_1_4, allowSpirv14) << comp;
184 }
185
checkSupport(Context & context) const186 void SpvAsmFloatControlsExtensionlessCase::checkSupport (Context& context) const
187 {
188 if (m_spirv14)
189 {
190 context.requireDeviceFunctionality("VK_KHR_spirv_1_4");
191 }
192 else
193 {
194 if (!context.contextSupports(vk::ApiVersion(1, 2, 0)))
195 TCU_THROW(NotSupportedError, "Test requires Vulkan 1.2");
196 }
197
198 if (m_fpWideness == 16)
199 {
200 context.requireDeviceFunctionality("VK_KHR_shader_float16_int8");
201 if (!isFloat16Int8FeaturesSupported(context, EXTFLOAT16INT8FEATURES_FLOAT16))
202 TCU_THROW(NotSupportedError, "Floating point number of width 16 bit are not supported");
203 }
204
205 if (m_fpWideness == 64)
206 {
207 context.requireDeviceCoreFeature(DEVICE_CORE_FEATURE_SHADER_FLOAT64);
208 }
209
210 if (!getFloatControlsProperty(context, m_fpWideness, m_featureName))
211 TCU_THROW(NotSupportedError, "Property is not supported");
212 }
213
createInstance(Context & context) const214 TestInstance* SpvAsmFloatControlsExtensionlessCase::createInstance (Context& context) const
215 {
216 return new SpvAsmFloatControlsExtensionlessInstance(context, getName());
217 }
218
createFloatControlsExtensionlessGroup(tcu::TestContext & testCtx)219 tcu::TestCaseGroup* createFloatControlsExtensionlessGroup (tcu::TestContext& testCtx)
220 {
221 const char* spirVersions[] = { "spirv1p4", "vulkan1_2" };
222 const int floatingPointWideness[] = { 16, 32, 64 };
223 const struct FpFeatures
224 {
225 const char* testName;
226 const char* featureName;
227 }
228 fpFeatures[] =
229 {
230 { "denorm_preserve", TEST_FEATURE_DENORM_PRESERVE },
231 { "denorm_flush_to_zero", TEST_FEATURE_DENORM_FLUSH_TO_ZERO },
232 { "signed_zero_inf_nan_preserve", TEST_FEATURE_SIGNED_ZERO_INF_NAN_PRESERVE },
233 { "rounding_mode_rte", TEST_FEATURE_ROUNDING_MODE_RTE },
234 { "rounding_mode_rtz", TEST_FEATURE_ROUNDING_MODE_RTZ },
235 };
236 de::MovePtr<tcu::TestCaseGroup> group (new tcu::TestCaseGroup(testCtx, "float_controls_extensionless", "Tests float controls without extension"));
237
238 for (int spirVersionsNdx = 0; spirVersionsNdx < DE_LENGTH_OF_ARRAY(spirVersions); ++spirVersionsNdx)
239 {
240 const bool spirv14 = (spirVersionsNdx == 0);
241 de::MovePtr<tcu::TestCaseGroup> spirVersionGroup (new tcu::TestCaseGroup(testCtx, spirVersions[spirVersionsNdx], ""));
242
243 for (int fpWidenessNdx = 0; fpWidenessNdx < DE_LENGTH_OF_ARRAY(floatingPointWideness); ++fpWidenessNdx)
244 for (int execModeNdx = 0; execModeNdx < DE_LENGTH_OF_ARRAY(fpFeatures); ++execModeNdx)
245 {
246 const int fpWideness = floatingPointWideness[fpWidenessNdx];
247 const std::string testName = fpFeatures[execModeNdx].testName;
248 const char* featureName = fpFeatures[execModeNdx].featureName;
249 const std::string fullTestName = "fp" + de::toString(fpWideness) + "_" + testName;
250
251 spirVersionGroup->addChild(new SpvAsmFloatControlsExtensionlessCase(testCtx, fullTestName.c_str(), "", featureName, fpWideness, spirv14));
252 }
253
254 group->addChild(spirVersionGroup.release());
255 }
256
257 return group.release();
258 }
259
260
261 } // SpirVAssembly
262 } // vkt
263