mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2025-01-13 18:00:05 +00:00
1287 lines
42 KiB
C++
1287 lines
42 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.
|
|
|
|
#include <sstream>
|
|
#include <string>
|
|
|
|
#include "gmock/gmock.h"
|
|
#include "test/unit_spirv.h"
|
|
#include "test/val/val_fixtures.h"
|
|
|
|
namespace spvtools {
|
|
namespace val {
|
|
namespace {
|
|
|
|
using ::testing::HasSubstr;
|
|
using ::testing::Not;
|
|
|
|
using ValidateBarriers = spvtest::ValidateBase<bool>;
|
|
|
|
std::string GenerateShaderCodeImpl(
|
|
const std::string& body, const std::string& capabilities_and_extensions,
|
|
const std::string& definitions, const std::string& execution_model,
|
|
const std::string& memory_model) {
|
|
std::ostringstream ss;
|
|
ss << R"(
|
|
OpCapability Shader
|
|
)";
|
|
|
|
ss << capabilities_and_extensions;
|
|
ss << memory_model << std::endl;
|
|
ss << "OpEntryPoint " << execution_model << " %main \"main\"\n";
|
|
if (execution_model == "Fragment") {
|
|
ss << "OpExecutionMode %main OriginUpperLeft\n";
|
|
} else if (execution_model == "Geometry") {
|
|
ss << "OpExecutionMode %main InputPoints\n";
|
|
ss << "OpExecutionMode %main OutputPoints\n";
|
|
} else if (execution_model == "GLCompute") {
|
|
ss << "OpExecutionMode %main LocalSize 1 1 1\n";
|
|
}
|
|
|
|
ss << R"(
|
|
%void = OpTypeVoid
|
|
%func = OpTypeFunction %void
|
|
%bool = OpTypeBool
|
|
%f32 = OpTypeFloat 32
|
|
%u32 = OpTypeInt 32 0
|
|
|
|
%f32_0 = OpConstant %f32 0
|
|
%f32_1 = OpConstant %f32 1
|
|
%u32_0 = OpConstant %u32 0
|
|
%u32_1 = OpConstant %u32 1
|
|
%u32_4 = OpConstant %u32 4
|
|
)";
|
|
ss << definitions;
|
|
ss << R"(
|
|
%cross_device = OpConstant %u32 0
|
|
%device = OpConstant %u32 1
|
|
%workgroup = OpConstant %u32 2
|
|
%subgroup = OpConstant %u32 3
|
|
%invocation = OpConstant %u32 4
|
|
%queuefamily = OpConstant %u32 5
|
|
|
|
%none = OpConstant %u32 0
|
|
%acquire = OpConstant %u32 2
|
|
%release = OpConstant %u32 4
|
|
%acquire_release = OpConstant %u32 8
|
|
%acquire_and_release = OpConstant %u32 6
|
|
%sequentially_consistent = OpConstant %u32 16
|
|
%acquire_release_uniform_workgroup = OpConstant %u32 328
|
|
%acquire_and_release_uniform = OpConstant %u32 70
|
|
%acquire_release_subgroup = OpConstant %u32 136
|
|
%uniform = OpConstant %u32 64
|
|
|
|
%main = OpFunction %void None %func
|
|
%main_entry = OpLabel
|
|
)";
|
|
|
|
ss << body;
|
|
|
|
ss << R"(
|
|
OpReturn
|
|
OpFunctionEnd)";
|
|
|
|
return ss.str();
|
|
}
|
|
|
|
std::string GenerateShaderCode(
|
|
const std::string& body,
|
|
const std::string& capabilities_and_extensions = "",
|
|
const std::string& execution_model = "GLCompute") {
|
|
const std::string int64_capability = R"(
|
|
OpCapability Int64
|
|
)";
|
|
const std::string int64_declarations = R"(
|
|
%u64 = OpTypeInt 64 0
|
|
%u64_0 = OpConstant %u64 0
|
|
%u64_1 = OpConstant %u64 1
|
|
)";
|
|
const std::string memory_model = "OpMemoryModel Logical GLSL450";
|
|
return GenerateShaderCodeImpl(
|
|
body, int64_capability + capabilities_and_extensions, int64_declarations,
|
|
execution_model, memory_model);
|
|
}
|
|
|
|
std::string GenerateWebGPUShaderCode(
|
|
const std::string& body,
|
|
const std::string& capabilities_and_extensions = "",
|
|
const std::string& execution_model = "GLCompute") {
|
|
const std::string vulkan_memory_capability = R"(
|
|
OpCapability VulkanMemoryModelKHR
|
|
)";
|
|
const std::string vulkan_memory_extension = R"(
|
|
OpExtension "SPV_KHR_vulkan_memory_model"
|
|
)";
|
|
const std::string memory_model = "OpMemoryModel Logical VulkanKHR";
|
|
return GenerateShaderCodeImpl(body,
|
|
vulkan_memory_capability +
|
|
capabilities_and_extensions +
|
|
vulkan_memory_extension,
|
|
"", execution_model, memory_model);
|
|
}
|
|
|
|
std::string GenerateKernelCode(
|
|
const std::string& body,
|
|
const std::string& capabilities_and_extensions = "") {
|
|
std::ostringstream ss;
|
|
ss << R"(
|
|
OpCapability Addresses
|
|
OpCapability Kernel
|
|
OpCapability Linkage
|
|
OpCapability Int64
|
|
OpCapability NamedBarrier
|
|
)";
|
|
|
|
ss << capabilities_and_extensions;
|
|
ss << R"(
|
|
OpMemoryModel Physical32 OpenCL
|
|
%void = OpTypeVoid
|
|
%func = OpTypeFunction %void
|
|
%bool = OpTypeBool
|
|
%f32 = OpTypeFloat 32
|
|
%u32 = OpTypeInt 32 0
|
|
%u64 = OpTypeInt 64 0
|
|
|
|
%f32_0 = OpConstant %f32 0
|
|
%f32_1 = OpConstant %f32 1
|
|
%f32_4 = OpConstant %f32 4
|
|
%u32_0 = OpConstant %u32 0
|
|
%u32_1 = OpConstant %u32 1
|
|
%u32_4 = OpConstant %u32 4
|
|
%u64_0 = OpConstant %u64 0
|
|
%u64_1 = OpConstant %u64 1
|
|
%u64_4 = OpConstant %u64 4
|
|
|
|
%cross_device = OpConstant %u32 0
|
|
%device = OpConstant %u32 1
|
|
%workgroup = OpConstant %u32 2
|
|
%subgroup = OpConstant %u32 3
|
|
%invocation = OpConstant %u32 4
|
|
|
|
%none = OpConstant %u32 0
|
|
%acquire = OpConstant %u32 2
|
|
%release = OpConstant %u32 4
|
|
%acquire_release = OpConstant %u32 8
|
|
%acquire_and_release = OpConstant %u32 6
|
|
%sequentially_consistent = OpConstant %u32 16
|
|
%acquire_release_uniform_workgroup = OpConstant %u32 328
|
|
%acquire_and_release_uniform = OpConstant %u32 70
|
|
%uniform = OpConstant %u32 64
|
|
|
|
%named_barrier = OpTypeNamedBarrier
|
|
|
|
%main = OpFunction %void None %func
|
|
%main_entry = OpLabel
|
|
)";
|
|
|
|
ss << body;
|
|
|
|
ss << R"(
|
|
OpReturn
|
|
OpFunctionEnd)";
|
|
|
|
return ss.str();
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierGLComputeSuccess) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %device %none
|
|
OpControlBarrier %workgroup %workgroup %acquire
|
|
OpControlBarrier %workgroup %device %release
|
|
OpControlBarrier %cross_device %cross_device %acquire_release
|
|
OpControlBarrier %cross_device %cross_device %sequentially_consistent
|
|
OpControlBarrier %cross_device %cross_device %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierKernelSuccess) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %device %none
|
|
OpControlBarrier %workgroup %workgroup %acquire
|
|
OpControlBarrier %workgroup %device %release
|
|
OpControlBarrier %cross_device %cross_device %acquire_release
|
|
OpControlBarrier %cross_device %cross_device %sequentially_consistent
|
|
OpControlBarrier %cross_device %cross_device %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierTesselationControlSuccess) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %device %none
|
|
OpControlBarrier %workgroup %workgroup %acquire
|
|
OpControlBarrier %workgroup %device %release
|
|
OpControlBarrier %cross_device %cross_device %acquire_release
|
|
OpControlBarrier %cross_device %cross_device %sequentially_consistent
|
|
OpControlBarrier %cross_device %cross_device %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "OpCapability Tessellation\n",
|
|
"TessellationControl"));
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierVulkanSuccess) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %workgroup %device %none
|
|
OpControlBarrier %workgroup %workgroup %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierWebGPUSuccess) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %workgroup %queuefamily %none
|
|
OpControlBarrier %workgroup %workgroup %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateWebGPUShaderCode(body), SPV_ENV_WEBGPU_0);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_WEBGPU_0));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierExecutionModelFragmentSpirv12) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %device %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "", "Fragment"),
|
|
SPV_ENV_UNIVERSAL_1_2);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_UNIVERSAL_1_2));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("OpControlBarrier requires one of the following Execution "
|
|
"Models: TessellationControl, GLCompute or Kernel"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierExecutionModelFragmentSpirv13) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %device %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "", "Fragment"),
|
|
SPV_ENV_UNIVERSAL_1_3);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierFloatExecutionScope) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %f32_1 %device %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: expected Execution Scope to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierU64ExecutionScope) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %u64_1 %device %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: expected Execution Scope to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierFloatMemoryScope) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %f32_1 %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: expected Memory Scope to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierU64MemoryScope) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %u64_1 %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: expected Memory Scope to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierFloatMemorySemantics) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %device %f32_0
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr(
|
|
"ControlBarrier: expected Memory Semantics to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierU64MemorySemantics) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %device %u64_0
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr(
|
|
"ControlBarrier: expected Memory Semantics to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierVulkanExecutionScopeDevice) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %workgroup %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: in Vulkan environment Execution Scope "
|
|
"is limited to Workgroup and Subgroup"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierWebGPUExecutionScopeDevice) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %workgroup %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateWebGPUShaderCode(body), SPV_ENV_WEBGPU_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_WEBGPU_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: in WebGPU environment Execution Scope "
|
|
"is limited to Workgroup and Subgroup"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierVulkanMemoryScopeSubgroup) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %subgroup %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: in Vulkan 1.0 environment Memory Scope is "
|
|
"limited to Device, Workgroup and Invocation"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierVulkan1p1MemoryScopeSubgroup) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %subgroup %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierVulkan1p1MemoryScopeCrossDevice) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %cross_device %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: in Vulkan environment, Memory Scope "
|
|
"cannot be CrossDevice"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierAcquireAndRelease) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %device %device %acquire_and_release_uniform
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: Memory Semantics can have at most one "
|
|
"of the following bits set: Acquire, Release, "
|
|
"AcquireRelease or SequentiallyConsistent"));
|
|
}
|
|
|
|
// TODO(atgoo@github.com): the corresponding check fails Vulkan CTS,
|
|
// reenable once fixed.
|
|
TEST_F(ValidateBarriers, DISABLED_OpControlBarrierVulkanSubgroupStorageClass) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %workgroup %device %acquire_release_subgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr(
|
|
"ControlBarrier: expected Memory Semantics to include a "
|
|
"Vulkan-supported storage class if Memory Semantics is not None"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierSubgroupExecutionFragment1p1) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %subgroup %acquire_release_subgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "", "Fragment"),
|
|
SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierWorkgroupExecutionFragment1p1) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %workgroup %workgroup %acquire_release
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "", "Fragment"),
|
|
SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("OpControlBarrier execution scope must be Subgroup for "
|
|
"Fragment, Vertex, Geometry and TessellationEvaluation "
|
|
"execution models"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierSubgroupExecutionFragment1p0) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %workgroup %acquire_release
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "", "Fragment"),
|
|
SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("OpControlBarrier requires one of the following Execution "
|
|
"Models: TessellationControl, GLCompute or Kernel"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierSubgroupExecutionVertex1p1) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %subgroup %acquire_release_subgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "", "Vertex"),
|
|
SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierWorkgroupExecutionVertex1p1) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %workgroup %workgroup %acquire_release
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "", "Vertex"),
|
|
SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("OpControlBarrier execution scope must be Subgroup for "
|
|
"Fragment, Vertex, Geometry and TessellationEvaluation "
|
|
"execution models"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierSubgroupExecutionVertex1p0) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %workgroup %acquire_release
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "", "Vertex"),
|
|
SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("OpControlBarrier requires one of the following Execution "
|
|
"Models: TessellationControl, GLCompute or Kernel"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierSubgroupExecutionGeometry1p1) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %subgroup %acquire_release_subgroup
|
|
)";
|
|
|
|
CompileSuccessfully(
|
|
GenerateShaderCode(body, "OpCapability Geometry\n", "Geometry"),
|
|
SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierWorkgroupExecutionGeometry1p1) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %workgroup %workgroup %acquire_release
|
|
)";
|
|
|
|
CompileSuccessfully(
|
|
GenerateShaderCode(body, "OpCapability Geometry\n", "Geometry"),
|
|
SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("OpControlBarrier execution scope must be Subgroup for "
|
|
"Fragment, Vertex, Geometry and TessellationEvaluation "
|
|
"execution models"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpControlBarrierSubgroupExecutionGeometry1p0) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %workgroup %acquire_release
|
|
)";
|
|
|
|
CompileSuccessfully(
|
|
GenerateShaderCode(body, "OpCapability Geometry\n", "Geometry"),
|
|
SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("OpControlBarrier requires one of the following Execution "
|
|
"Models: TessellationControl, GLCompute or Kernel"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers,
|
|
OpControlBarrierSubgroupExecutionTessellationEvaluation1p1) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %subgroup %acquire_release_subgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "OpCapability Tessellation\n",
|
|
"TessellationEvaluation"),
|
|
SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers,
|
|
OpControlBarrierWorkgroupExecutionTessellationEvaluation1p1) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %workgroup %workgroup %acquire_release
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "OpCapability Tessellation\n",
|
|
"TessellationEvaluation"),
|
|
SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("OpControlBarrier execution scope must be Subgroup for "
|
|
"Fragment, Vertex, Geometry and TessellationEvaluation "
|
|
"execution models"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers,
|
|
OpControlBarrierSubgroupExecutionTessellationEvaluation1p0) {
|
|
const std::string body = R"(
|
|
OpControlBarrier %subgroup %workgroup %acquire_release
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body, "OpCapability Tessellation\n",
|
|
"TessellationEvaluation"),
|
|
SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("OpControlBarrier requires one of the following Execution "
|
|
"Models: TessellationControl, GLCompute or Kernel"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierSuccess) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %cross_device %acquire_release_uniform_workgroup
|
|
OpMemoryBarrier %device %uniform
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierKernelSuccess) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %cross_device %acquire_release_uniform_workgroup
|
|
OpMemoryBarrier %device %uniform
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierVulkanSuccess) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %workgroup %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierFloatMemoryScope) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %f32_1 %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: expected Memory Scope to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierU64MemoryScope) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %u64_1 %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: expected Memory Scope to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierFloatMemorySemantics) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %device %f32_0
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: expected Memory Semantics to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierU64MemorySemantics) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %device %u64_0
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: expected Memory Semantics to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierVulkanMemoryScopeSubgroup) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %subgroup %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: in Vulkan 1.0 environment Memory Scope is "
|
|
"limited to Device, Workgroup and Invocation"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierVulkan1p1MemoryScopeSubgroup) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %subgroup %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierAcquireAndRelease) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %device %acquire_and_release_uniform
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body));
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: Memory Semantics can have at most one "
|
|
"of the following bits set: Acquire, Release, "
|
|
"AcquireRelease or SequentiallyConsistent"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierVulkanMemorySemanticsNone) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %device %none
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: Vulkan specification requires Memory Semantics "
|
|
"to have one of the following bits set: Acquire, Release, "
|
|
"AcquireRelease or SequentiallyConsistent"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierVulkanMemorySemanticsAcquire) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %device %acquire
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: expected Memory Semantics to include a "
|
|
"Vulkan-supported storage class"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryBarrierVulkanSubgroupStorageClass) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %device %acquire_release_subgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: expected Memory Semantics to include a "
|
|
"Vulkan-supported storage class"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpNamedBarrierInitializeSuccess) {
|
|
const std::string body = R"(
|
|
%barrier = OpNamedBarrierInitialize %named_barrier %u32_4
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpNamedBarrierInitializeWrongResultType) {
|
|
const std::string body = R"(
|
|
%barrier = OpNamedBarrierInitialize %u32 %u32_4
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("NamedBarrierInitialize: expected Result Type to be "
|
|
"OpTypeNamedBarrier"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpNamedBarrierInitializeFloatSubgroupCount) {
|
|
const std::string body = R"(
|
|
%barrier = OpNamedBarrierInitialize %named_barrier %f32_4
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("NamedBarrierInitialize: expected Subgroup Count to be "
|
|
"a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpNamedBarrierInitializeU64SubgroupCount) {
|
|
const std::string body = R"(
|
|
%barrier = OpNamedBarrierInitialize %named_barrier %u64_4
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("NamedBarrierInitialize: expected Subgroup Count to be "
|
|
"a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryNamedBarrierSuccess) {
|
|
const std::string body = R"(
|
|
%barrier = OpNamedBarrierInitialize %named_barrier %u32_4
|
|
OpMemoryNamedBarrier %barrier %workgroup %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryNamedBarrierNotNamedBarrier) {
|
|
const std::string body = R"(
|
|
OpMemoryNamedBarrier %u32_1 %workgroup %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("MemoryNamedBarrier: expected Named Barrier to be of "
|
|
"type OpTypeNamedBarrier"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryNamedBarrierFloatMemoryScope) {
|
|
const std::string body = R"(
|
|
%barrier = OpNamedBarrierInitialize %named_barrier %u32_4
|
|
OpMemoryNamedBarrier %barrier %f32_1 %acquire_release_uniform_workgroup
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr(
|
|
"MemoryNamedBarrier: expected Memory Scope to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryNamedBarrierFloatMemorySemantics) {
|
|
const std::string body = R"(
|
|
%barrier = OpNamedBarrierInitialize %named_barrier %u32_4
|
|
OpMemoryNamedBarrier %barrier %workgroup %f32_0
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr(
|
|
"MemoryNamedBarrier: expected Memory Semantics to be a 32-bit int"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OpMemoryNamedBarrierAcquireAndRelease) {
|
|
const std::string body = R"(
|
|
%barrier = OpNamedBarrierInitialize %named_barrier %u32_4
|
|
OpMemoryNamedBarrier %barrier %workgroup %acquire_and_release_uniform
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_1);
|
|
ASSERT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_1));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("MemoryNamedBarrier: Memory Semantics can have at most "
|
|
"one of the following bits set: Acquire, Release, "
|
|
"AcquireRelease or SequentiallyConsistent"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, TypeAsMemoryScope) {
|
|
const std::string body = R"(
|
|
OpMemoryBarrier %u32 %u32_0
|
|
)";
|
|
|
|
CompileSuccessfully(GenerateKernelCode(body));
|
|
EXPECT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions());
|
|
EXPECT_THAT(getDiagnosticString(), HasSubstr("Operand 5[%uint] cannot be a "
|
|
"type"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers,
|
|
OpControlBarrierVulkanMemoryModelBanSequentiallyConsistent) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability VulkanMemoryModelKHR
|
|
OpExtension "SPV_KHR_vulkan_memory_model"
|
|
OpMemoryModel Logical VulkanKHR
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeInt 32 0
|
|
%4 = OpConstant %3 16
|
|
%5 = OpTypeFunction %2
|
|
%6 = OpConstant %3 5
|
|
%1 = OpFunction %2 None %5
|
|
%7 = OpLabel
|
|
OpControlBarrier %6 %6 %4
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("SequentiallyConsistent memory semantics cannot be "
|
|
"used with the VulkanKHR memory model."));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers,
|
|
OpMemoryBarrierVulkanMemoryModelBanSequentiallyConsistent) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability VulkanMemoryModelKHR
|
|
OpExtension "SPV_KHR_vulkan_memory_model"
|
|
OpMemoryModel Logical VulkanKHR
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeInt 32 0
|
|
%4 = OpConstant %3 16
|
|
%5 = OpTypeFunction %2
|
|
%6 = OpConstant %3 5
|
|
%1 = OpFunction %2 None %5
|
|
%7 = OpLabel
|
|
OpMemoryBarrier %6 %4
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("SequentiallyConsistent memory semantics cannot be "
|
|
"used with the VulkanKHR memory model."));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, OutputMemoryKHRRequireVulkanMemoryModelKHR) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeInt 32 0
|
|
%semantics = OpConstant %3 4104
|
|
%5 = OpTypeFunction %2
|
|
%device = OpConstant %3 1
|
|
%1 = OpFunction %2 None %5
|
|
%7 = OpLabel
|
|
OpControlBarrier %device %device %semantics
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: Memory Semantics OutputMemoryKHR "
|
|
"requires capability VulkanMemoryModelKHR"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, MakeAvailableKHRRequireVulkanMemoryModelKHR) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeInt 32 0
|
|
%semantics = OpConstant %3 8264
|
|
%5 = OpTypeFunction %2
|
|
%device = OpConstant %3 1
|
|
%1 = OpFunction %2 None %5
|
|
%7 = OpLabel
|
|
OpControlBarrier %device %device %semantics
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: Memory Semantics MakeAvailableKHR "
|
|
"requires capability VulkanMemoryModelKHR"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, MakeVisibleKHRRequireVulkanMemoryModelKHR) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%2 = OpTypeVoid
|
|
%3 = OpTypeInt 32 0
|
|
%semantics = OpConstant %3 16456
|
|
%5 = OpTypeFunction %2
|
|
%device = OpConstant %3 1
|
|
%1 = OpFunction %2 None %5
|
|
%7 = OpLabel
|
|
OpControlBarrier %device %device %semantics
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: Memory Semantics MakeVisibleKHR "
|
|
"requires capability VulkanMemoryModelKHR"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, MakeAvailableKHRRequiresReleaseSemantics) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability VulkanMemoryModelKHR
|
|
OpExtension "SPV_KHR_vulkan_memory_model"
|
|
OpMemoryModel Logical VulkanKHR
|
|
OpEntryPoint Fragment %func "func"
|
|
OpExecutionMode %func OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%workgroup = OpConstant %int 2
|
|
%semantics = OpConstant %int 8448
|
|
%functy = OpTypeFunction %void
|
|
%func = OpFunction %void None %functy
|
|
%1 = OpLabel
|
|
OpControlBarrier %workgroup %workgroup %semantics
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: MakeAvailableKHR Memory Semantics also "
|
|
"requires either Release or AcquireRelease Memory Semantics"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, MakeVisibleKHRRequiresAcquireSemantics) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability VulkanMemoryModelKHR
|
|
OpExtension "SPV_KHR_vulkan_memory_model"
|
|
OpMemoryModel Logical VulkanKHR
|
|
OpEntryPoint Fragment %func "func"
|
|
OpExecutionMode %func OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%workgroup = OpConstant %int 2
|
|
%semantics = OpConstant %int 16640
|
|
%functy = OpTypeFunction %void
|
|
%func = OpFunction %void None %functy
|
|
%1 = OpLabel
|
|
OpControlBarrier %workgroup %workgroup %semantics
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("ControlBarrier: MakeVisibleKHR Memory Semantics also requires "
|
|
"either Acquire or AcquireRelease Memory Semantics"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, MakeAvailableKHRRequiresStorageSemantics) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability VulkanMemoryModelKHR
|
|
OpExtension "SPV_KHR_vulkan_memory_model"
|
|
OpMemoryModel Logical VulkanKHR
|
|
OpEntryPoint Fragment %func "func"
|
|
OpExecutionMode %func OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%workgroup = OpConstant %int 2
|
|
%semantics = OpConstant %int 8196
|
|
%functy = OpTypeFunction %void
|
|
%func = OpFunction %void None %functy
|
|
%1 = OpLabel
|
|
OpMemoryBarrier %workgroup %semantics
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: expected Memory Semantics to include a "
|
|
"storage class"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, MakeVisibleKHRRequiresStorageSemantics) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability VulkanMemoryModelKHR
|
|
OpExtension "SPV_KHR_vulkan_memory_model"
|
|
OpMemoryModel Logical VulkanKHR
|
|
OpEntryPoint Fragment %func "func"
|
|
OpExecutionMode %func OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%workgroup = OpConstant %int 2
|
|
%semantics = OpConstant %int 16386
|
|
%functy = OpTypeFunction %void
|
|
%func = OpFunction %void None %functy
|
|
%1 = OpLabel
|
|
OpMemoryBarrier %workgroup %semantics
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("MemoryBarrier: expected Memory Semantics to include a "
|
|
"storage class"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, SemanticsSpecConstantShader) {
|
|
const std::string spirv = R"(
|
|
OpCapability Shader
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %func "func"
|
|
OpExecutionMode %func OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%ptr_int_workgroup = OpTypePointer Workgroup %int
|
|
%var = OpVariable %ptr_int_workgroup Workgroup
|
|
%voidfn = OpTypeFunction %void
|
|
%spec_const = OpSpecConstant %int 0
|
|
%workgroup = OpConstant %int 2
|
|
%func = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
OpMemoryBarrier %workgroup %spec_const
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(spirv);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Memory Semantics ids must be OpConstant when Shader "
|
|
"capability is present"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, SemanticsSpecConstantKernel) {
|
|
const std::string spirv = R"(
|
|
OpCapability Kernel
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical OpenCL
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%ptr_int_workgroup = OpTypePointer Workgroup %int
|
|
%var = OpVariable %ptr_int_workgroup Workgroup
|
|
%voidfn = OpTypeFunction %void
|
|
%spec_const = OpSpecConstant %int 0
|
|
%workgroup = OpConstant %int 2
|
|
%func = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
OpMemoryBarrier %workgroup %spec_const
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(spirv);
|
|
EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, ScopeSpecConstantShader) {
|
|
const std::string spirv = R"(
|
|
OpCapability Shader
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %func "func"
|
|
OpExecutionMode %func OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%ptr_int_workgroup = OpTypePointer Workgroup %int
|
|
%var = OpVariable %ptr_int_workgroup Workgroup
|
|
%voidfn = OpTypeFunction %void
|
|
%spec_const = OpSpecConstant %int 0
|
|
%relaxed = OpConstant %int 0
|
|
%func = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
OpMemoryBarrier %spec_const %relaxed
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(spirv);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
|
|
EXPECT_THAT(getDiagnosticString(),
|
|
HasSubstr("Scope ids must be OpConstant when Shader "
|
|
"capability is present"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, ScopeSpecConstantKernel) {
|
|
const std::string spirv = R"(
|
|
OpCapability Kernel
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical OpenCL
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%ptr_int_workgroup = OpTypePointer Workgroup %int
|
|
%var = OpVariable %ptr_int_workgroup Workgroup
|
|
%voidfn = OpTypeFunction %void
|
|
%spec_const = OpSpecConstant %int 0
|
|
%relaxed = OpConstant %int 0
|
|
%func = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
OpMemoryBarrier %spec_const %relaxed
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(spirv);
|
|
EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, VulkanMemoryModelDeviceScopeBad) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability VulkanMemoryModelKHR
|
|
OpExtension "SPV_KHR_vulkan_memory_model"
|
|
OpMemoryModel Logical VulkanKHR
|
|
OpEntryPoint Fragment %func "func"
|
|
OpExecutionMode %func OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%device = OpConstant %int 1
|
|
%semantics = OpConstant %int 0
|
|
%functy = OpTypeFunction %void
|
|
%func = OpFunction %void None %functy
|
|
%1 = OpLabel
|
|
OpMemoryBarrier %device %semantics
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
|
|
EXPECT_EQ(SPV_ERROR_INVALID_DATA,
|
|
ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
|
|
EXPECT_THAT(
|
|
getDiagnosticString(),
|
|
HasSubstr("Use of device scope with VulkanKHR memory model requires the "
|
|
"VulkanMemoryModelDeviceScopeKHR capability"));
|
|
}
|
|
|
|
TEST_F(ValidateBarriers, VulkanMemoryModelDeviceScopeGood) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability VulkanMemoryModelKHR
|
|
OpCapability VulkanMemoryModelDeviceScopeKHR
|
|
OpExtension "SPV_KHR_vulkan_memory_model"
|
|
OpMemoryModel Logical VulkanKHR
|
|
OpEntryPoint Fragment %func "func"
|
|
OpExecutionMode %func OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%int = OpTypeInt 32 0
|
|
%device = OpConstant %int 1
|
|
%semantics = OpConstant %int 0
|
|
%functy = OpTypeFunction %void
|
|
%func = OpFunction %void None %functy
|
|
%1 = OpLabel
|
|
OpMemoryBarrier %device %semantics
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
|
|
EXPECT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace val
|
|
} // namespace spvtools
|