[spirv-reader] Emit function variable initializers
Bug: tint:3 Change-Id: I1c7fa2c282e6dd95ea048df83f608d4462150394 Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/18820 Reviewed-by: dan sinclair <dsinclair@google.com>
This commit is contained in:
parent
4fa1ceb094
commit
4194d0f948
|
@ -18,6 +18,11 @@
|
||||||
#include "source/opt/function.h"
|
#include "source/opt/function.h"
|
||||||
#include "source/opt/instruction.h"
|
#include "source/opt/instruction.h"
|
||||||
#include "source/opt/module.h"
|
#include "source/opt/module.h"
|
||||||
|
#include "src/ast/bool_literal.h"
|
||||||
|
#include "src/ast/float_literal.h"
|
||||||
|
#include "src/ast/int_literal.h"
|
||||||
|
#include "src/ast/scalar_constructor_expression.h"
|
||||||
|
#include "src/ast/uint_literal.h"
|
||||||
#include "src/ast/variable.h"
|
#include "src/ast/variable.h"
|
||||||
#include "src/ast/variable_decl_statement.h"
|
#include "src/ast/variable_decl_statement.h"
|
||||||
#include "src/reader/spirv/fail_stream.h"
|
#include "src/reader/spirv/fail_stream.h"
|
||||||
|
@ -32,6 +37,8 @@ FunctionEmitter::FunctionEmitter(ParserImpl* pi,
|
||||||
: parser_impl_(*pi),
|
: parser_impl_(*pi),
|
||||||
ast_module_(pi->get_module()),
|
ast_module_(pi->get_module()),
|
||||||
ir_context_(*(pi->ir_context())),
|
ir_context_(*(pi->ir_context())),
|
||||||
|
def_use_mgr_(ir_context_.get_def_use_mgr()),
|
||||||
|
constant_mgr_(ir_context_.get_constant_mgr()),
|
||||||
type_mgr_(ir_context_.get_type_mgr()),
|
type_mgr_(ir_context_.get_type_mgr()),
|
||||||
fail_stream_(pi->fail_stream()),
|
fail_stream_(pi->fail_stream()),
|
||||||
namer_(pi->namer()),
|
namer_(pi->namer()),
|
||||||
|
@ -142,6 +149,13 @@ bool FunctionEmitter::EmitFunctionVariables() {
|
||||||
auto var =
|
auto var =
|
||||||
parser_impl_.MakeVariable(inst.result_id(),
|
parser_impl_.MakeVariable(inst.result_id(),
|
||||||
ast::StorageClass::kNone, var_store_type);
|
ast::StorageClass::kNone, var_store_type);
|
||||||
|
if (inst.NumInOperands() > 1) {
|
||||||
|
// SPIR-V initializers are always constants.
|
||||||
|
// (OpenCL also allows the ID of an OpVariable, but we don't handle that
|
||||||
|
// here.)
|
||||||
|
var->set_constructor(
|
||||||
|
parser_impl_.MakeConstantExpression(inst.GetSingleWordInOperand(1)));
|
||||||
|
}
|
||||||
// TODO(dneto): Add the initializer via Variable::set_constructor.
|
// TODO(dneto): Add the initializer via Variable::set_constructor.
|
||||||
auto var_decl_stmt = std::make_unique<ast::VariableDeclStatement>(std::move(var));
|
auto var_decl_stmt = std::make_unique<ast::VariableDeclStatement>(std::move(var));
|
||||||
ast_body_.emplace_back(std::move(var_decl_stmt));
|
ast_body_.emplace_back(std::move(var_decl_stmt));
|
||||||
|
|
|
@ -15,9 +15,11 @@
|
||||||
#ifndef SRC_READER_SPIRV_FUNCTION_H_
|
#ifndef SRC_READER_SPIRV_FUNCTION_H_
|
||||||
#define SRC_READER_SPIRV_FUNCTION_H_
|
#define SRC_READER_SPIRV_FUNCTION_H_
|
||||||
|
|
||||||
|
#include "source/opt/constants.h"
|
||||||
#include "source/opt/function.h"
|
#include "source/opt/function.h"
|
||||||
#include "source/opt/ir_context.h"
|
#include "source/opt/ir_context.h"
|
||||||
#include "source/opt/type_manager.h"
|
#include "source/opt/type_manager.h"
|
||||||
|
#include "src/ast/expression.h"
|
||||||
#include "src/ast/module.h"
|
#include "src/ast/module.h"
|
||||||
#include "src/reader/spirv/fail_stream.h"
|
#include "src/reader/spirv/fail_stream.h"
|
||||||
#include "src/reader/spirv/namer.h"
|
#include "src/reader/spirv/namer.h"
|
||||||
|
@ -73,6 +75,8 @@ class FunctionEmitter {
|
||||||
ParserImpl& parser_impl_;
|
ParserImpl& parser_impl_;
|
||||||
ast::Module& ast_module_;
|
ast::Module& ast_module_;
|
||||||
spvtools::opt::IRContext& ir_context_;
|
spvtools::opt::IRContext& ir_context_;
|
||||||
|
spvtools::opt::analysis::DefUseManager* def_use_mgr_;
|
||||||
|
spvtools::opt::analysis::ConstantManager* constant_mgr_;
|
||||||
spvtools::opt::analysis::TypeManager* type_mgr_;
|
spvtools::opt::analysis::TypeManager* type_mgr_;
|
||||||
FailStream& fail_stream_;
|
FailStream& fail_stream_;
|
||||||
Namer& namer_;
|
Namer& namer_;
|
||||||
|
|
|
@ -43,13 +43,30 @@ std::string CommonTypes() {
|
||||||
return R"(
|
return R"(
|
||||||
%void = OpTypeVoid
|
%void = OpTypeVoid
|
||||||
%voidfn = OpTypeFunction %void
|
%voidfn = OpTypeFunction %void
|
||||||
|
|
||||||
|
%bool = OpTypeBool
|
||||||
%float = OpTypeFloat 32
|
%float = OpTypeFloat 32
|
||||||
%uint = OpTypeInt 32 0
|
%uint = OpTypeInt 32 0
|
||||||
%int = OpTypeInt 32 1
|
%int = OpTypeInt 32 1
|
||||||
%float_0 = OpConstant %float 0.0
|
|
||||||
|
%ptr_bool = OpTypePointer Function %bool
|
||||||
%ptr_float = OpTypePointer Function %float
|
%ptr_float = OpTypePointer Function %float
|
||||||
%ptr_uint = OpTypePointer Function %uint
|
%ptr_uint = OpTypePointer Function %uint
|
||||||
%ptr_int = OpTypePointer Function %int
|
%ptr_int = OpTypePointer Function %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
|
||||||
|
|
||||||
|
%v2float = OpTypeVector %float 2
|
||||||
|
%m3v2float = OpTypeMatrix %v2float 3
|
||||||
|
|
||||||
|
%arr2uint = OpTypeArray %uint %uint_2
|
||||||
|
%strct = OpTypeStruct %uint %float %arr2uint
|
||||||
)";
|
)";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -167,6 +184,296 @@ VariableDeclStatement{
|
||||||
)"));
|
)"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_F(SpvParserTest, EmitFunctionVariables_ScalarInitializers) {
|
||||||
|
auto p = parser(
|
||||||
|
test::Assemble(Names({"a", "b", "c", "d", "e"}) + CommonTypes() + R"(
|
||||||
|
%100 = OpFunction %void None %voidfn
|
||||||
|
%entry = OpLabel
|
||||||
|
%a = OpVariable %ptr_bool Function %true
|
||||||
|
%b = OpVariable %ptr_bool Function %false
|
||||||
|
%c = OpVariable %ptr_int Function %int_m1
|
||||||
|
%d = OpVariable %ptr_uint Function %uint_1
|
||||||
|
%e = OpVariable %ptr_float Function %float_1p5
|
||||||
|
OpReturn
|
||||||
|
OpFunctionEnd
|
||||||
|
)"));
|
||||||
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
||||||
|
FunctionEmitter fe(p, *spirv_function(100));
|
||||||
|
EXPECT_TRUE(fe.EmitFunctionVariables());
|
||||||
|
|
||||||
|
EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
a
|
||||||
|
none
|
||||||
|
__bool
|
||||||
|
{
|
||||||
|
ScalarConstructor{true}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
b
|
||||||
|
none
|
||||||
|
__bool
|
||||||
|
{
|
||||||
|
ScalarConstructor{false}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
c
|
||||||
|
none
|
||||||
|
__i32
|
||||||
|
{
|
||||||
|
ScalarConstructor{-1}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
d
|
||||||
|
none
|
||||||
|
__u32
|
||||||
|
{
|
||||||
|
ScalarConstructor{1}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
e
|
||||||
|
none
|
||||||
|
__f32
|
||||||
|
{
|
||||||
|
ScalarConstructor{1.500000}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)"));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(SpvParserTest, EmitFunctionVariables_ScalarNullInitializers) {
|
||||||
|
auto p =
|
||||||
|
parser(test::Assemble(Names({"a", "b", "c", "d"}) + CommonTypes() + R"(
|
||||||
|
%null_bool = OpConstantNull %bool
|
||||||
|
%null_int = OpConstantNull %int
|
||||||
|
%null_uint = OpConstantNull %uint
|
||||||
|
%null_float = OpConstantNull %float
|
||||||
|
|
||||||
|
%100 = OpFunction %void None %voidfn
|
||||||
|
%entry = OpLabel
|
||||||
|
%a = OpVariable %ptr_bool Function %null_bool
|
||||||
|
%b = OpVariable %ptr_int Function %null_int
|
||||||
|
%c = OpVariable %ptr_uint Function %null_uint
|
||||||
|
%d = OpVariable %ptr_float Function %null_float
|
||||||
|
OpReturn
|
||||||
|
OpFunctionEnd
|
||||||
|
)"));
|
||||||
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions()) << p->error();
|
||||||
|
FunctionEmitter fe(p, *spirv_function(100));
|
||||||
|
EXPECT_TRUE(fe.EmitFunctionVariables());
|
||||||
|
|
||||||
|
EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
a
|
||||||
|
none
|
||||||
|
__bool
|
||||||
|
{
|
||||||
|
ScalarConstructor{false}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
b
|
||||||
|
none
|
||||||
|
__i32
|
||||||
|
{
|
||||||
|
ScalarConstructor{0}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
c
|
||||||
|
none
|
||||||
|
__u32
|
||||||
|
{
|
||||||
|
ScalarConstructor{0}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
d
|
||||||
|
none
|
||||||
|
__f32
|
||||||
|
{
|
||||||
|
ScalarConstructor{0.000000}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)"));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(SpvParserTest, EmitFunctionVariables_VectorInitializer) {
|
||||||
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
||||||
|
%ptr = OpTypePointer Function %v2float
|
||||||
|
%two = OpConstant %float 2.0
|
||||||
|
%const = OpConstantComposite %v2float %float_1p5 %two
|
||||||
|
|
||||||
|
%100 = OpFunction %void None %voidfn
|
||||||
|
%entry = OpLabel
|
||||||
|
%200 = OpVariable %ptr Function %const
|
||||||
|
OpReturn
|
||||||
|
OpFunctionEnd
|
||||||
|
)"));
|
||||||
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
||||||
|
FunctionEmitter fe(p, *spirv_function(100));
|
||||||
|
EXPECT_TRUE(fe.EmitFunctionVariables());
|
||||||
|
|
||||||
|
EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
x_200
|
||||||
|
none
|
||||||
|
__vec_2__f32
|
||||||
|
{
|
||||||
|
TypeConstructor{
|
||||||
|
__vec_2__f32
|
||||||
|
ScalarConstructor{1.500000}
|
||||||
|
ScalarConstructor{2.000000}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)")) << ToString(fe.ast_body());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(SpvParserTest, EmitFunctionVariables_MatrixInitializer) {
|
||||||
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
||||||
|
%ptr = OpTypePointer Function %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
|
||||||
|
|
||||||
|
%100 = OpFunction %void None %voidfn
|
||||||
|
%entry = OpLabel
|
||||||
|
%200 = OpVariable %ptr Function %const
|
||||||
|
OpReturn
|
||||||
|
OpFunctionEnd
|
||||||
|
)"));
|
||||||
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
||||||
|
FunctionEmitter fe(p, *spirv_function(100));
|
||||||
|
EXPECT_TRUE(fe.EmitFunctionVariables());
|
||||||
|
|
||||||
|
EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
x_200
|
||||||
|
none
|
||||||
|
__mat_2_3__f32
|
||||||
|
{
|
||||||
|
TypeConstructor{
|
||||||
|
__mat_2_3__f32
|
||||||
|
TypeConstructor{
|
||||||
|
__vec_2__f32
|
||||||
|
ScalarConstructor{1.500000}
|
||||||
|
ScalarConstructor{2.000000}
|
||||||
|
}
|
||||||
|
TypeConstructor{
|
||||||
|
__vec_2__f32
|
||||||
|
ScalarConstructor{2.000000}
|
||||||
|
ScalarConstructor{3.000000}
|
||||||
|
}
|
||||||
|
TypeConstructor{
|
||||||
|
__vec_2__f32
|
||||||
|
ScalarConstructor{3.000000}
|
||||||
|
ScalarConstructor{4.000000}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)"));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(SpvParserTest, EmitFunctionVariables_ArrayInitializer) {
|
||||||
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
||||||
|
%ptr = OpTypePointer Function %arr2uint
|
||||||
|
%two = OpConstant %uint 2
|
||||||
|
%const = OpConstantComposite %arr2uint %uint_1 %two
|
||||||
|
|
||||||
|
%100 = OpFunction %void None %voidfn
|
||||||
|
%entry = OpLabel
|
||||||
|
%200 = OpVariable %ptr Function %const
|
||||||
|
OpReturn
|
||||||
|
OpFunctionEnd
|
||||||
|
)"));
|
||||||
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
||||||
|
FunctionEmitter fe(p, *spirv_function(100));
|
||||||
|
EXPECT_TRUE(fe.EmitFunctionVariables());
|
||||||
|
|
||||||
|
EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
x_200
|
||||||
|
none
|
||||||
|
__array__u32_2
|
||||||
|
{
|
||||||
|
TypeConstructor{
|
||||||
|
__array__u32_2
|
||||||
|
ScalarConstructor{1}
|
||||||
|
ScalarConstructor{2}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)"));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(SpvParserTest, EmitFunctionVariables_StructInitializer) {
|
||||||
|
auto p = parser(test::Assemble(CommonTypes() + R"(
|
||||||
|
%ptr = OpTypePointer Function %strct
|
||||||
|
%two = OpConstant %uint 2
|
||||||
|
%arrconst = OpConstantComposite %arr2uint %uint_1 %two
|
||||||
|
%const = OpConstantComposite %strct %uint_1 %float_1p5 %arrconst
|
||||||
|
|
||||||
|
%100 = OpFunction %void None %voidfn
|
||||||
|
%entry = OpLabel
|
||||||
|
%200 = OpVariable %ptr Function %const
|
||||||
|
OpReturn
|
||||||
|
OpFunctionEnd
|
||||||
|
)"));
|
||||||
|
ASSERT_TRUE(p->BuildAndParseInternalModuleExceptFunctions());
|
||||||
|
FunctionEmitter fe(p, *spirv_function(100));
|
||||||
|
EXPECT_TRUE(fe.EmitFunctionVariables());
|
||||||
|
|
||||||
|
EXPECT_THAT(ToString(fe.ast_body()), HasSubstr(R"(VariableDeclStatement{
|
||||||
|
Variable{
|
||||||
|
x_200
|
||||||
|
none
|
||||||
|
__struct_S
|
||||||
|
{
|
||||||
|
TypeConstructor{
|
||||||
|
__struct_S
|
||||||
|
ScalarConstructor{1}
|
||||||
|
ScalarConstructor{1.500000}
|
||||||
|
TypeConstructor{
|
||||||
|
__array__u32_2
|
||||||
|
ScalarConstructor{1}
|
||||||
|
ScalarConstructor{2}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)"));
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
} // namespace spirv
|
} // namespace spirv
|
||||||
} // namespace reader
|
} // namespace reader
|
||||||
|
|
|
@ -25,6 +25,7 @@
|
||||||
|
|
||||||
#include "source/opt/basic_block.h"
|
#include "source/opt/basic_block.h"
|
||||||
#include "source/opt/build_module.h"
|
#include "source/opt/build_module.h"
|
||||||
|
#include "source/opt/constants.h"
|
||||||
#include "source/opt/decoration_manager.h"
|
#include "source/opt/decoration_manager.h"
|
||||||
#include "source/opt/function.h"
|
#include "source/opt/function.h"
|
||||||
#include "source/opt/instruction.h"
|
#include "source/opt/instruction.h"
|
||||||
|
@ -32,8 +33,12 @@
|
||||||
#include "source/opt/type_manager.h"
|
#include "source/opt/type_manager.h"
|
||||||
#include "source/opt/types.h"
|
#include "source/opt/types.h"
|
||||||
#include "spirv-tools/libspirv.hpp"
|
#include "spirv-tools/libspirv.hpp"
|
||||||
|
#include "src/ast/bool_literal.h"
|
||||||
#include "src/ast/builtin_decoration.h"
|
#include "src/ast/builtin_decoration.h"
|
||||||
#include "src/ast/decorated_variable.h"
|
#include "src/ast/decorated_variable.h"
|
||||||
|
#include "src/ast/float_literal.h"
|
||||||
|
#include "src/ast/int_literal.h"
|
||||||
|
#include "src/ast/scalar_constructor_expression.h"
|
||||||
#include "src/ast/struct.h"
|
#include "src/ast/struct.h"
|
||||||
#include "src/ast/struct_decoration.h"
|
#include "src/ast/struct_decoration.h"
|
||||||
#include "src/ast/struct_member.h"
|
#include "src/ast/struct_member.h"
|
||||||
|
@ -51,6 +56,8 @@
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
#include "src/ast/type/vector_type.h"
|
#include "src/ast/type/vector_type.h"
|
||||||
#include "src/ast/type/void_type.h"
|
#include "src/ast/type/void_type.h"
|
||||||
|
#include "src/ast/type_constructor_expression.h"
|
||||||
|
#include "src/ast/uint_literal.h"
|
||||||
#include "src/ast/variable.h"
|
#include "src/ast/variable.h"
|
||||||
#include "src/ast/variable_decl_statement.h"
|
#include "src/ast/variable_decl_statement.h"
|
||||||
#include "src/ast/variable_decoration.h"
|
#include "src/ast/variable_decoration.h"
|
||||||
|
@ -750,6 +757,80 @@ std::unique_ptr<ast::Variable> ParserImpl::MakeVariable(uint32_t id,
|
||||||
return ast_var;
|
return ast_var;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<ast::Expression> ParserImpl::MakeConstantExpression(
|
||||||
|
uint32_t id) {
|
||||||
|
if (!success_) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
const auto* inst = def_use_mgr_->GetDef(id);
|
||||||
|
if (inst == nullptr) {
|
||||||
|
Fail() << "ID " << id << " is not a registered instruction";
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
auto* ast_type = ConvertType(inst->type_id());
|
||||||
|
if (ast_type == nullptr) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
// TODO(dneto): Handle spec constants too?
|
||||||
|
const auto* spirv_const = constant_mgr_->FindDeclaredConstant(id);
|
||||||
|
if (spirv_const == nullptr) {
|
||||||
|
Fail() << "ID " << id << " is not a constant";
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
// TODO(dneto): Note: NullConstant for int, uint, float map to a regular 0.
|
||||||
|
// So canonicalization should map that way too.
|
||||||
|
// Currently "null<type>" is missing from the WGSL parser.
|
||||||
|
// See https://bugs.chromium.org/p/tint/issues/detail?id=34
|
||||||
|
if (ast_type->IsU32()) {
|
||||||
|
return std::make_unique<ast::ScalarConstructorExpression>(
|
||||||
|
std::make_unique<ast::UintLiteral>(ast_type, spirv_const->GetU32()));
|
||||||
|
}
|
||||||
|
if (ast_type->IsI32()) {
|
||||||
|
return std::make_unique<ast::ScalarConstructorExpression>(
|
||||||
|
std::make_unique<ast::IntLiteral>(ast_type, spirv_const->GetS32()));
|
||||||
|
}
|
||||||
|
if (ast_type->IsF32()) {
|
||||||
|
return std::make_unique<ast::ScalarConstructorExpression>(
|
||||||
|
std::make_unique<ast::FloatLiteral>(ast_type, spirv_const->GetFloat()));
|
||||||
|
}
|
||||||
|
if (ast_type->IsBool()) {
|
||||||
|
const bool value = spirv_const->AsNullConstant() ? false :
|
||||||
|
spirv_const->AsBoolConstant()->value();
|
||||||
|
return std::make_unique<ast::ScalarConstructorExpression>(
|
||||||
|
std::make_unique<ast::BoolLiteral>(
|
||||||
|
ast_type, value));
|
||||||
|
}
|
||||||
|
auto spirv_composite_const = spirv_const->AsCompositeConstant();
|
||||||
|
if (spirv_composite_const != nullptr) {
|
||||||
|
// Handle vector, matrix, array, and struct
|
||||||
|
|
||||||
|
// TODO(dneto): Handle the spirv_composite_const->IsZero() case specially.
|
||||||
|
// See https://github.com/gpuweb/gpuweb/issues/685
|
||||||
|
|
||||||
|
// Generate a composite from explicit components.
|
||||||
|
ast::ExpressionList ast_components;
|
||||||
|
for (const auto* component : spirv_composite_const->GetComponents()) {
|
||||||
|
auto* def = constant_mgr_->GetDefiningInstruction(component);
|
||||||
|
if (def == nullptr) {
|
||||||
|
Fail() << "internal error: SPIR-V constant doesn't have defining "
|
||||||
|
"instruction";
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
auto ast_component = MakeConstantExpression(def->result_id());
|
||||||
|
if (!success_) {
|
||||||
|
// We've already emitted a diagnostic.
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
ast_components.emplace_back(std::move(ast_component));
|
||||||
|
}
|
||||||
|
return std::make_unique<ast::TypeConstructorExpression>(
|
||||||
|
ast_type, std::move(ast_components));
|
||||||
|
}
|
||||||
|
Fail() << "Unhandled constant type " << inst->type_id() << " for value ID "
|
||||||
|
<< id;
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
bool ParserImpl::EmitFunctions() {
|
bool ParserImpl::EmitFunctions() {
|
||||||
if (!success_) {
|
if (!success_) {
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -221,6 +221,11 @@ class ParserImpl : Reader {
|
||||||
ast::StorageClass sc,
|
ast::StorageClass sc,
|
||||||
ast::type::Type* type);
|
ast::type::Type* type);
|
||||||
|
|
||||||
|
/// Creates an AST expression node for a SPIR-V constant.
|
||||||
|
/// @param id the SPIR-V ID of the constant
|
||||||
|
/// @returns a new Literal node
|
||||||
|
std::unique_ptr<ast::Expression> MakeConstantExpression(uint32_t id);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/// Converts a specific SPIR-V type to a Tint type. Integer case
|
/// Converts a specific SPIR-V type to a Tint type. Integer case
|
||||||
ast::type::Type* ConvertType(const spvtools::opt::analysis::Integer* int_ty);
|
ast::type::Type* ConvertType(const spvtools::opt::analysis::Integer* int_ty);
|
||||||
|
|
Loading…
Reference in New Issue