// Copyright 2021 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 "src/program_builder.h" #include "src/ast/assignment_statement.h" #include "src/ast/call_statement.h" #include "src/ast/variable_decl_statement.h" #include "src/debug.h" #include "src/demangler.h" #include "src/sem/expression.h" namespace tint { ProgramBuilder::ProgramBuilder() : id_(ProgramID::New()), ast_(ast_nodes_.Create(id_, Source{})) {} ProgramBuilder::ProgramBuilder(ProgramBuilder&& rhs) : id_(std::move(rhs.id_)), types_(std::move(rhs.types_)), ast_nodes_(std::move(rhs.ast_nodes_)), sem_nodes_(std::move(rhs.sem_nodes_)), ast_(rhs.ast_), sem_(std::move(rhs.sem_)), symbols_(std::move(rhs.symbols_)) { rhs.MarkAsMoved(); } ProgramBuilder::~ProgramBuilder() = default; ProgramBuilder& ProgramBuilder::operator=(ProgramBuilder&& rhs) { rhs.MarkAsMoved(); AssertNotMoved(); id_ = std::move(rhs.id_); types_ = std::move(rhs.types_); ast_nodes_ = std::move(rhs.ast_nodes_); sem_nodes_ = std::move(rhs.sem_nodes_); ast_ = rhs.ast_; sem_ = std::move(rhs.sem_); symbols_ = std::move(rhs.symbols_); return *this; } ProgramBuilder ProgramBuilder::Wrap(const Program* program) { ProgramBuilder builder; builder.id_ = program->ID(); builder.types_ = sem::Manager::Wrap(program->Types()); builder.ast_ = builder.create( program->AST().source(), program->AST().GlobalDeclarations()); builder.sem_ = sem::Info::Wrap(program->Sem()); builder.symbols_ = program->Symbols(); builder.diagnostics_ = program->Diagnostics(); return builder; } bool ProgramBuilder::IsValid() const { return !diagnostics_.contains_errors(); } std::string ProgramBuilder::str(const ast::Node* node) const { return Demangler().Demangle(Symbols(), node->str(Sem())); } void ProgramBuilder::MarkAsMoved() { AssertNotMoved(); moved_ = true; } void ProgramBuilder::AssertNotMoved() const { if (moved_) { TINT_ICE(const_cast(this)->Diagnostics()) << "Attempting to use ProgramBuilder after it has been moved"; } } sem::Type* ProgramBuilder::TypeOf(const ast::Expression* expr) const { auto* sem = Sem().Get(expr); return sem ? sem->Type() : nullptr; } const sem::Type* ProgramBuilder::TypeOf(const ast::Type* type) const { return Sem().Get(type); } ast::ConstructorExpression* ProgramBuilder::ConstructValueFilledWith( const ast::Type* type, int elem_value) { CloneContext ctx(this); if (type->Is()) { return create( create(elem_value == 0 ? false : true)); } if (type->Is()) { return create( create(static_cast(elem_value))); } if (type->Is()) { return create( create(static_cast(elem_value))); } if (type->Is()) { return create( create(static_cast(elem_value))); } if (auto* v = type->As()) { ast::ExpressionList el(v->size()); for (size_t i = 0; i < el.size(); i++) { el[i] = ConstructValueFilledWith(ctx.Clone(v->type()), elem_value); } return create(const_cast(type), std::move(el)); } if (auto* m = type->As()) { ast::ExpressionList el(m->columns()); for (size_t i = 0; i < el.size(); i++) { auto* col_vec_type = create(ctx.Clone(m->type()), m->rows()); el[i] = ConstructValueFilledWith(col_vec_type, elem_value); } return create(const_cast(type), std::move(el)); } if (auto* tn = type->As()) { if (auto* lookup = AST().LookupType(tn->name())) { if (auto* alias = lookup->As()) { return ConstructValueFilledWith(ctx.Clone(alias->type()), elem_value); } } TINT_ICE(diagnostics_) << "unable to find NamedType '" << Symbols().NameFor(tn->name()) << "'"; return nullptr; } TINT_ICE(diagnostics_) << "unhandled type: " << type->TypeInfo().name; return nullptr; } typ::Type ProgramBuilder::TypesBuilder::MaybeCreateTypename( typ::Type type) const { if (auto* nt = As(type.ast)) { return {type_name(nt->name()), type.sem}; } return type; } ProgramBuilder::TypesBuilder::TypesBuilder(ProgramBuilder* pb) : builder(pb) {} ast::Statement* ProgramBuilder::WrapInStatement(ast::Literal* lit) { return WrapInStatement(create(lit)); } ast::Statement* ProgramBuilder::WrapInStatement(ast::Expression* expr) { // Create a temporary variable of inferred type from expr. return Decl(Var(symbols_.New(), nullptr, ast::StorageClass::kFunction, expr)); } ast::VariableDeclStatement* ProgramBuilder::WrapInStatement(ast::Variable* v) { return create(v); } ast::Statement* ProgramBuilder::WrapInStatement(ast::Statement* stmt) { return stmt; } ast::Function* ProgramBuilder::WrapInFunction(ast::StatementList stmts) { return Func("test_function", {}, ty.void_(), std::move(stmts), {create(ast::PipelineStage::kCompute)}); } } // namespace tint