2284 lines
90 KiB
C++
2284 lines
90 KiB
C++
// Copyright (c) 2018 Google LLC.
|
|
//
|
|
// 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.
|
|
|
|
// Tests validation rules of GLSL.450.std and OpenCL.std extended instructions.
|
|
// Doesn't test OpenCL.std vector size 2, 3, 4, 8 or 16 rules (not supported
|
|
// by standard SPIR-V).
|
|
|
|
#include <cstring>
|
|
#include <sstream>
|
|
#include <string>
|
|
#include <tuple>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "gmock/gmock.h"
|
|
#include "test/unit_spirv.h"
|
|
#include "test/val/val_fixtures.h"
|
|
|
|
namespace spvtools {
|
|
namespace val {
|
|
namespace {
|
|
|
|
struct TestResult {
|
|
TestResult(spv_result_t in_validation_result = SPV_SUCCESS,
|
|
const char* in_error_str = nullptr,
|
|
const char* in_error_str2 = nullptr)
|
|
: validation_result(in_validation_result),
|
|
error_str(in_error_str),
|
|
error_str2(in_error_str2) {}
|
|
spv_result_t validation_result;
|
|
const char* error_str;
|
|
const char* error_str2;
|
|
};
|
|
|
|
using ::testing::Combine;
|
|
using ::testing::HasSubstr;
|
|
using ::testing::Not;
|
|
using ::testing::Values;
|
|
using ::testing::ValuesIn;
|
|
|
|
using ValidateBuiltIns = spvtest::ValidateBase<bool>;
|
|
using ValidateVulkanCombineBuiltInExecutionModelDataTypeResult =
|
|
spvtest::ValidateBase<std::tuple<const char*, const char*, const char*,
|
|
const char*, TestResult>>;
|
|
using ValidateVulkanCombineBuiltInArrayedVariable = spvtest::ValidateBase<
|
|
std::tuple<const char*, const char*, const char*, const char*, TestResult>>;
|
|
|
|
struct EntryPoint {
|
|
std::string name;
|
|
std::string execution_model;
|
|
std::string execution_modes;
|
|
std::string body;
|
|
std::string interfaces;
|
|
};
|
|
|
|
class CodeGenerator {
|
|
public:
|
|
std::string Build() const;
|
|
|
|
std::vector<EntryPoint> entry_points_;
|
|
std::string capabilities_;
|
|
std::string extensions_;
|
|
std::string memory_model_;
|
|
std::string before_types_;
|
|
std::string types_;
|
|
std::string after_types_;
|
|
std::string add_at_the_end_;
|
|
};
|
|
|
|
std::string CodeGenerator::Build() const {
|
|
std::ostringstream ss;
|
|
|
|
ss << capabilities_;
|
|
ss << extensions_;
|
|
ss << memory_model_;
|
|
|
|
for (const EntryPoint& entry_point : entry_points_) {
|
|
ss << "OpEntryPoint " << entry_point.execution_model << " %"
|
|
<< entry_point.name << " \"" << entry_point.name << "\" "
|
|
<< entry_point.interfaces << "\n";
|
|
}
|
|
|
|
for (const EntryPoint& entry_point : entry_points_) {
|
|
ss << entry_point.execution_modes << "\n";
|
|
}
|
|
|
|
ss << before_types_;
|
|
ss << types_;
|
|
ss << after_types_;
|
|
|
|
for (const EntryPoint& entry_point : entry_points_) {
|
|
ss << "\n";
|
|
ss << "%" << entry_point.name << " = OpFunction %void None %func\n";
|
|
ss << "%" << entry_point.name << "_entry = OpLabel\n";
|
|
ss << entry_point.body;
|
|
ss << "\nOpReturn\nOpFunctionEnd\n";
|
|
}
|
|
|
|
ss << add_at_the_end_;
|
|
|
|
return ss.str();
|
|
}
|
|
|
|
std::string GetDefaultShaderCapabilities() {
|
|
return R"(
|
|
OpCapability Shader
|
|
OpCapability Geometry
|
|
OpCapability Tessellation
|
|
OpCapability Float64
|
|
OpCapability Int64
|
|
OpCapability MultiViewport
|
|
OpCapability SampleRateShading
|
|
)";
|
|
}
|
|
|
|
std::string GetDefaultShaderTypes() {
|
|
return R"(
|
|
%void = OpTypeVoid
|
|
%func = OpTypeFunction %void
|
|
%bool = OpTypeBool
|
|
%f32 = OpTypeFloat 32
|
|
%f64 = OpTypeFloat 64
|
|
%u32 = OpTypeInt 32 0
|
|
%u64 = OpTypeInt 64 0
|
|
%f32vec2 = OpTypeVector %f32 2
|
|
%f32vec3 = OpTypeVector %f32 3
|
|
%f32vec4 = OpTypeVector %f32 4
|
|
%f64vec2 = OpTypeVector %f64 2
|
|
%f64vec3 = OpTypeVector %f64 3
|
|
%f64vec4 = OpTypeVector %f64 4
|
|
%u32vec2 = OpTypeVector %u32 2
|
|
%u32vec3 = OpTypeVector %u32 3
|
|
%u64vec3 = OpTypeVector %u64 3
|
|
%u32vec4 = OpTypeVector %u32 4
|
|
%u64vec2 = OpTypeVector %u64 2
|
|
|
|
%f32_0 = OpConstant %f32 0
|
|
%f32_1 = OpConstant %f32 1
|
|
%f32_2 = OpConstant %f32 2
|
|
%f32_3 = OpConstant %f32 3
|
|
%f32_4 = OpConstant %f32 4
|
|
%f32_h = OpConstant %f32 0.5
|
|
%f32vec2_01 = OpConstantComposite %f32vec2 %f32_0 %f32_1
|
|
%f32vec2_12 = OpConstantComposite %f32vec2 %f32_1 %f32_2
|
|
%f32vec3_012 = OpConstantComposite %f32vec3 %f32_0 %f32_1 %f32_2
|
|
%f32vec3_123 = OpConstantComposite %f32vec3 %f32_1 %f32_2 %f32_3
|
|
%f32vec4_0123 = OpConstantComposite %f32vec4 %f32_0 %f32_1 %f32_2 %f32_3
|
|
%f32vec4_1234 = OpConstantComposite %f32vec4 %f32_1 %f32_2 %f32_3 %f32_4
|
|
|
|
%f64_0 = OpConstant %f64 0
|
|
%f64_1 = OpConstant %f64 1
|
|
%f64_2 = OpConstant %f64 2
|
|
%f64_3 = OpConstant %f64 3
|
|
%f64vec2_01 = OpConstantComposite %f64vec2 %f64_0 %f64_1
|
|
%f64vec3_012 = OpConstantComposite %f64vec3 %f64_0 %f64_1 %f64_2
|
|
%f64vec4_0123 = OpConstantComposite %f64vec4 %f64_0 %f64_1 %f64_2 %f64_3
|
|
|
|
%u32_0 = OpConstant %u32 0
|
|
%u32_1 = OpConstant %u32 1
|
|
%u32_2 = OpConstant %u32 2
|
|
%u32_3 = OpConstant %u32 3
|
|
%u32_4 = OpConstant %u32 4
|
|
|
|
%u64_0 = OpConstant %u64 0
|
|
%u64_1 = OpConstant %u64 1
|
|
%u64_2 = OpConstant %u64 2
|
|
%u64_3 = OpConstant %u64 3
|
|
|
|
%u32vec2_01 = OpConstantComposite %u32vec2 %u32_0 %u32_1
|
|
%u32vec2_12 = OpConstantComposite %u32vec2 %u32_1 %u32_2
|
|
%u32vec4_0123 = OpConstantComposite %u32vec4 %u32_0 %u32_1 %u32_2 %u32_3
|
|
%u64vec2_01 = OpConstantComposite %u64vec2 %u64_0 %u64_1
|
|
|
|
%u32arr2 = OpTypeArray %u32 %u32_2
|
|
%u32arr3 = OpTypeArray %u32 %u32_3
|
|
%u32arr4 = OpTypeArray %u32 %u32_4
|
|
%u64arr2 = OpTypeArray %u64 %u32_2
|
|
%u64arr3 = OpTypeArray %u64 %u32_3
|
|
%u64arr4 = OpTypeArray %u64 %u32_4
|
|
%f32arr2 = OpTypeArray %f32 %u32_2
|
|
%f32arr3 = OpTypeArray %f32 %u32_3
|
|
%f32arr4 = OpTypeArray %f32 %u32_4
|
|
%f64arr2 = OpTypeArray %f64 %u32_2
|
|
%f64arr3 = OpTypeArray %f64 %u32_3
|
|
%f64arr4 = OpTypeArray %f64 %u32_4
|
|
|
|
%f32vec3arr3 = OpTypeArray %f32vec3 %u32_3
|
|
%f32vec4arr3 = OpTypeArray %f32vec4 %u32_3
|
|
%f64vec4arr3 = OpTypeArray %f64vec4 %u32_3
|
|
)";
|
|
}
|
|
|
|
CodeGenerator GetDefaultShaderCodeGenerator() {
|
|
CodeGenerator generator;
|
|
generator.capabilities_ = GetDefaultShaderCapabilities();
|
|
generator.memory_model_ = "OpMemoryModel Logical GLSL450\n";
|
|
generator.types_ = GetDefaultShaderTypes();
|
|
return generator;
|
|
}
|
|
|
|
TEST_P(ValidateVulkanCombineBuiltInExecutionModelDataTypeResult, InMain) {
|
|
const char* const built_in = std::get<0>(GetParam());
|
|
const char* const execution_model = std::get<1>(GetParam());
|
|
const char* const storage_class = std::get<2>(GetParam());
|
|
const char* const data_type = std::get<3>(GetParam());
|
|
const TestResult& test_result = std::get<4>(GetParam());
|
|
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = "OpMemberDecorate %built_in_type 0 BuiltIn ";
|
|
generator.before_types_ += built_in;
|
|
generator.before_types_ += "\n";
|
|
|
|
std::ostringstream after_types;
|
|
after_types << "%built_in_type = OpTypeStruct " << data_type << "\n";
|
|
after_types << "%built_in_ptr = OpTypePointer " << storage_class
|
|
<< " %built_in_type\n";
|
|
after_types << "%built_in_var = OpVariable %built_in_ptr " << storage_class
|
|
<< "\n";
|
|
after_types << "%data_ptr = OpTypePointer " << storage_class << " "
|
|
<< data_type << "\n";
|
|
generator.after_types_ = after_types.str();
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = execution_model;
|
|
if (strncmp(storage_class, "Input", 5) == 0 ||
|
|
strncmp(storage_class, "Output", 6) == 0) {
|
|
entry_point.interfaces = "%built_in_var";
|
|
}
|
|
|
|
std::ostringstream execution_modes;
|
|
if (0 == std::strcmp(execution_model, "Fragment")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " OriginUpperLeft\n";
|
|
if (0 == std::strcmp(built_in, "FragDepth")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " DepthReplacing\n";
|
|
}
|
|
}
|
|
if (0 == std::strcmp(execution_model, "Geometry")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " InputPoints\n";
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " OutputPoints\n";
|
|
}
|
|
if (0 == std::strcmp(execution_model, "GLCompute")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " LocalSize 1 1 1\n";
|
|
}
|
|
entry_point.execution_modes = execution_modes.str();
|
|
|
|
entry_point.body = R"(
|
|
%ptr = OpAccessChain %data_ptr %built_in_var %u32_0
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(test_result.validation_result,
|
|
ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
if (test_result.error_str) {
|
|
EXPECT_THAT(getDiagnosticString(), HasSubstr(test_result.error_str));
|
|
}
|
|
if (test_result.error_str2) {
|
|
EXPECT_THAT(getDiagnosticString(), HasSubstr(test_result.error_str2));
|
|
}
|
|
}
|
|
|
|
TEST_P(ValidateVulkanCombineBuiltInExecutionModelDataTypeResult, InFunction) {
|
|
const char* const built_in = std::get<0>(GetParam());
|
|
const char* const execution_model = std::get<1>(GetParam());
|
|
const char* const storage_class = std::get<2>(GetParam());
|
|
const char* const data_type = std::get<3>(GetParam());
|
|
const TestResult& test_result = std::get<4>(GetParam());
|
|
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = "OpMemberDecorate %built_in_type 0 BuiltIn ";
|
|
generator.before_types_ += built_in;
|
|
generator.before_types_ += "\n";
|
|
|
|
std::ostringstream after_types;
|
|
after_types << "%built_in_type = OpTypeStruct " << data_type << "\n";
|
|
after_types << "%built_in_ptr = OpTypePointer " << storage_class
|
|
<< " %built_in_type\n";
|
|
after_types << "%built_in_var = OpVariable %built_in_ptr " << storage_class
|
|
<< "\n";
|
|
after_types << "%data_ptr = OpTypePointer " << storage_class << " "
|
|
<< data_type << "\n";
|
|
generator.after_types_ = after_types.str();
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = execution_model;
|
|
if (strncmp(storage_class, "Input", 5) == 0 ||
|
|
strncmp(storage_class, "Output", 6) == 0) {
|
|
entry_point.interfaces = "%built_in_var";
|
|
}
|
|
|
|
std::ostringstream execution_modes;
|
|
if (0 == std::strcmp(execution_model, "Fragment")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " OriginUpperLeft\n";
|
|
if (0 == std::strcmp(built_in, "FragDepth")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " DepthReplacing\n";
|
|
}
|
|
}
|
|
if (0 == std::strcmp(execution_model, "Geometry")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " InputPoints\n";
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " OutputPoints\n";
|
|
}
|
|
if (0 == std::strcmp(execution_model, "GLCompute")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " LocalSize 1 1 1\n";
|
|
}
|
|
entry_point.execution_modes = execution_modes.str();
|
|
|
|
entry_point.body = R"(
|
|
%val2 = OpFunctionCall %void %foo
|
|
)";
|
|
|
|
generator.add_at_the_end_ = R"(
|
|
%foo = OpFunction %void None %func
|
|
%foo_entry = OpLabel
|
|
%ptr = OpAccessChain %data_ptr %built_in_var %u32_0
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(test_result.validation_result,
|
|
ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
if (test_result.error_str) {
|
|
EXPECT_THAT(getDiagnosticString(), HasSubstr(test_result.error_str));
|
|
}
|
|
if (test_result.error_str2) {
|
|
EXPECT_THAT(getDiagnosticString(), HasSubstr(test_result.error_str2));
|
|
}
|
|
}
|
|
|
|
TEST_P(ValidateVulkanCombineBuiltInExecutionModelDataTypeResult, Variable) {
|
|
const char* const built_in = std::get<0>(GetParam());
|
|
const char* const execution_model = std::get<1>(GetParam());
|
|
const char* const storage_class = std::get<2>(GetParam());
|
|
const char* const data_type = std::get<3>(GetParam());
|
|
const TestResult& test_result = std::get<4>(GetParam());
|
|
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = "OpDecorate %built_in_var BuiltIn ";
|
|
generator.before_types_ += built_in;
|
|
generator.before_types_ += "\n";
|
|
|
|
std::ostringstream after_types;
|
|
after_types << "%built_in_ptr = OpTypePointer " << storage_class << " "
|
|
<< data_type << "\n";
|
|
after_types << "%built_in_var = OpVariable %built_in_ptr " << storage_class
|
|
<< "\n";
|
|
generator.after_types_ = after_types.str();
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = execution_model;
|
|
if (strncmp(storage_class, "Input", 5) == 0 ||
|
|
strncmp(storage_class, "Output", 6) == 0) {
|
|
entry_point.interfaces = "%built_in_var";
|
|
}
|
|
// Any kind of reference would do.
|
|
entry_point.body = R"(
|
|
%val = OpBitcast %u64 %built_in_var
|
|
)";
|
|
|
|
std::ostringstream execution_modes;
|
|
if (0 == std::strcmp(execution_model, "Fragment")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " OriginUpperLeft\n";
|
|
if (0 == std::strcmp(built_in, "FragDepth")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " DepthReplacing\n";
|
|
}
|
|
}
|
|
if (0 == std::strcmp(execution_model, "Geometry")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " InputPoints\n";
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " OutputPoints\n";
|
|
}
|
|
if (0 == std::strcmp(execution_model, "GLCompute")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " LocalSize 1 1 1\n";
|
|
}
|
|
entry_point.execution_modes = execution_modes.str();
|
|
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(test_result.validation_result,
|
|
ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
if (test_result.error_str) {
|
|
EXPECT_THAT(getDiagnosticString(), HasSubstr(test_result.error_str));
|
|
}
|
|
if (test_result.error_str2) {
|
|
EXPECT_THAT(getDiagnosticString(), HasSubstr(test_result.error_str2));
|
|
}
|
|
}
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceOutputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("ClipDistance", "CullDistance"),
|
|
Values("Vertex", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Output"), Values("%f32arr2", "%f32arr4"),
|
|
Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceInputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("ClipDistance", "CullDistance"),
|
|
Values("Fragment", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%f32arr2", "%f32arr4"),
|
|
Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceFragmentOutput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("ClipDistance", "CullDistance"), Values("Fragment"),
|
|
Values("Output"), Values("%f32arr4"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec doesn't allow BuiltIn ClipDistance/CullDistance "
|
|
"to be used for variables with Output storage class if "
|
|
"execution model is Fragment.",
|
|
"which is called with execution model Fragment."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
VertexIdAndInstanceIdVertexInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("VertexId", "InstanceId"), Values("Vertex"), Values("Input"),
|
|
Values("%u32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec doesn't allow BuiltIn VertexId/InstanceId to be "
|
|
"used."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceVertexInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("ClipDistance", "CullDistance"), Values("Vertex"),
|
|
Values("Input"), Values("%f32arr4"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec doesn't allow BuiltIn ClipDistance/CullDistance "
|
|
"to be used for variables with Input storage class if "
|
|
"execution model is Vertex.",
|
|
"which is called with execution model Vertex."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("ClipDistance", "CullDistance"), Values("GLCompute"),
|
|
Values("Input", "Output"), Values("%f32arr4"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Fragment, Vertex, TessellationControl, "
|
|
"TessellationEvaluation or Geometry execution models"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceNotArray,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("ClipDistance", "CullDistance"), Values("Fragment"),
|
|
Values("Input"), Values("%f32vec2", "%f32vec4", "%f32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float array",
|
|
"is not an array"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceNotFloatArray,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("ClipDistance", "CullDistance"), Values("Fragment"),
|
|
Values("Input"), Values("%u32arr2", "%u64arr4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float array",
|
|
"components are not float scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceNotF32Array,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("ClipDistance", "CullDistance"), Values("Fragment"),
|
|
Values("Input"), Values("%f64arr2", "%f64arr4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float array",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragCoordSuccess, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FragCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f32vec4"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragCoordNotFragment,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("FragCoord"),
|
|
Values("Vertex", "GLCompute", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%f32vec4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Fragment execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragCoordNotInput, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FragCoord"), Values("Fragment"), Values("Output"),
|
|
Values("%f32vec4"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Input storage class",
|
|
"uses storage class Output"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragCoordNotFloatVector,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FragCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f32arr4", "%u32vec4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float vector",
|
|
"is not a float vector"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragCoordNotFloatVec4,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FragCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float vector",
|
|
"has 3 components"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragCoordNotF32Vec4,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FragCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f64vec4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float vector",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragDepthSuccess, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FragDepth"), Values("Fragment"), Values("Output"),
|
|
Values("%f32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragDepthNotFragment,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("FragDepth"),
|
|
Values("Vertex", "GLCompute", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Output"), Values("%f32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Fragment execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragDepthNotOutput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FragDepth"), Values("Fragment"), Values("Input"),
|
|
Values("%f32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Output storage class",
|
|
"uses storage class Input"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragDepthNotFloatScalar,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FragDepth"), Values("Fragment"), Values("Output"),
|
|
Values("%f32vec4", "%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float scalar",
|
|
"is not a float scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FragDepthNotF32, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FragDepth"), Values("Fragment"), Values("Output"),
|
|
Values("%f64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float scalar",
|
|
"has bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FrontFacingAndHelperInvocationSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FrontFacing", "HelperInvocation"), Values("Fragment"),
|
|
Values("Input"), Values("%bool"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FrontFacingAndHelperInvocationNotFragment,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("FrontFacing", "HelperInvocation"),
|
|
Values("Vertex", "GLCompute", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%bool"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Fragment execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FrontFacingAndHelperInvocationNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FrontFacing", "HelperInvocation"), Values("Fragment"),
|
|
Values("Output"), Values("%bool"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Input storage class",
|
|
"uses storage class Output"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
FrontFacingAndHelperInvocationNotBool,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("FrontFacing", "HelperInvocation"), Values("Fragment"),
|
|
Values("Input"), Values("%f32", "%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a bool scalar",
|
|
"is not a bool scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ComputeShaderInputInt32Vec3Success,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("GlobalInvocationId", "LocalInvocationId", "NumWorkgroups",
|
|
"WorkgroupId"),
|
|
Values("GLCompute"), Values("Input"), Values("%u32vec3"),
|
|
Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ComputeShaderInputInt32Vec3NotGLCompute,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("GlobalInvocationId", "LocalInvocationId", "NumWorkgroups",
|
|
"WorkgroupId"),
|
|
Values("Vertex", "Fragment", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%u32vec3"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be used only with GLCompute execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ComputeShaderInputInt32Vec3NotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("GlobalInvocationId", "LocalInvocationId", "NumWorkgroups",
|
|
"WorkgroupId"),
|
|
Values("GLCompute"), Values("Output"), Values("%u32vec3"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Input storage class",
|
|
"uses storage class Output"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ComputeShaderInputInt32Vec3NotIntVector,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("GlobalInvocationId", "LocalInvocationId", "NumWorkgroups",
|
|
"WorkgroupId"),
|
|
Values("GLCompute"), Values("Input"),
|
|
Values("%u32arr3", "%f32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 3-component 32-bit int vector",
|
|
"is not an int vector"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ComputeShaderInputInt32Vec3NotIntVec3,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("GlobalInvocationId", "LocalInvocationId", "NumWorkgroups",
|
|
"WorkgroupId"),
|
|
Values("GLCompute"), Values("Input"), Values("%u32vec4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 3-component 32-bit int vector",
|
|
"has 4 components"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ComputeShaderInputInt32Vec3NotInt32Vec,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("GlobalInvocationId", "LocalInvocationId", "NumWorkgroups",
|
|
"WorkgroupId"),
|
|
Values("GLCompute"), Values("Input"), Values("%u64vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 3-component 32-bit int vector",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InvocationIdSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("InvocationId"), Values("Geometry", "TessellationControl"),
|
|
Values("Input"), Values("%u32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InvocationIdInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("InvocationId"),
|
|
Values("Vertex", "Fragment", "GLCompute", "TessellationEvaluation"),
|
|
Values("Input"), Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with TessellationControl or "
|
|
"Geometry execution models"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InvocationIdNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("InvocationId"), Values("Geometry", "TessellationControl"),
|
|
Values("Output"), Values("%u32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Input storage class",
|
|
"uses storage class Output"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InvocationIdNotIntScalar,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("InvocationId"), Values("Geometry", "TessellationControl"),
|
|
Values("Input"), Values("%f32", "%u32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"is not an int scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InvocationIdNotInt32,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("InvocationId"), Values("Geometry", "TessellationControl"),
|
|
Values("Input"), Values("%u64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"has bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InstanceIndexSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("InstanceIndex"), Values("Vertex"), Values("Input"),
|
|
Values("%u32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InstanceIndexInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("InstanceIndex"),
|
|
Values("Geometry", "Fragment", "GLCompute", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Vertex execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InstanceIndexNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("InstanceIndex"), Values("Vertex"), Values("Output"),
|
|
Values("%u32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Input storage class",
|
|
"uses storage class Output"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InstanceIndexNotIntScalar,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("InstanceIndex"), Values("Vertex"), Values("Input"),
|
|
Values("%f32", "%u32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"is not an int scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
InstanceIndexNotInt32,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("InstanceIndex"), Values("Vertex"), Values("Input"),
|
|
Values("%u64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"has bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
LayerAndViewportIndexInputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Layer", "ViewportIndex"), Values("Fragment"),
|
|
Values("Input"), Values("%u32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
LayerAndViewportIndexOutputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Layer", "ViewportIndex"), Values("Geometry"),
|
|
Values("Output"), Values("%u32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
LayerAndViewportIndexInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Layer", "ViewportIndex"),
|
|
Values("TessellationControl", "GLCompute"), Values("Input"),
|
|
Values("%u32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Vertex, TessellationEvaluation, "
|
|
"Geometry, or Fragment execution models"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
LayerAndViewportIndexExecutionModelEnabledByCapability,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Layer", "ViewportIndex"),
|
|
Values("Vertex", "TessellationEvaluation"), Values("Output"),
|
|
Values("%u32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"requires the ShaderViewportIndexLayerEXT capability"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
LayerAndViewportIndexFragmentNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("Layer", "ViewportIndex"), Values("Fragment"), Values("Output"),
|
|
Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"Output storage class if execution model is Fragment",
|
|
"which is called with execution model Fragment"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
LayerAndViewportIndexGeometryNotOutput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("Layer", "ViewportIndex"),
|
|
Values("Vertex", "TessellationEvaluation", "Geometry"), Values("Input"),
|
|
Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"Input storage class if execution model is Vertex, "
|
|
"TessellationEvaluation, or Geometry",
|
|
"which is called with execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
LayerAndViewportIndexNotIntScalar,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Layer", "ViewportIndex"), Values("Fragment"),
|
|
Values("Input"), Values("%f32", "%u32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"is not an int scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
LayerAndViewportIndexNotInt32,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Layer", "ViewportIndex"), Values("Fragment"),
|
|
Values("Input"), Values("%u64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"has bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PatchVerticesSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PatchVertices"),
|
|
Values("TessellationEvaluation", "TessellationControl"),
|
|
Values("Input"), Values("%u32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PatchVerticesInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PatchVertices"),
|
|
Values("Vertex", "Fragment", "GLCompute", "Geometry"),
|
|
Values("Input"), Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with TessellationControl or "
|
|
"TessellationEvaluation execution models"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PatchVerticesNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PatchVertices"),
|
|
Values("TessellationEvaluation", "TessellationControl"),
|
|
Values("Output"), Values("%u32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Input storage class",
|
|
"uses storage class Output"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PatchVerticesNotIntScalar,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PatchVertices"),
|
|
Values("TessellationEvaluation", "TessellationControl"),
|
|
Values("Input"), Values("%f32", "%u32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"is not an int scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PatchVerticesNotInt32,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PatchVertices"),
|
|
Values("TessellationEvaluation", "TessellationControl"),
|
|
Values("Input"), Values("%u64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"has bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointCoordSuccess, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f32vec2"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointCoordNotFragment,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("PointCoord"),
|
|
Values("Vertex", "GLCompute", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%f32vec2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Fragment execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointCoordNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointCoord"), Values("Fragment"), Values("Output"),
|
|
Values("%f32vec2"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Input storage class",
|
|
"uses storage class Output"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointCoordNotFloatVector,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f32arr2", "%u32vec2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float vector",
|
|
"is not a float vector"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointCoordNotFloatVec3,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float vector",
|
|
"has 3 components"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointCoordNotF32Vec4,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f64vec2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float vector",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointSizeOutputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointSize"),
|
|
Values("Vertex", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Output"), Values("%f32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointSizeInputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointSize"),
|
|
Values("Geometry", "TessellationControl", "TessellationEvaluation"),
|
|
Values("Input"), Values("%f32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointSizeVertexInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointSize"), Values("Vertex"), Values("Input"),
|
|
Values("%f32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec doesn't allow BuiltIn PointSize "
|
|
"to be used for variables with Input storage class if "
|
|
"execution model is Vertex.",
|
|
"which is called with execution model Vertex."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointSizeInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointSize"), Values("GLCompute", "Fragment"),
|
|
Values("Input", "Output"), Values("%f32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Vertex, TessellationControl, "
|
|
"TessellationEvaluation or Geometry execution models"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointSizeNotFloatScalar,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointSize"), Values("Vertex"), Values("Output"),
|
|
Values("%f32vec4", "%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float scalar",
|
|
"is not a float scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointSizeNotF32, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PointSize"), Values("Vertex"), Values("Output"),
|
|
Values("%f64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float scalar",
|
|
"has bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PositionOutputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Position"),
|
|
Values("Vertex", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Output"), Values("%f32vec4"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PositionInputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Position"),
|
|
Values("Geometry", "TessellationControl", "TessellationEvaluation"),
|
|
Values("Input"), Values("%f32vec4"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PositionVertexInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Position"), Values("Vertex"), Values("Input"),
|
|
Values("%f32vec4"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec doesn't allow BuiltIn Position "
|
|
"to be used for variables with Input storage class if "
|
|
"execution model is Vertex.",
|
|
"which is called with execution model Vertex."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PositionInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Position"), Values("GLCompute", "Fragment"),
|
|
Values("Input", "Output"), Values("%f32vec4"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Vertex, TessellationControl, "
|
|
"TessellationEvaluation or Geometry execution models"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PositionNotFloatVector,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Position"), Values("Geometry"), Values("Input"),
|
|
Values("%f32arr4", "%u32vec4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float vector",
|
|
"is not a float vector"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PositionNotFloatVec4,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Position"), Values("Geometry"), Values("Input"),
|
|
Values("%f32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float vector",
|
|
"has 3 components"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PositionNotF32Vec4,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("Position"), Values("Geometry"), Values("Input"),
|
|
Values("%f64vec4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float vector",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PrimitiveIdInputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PrimitiveId"),
|
|
Values("Fragment", "TessellationControl", "TessellationEvaluation",
|
|
"Geometry"),
|
|
Values("Input"), Values("%u32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PrimitiveIdOutputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PrimitiveId"), Values("Geometry"), Values("Output"),
|
|
Values("%u32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PrimitiveIdInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PrimitiveId"), Values("Vertex", "GLCompute"),
|
|
Values("Input"), Values("%u32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Fragment, TessellationControl, "
|
|
"TessellationEvaluation or Geometry execution models"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PrimitiveIdFragmentNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("PrimitiveId"), Values("Fragment"), Values("Output"),
|
|
Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"Output storage class if execution model is Fragment",
|
|
"which is called with execution model Fragment"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PrimitiveIdGeometryNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PrimitiveId"),
|
|
Values("TessellationControl", "TessellationEvaluation"),
|
|
Values("Output"), Values("%u32"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Output storage class if execution model is Tessellation",
|
|
"which is called with execution model Tessellation"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PrimitiveIdNotIntScalar,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PrimitiveId"), Values("Fragment"), Values("Input"),
|
|
Values("%f32", "%u32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"is not an int scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PrimitiveIdNotInt32,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("PrimitiveId"), Values("Fragment"), Values("Input"),
|
|
Values("%u64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"has bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleIdSuccess, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SampleId"), Values("Fragment"), Values("Input"),
|
|
Values("%u32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleIdInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("SampleId"),
|
|
Values("Vertex", "GLCompute", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Fragment execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleIdNotInput, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("SampleId"), Values("Fragment"), Values("Output"),
|
|
Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec allows BuiltIn SampleId to be only used "
|
|
"for variables with Input storage class"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleIdNotIntScalar,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SampleId"), Values("Fragment"), Values("Input"),
|
|
Values("%f32", "%u32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"is not an int scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleIdNotInt32, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SampleId"), Values("Fragment"), Values("Input"),
|
|
Values("%u64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"has bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleMaskSuccess, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SampleMask"), Values("Fragment"), Values("Input", "Output"),
|
|
Values("%u32arr2", "%u32arr4"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleMaskInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("SampleMask"),
|
|
Values("Vertex", "GLCompute", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%u32arr2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Fragment execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleMaskWrongStorageClass,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SampleMask"), Values("Fragment"), Values("Workgroup"),
|
|
Values("%u32arr2"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec allows BuiltIn SampleMask to be only used for "
|
|
"variables with Input or Output storage class"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleMaskNotArray,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SampleMask"), Values("Fragment"), Values("Input"),
|
|
Values("%f32", "%u32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int array",
|
|
"is not an array"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleMaskNotIntArray,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SampleMask"), Values("Fragment"), Values("Input"),
|
|
Values("%f32arr2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int array",
|
|
"components are not int scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SampleMaskNotInt32Array,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SampleMask"), Values("Fragment"), Values("Input"),
|
|
Values("%u64arr2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int array",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SamplePositionSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SamplePosition"), Values("Fragment"), Values("Input"),
|
|
Values("%f32vec2"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SamplePositionNotFragment,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("SamplePosition"),
|
|
Values("Vertex", "GLCompute", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%f32vec2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Fragment execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SamplePositionNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SamplePosition"), Values("Fragment"), Values("Output"),
|
|
Values("%f32vec2"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Input storage class",
|
|
"uses storage class Output"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SamplePositionNotFloatVector,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SamplePosition"), Values("Fragment"), Values("Input"),
|
|
Values("%f32arr2", "%u32vec4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float vector",
|
|
"is not a float vector"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SamplePositionNotFloatVec2,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SamplePosition"), Values("Fragment"), Values("Input"),
|
|
Values("%f32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float vector",
|
|
"has 3 components"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
SamplePositionNotF32Vec2,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("SamplePosition"), Values("Fragment"), Values("Input"),
|
|
Values("%f64vec2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float vector",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessCoordSuccess, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessCoord"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%f32vec3"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessCoordNotFragment,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("TessCoord"),
|
|
Values("Vertex", "GLCompute", "Geometry", "TessellationControl",
|
|
"Fragment"),
|
|
Values("Input"), Values("%f32vec3"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be used only with TessellationEvaluation execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessCoordNotInput, ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessCoord"), Values("Fragment"), Values("Output"),
|
|
Values("%f32vec3"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"to be only used for variables with Input storage class",
|
|
"uses storage class Output"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessCoordNotFloatVector,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f32arr3", "%u32vec4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 3-component 32-bit float vector",
|
|
"is not a float vector"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessCoordNotFloatVec3,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f32vec2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 3-component 32-bit float vector",
|
|
"has 2 components"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessCoordNotF32Vec3,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessCoord"), Values("Fragment"), Values("Input"),
|
|
Values("%f64vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 3-component 32-bit float vector",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelOuterTeseInputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelOuter"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%f32arr4"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelOuterTescOutputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelOuter"), Values("TessellationControl"),
|
|
Values("Output"), Values("%f32arr4"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelOuterInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelOuter"),
|
|
Values("Vertex", "GLCompute", "Geometry", "Fragment"),
|
|
Values("Input"), Values("%f32arr4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with TessellationControl or "
|
|
"TessellationEvaluation execution models."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelOuterOutputTese,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelOuter"), Values("TessellationEvaluation"),
|
|
Values("Output"), Values("%f32arr4"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec doesn't allow TessLevelOuter/TessLevelInner to be "
|
|
"used for variables with Output storage class if execution "
|
|
"model is TessellationEvaluation."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelOuterInputTesc,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelOuter"), Values("TessellationControl"),
|
|
Values("Input"), Values("%f32arr4"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec doesn't allow TessLevelOuter/TessLevelInner to be "
|
|
"used for variables with Input storage class if execution "
|
|
"model is TessellationControl."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelOuterNotArray,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelOuter"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%f32vec4", "%f32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float array",
|
|
"is not an array"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelOuterNotFloatArray,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelOuter"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%u32arr4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float array",
|
|
"components are not float scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelOuterNotFloatArr4,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelOuter"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%f32arr3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float array",
|
|
"has 3 components"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelOuterNotF32Arr4,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelOuter"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%f64arr4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float array",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelInnerTeseInputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelInner"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%f32arr2"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelInnerTescOutputSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelInner"), Values("TessellationControl"),
|
|
Values("Output"), Values("%f32arr2"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelInnerInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelInner"),
|
|
Values("Vertex", "GLCompute", "Geometry", "Fragment"),
|
|
Values("Input"), Values("%f32arr2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with TessellationControl or "
|
|
"TessellationEvaluation execution models."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelInnerOutputTese,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelInner"), Values("TessellationEvaluation"),
|
|
Values("Output"), Values("%f32arr2"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec doesn't allow TessLevelOuter/TessLevelInner to be "
|
|
"used for variables with Output storage class if execution "
|
|
"model is TessellationEvaluation."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelInnerInputTesc,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelInner"), Values("TessellationControl"),
|
|
Values("Input"), Values("%f32arr2"),
|
|
Values(TestResult(
|
|
SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec doesn't allow TessLevelOuter/TessLevelInner to be "
|
|
"used for variables with Input storage class if execution "
|
|
"model is TessellationControl."))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelInnerNotArray,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelInner"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%f32vec2", "%f32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float array",
|
|
"is not an array"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelInnerNotFloatArray,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelInner"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%u32arr2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float array",
|
|
"components are not float scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelInnerNotFloatArr2,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelInner"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%f32arr3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float array",
|
|
"has 3 components"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
TessLevelInnerNotF32Arr2,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("TessLevelInner"), Values("TessellationEvaluation"),
|
|
Values("Input"), Values("%f64arr2"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 2-component 32-bit float array",
|
|
"has components with bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
VertexIndexSuccess,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("VertexIndex"), Values("Vertex"), Values("Input"),
|
|
Values("%u32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
VertexIndexInvalidExecutionModel,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("VertexIndex"),
|
|
Values("Fragment", "GLCompute", "Geometry", "TessellationControl",
|
|
"TessellationEvaluation"),
|
|
Values("Input"), Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"to be used only with Vertex execution model"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
VertexIndexNotInput,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(
|
|
Values("VertexIndex"), Values("Vertex"), Values("Output"),
|
|
Values("%u32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"Vulkan spec allows BuiltIn VertexIndex to be only "
|
|
"used for variables with Input storage class"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
VertexIndexNotIntScalar,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("VertexIndex"), Values("Vertex"), Values("Input"),
|
|
Values("%f32", "%u32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"is not an int scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
VertexIndexNotInt32,
|
|
ValidateVulkanCombineBuiltInExecutionModelDataTypeResult,
|
|
Combine(Values("VertexIndex"), Values("Vertex"), Values("Input"),
|
|
Values("%u64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit int scalar",
|
|
"has bit width 64"))), );
|
|
|
|
TEST_P(ValidateVulkanCombineBuiltInArrayedVariable, Variable) {
|
|
const char* const built_in = std::get<0>(GetParam());
|
|
const char* const execution_model = std::get<1>(GetParam());
|
|
const char* const storage_class = std::get<2>(GetParam());
|
|
const char* const data_type = std::get<3>(GetParam());
|
|
const TestResult& test_result = std::get<4>(GetParam());
|
|
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = "OpDecorate %built_in_var BuiltIn ";
|
|
generator.before_types_ += built_in;
|
|
generator.before_types_ += "\n";
|
|
|
|
std::ostringstream after_types;
|
|
after_types << "%built_in_array = OpTypeArray " << data_type << " %u32_3\n";
|
|
after_types << "%built_in_ptr = OpTypePointer " << storage_class
|
|
<< " %built_in_array\n";
|
|
after_types << "%built_in_var = OpVariable %built_in_ptr " << storage_class
|
|
<< "\n";
|
|
generator.after_types_ = after_types.str();
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = execution_model;
|
|
entry_point.interfaces = "%built_in_var";
|
|
// Any kind of reference would do.
|
|
entry_point.body = R"(
|
|
%val = OpBitcast %u64 %built_in_var
|
|
)";
|
|
|
|
std::ostringstream execution_modes;
|
|
if (0 == std::strcmp(execution_model, "Fragment")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " OriginUpperLeft\n";
|
|
if (0 == std::strcmp(built_in, "FragDepth")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " DepthReplacing\n";
|
|
}
|
|
}
|
|
if (0 == std::strcmp(execution_model, "Geometry")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " InputPoints\n";
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " OutputPoints\n";
|
|
}
|
|
if (0 == std::strcmp(execution_model, "GLCompute")) {
|
|
execution_modes << "OpExecutionMode %" << entry_point.name
|
|
<< " LocalSize 1 1 1\n";
|
|
}
|
|
entry_point.execution_modes = execution_modes.str();
|
|
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(test_result.validation_result,
|
|
ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
if (test_result.error_str) {
|
|
EXPECT_THAT(getDiagnosticString(), HasSubstr(test_result.error_str));
|
|
}
|
|
if (test_result.error_str2) {
|
|
EXPECT_THAT(getDiagnosticString(), HasSubstr(test_result.error_str2));
|
|
}
|
|
}
|
|
|
|
INSTANTIATE_TEST_CASE_P(PointSizeArrayedF32TessControl,
|
|
ValidateVulkanCombineBuiltInArrayedVariable,
|
|
Combine(Values("PointSize"),
|
|
Values("TessellationControl"), Values("Input"),
|
|
Values("%f32"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointSizeArrayedF64TessControl, ValidateVulkanCombineBuiltInArrayedVariable,
|
|
Combine(Values("PointSize"), Values("TessellationControl"), Values("Input"),
|
|
Values("%f64"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float scalar",
|
|
"has bit width 64"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PointSizeArrayedF32Vertex, ValidateVulkanCombineBuiltInArrayedVariable,
|
|
Combine(Values("PointSize"), Values("Vertex"), Values("Output"),
|
|
Values("%f32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float scalar",
|
|
"is not a float scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(PositionArrayedF32Vec4TessControl,
|
|
ValidateVulkanCombineBuiltInArrayedVariable,
|
|
Combine(Values("Position"),
|
|
Values("TessellationControl"), Values("Input"),
|
|
Values("%f32vec4"), Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PositionArrayedF32Vec3TessControl,
|
|
ValidateVulkanCombineBuiltInArrayedVariable,
|
|
Combine(Values("Position"), Values("TessellationControl"), Values("Input"),
|
|
Values("%f32vec3"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float vector",
|
|
"has 3 components"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
PositionArrayedF32Vec4Vertex, ValidateVulkanCombineBuiltInArrayedVariable,
|
|
Combine(Values("Position"), Values("Vertex"), Values("Output"),
|
|
Values("%f32"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 4-component 32-bit float vector",
|
|
"is not a float vector"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceOutputSuccess,
|
|
ValidateVulkanCombineBuiltInArrayedVariable,
|
|
Combine(Values("ClipDistance", "CullDistance"),
|
|
Values("Geometry", "TessellationControl", "TessellationEvaluation"),
|
|
Values("Output"), Values("%f32arr2", "%f32arr4"),
|
|
Values(TestResult())), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceVertexInput, ValidateVulkanCombineBuiltInArrayedVariable,
|
|
Combine(Values("ClipDistance", "CullDistance"), Values("Fragment"),
|
|
Values("Input"), Values("%f32arr4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float array",
|
|
"components are not float scalar"))), );
|
|
|
|
INSTANTIATE_TEST_CASE_P(
|
|
ClipAndCullDistanceNotArray, ValidateVulkanCombineBuiltInArrayedVariable,
|
|
Combine(Values("ClipDistance", "CullDistance"),
|
|
Values("Geometry", "TessellationControl", "TessellationEvaluation"),
|
|
Values("Input"), Values("%f32vec2", "%f32vec4"),
|
|
Values(TestResult(SPV_ERROR_INVALID_DATA,
|
|
"needs to be a 32-bit float array",
|
|
"components are not float scalar"))), );
|
|
|
|
TEST_F(ValidateBuiltIns, WorkgroupSizeSuccess) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %workgroup_size BuiltIn WorkgroupSize
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%workgroup_size = OpConstantComposite %u32vec3 %u32_1 %u32_1 %u32_1
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "GLCompute";
|
|
entry_point.body = R"(
|
|
%copy = OpCopyObject %u32vec3 %workgroup_size
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, WorkgroupSizeFragment) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %workgroup_size BuiltIn WorkgroupSize
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%workgroup_size = OpConstantComposite %u32vec3 %u32_1 %u32_1 %u32_1
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "Fragment";
|
|
entry_point.execution_modes = "OpExecutionMode %main OriginUpperLeft";
|
|
entry_point.body = R"(
|
|
%copy = OpCopyObject %u32vec3 %workgroup_size
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Vulkan spec allows BuiltIn WorkgroupSize to be used "
|
|
"only with GLCompute execution model"));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("is referencing ID <2> (OpConstantComposite) which is "
|
|
"decorated with BuiltIn WorkgroupSize in function <1> "
|
|
"called with execution model Fragment"));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, WorkgroupSizeNotConstant) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %copy BuiltIn WorkgroupSize
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%workgroup_size = OpConstantComposite %u32vec3 %u32_1 %u32_1 %u32_1
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "GLCompute";
|
|
entry_point.body = R"(
|
|
%copy = OpCopyObject %u32vec3 %workgroup_size
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Vulkan spec requires BuiltIn WorkgroupSize to be a "
|
|
"constant. ID <2> (OpCopyObject) is not a constant"));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, WorkgroupSizeNotVector) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %workgroup_size BuiltIn WorkgroupSize
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%workgroup_size = OpConstant %u32 16
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "GLCompute";
|
|
entry_point.body = R"(
|
|
%copy = OpCopyObject %u32 %workgroup_size
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("According to the Vulkan spec BuiltIn WorkgroupSize "
|
|
"variable needs to be a 3-component 32-bit int vector. "
|
|
"ID <2> (OpConstant) is not an int vector."));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, WorkgroupSizeNotIntVector) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %workgroup_size BuiltIn WorkgroupSize
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%workgroup_size = OpConstantComposite %f32vec3 %f32_1 %f32_1 %f32_1
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "GLCompute";
|
|
entry_point.body = R"(
|
|
%copy = OpCopyObject %f32vec3 %workgroup_size
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("According to the Vulkan spec BuiltIn WorkgroupSize "
|
|
"variable needs to be a 3-component 32-bit int vector. "
|
|
"ID <2> (OpConstantComposite) is not an int vector."));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, WorkgroupSizeNotVec3) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %workgroup_size BuiltIn WorkgroupSize
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%workgroup_size = OpConstantComposite %u32vec2 %u32_1 %u32_1
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "GLCompute";
|
|
entry_point.body = R"(
|
|
%copy = OpCopyObject %u32vec2 %workgroup_size
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("According to the Vulkan spec BuiltIn WorkgroupSize "
|
|
"variable needs to be a 3-component 32-bit int vector. "
|
|
"ID <2> (OpConstantComposite) has 2 components."));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, WorkgroupSizeNotInt32Vec) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %workgroup_size BuiltIn WorkgroupSize
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%workgroup_size = OpConstantComposite %u64vec3 %u64_1 %u64_1 %u64_1
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "GLCompute";
|
|
entry_point.body = R"(
|
|
%copy = OpCopyObject %u64vec3 %workgroup_size
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("According to the Vulkan spec BuiltIn WorkgroupSize variable "
|
|
"needs to be a 3-component 32-bit int vector. ID <2> "
|
|
"(OpConstantComposite) has components with bit width 64."));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, WorkgroupSizePrivateVar) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %workgroup_size BuiltIn WorkgroupSize
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%workgroup_size = OpConstantComposite %u32vec3 %u32_1 %u32_1 %u32_1
|
|
%private_ptr_u32vec3 = OpTypePointer Private %u32vec3
|
|
%var = OpVariable %private_ptr_u32vec3 Private %workgroup_size
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "GLCompute";
|
|
entry_point.body = R"(
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, GeometryPositionInOutSuccess) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
|
|
generator.before_types_ = R"(
|
|
OpMemberDecorate %input_type 0 BuiltIn Position
|
|
OpMemberDecorate %output_type 0 BuiltIn Position
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%input_type = OpTypeStruct %f32vec4
|
|
%arrayed_input_type = OpTypeArray %input_type %u32_3
|
|
%input_ptr = OpTypePointer Input %arrayed_input_type
|
|
%input = OpVariable %input_ptr Input
|
|
%input_f32vec4_ptr = OpTypePointer Input %f32vec4
|
|
%output_type = OpTypeStruct %f32vec4
|
|
%arrayed_output_type = OpTypeArray %output_type %u32_3
|
|
%output_ptr = OpTypePointer Output %arrayed_output_type
|
|
%output = OpVariable %output_ptr Output
|
|
%output_f32vec4_ptr = OpTypePointer Output %f32vec4
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "Geometry";
|
|
entry_point.interfaces = "%input %output";
|
|
entry_point.body = R"(
|
|
%input_pos = OpAccessChain %input_f32vec4_ptr %input %u32_0 %u32_0
|
|
%output_pos = OpAccessChain %output_f32vec4_ptr %output %u32_0 %u32_0
|
|
%pos = OpLoad %f32vec4 %input_pos
|
|
OpStore %output_pos %pos
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
generator.entry_points_[0].execution_modes =
|
|
"OpExecutionMode %main InputPoints\nOpExecutionMode %main OutputPoints\n";
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, WorkgroupIdNotVec3) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %workgroup_size BuiltIn WorkgroupSize
|
|
OpDecorate %workgroup_id BuiltIn WorkgroupId
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%workgroup_size = OpConstantComposite %u32vec3 %u32_1 %u32_1 %u32_1
|
|
%input_ptr = OpTypePointer Input %u32vec2
|
|
%workgroup_id = OpVariable %input_ptr Input
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "GLCompute";
|
|
entry_point.interfaces = "%workgroup_id";
|
|
entry_point.body = R"(
|
|
%copy_size = OpCopyObject %u32vec3 %workgroup_size
|
|
%load_id = OpLoad %u32vec2 %workgroup_id
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("According to the Vulkan spec BuiltIn WorkgroupId "
|
|
"variable needs to be a 3-component 32-bit int vector. "
|
|
"ID <2> (OpVariable) has 2 components."));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, TwoBuiltInsFirstFails) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
|
|
generator.before_types_ = R"(
|
|
OpMemberDecorate %input_type 0 BuiltIn FragCoord
|
|
OpMemberDecorate %output_type 0 BuiltIn Position
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%input_type = OpTypeStruct %f32vec4
|
|
%input_ptr = OpTypePointer Input %input_type
|
|
%input = OpVariable %input_ptr Input
|
|
%input_f32vec4_ptr = OpTypePointer Input %f32vec4
|
|
%output_type = OpTypeStruct %f32vec4
|
|
%output_ptr = OpTypePointer Output %output_type
|
|
%output = OpVariable %output_ptr Output
|
|
%output_f32vec4_ptr = OpTypePointer Output %f32vec4
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "Geometry";
|
|
entry_point.interfaces = "%input %output";
|
|
entry_point.body = R"(
|
|
%input_pos = OpAccessChain %input_f32vec4_ptr %input %u32_0
|
|
%output_pos = OpAccessChain %output_f32vec4_ptr %output %u32_0
|
|
%pos = OpLoad %f32vec4 %input_pos
|
|
OpStore %output_pos %pos
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
generator.entry_points_[0].execution_modes =
|
|
"OpExecutionMode %main InputPoints\nOpExecutionMode %main OutputPoints\n";
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Vulkan spec allows BuiltIn FragCoord to be used only "
|
|
"with Fragment execution model"));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, TwoBuiltInsSecondFails) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
|
|
generator.before_types_ = R"(
|
|
OpMemberDecorate %input_type 0 BuiltIn Position
|
|
OpMemberDecorate %output_type 0 BuiltIn FragCoord
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%input_type = OpTypeStruct %f32vec4
|
|
%input_ptr = OpTypePointer Input %input_type
|
|
%input = OpVariable %input_ptr Input
|
|
%input_f32vec4_ptr = OpTypePointer Input %f32vec4
|
|
%output_type = OpTypeStruct %f32vec4
|
|
%output_ptr = OpTypePointer Output %output_type
|
|
%output = OpVariable %output_ptr Output
|
|
%output_f32vec4_ptr = OpTypePointer Output %f32vec4
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "Geometry";
|
|
entry_point.interfaces = "%input %output";
|
|
entry_point.body = R"(
|
|
%input_pos = OpAccessChain %input_f32vec4_ptr %input %u32_0
|
|
%output_pos = OpAccessChain %output_f32vec4_ptr %output %u32_0
|
|
%pos = OpLoad %f32vec4 %input_pos
|
|
OpStore %output_pos %pos
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
generator.entry_points_[0].execution_modes =
|
|
"OpExecutionMode %main InputPoints\nOpExecutionMode %main OutputPoints\n";
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Vulkan spec allows BuiltIn FragCoord to be only used "
|
|
"for variables with Input storage class"));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, VertexPositionVariableSuccess) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpDecorate %position BuiltIn Position
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%f32vec4_ptr_output = OpTypePointer Output %f32vec4
|
|
%position = OpVariable %f32vec4_ptr_output Output
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "Vertex";
|
|
entry_point.interfaces = "%position";
|
|
entry_point.body = R"(
|
|
OpStore %position %f32vec4_0123
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, FragmentPositionTwoEntryPoints) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpMemberDecorate %output_type 0 BuiltIn Position
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%output_type = OpTypeStruct %f32vec4
|
|
%output_ptr = OpTypePointer Output %output_type
|
|
%output = OpVariable %output_ptr Output
|
|
%output_f32vec4_ptr = OpTypePointer Output %f32vec4
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "vmain";
|
|
entry_point.execution_model = "Vertex";
|
|
entry_point.interfaces = "%output";
|
|
entry_point.body = R"(
|
|
%val1 = OpFunctionCall %void %foo
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
entry_point.name = "fmain";
|
|
entry_point.execution_model = "Fragment";
|
|
entry_point.interfaces = "%output";
|
|
entry_point.execution_modes = "OpExecutionMode %fmain OriginUpperLeft";
|
|
entry_point.body = R"(
|
|
%val2 = OpFunctionCall %void %foo
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
generator.add_at_the_end_ = R"(
|
|
%foo = OpFunction %void None %func
|
|
%foo_entry = OpLabel
|
|
%position = OpAccessChain %output_f32vec4_ptr %output %u32_0
|
|
OpStore %position %f32vec4_0123
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Vulkan spec allows BuiltIn Position to be used only "
|
|
"with Vertex, TessellationControl, "
|
|
"TessellationEvaluation or Geometry execution models"));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("called with execution model Fragment"));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, FragmentFragDepthNoDepthReplacing) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpMemberDecorate %output_type 0 BuiltIn FragDepth
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%output_type = OpTypeStruct %f32
|
|
%output_ptr = OpTypePointer Output %output_type
|
|
%output = OpVariable %output_ptr Output
|
|
%output_f32_ptr = OpTypePointer Output %f32
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main";
|
|
entry_point.execution_model = "Fragment";
|
|
entry_point.interfaces = "%output";
|
|
entry_point.execution_modes = "OpExecutionMode %main OriginUpperLeft";
|
|
entry_point.body = R"(
|
|
%val2 = OpFunctionCall %void %foo
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
generator.add_at_the_end_ = R"(
|
|
%foo = OpFunction %void None %func
|
|
%foo_entry = OpLabel
|
|
%frag_depth = OpAccessChain %output_f32_ptr %output %u32_0
|
|
OpStore %frag_depth %f32_1
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Vulkan spec requires DepthReplacing execution mode to "
|
|
"be declared when using BuiltIn FragDepth"));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, FragmentFragDepthOneMainHasDepthReplacingOtherHasnt) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.before_types_ = R"(
|
|
OpMemberDecorate %output_type 0 BuiltIn FragDepth
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%output_type = OpTypeStruct %f32
|
|
%output_ptr = OpTypePointer Output %output_type
|
|
%output = OpVariable %output_ptr Output
|
|
%output_f32_ptr = OpTypePointer Output %f32
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main_d_r";
|
|
entry_point.execution_model = "Fragment";
|
|
entry_point.interfaces = "%output";
|
|
entry_point.execution_modes =
|
|
"OpExecutionMode %main_d_r OriginUpperLeft\n"
|
|
"OpExecutionMode %main_d_r DepthReplacing";
|
|
entry_point.body = R"(
|
|
%val2 = OpFunctionCall %void %foo
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
entry_point.name = "main_no_d_r";
|
|
entry_point.execution_model = "Fragment";
|
|
entry_point.interfaces = "%output";
|
|
entry_point.execution_modes = "OpExecutionMode %main_no_d_r OriginUpperLeft";
|
|
entry_point.body = R"(
|
|
%val3 = OpFunctionCall %void %foo
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
generator.add_at_the_end_ = R"(
|
|
%foo = OpFunction %void None %func
|
|
%foo_entry = OpLabel
|
|
%frag_depth = OpAccessChain %output_f32_ptr %output %u32_0
|
|
OpStore %frag_depth %f32_1
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Vulkan spec requires DepthReplacing execution mode to "
|
|
"be declared when using BuiltIn FragDepth"));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, AllowInstanceIdWithIntersectionShader) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.capabilities_ += R"(
|
|
OpCapability RayTracingNV
|
|
)";
|
|
|
|
generator.extensions_ = R"(
|
|
OpExtension "SPV_NV_ray_tracing"
|
|
)";
|
|
|
|
generator.before_types_ = R"(
|
|
OpMemberDecorate %input_type 0 BuiltIn InstanceId
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%input_type = OpTypeStruct %u32
|
|
%input_ptr = OpTypePointer Input %input_type
|
|
%input = OpVariable %input_ptr Input
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main_d_r";
|
|
entry_point.execution_model = "IntersectionNV";
|
|
entry_point.interfaces = "%input";
|
|
entry_point.body = R"(
|
|
%val2 = OpFunctionCall %void %foo
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
generator.add_at_the_end_ = R"(
|
|
%foo = OpFunction %void None %func
|
|
%foo_entry = OpLabel
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
EXPECT_THAT(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
}
|
|
|
|
TEST_F(ValidateBuiltIns, DisallowInstanceIdWithRayGenShader) {
|
|
CodeGenerator generator = GetDefaultShaderCodeGenerator();
|
|
generator.capabilities_ += R"(
|
|
OpCapability RayTracingNV
|
|
)";
|
|
|
|
generator.extensions_ = R"(
|
|
OpExtension "SPV_NV_ray_tracing"
|
|
)";
|
|
|
|
generator.before_types_ = R"(
|
|
OpMemberDecorate %input_type 0 BuiltIn InstanceId
|
|
)";
|
|
|
|
generator.after_types_ = R"(
|
|
%input_type = OpTypeStruct %u32
|
|
%input_ptr = OpTypePointer Input %input_type
|
|
%input_ptr_u32 = OpTypePointer Input %u32
|
|
%input = OpVariable %input_ptr Input
|
|
)";
|
|
|
|
EntryPoint entry_point;
|
|
entry_point.name = "main_d_r";
|
|
entry_point.execution_model = "RayGenerationNV";
|
|
entry_point.interfaces = "%input";
|
|
entry_point.body = R"(
|
|
%input_member = OpAccessChain %input_ptr_u32 %input %u32_0
|
|
)";
|
|
generator.entry_points_.push_back(std::move(entry_point));
|
|
|
|
CompileSuccessfully(generator.Build(), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Vulkan spec allows BuiltIn InstanceId to be used "
|
|
"only with IntersectionNV, ClosestHitNV and "
|
|
"AnyHitNV execution models"));
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace val
|
|
} // namespace spvtools
|