mirror of
https://github.com/encounter/dawn-cmake.git
synced 2025-05-13 10:51:35 +00:00
This CL changes the set decoration to a group decoration. We still parse `set` until downstream users can be updated. Bug: tint:338 Change-Id: I610d1ed769b3a26c117ad6d875f8a99a3d5b7754 Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/37920 Auto-Submit: dan sinclair <dsinclair@chromium.org> Commit-Queue: Ben Clayton <bclayton@google.com> Reviewed-by: Ben Clayton <bclayton@google.com>
2076 lines
58 KiB
C++
2076 lines
58 KiB
C++
// Copyright 2020 The Tint Authors.
|
|
//
|
|
// 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 <string>
|
|
|
|
#include "gmock/gmock.h"
|
|
#include "src/demangler.h"
|
|
#include "src/reader/spirv/function.h"
|
|
#include "src/reader/spirv/parser_impl.h"
|
|
#include "src/reader/spirv/parser_impl_test_helper.h"
|
|
#include "src/reader/spirv/spirv_tools_helpers_test.h"
|
|
|
|
namespace tint {
|
|
namespace reader {
|
|
namespace spirv {
|
|
namespace {
|
|
|
|
using SpvModuleScopeVarParserTest = SpvParserTest;
|
|
|
|
using ::testing::Eq;
|
|
using ::testing::HasSubstr;
|
|
using ::testing::Not;
|
|
|
|
std::string CommonTypes() {
|
|
return R"(
|
|
%void = OpTypeVoid
|
|
%voidfn = OpTypeFunction %void
|
|
|
|
%bool = OpTypeBool
|
|
%float = OpTypeFloat 32
|
|
%uint = OpTypeInt 32 0
|
|
%int = OpTypeInt 32 1
|
|
|
|
%ptr_bool = OpTypePointer Private %bool
|
|
%ptr_float = OpTypePointer Private %float
|
|
%ptr_uint = OpTypePointer Private %uint
|
|
%ptr_int = OpTypePointer Private %int
|
|
|
|
%true = OpConstantTrue %bool
|
|
%false = OpConstantFalse %bool
|
|
%float_0 = OpConstant %float 0.0
|
|
%float_1p5 = OpConstant %float 1.5
|
|
%uint_1 = OpConstant %uint 1
|
|
%int_m1 = OpConstant %int -1
|
|
%uint_2 = OpConstant %uint 2
|
|
|
|
%v2bool = OpTypeVector %bool 2
|
|
%v2uint = OpTypeVector %uint 2
|
|
%v2int = OpTypeVector %int 2
|
|
%v2float = OpTypeVector %float 2
|
|
%m3v2float = OpTypeMatrix %v2float 3
|
|
|
|
%arr2uint = OpTypeArray %uint %uint_2
|
|
%strct = OpTypeStruct %uint %float %arr2uint
|
|
)";
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, NoVar) {
|
|
auto p = parser(test::Assemble(""));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_ast = p->module().to_str();
|
|
EXPECT_THAT(module_ast, Not(HasSubstr("Variable")));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BadStorageClass_NotAWebGPUStorageClass) {
|
|
auto p = parser(test::Assemble(R"(
|
|
%float = OpTypeFloat 32
|
|
%ptr = OpTypePointer CrossWorkgroup %float
|
|
%52 = OpVariable %ptr CrossWorkgroup
|
|
)"));
|
|
EXPECT_TRUE(p->BuildInternalModule());
|
|
// Normally we should run ParserImpl::RegisterTypes before emitting
|
|
// variables. But defensive coding in EmitModuleScopeVariables lets
|
|
// us catch this error.
|
|
EXPECT_FALSE(p->EmitModuleScopeVariables()) << p->error();
|
|
EXPECT_THAT(p->error(), HasSubstr("unknown SPIR-V storage class: 5"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BadStorageClass_Function) {
|
|
auto p = parser(test::Assemble(R"(
|
|
%float = OpTypeFloat 32
|
|
%ptr = OpTypePointer Function %float
|
|
%52 = OpVariable %ptr Function
|
|
)"));
|
|
EXPECT_TRUE(p->BuildInternalModule());
|
|
// Normally we should run ParserImpl::RegisterTypes before emitting
|
|
// variables. But defensive coding in EmitModuleScopeVariables lets
|
|
// us catch this error.
|
|
EXPECT_FALSE(p->EmitModuleScopeVariables()) << p->error();
|
|
EXPECT_THAT(p->error(),
|
|
HasSubstr("invalid SPIR-V storage class 7 for module scope "
|
|
"variable: %52 = OpVariable %2 Function"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BadPointerType) {
|
|
auto p = parser(test::Assemble(R"(
|
|
%float = OpTypeFloat 32
|
|
%fn_ty = OpTypeFunction %float
|
|
%3 = OpTypePointer Private %fn_ty
|
|
%52 = OpVariable %3 Private
|
|
)"));
|
|
EXPECT_TRUE(p->BuildInternalModule());
|
|
// Normally we should run ParserImpl::RegisterTypes before emitting
|
|
// variables. But defensive coding in EmitModuleScopeVariables lets
|
|
// us catch this error.
|
|
EXPECT_FALSE(p->EmitModuleScopeVariables());
|
|
EXPECT_THAT(p->error(), HasSubstr("internal error: failed to register Tint "
|
|
"AST type for SPIR-V type with ID: 3"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, NonPointerType) {
|
|
auto p = parser(test::Assemble(R"(
|
|
%float = OpTypeFloat 32
|
|
%5 = OpTypeFunction %float
|
|
%3 = OpTypePointer Private %5
|
|
%52 = OpVariable %float Private
|
|
)"));
|
|
EXPECT_TRUE(p->BuildInternalModule());
|
|
EXPECT_FALSE(p->RegisterTypes());
|
|
EXPECT_THAT(
|
|
p->error(),
|
|
HasSubstr("SPIR-V pointer type with ID 3 has invalid pointee type 5"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, AnonWorkgroupVar) {
|
|
auto p = parser(test::Assemble(R"(
|
|
%float = OpTypeFloat 32
|
|
%ptr = OpTypePointer Workgroup %float
|
|
%52 = OpVariable %ptr Workgroup
|
|
)"));
|
|
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Variable{
|
|
x_52
|
|
workgroup
|
|
__f32
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, NamedWorkgroupVar) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %52 "the_counter"
|
|
%float = OpTypeFloat 32
|
|
%ptr = OpTypePointer Workgroup %float
|
|
%52 = OpVariable %ptr Workgroup
|
|
)"));
|
|
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Variable{
|
|
the_counter
|
|
workgroup
|
|
__f32
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, PrivateVar) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %52 "my_own_private_idaho"
|
|
%float = OpTypeFloat 32
|
|
%ptr = OpTypePointer Private %float
|
|
%52 = OpVariable %ptr Private
|
|
)"));
|
|
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Variable{
|
|
my_own_private_idaho
|
|
private
|
|
__f32
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinVertexIndex) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpDecorate %52 BuiltIn VertexIndex
|
|
%uint = OpTypeInt 32 0
|
|
%ptr = OpTypePointer Input %uint
|
|
%52 = OpVariable %ptr Input
|
|
)"));
|
|
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Variable{
|
|
Decorations{
|
|
BuiltinDecoration{vertex_idx}
|
|
}
|
|
x_52
|
|
in
|
|
__u32
|
|
})"));
|
|
}
|
|
|
|
std::string PerVertexPreamble() {
|
|
return R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage ; so we don't have to declare an entry point
|
|
OpMemoryModel Logical Simple
|
|
|
|
OpMemberDecorate %10 0 BuiltIn Position
|
|
OpMemberDecorate %10 1 BuiltIn PointSize
|
|
OpMemberDecorate %10 2 BuiltIn ClipDistance
|
|
OpMemberDecorate %10 3 BuiltIn CullDistance
|
|
%void = OpTypeVoid
|
|
%voidfn = OpTypeFunction %void
|
|
%float = OpTypeFloat 32
|
|
%12 = OpTypeVector %float 4
|
|
%uint = OpTypeInt 32 0
|
|
%uint_0 = OpConstant %uint 0
|
|
%uint_1 = OpConstant %uint 1
|
|
%arr = OpTypeArray %float %uint_1
|
|
%10 = OpTypeStruct %12 %float %arr %arr
|
|
%11 = OpTypePointer Output %10
|
|
%1 = OpVariable %11 Output
|
|
)";
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinPosition_MapsToModuleScopeVec4Var) {
|
|
// In Vulkan SPIR-V, Position is the first member of gl_PerVertex
|
|
const std::string assembly = PerVertexPreamble();
|
|
auto p = parser(test::Assemble(assembly));
|
|
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule()) << assembly;
|
|
EXPECT_TRUE(p->error().empty()) << p->error();
|
|
const auto& position_info = p->GetBuiltInPositionInfo();
|
|
EXPECT_EQ(position_info.struct_type_id, 10u);
|
|
EXPECT_EQ(position_info.position_member_index, 0u);
|
|
EXPECT_EQ(position_info.position_member_type_id, 12u);
|
|
EXPECT_EQ(position_info.pointer_type_id, 11u);
|
|
EXPECT_EQ(position_info.storage_class, SpvStorageClassOutput);
|
|
EXPECT_EQ(position_info.per_vertex_var_id, 1u);
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Variable{
|
|
Decorations{
|
|
BuiltinDecoration{position}
|
|
}
|
|
gl_Position
|
|
out
|
|
__vec_4__f32
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPosition_StoreWholeStruct_NotSupported) {
|
|
// Glslang does not generate this code pattern.
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%nil = OpConstantNull %10 ; the whole struct
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
OpStore %1 %nil ; store the whole struct
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule()) << assembly;
|
|
EXPECT_THAT(p->error(), Eq("storing to the whole per-vertex structure is not "
|
|
"supported: OpStore %1 %9"))
|
|
<< p->error();
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPosition_IntermediateWholeStruct_NotSupported) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%1000 = OpUndef %10
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule()) << assembly;
|
|
EXPECT_THAT(p->error(), Eq("operations producing a per-vertex structure are "
|
|
"not supported: %1000 = OpUndef %10"))
|
|
<< p->error();
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPosition_IntermediatePtrWholeStruct_NotSupported) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%1000 = OpUndef %11
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule());
|
|
EXPECT_THAT(p->error(),
|
|
Eq("operations producing a pointer to a per-vertex structure are "
|
|
"not supported: %1000 = OpUndef %11"))
|
|
<< p->error();
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinPosition_StorePosition) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr_v4float = OpTypePointer Output %12
|
|
%nil = OpConstantNull %12
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%100 = OpAccessChain %ptr_v4float %1 %uint_0 ; address of the Position member
|
|
OpStore %100 %nil
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Assignment{
|
|
Identifier[not set]{gl_Position}
|
|
TypeConstructor[not set]{
|
|
__vec_4__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPosition_StorePosition_PerVertexStructOutOfOrderDecl) {
|
|
const std::string assembly = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage ; so we don't have to declare an entry point
|
|
OpMemoryModel Logical Simple
|
|
|
|
; scramble the member indices
|
|
OpMemberDecorate %10 0 BuiltIn ClipDistance
|
|
OpMemberDecorate %10 1 BuiltIn CullDistance
|
|
OpMemberDecorate %10 2 BuiltIn Position
|
|
OpMemberDecorate %10 3 BuiltIn PointSize
|
|
%void = OpTypeVoid
|
|
%voidfn = OpTypeFunction %void
|
|
%float = OpTypeFloat 32
|
|
%12 = OpTypeVector %float 4
|
|
%uint = OpTypeInt 32 0
|
|
%uint_0 = OpConstant %uint 0
|
|
%uint_1 = OpConstant %uint 1
|
|
%uint_2 = OpConstant %uint 2
|
|
%arr = OpTypeArray %float %uint_1
|
|
%10 = OpTypeStruct %arr %arr %12 %float
|
|
%11 = OpTypePointer Output %10
|
|
%1 = OpVariable %11 Output
|
|
|
|
%ptr_v4float = OpTypePointer Output %12
|
|
%nil = OpConstantNull %12
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%100 = OpAccessChain %ptr_v4float %1 %uint_2 ; address of the Position member
|
|
OpStore %100 %nil
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Assignment{
|
|
Identifier[not set]{gl_Position}
|
|
TypeConstructor[not set]{
|
|
__vec_4__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPosition_StorePositionMember_OneAccessChain) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr_float = OpTypePointer Output %float
|
|
%nil = OpConstantNull %float
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%100 = OpAccessChain %ptr_float %1 %uint_0 %uint_1 ; address of the Position.y member
|
|
OpStore %100 %nil
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Assignment{
|
|
MemberAccessor[not set]{
|
|
Identifier[not set]{gl_Position}
|
|
Identifier[not set]{y}
|
|
}
|
|
ScalarConstructor[not set]{0.000000}
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPosition_StorePositionMember_TwoAccessChain) {
|
|
// The algorithm is smart enough to collapse it down.
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr = OpTypePointer Output %12
|
|
%ptr_float = OpTypePointer Output %float
|
|
%nil = OpConstantNull %float
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%100 = OpAccessChain %ptr %1 %uint_0 ; address of the Position member
|
|
%101 = OpAccessChain %ptr_float %100 %uint_1 ; address of the Position.y member
|
|
OpStore %101 %nil
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
{
|
|
Assignment{
|
|
MemberAccessor[not set]{
|
|
Identifier[not set]{gl_Position}
|
|
Identifier[not set]{y}
|
|
}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
Return{}
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinPointSize_Write1_IsErased) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr = OpTypePointer Output %float
|
|
%one = OpConstant %float 1.0
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%100 = OpAccessChain %ptr %1 %uint_1 ; address of the PointSize member
|
|
OpStore %100 %one
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_EQ(module_str, R"(Module{
|
|
Variable{
|
|
Decorations{
|
|
BuiltinDecoration{position}
|
|
}
|
|
gl_Position
|
|
out
|
|
__vec_4__f32
|
|
}
|
|
Function x_14 -> __void
|
|
()
|
|
{
|
|
Return{}
|
|
}
|
|
}
|
|
)") << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinPointSize_WriteNon1_IsError) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr = OpTypePointer Output %float
|
|
%999 = OpConstant %float 2.0
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%100 = OpAccessChain %ptr %1 %uint_1 ; address of the PointSize member
|
|
OpStore %100 %999
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule());
|
|
EXPECT_THAT(p->error(),
|
|
HasSubstr("cannot store a value other than constant 1.0 to "
|
|
"PointSize builtin: OpStore %100 %999"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinPointSize_ReadReplaced) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr = OpTypePointer Output %float
|
|
%nil = OpConstantNull %12
|
|
%private_ptr = OpTypePointer Private %float
|
|
%900 = OpVariable %private_ptr Private
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%100 = OpAccessChain %ptr %1 %uint_1 ; address of the PointSize member
|
|
%99 = OpLoad %float %100
|
|
OpStore %900 %99
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_EQ(module_str, R"(Module{
|
|
Variable{
|
|
x_900
|
|
private
|
|
__f32
|
|
}
|
|
Variable{
|
|
Decorations{
|
|
BuiltinDecoration{position}
|
|
}
|
|
gl_Position
|
|
out
|
|
__vec_4__f32
|
|
}
|
|
Function x_15 -> __void
|
|
()
|
|
{
|
|
Assignment{
|
|
Identifier[not set]{x_900}
|
|
ScalarConstructor[not set]{1.000000}
|
|
}
|
|
Return{}
|
|
}
|
|
}
|
|
)") << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPointSize_WriteViaCopyObjectPriorAccess_Unsupported) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr = OpTypePointer Output %float
|
|
%nil = OpConstantNull %12
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%20 = OpCopyObject %11 %1
|
|
%100 = OpAccessChain %20 %1 %uint_1 ; address of the PointSize member
|
|
OpStore %100 %nil
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule()) << p->error();
|
|
EXPECT_THAT(
|
|
p->error(),
|
|
HasSubstr("operations producing a pointer to a per-vertex structure are "
|
|
"not supported: %20 = OpCopyObject %11 %1"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPointSize_WriteViaCopyObjectPostAccessChainErased) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr = OpTypePointer Output %12
|
|
%one = OpConstant %float 1.0
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%100 = OpAccessChain %ptr %1 %uint_1 ; address of the PointSize member
|
|
%101 = OpCopyObject %ptr %100
|
|
OpStore %101 %one
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_EQ(module_str, R"(Module{
|
|
Variable{
|
|
Decorations{
|
|
BuiltinDecoration{position}
|
|
}
|
|
gl_Position
|
|
out
|
|
__vec_4__f32
|
|
}
|
|
Function x_14 -> __void
|
|
()
|
|
{
|
|
Return{}
|
|
}
|
|
}
|
|
)") << module_str;
|
|
}
|
|
|
|
std::string LoosePointSizePreamble() {
|
|
return R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage ; so we don't have to declare an entry point
|
|
OpMemoryModel Logical Simple
|
|
|
|
OpDecorate %1 BuiltIn PointSize
|
|
%void = OpTypeVoid
|
|
%voidfn = OpTypeFunction %void
|
|
%float = OpTypeFloat 32
|
|
%uint = OpTypeInt 32 0
|
|
%uint_0 = OpConstant %uint 0
|
|
%uint_1 = OpConstant %uint 1
|
|
%11 = OpTypePointer Output %float
|
|
%1 = OpVariable %11 Output
|
|
)";
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinPointSize_Loose_Write1_IsErased) {
|
|
const std::string assembly = LoosePointSizePreamble() + R"(
|
|
%ptr = OpTypePointer Output %float
|
|
%one = OpConstant %float 1.0
|
|
|
|
%500 = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
OpStore %1 %one
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_EQ(module_str, R"(Module{
|
|
Function x_500 -> __void
|
|
()
|
|
{
|
|
Return{}
|
|
}
|
|
}
|
|
)") << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinPointSize_Loose_WriteNon1_IsError) {
|
|
const std::string assembly = LoosePointSizePreamble() + R"(
|
|
%ptr = OpTypePointer Output %float
|
|
%999 = OpConstant %float 2.0
|
|
|
|
%500 = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
OpStore %1 %999
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule());
|
|
EXPECT_THAT(p->error(),
|
|
HasSubstr("cannot store a value other than constant 1.0 to "
|
|
"PointSize builtin: OpStore %1 %999"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinPointSize_Loose_ReadReplaced) {
|
|
const std::string assembly = LoosePointSizePreamble() + R"(
|
|
%ptr = OpTypePointer Private %float
|
|
%900 = OpVariable %ptr Private
|
|
|
|
%500 = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%99 = OpLoad %float %1
|
|
OpStore %900 %99
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_EQ(module_str, R"(Module{
|
|
Variable{
|
|
x_900
|
|
private
|
|
__f32
|
|
}
|
|
Function x_500 -> __void
|
|
()
|
|
{
|
|
Assignment{
|
|
Identifier[not set]{x_900}
|
|
ScalarConstructor[not set]{1.000000}
|
|
}
|
|
Return{}
|
|
}
|
|
}
|
|
)") << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPointSize_Loose_WriteViaCopyObjectPriorAccess_Erased) {
|
|
const std::string assembly = LoosePointSizePreamble() + R"(
|
|
%one = OpConstant %float 1.0
|
|
|
|
%500 = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%20 = OpCopyObject %11 %1
|
|
%100 = OpAccessChain %11 %20
|
|
OpStore %100 %one
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_EQ(module_str, R"(Module{
|
|
Function x_500 -> __void
|
|
()
|
|
{
|
|
Return{}
|
|
}
|
|
}
|
|
)") << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPointSize_Loose_WriteViaCopyObjectPostAccessChainErased) {
|
|
const std::string assembly = LoosePointSizePreamble() + R"(
|
|
%one = OpConstant %float 1.0
|
|
|
|
%500 = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%100 = OpAccessChain %11 %1
|
|
%101 = OpCopyObject %11 %100
|
|
OpStore %101 %one
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_TRUE(p->BuildAndParseInternalModule()) << p->error();
|
|
EXPECT_TRUE(p->error().empty()) << p->error();
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_EQ(module_str, R"(Module{
|
|
Function x_500 -> __void
|
|
()
|
|
{
|
|
Return{}
|
|
}
|
|
}
|
|
)") << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinClipDistance_NotSupported) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr_float = OpTypePointer Output %float
|
|
%nil = OpConstantNull %float
|
|
%uint_2 = OpConstant %uint 2
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
; address of the first entry in ClipDistance
|
|
%100 = OpAccessChain %ptr_float %1 %uint_2 %uint_0
|
|
OpStore %100 %nil
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule());
|
|
EXPECT_EQ(p->error(),
|
|
"accessing per-vertex member 2 is not supported. Only Position is "
|
|
"supported, and PointSize is ignored");
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinCullDistance_NotSupported) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr_float = OpTypePointer Output %float
|
|
%nil = OpConstantNull %float
|
|
%uint_3 = OpConstant %uint 3
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
; address of the first entry in CullDistance
|
|
%100 = OpAccessChain %ptr_float %1 %uint_3 %uint_0
|
|
OpStore %100 %nil
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule());
|
|
EXPECT_EQ(p->error(),
|
|
"accessing per-vertex member 3 is not supported. Only Position is "
|
|
"supported, and PointSize is ignored");
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BuiltinPerVertex_MemberIndex_NotConstant) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr_float = OpTypePointer Output %float
|
|
%nil = OpConstantNull %float
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
%sum = OpIAdd %uint %uint_0 %uint_0
|
|
%100 = OpAccessChain %ptr_float %1 %sum
|
|
OpStore %100 %nil
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule());
|
|
EXPECT_THAT(p->error(),
|
|
Eq("first index of access chain into per-vertex structure is not "
|
|
"a constant: %100 = OpAccessChain %9 %1 %16"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BuiltinPerVertex_MemberIndex_NotConstantInteger) {
|
|
const std::string assembly = PerVertexPreamble() + R"(
|
|
%ptr_float = OpTypePointer Output %float
|
|
%nil = OpConstantNull %float
|
|
|
|
%main = OpFunction %void None %voidfn
|
|
%entry = OpLabel
|
|
; nil is bad here!
|
|
%100 = OpAccessChain %ptr_float %1 %nil
|
|
OpStore %100 %nil
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
auto p = parser(test::Assemble(assembly));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModule());
|
|
EXPECT_THAT(p->error(),
|
|
Eq("first index of access chain into per-vertex structure is not "
|
|
"a constant integer: %100 = OpAccessChain %9 %1 %13"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ScalarInitializers) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%1 = OpVariable %ptr_bool Private %true
|
|
%2 = OpVariable %ptr_bool Private %false
|
|
%3 = OpVariable %ptr_int Private %int_m1
|
|
%4 = OpVariable %ptr_uint Private %uint_1
|
|
%5 = OpVariable %ptr_float Private %float_1p5
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_1
|
|
private
|
|
__bool
|
|
{
|
|
ScalarConstructor[not set]{true}
|
|
}
|
|
}
|
|
Variable{
|
|
x_2
|
|
private
|
|
__bool
|
|
{
|
|
ScalarConstructor[not set]{false}
|
|
}
|
|
}
|
|
Variable{
|
|
x_3
|
|
private
|
|
__i32
|
|
{
|
|
ScalarConstructor[not set]{-1}
|
|
}
|
|
}
|
|
Variable{
|
|
x_4
|
|
private
|
|
__u32
|
|
{
|
|
ScalarConstructor[not set]{1}
|
|
}
|
|
}
|
|
Variable{
|
|
x_5
|
|
private
|
|
__f32
|
|
{
|
|
ScalarConstructor[not set]{1.500000}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ScalarNullInitializers) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%null_bool = OpConstantNull %bool
|
|
%null_int = OpConstantNull %int
|
|
%null_uint = OpConstantNull %uint
|
|
%null_float = OpConstantNull %float
|
|
|
|
%1 = OpVariable %ptr_bool Private %null_bool
|
|
%2 = OpVariable %ptr_int Private %null_int
|
|
%3 = OpVariable %ptr_uint Private %null_uint
|
|
%4 = OpVariable %ptr_float Private %null_float
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_1
|
|
private
|
|
__bool
|
|
{
|
|
ScalarConstructor[not set]{false}
|
|
}
|
|
}
|
|
Variable{
|
|
x_2
|
|
private
|
|
__i32
|
|
{
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
Variable{
|
|
x_3
|
|
private
|
|
__u32
|
|
{
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
Variable{
|
|
x_4
|
|
private
|
|
__f32
|
|
{
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ScalarUndefInitializers) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%undef_bool = OpUndef %bool
|
|
%undef_int = OpUndef %int
|
|
%undef_uint = OpUndef %uint
|
|
%undef_float = OpUndef %float
|
|
|
|
%1 = OpVariable %ptr_bool Private %undef_bool
|
|
%2 = OpVariable %ptr_int Private %undef_int
|
|
%3 = OpVariable %ptr_uint Private %undef_uint
|
|
%4 = OpVariable %ptr_float Private %undef_float
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_1
|
|
private
|
|
__bool
|
|
{
|
|
ScalarConstructor[not set]{false}
|
|
}
|
|
}
|
|
Variable{
|
|
x_2
|
|
private
|
|
__i32
|
|
{
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
Variable{
|
|
x_3
|
|
private
|
|
__u32
|
|
{
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
Variable{
|
|
x_4
|
|
private
|
|
__f32
|
|
{
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, VectorInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %v2float
|
|
%two = OpConstant %float 2.0
|
|
%const = OpConstantComposite %v2float %float_1p5 %two
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__vec_2__f32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{1.500000}
|
|
ScalarConstructor[not set]{2.000000}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, VectorBoolNullInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %v2bool
|
|
%const = OpConstantNull %v2bool
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__vec_2__bool
|
|
{
|
|
TypeConstructor[not set]{
|
|
__vec_2__bool
|
|
ScalarConstructor[not set]{false}
|
|
ScalarConstructor[not set]{false}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, VectorBoolUndefInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %v2bool
|
|
%const = OpUndef %v2bool
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__vec_2__bool
|
|
{
|
|
TypeConstructor[not set]{
|
|
__vec_2__bool
|
|
ScalarConstructor[not set]{false}
|
|
ScalarConstructor[not set]{false}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, VectorUintNullInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %v2uint
|
|
%const = OpConstantNull %v2uint
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__vec_2__u32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__vec_2__u32
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, VectorUintUndefInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %v2uint
|
|
%const = OpUndef %v2uint
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__vec_2__u32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__vec_2__u32
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, VectorIntNullInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %v2int
|
|
%const = OpConstantNull %v2int
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__vec_2__i32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__vec_2__i32
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, VectorIntUndefInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %v2int
|
|
%const = OpUndef %v2int
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__vec_2__i32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__vec_2__i32
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, VectorFloatNullInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %v2float
|
|
%const = OpConstantNull %v2float
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__vec_2__f32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, VectorFloatUndefInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %v2float
|
|
%const = OpUndef %v2float
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__vec_2__f32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, MatrixInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %m3v2float
|
|
%two = OpConstant %float 2.0
|
|
%three = OpConstant %float 3.0
|
|
%four = OpConstant %float 4.0
|
|
%v0 = OpConstantComposite %v2float %float_1p5 %two
|
|
%v1 = OpConstantComposite %v2float %two %three
|
|
%v2 = OpConstantComposite %v2float %three %four
|
|
%const = OpConstantComposite %m3v2float %v0 %v1 %v2
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__mat_2_3__f32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__mat_2_3__f32
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{1.500000}
|
|
ScalarConstructor[not set]{2.000000}
|
|
}
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{2.000000}
|
|
ScalarConstructor[not set]{3.000000}
|
|
}
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{3.000000}
|
|
ScalarConstructor[not set]{4.000000}
|
|
}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, MatrixNullInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %m3v2float
|
|
%const = OpConstantNull %m3v2float
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__mat_2_3__f32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__mat_2_3__f32
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, MatrixUndefInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %m3v2float
|
|
%const = OpUndef %m3v2float
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__mat_2_3__f32
|
|
{
|
|
TypeConstructor[not set]{
|
|
__mat_2_3__f32
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
TypeConstructor[not set]{
|
|
__vec_2__f32
|
|
ScalarConstructor[not set]{0.000000}
|
|
ScalarConstructor[not set]{0.000000}
|
|
}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ArrayInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %arr2uint
|
|
%two = OpConstant %uint 2
|
|
%const = OpConstantComposite %arr2uint %uint_1 %two
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__array__u32_2
|
|
{
|
|
TypeConstructor[not set]{
|
|
__array__u32_2
|
|
ScalarConstructor[not set]{1}
|
|
ScalarConstructor[not set]{2}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ArrayNullInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %arr2uint
|
|
%const = OpConstantNull %arr2uint
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__array__u32_2
|
|
{
|
|
TypeConstructor[not set]{
|
|
__array__u32_2
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ArrayUndefInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %arr2uint
|
|
%const = OpUndef %arr2uint
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__array__u32_2
|
|
{
|
|
TypeConstructor[not set]{
|
|
__array__u32_2
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
})"));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, StructInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %strct
|
|
%two = OpConstant %uint 2
|
|
%arrconst = OpConstantComposite %arr2uint %uint_1 %two
|
|
%const = OpConstantComposite %strct %uint_1 %float_1p5 %arrconst
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__struct_S
|
|
{
|
|
TypeConstructor[not set]{
|
|
__struct_S
|
|
ScalarConstructor[not set]{1}
|
|
ScalarConstructor[not set]{1.500000}
|
|
TypeConstructor[not set]{
|
|
__array__u32_2
|
|
ScalarConstructor[not set]{1}
|
|
ScalarConstructor[not set]{2}
|
|
}
|
|
}
|
|
}
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, StructNullInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %strct
|
|
%const = OpConstantNull %strct
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__struct_S
|
|
{
|
|
TypeConstructor[not set]{
|
|
__struct_S
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0.000000}
|
|
TypeConstructor[not set]{
|
|
__array__u32_2
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
}
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, StructUndefInitializer) {
|
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
|
%ptr = OpTypePointer Private %strct
|
|
%const = OpUndef %strct
|
|
%200 = OpVariable %ptr Private %const
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(Variable{
|
|
x_200
|
|
private
|
|
__struct_S
|
|
{
|
|
TypeConstructor[not set]{
|
|
__struct_S
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0.000000}
|
|
TypeConstructor[not set]{
|
|
__array__u32_2
|
|
ScalarConstructor[not set]{0}
|
|
ScalarConstructor[not set]{0}
|
|
}
|
|
}
|
|
}
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, LocationDecoration_Valid) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %myvar Location 3
|
|
)" + CommonTypes() + R"(
|
|
%ptr = OpTypePointer Input %uint
|
|
%myvar = OpVariable %ptr Input
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Variable{
|
|
Decorations{
|
|
LocationDecoration{3}
|
|
}
|
|
myvar
|
|
in
|
|
__u32
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
LocationDecoration_MissingOperandWontAssemble) {
|
|
const auto assembly = R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %myvar Location
|
|
)" + CommonTypes() + R"(
|
|
%ptr = OpTypePointer Input %uint
|
|
%myvar = OpVariable %ptr Input
|
|
)";
|
|
EXPECT_THAT(test::AssembleFailure(assembly),
|
|
Eq("4:4: Expected operand, found next instruction instead."));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
LocationDecoration_TwoOperandsWontAssemble) {
|
|
const auto assembly = R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %myvar Location 3 4
|
|
)" + CommonTypes() + R"(
|
|
%ptr = OpTypePointer Input %uint
|
|
%myvar = OpVariable %ptr Input
|
|
)";
|
|
EXPECT_THAT(
|
|
test::AssembleFailure(assembly),
|
|
Eq("2:34: Expected <opcode> or <result-id> at the beginning of an "
|
|
"instruction, found '4'."));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, DescriptorGroupDecoration_Valid) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %myvar DescriptorSet 3
|
|
OpDecorate %strct Block
|
|
)" + CommonTypes() + R"(
|
|
%ptr_sb_strct = OpTypePointer StorageBuffer %strct
|
|
%myvar = OpVariable %ptr_sb_strct StorageBuffer
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Variable{
|
|
Decorations{
|
|
GroupDecoration{3}
|
|
}
|
|
myvar
|
|
storage_buffer
|
|
__access_control_read_write__struct_S
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
DescriptorGroupDecoration_MissingOperandWontAssemble) {
|
|
const auto assembly = R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %myvar DescriptorSet
|
|
OpDecorate %strct Block
|
|
)" + CommonTypes() + R"(
|
|
%ptr_sb_strct = OpTypePointer StorageBuffer %strct
|
|
%myvar = OpVariable %ptr_sb_strct StorageBuffer
|
|
)";
|
|
EXPECT_THAT(test::AssembleFailure(assembly),
|
|
Eq("3:5: Expected operand, found next instruction instead."));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
DescriptorGroupDecoration_TwoOperandsWontAssemble) {
|
|
const auto assembly = R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %myvar DescriptorSet 3 4
|
|
OpDecorate %strct Block
|
|
)" + CommonTypes() + R"(
|
|
%ptr_sb_strct = OpTypePointer StorageBuffer %strct
|
|
%myvar = OpVariable %ptr_sb_strct StorageBuffer
|
|
)";
|
|
EXPECT_THAT(
|
|
test::AssembleFailure(assembly),
|
|
Eq("2:39: Expected <opcode> or <result-id> at the beginning of an "
|
|
"instruction, found '4'."));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BindingDecoration_Valid) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %myvar Binding 3
|
|
OpDecorate %strct Block
|
|
)" + CommonTypes() + R"(
|
|
%ptr_sb_strct = OpTypePointer StorageBuffer %strct
|
|
%myvar = OpVariable %ptr_sb_strct StorageBuffer
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
Variable{
|
|
Decorations{
|
|
BindingDecoration{3}
|
|
}
|
|
myvar
|
|
storage_buffer
|
|
__access_control_read_write__struct_S
|
|
})"))
|
|
<< module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
BindingDecoration_MissingOperandWontAssemble) {
|
|
const auto assembly = R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %myvar Binding
|
|
OpDecorate %strct Block
|
|
)" + CommonTypes() + R"(
|
|
%ptr_sb_strct = OpTypePointer StorageBuffer %strct
|
|
%myvar = OpVariable %ptr_sb_strct StorageBuffer
|
|
)";
|
|
EXPECT_THAT(test::AssembleFailure(assembly),
|
|
Eq("3:5: Expected operand, found next instruction instead."));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, BindingDecoration_TwoOperandsWontAssemble) {
|
|
const auto assembly = R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %myvar Binding 3 4
|
|
OpDecorate %strct Block
|
|
)" + CommonTypes() + R"(
|
|
%ptr_sb_strct = OpTypePointer StorageBuffer %strct
|
|
%myvar = OpVariable %ptr_sb_strct StorageBuffer
|
|
)";
|
|
EXPECT_THAT(
|
|
test::AssembleFailure(assembly),
|
|
Eq("2:33: Expected <opcode> or <result-id> at the beginning of an "
|
|
"instruction, found '4'."));
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
StructMember_NonReadableDecoration_Dropped) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %strct Block
|
|
OpMemberDecorate %strct 0 NonReadable
|
|
)" + CommonTypes() + R"(
|
|
%ptr_sb_strct = OpTypePointer StorageBuffer %strct
|
|
%myvar = OpVariable %ptr_sb_strct StorageBuffer
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
S Struct{
|
|
[[block]]
|
|
StructMember{field0: __u32}
|
|
StructMember{field1: __f32}
|
|
StructMember{field2: __array__u32_2}
|
|
}
|
|
Variable{
|
|
myvar
|
|
storage_buffer
|
|
__access_control_read_write__struct_S
|
|
}
|
|
)")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ColMajorDecoration_Dropped) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %s Block
|
|
OpMemberDecorate %s 0 ColMajor
|
|
%float = OpTypeFloat 32
|
|
%v2float = OpTypeVector %float 2
|
|
%m3v2float = OpTypeMatrix %v2float 3
|
|
|
|
%s = OpTypeStruct %m3v2float
|
|
%ptr_sb_s = OpTypePointer StorageBuffer %s
|
|
%myvar = OpVariable %ptr_sb_s StorageBuffer
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
S Struct{
|
|
[[block]]
|
|
StructMember{field0: __mat_2_3__f32}
|
|
}
|
|
Variable{
|
|
myvar
|
|
storage_buffer
|
|
__access_control_read_write__struct_S
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, MatrixStrideDecoration_Dropped) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %s Block
|
|
OpMemberDecorate %s 0 MatrixStride 8
|
|
%float = OpTypeFloat 32
|
|
%v2float = OpTypeVector %float 2
|
|
%m3v2float = OpTypeMatrix %v2float 3
|
|
|
|
%s = OpTypeStruct %m3v2float
|
|
%ptr_sb_s = OpTypePointer StorageBuffer %s
|
|
%myvar = OpVariable %ptr_sb_s StorageBuffer
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
S Struct{
|
|
[[block]]
|
|
StructMember{field0: __mat_2_3__f32}
|
|
}
|
|
Variable{
|
|
myvar
|
|
storage_buffer
|
|
__access_control_read_write__struct_S
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, RowMajorDecoration_IsError) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %s Block
|
|
OpMemberDecorate %s 0 RowMajor
|
|
%float = OpTypeFloat 32
|
|
%v2float = OpTypeVector %float 2
|
|
%m3v2float = OpTypeMatrix %v2float 3
|
|
|
|
%s = OpTypeStruct %m3v2float
|
|
%ptr_sb_s = OpTypePointer StorageBuffer %s
|
|
%myvar = OpVariable %ptr_sb_s StorageBuffer
|
|
)"));
|
|
EXPECT_FALSE(p->BuildAndParseInternalModuleExceptFunctions());
|
|
EXPECT_THAT(
|
|
p->error(),
|
|
Eq(R"(WGSL does not support row-major matrices: can't translate member 0 of %2 = OpTypeStruct %5)"))
|
|
<< p->error();
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, StorageBuffer_NonWritable_AllMembers) {
|
|
// Variable should have access(read)
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %s Block
|
|
OpMemberDecorate %s 0 NonWritable
|
|
OpMemberDecorate %s 1 NonWritable
|
|
%float = OpTypeFloat 32
|
|
|
|
%s = OpTypeStruct %float %float
|
|
%ptr_sb_s = OpTypePointer StorageBuffer %s
|
|
%myvar = OpVariable %ptr_sb_s StorageBuffer
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
S Struct{
|
|
[[block]]
|
|
StructMember{field0: __f32}
|
|
StructMember{field1: __f32}
|
|
}
|
|
Variable{
|
|
myvar
|
|
storage_buffer
|
|
__access_control_read_only__struct_S
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, StorageBuffer_NonWritable_NotAllMembers) {
|
|
// Variable should have access(read_write)
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %s Block
|
|
OpMemberDecorate %s 0 NonWritable
|
|
%float = OpTypeFloat 32
|
|
|
|
%s = OpTypeStruct %float %float
|
|
%ptr_sb_s = OpTypePointer StorageBuffer %s
|
|
%myvar = OpVariable %ptr_sb_s StorageBuffer
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
S Struct{
|
|
[[block]]
|
|
StructMember{field0: __f32}
|
|
StructMember{field1: __f32}
|
|
}
|
|
Variable{
|
|
myvar
|
|
storage_buffer
|
|
__access_control_read_write__struct_S
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(
|
|
SpvModuleScopeVarParserTest,
|
|
StorageBuffer_NonWritable_NotAllMembers_DuplicatedOnSameMember) { // NOLINT
|
|
// Variable should have access(read_write)
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %myvar "myvar"
|
|
OpDecorate %s Block
|
|
OpMemberDecorate %s 0 NonWritable
|
|
OpMemberDecorate %s 0 NonWritable ; same member. Don't double-count it
|
|
%float = OpTypeFloat 32
|
|
|
|
%s = OpTypeStruct %float %float
|
|
%ptr_sb_s = OpTypePointer StorageBuffer %s
|
|
%myvar = OpVariable %ptr_sb_s StorageBuffer
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
S Struct{
|
|
[[block]]
|
|
StructMember{field0: __f32}
|
|
StructMember{field1: __f32}
|
|
}
|
|
Variable{
|
|
myvar
|
|
storage_buffer
|
|
__access_control_read_write__struct_S
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ScalarSpecConstant_DeclareConst_True) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %c "myconst"
|
|
OpDecorate %c SpecId 12
|
|
%bool = OpTypeBool
|
|
%c = OpSpecConstantTrue %bool
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
VariableConst{
|
|
Decorations{
|
|
ConstantIdDecoration{12}
|
|
}
|
|
myconst
|
|
none
|
|
__bool
|
|
{
|
|
ScalarConstructor[not set]{true}
|
|
}
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ScalarSpecConstant_DeclareConst_False) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %c "myconst"
|
|
OpDecorate %c SpecId 12
|
|
%bool = OpTypeBool
|
|
%c = OpSpecConstantFalse %bool
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
VariableConst{
|
|
Decorations{
|
|
ConstantIdDecoration{12}
|
|
}
|
|
myconst
|
|
none
|
|
__bool
|
|
{
|
|
ScalarConstructor[not set]{false}
|
|
}
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ScalarSpecConstant_DeclareConst_U32) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %c "myconst"
|
|
OpDecorate %c SpecId 12
|
|
%uint = OpTypeInt 32 0
|
|
%c = OpSpecConstant %uint 42
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
VariableConst{
|
|
Decorations{
|
|
ConstantIdDecoration{12}
|
|
}
|
|
myconst
|
|
none
|
|
__u32
|
|
{
|
|
ScalarConstructor[not set]{42}
|
|
}
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ScalarSpecConstant_DeclareConst_I32) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %c "myconst"
|
|
OpDecorate %c SpecId 12
|
|
%int = OpTypeInt 32 1
|
|
%c = OpSpecConstant %int 42
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
VariableConst{
|
|
Decorations{
|
|
ConstantIdDecoration{12}
|
|
}
|
|
myconst
|
|
none
|
|
__i32
|
|
{
|
|
ScalarConstructor[not set]{42}
|
|
}
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ScalarSpecConstant_DeclareConst_F32) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %c "myconst"
|
|
OpDecorate %c SpecId 12
|
|
%float = OpTypeFloat 32
|
|
%c = OpSpecConstant %float 2.5
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
VariableConst{
|
|
Decorations{
|
|
ConstantIdDecoration{12}
|
|
}
|
|
myconst
|
|
none
|
|
__f32
|
|
{
|
|
ScalarConstructor[not set]{2.500000}
|
|
}
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest,
|
|
ScalarSpecConstant_DeclareConst_F32_WithoutSpecId) {
|
|
// When we don't have a spec ID, declare an undecorated module-scope constant.
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %c "myconst"
|
|
%float = OpTypeFloat 32
|
|
%c = OpSpecConstant %float 2.5
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
const auto module_str =
|
|
Demangler().Demangle(p->get_module(), p->get_module().to_str());
|
|
EXPECT_THAT(module_str, HasSubstr(R"(
|
|
VariableConst{
|
|
myconst
|
|
none
|
|
__f32
|
|
{
|
|
ScalarConstructor[not set]{2.500000}
|
|
}
|
|
}
|
|
})")) << module_str;
|
|
}
|
|
|
|
TEST_F(SpvModuleScopeVarParserTest, ScalarSpecConstant_UsedInFunction) {
|
|
auto p = parser(test::Assemble(R"(
|
|
OpName %c "myconst"
|
|
%float = OpTypeFloat 32
|
|
%c = OpSpecConstant %float 2.5
|
|
%floatfn = OpTypeFunction %float
|
|
%100 = OpFunction %float None %floatfn
|
|
%entry = OpLabel
|
|
%1 = OpIAdd %float %c %c
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)"));
|
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
|
FunctionEmitter fe(p.get(), *spirv_function(p.get(), 100));
|
|
EXPECT_TRUE(fe.EmitBody()) << p->error();
|
|
EXPECT_TRUE(p->error().empty());
|
|
EXPECT_THAT(ToString(p->get_module(), fe.ast_body()), HasSubstr(R"(
|
|
VariableConst{
|
|
x_1
|
|
none
|
|
__f32
|
|
{
|
|
Binary[not set]{
|
|
Identifier[not set]{myconst}
|
|
add
|
|
Identifier[not set]{myconst}
|
|
}
|
|
}
|
|
})"))
|
|
<< ToString(p->get_module(), fe.ast_body());
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace spirv
|
|
} // namespace reader
|
|
} // namespace tint
|