// Copyright (C) 2020 Free Software Foundation, Inc. // This file is part of GCC. // GCC is free software; you can redistribute it and/or modify it under // the terms of the GNU General Public License as published by the Free // Software Foundation; either version 3, or (at your option) any later // version. // GCC is distributed in the hope that it will be useful, but WITHOUT ANY // WARRANTY; without even the implied warranty of MERCHANTABILITY or // FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License // for more details. // You should have received a copy of the GNU General Public License // along with GCC; see the file COPYING3. If not see // . #ifndef RUST_AST_LOWER_ITEM #define RUST_AST_LOWER_ITEM #include "rust-diagnostics.h" #include "rust-ast-lower-base.h" #include "rust-ast-lower-type.h" #include "rust-ast-lower-implitem.h" #include "rust-ast-lower-stmt.h" #include "rust-ast-lower-expr.h" #include "rust-ast-lower-pattern.h" #include "rust-ast-lower-block.h" namespace Rust { namespace HIR { class ASTLoweringItem : public ASTLoweringBase { using Rust::HIR::ASTLoweringBase::visit; public: static HIR::Item *translate (AST::Item *item) { ASTLoweringItem resolver; item->accept_vis (resolver); // this is useful for debugging // if (resolver.translated == nullptr) // { // rust_fatal_error (item->get_locus_slow (), "failed to lower: %s", // item->as_string ().c_str ()); // return nullptr; // } return resolver.translated; } void visit (AST::TypeAlias &alias) override { std::vector > where_clause_items; HIR::WhereClause where_clause (std::move (where_clause_items)); HIR::Visibility vis = HIR::Visibility::create_public (); std::vector > generic_params; if (alias.has_generics ()) generic_params = lower_generic_params (alias.get_generic_params ()); HIR::Type *existing_type = ASTLoweringType::translate (alias.get_type_aliased ().get ()); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, alias.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id (crate_num)); translated = new HIR::TypeAlias (mapping, alias.get_new_type_name (), std::move (generic_params), std::move (where_clause), std::unique_ptr (existing_type), std::move (vis), alias.get_outer_attrs (), alias.get_locus ()); mappings->insert_defid_mapping (mapping.get_defid (), translated); mappings->insert_hir_item (mapping.get_crate_num (), mapping.get_hirid (), translated); mappings->insert_location (crate_num, mapping.get_hirid (), alias.get_locus ()); } void visit (AST::TupleStruct &struct_decl) override { std::vector > generic_params; if (struct_decl.has_generics ()) { generic_params = lower_generic_params (struct_decl.get_generic_params ()); } std::vector > where_clause_items; HIR::WhereClause where_clause (std::move (where_clause_items)); HIR::Visibility vis = HIR::Visibility::create_public (); std::vector fields; struct_decl.iterate ([&] (AST::TupleField &field) mutable -> bool { HIR::Visibility vis = HIR::Visibility::create_public (); HIR::Type *type = ASTLoweringType::translate (field.get_field_type ().get ()); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, field.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id ( crate_num)); // FIXME // AST::TupleField is missing Location info Location field_locus; HIR::TupleField translated_field (mapping, std::unique_ptr (type), vis, field_locus, field.get_outer_attrs ()); fields.push_back (std::move (translated_field)); return true; }); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, struct_decl.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id (crate_num)); translated = new HIR::TupleStruct (mapping, std::move (fields), struct_decl.get_identifier (), std::move (generic_params), std::move (where_clause), vis, struct_decl.get_outer_attrs (), struct_decl.get_locus ()); mappings->insert_defid_mapping (mapping.get_defid (), translated); mappings->insert_hir_item (mapping.get_crate_num (), mapping.get_hirid (), translated); mappings->insert_location (crate_num, mapping.get_hirid (), struct_decl.get_locus ()); } void visit (AST::StructStruct &struct_decl) override { std::vector > generic_params; if (struct_decl.has_generics ()) { generic_params = lower_generic_params (struct_decl.get_generic_params ()); } std::vector > where_clause_items; HIR::WhereClause where_clause (std::move (where_clause_items)); HIR::Visibility vis = HIR::Visibility::create_public (); bool is_unit = struct_decl.is_unit_struct (); std::vector fields; struct_decl.iterate ([&] (AST::StructField &field) mutable -> bool { HIR::Visibility vis = HIR::Visibility::create_public (); HIR::Type *type = ASTLoweringType::translate (field.get_field_type ().get ()); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, field.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id ( crate_num)); // FIXME // AST::StructField is missing Location info Location field_locus; HIR::StructField translated_field (mapping, field.get_field_name (), std::unique_ptr (type), vis, field_locus, field.get_outer_attrs ()); fields.push_back (std::move (translated_field)); return true; }); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, struct_decl.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id (crate_num)); translated = new HIR::StructStruct (mapping, std::move (fields), struct_decl.get_identifier (), std::move (generic_params), std::move (where_clause), is_unit, vis, struct_decl.get_outer_attrs (), struct_decl.get_locus ()); mappings->insert_defid_mapping (mapping.get_defid (), translated); mappings->insert_hir_item (mapping.get_crate_num (), mapping.get_hirid (), translated); mappings->insert_location (crate_num, mapping.get_hirid (), struct_decl.get_locus ()); } void visit (AST::StaticItem &var) override { HIR::Visibility vis = HIR::Visibility::create_public (); HIR::Type *type = ASTLoweringType::translate (var.get_type ().get ()); HIR::Expr *expr = ASTLoweringExpr::translate (var.get_expr ().get ()); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, var.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id (crate_num)); translated = new HIR::StaticItem (mapping, var.get_identifier (), var.is_mutable (), std::unique_ptr (type), std::unique_ptr (expr), vis, var.get_outer_attrs (), var.get_locus ()); mappings->insert_defid_mapping (mapping.get_defid (), translated); mappings->insert_hir_item (mapping.get_crate_num (), mapping.get_hirid (), translated); mappings->insert_location (crate_num, mapping.get_hirid (), var.get_locus ()); } void visit (AST::ConstantItem &constant) override { HIR::Visibility vis = HIR::Visibility::create_public (); HIR::Type *type = ASTLoweringType::translate (constant.get_type ().get ()); HIR::Expr *expr = ASTLoweringExpr::translate (constant.get_expr ().get ()); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, constant.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id (crate_num)); translated = new HIR::ConstantItem (mapping, constant.get_identifier (), vis, std::unique_ptr (type), std::unique_ptr (expr), constant.get_outer_attrs (), constant.get_locus ()); mappings->insert_defid_mapping (mapping.get_defid (), translated); mappings->insert_hir_item (mapping.get_crate_num (), mapping.get_hirid (), translated); mappings->insert_location (crate_num, mapping.get_hirid (), constant.get_locus ()); } void visit (AST::Function &function) override { // ignore for now and leave empty std::vector > where_clause_items; HIR::WhereClause where_clause (std::move (where_clause_items)); HIR::FunctionQualifiers qualifiers ( HIR::FunctionQualifiers::AsyncConstStatus::NONE, false); HIR::Visibility vis = HIR::Visibility::create_public (); // need std::vector > generic_params; if (function.has_generics ()) { generic_params = lower_generic_params (function.get_generic_params ()); } Identifier function_name = function.get_function_name (); Location locus = function.get_locus (); std::unique_ptr return_type = function.has_return_type () ? std::unique_ptr ( ASTLoweringType::translate (function.get_return_type ().get ())) : nullptr; std::vector function_params; for (auto ¶m : function.get_function_params ()) { auto translated_pattern = std::unique_ptr ( ASTLoweringPattern::translate (param.get_pattern ().get ())); auto translated_type = std::unique_ptr ( ASTLoweringType::translate (param.get_type ().get ())); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, param.get_node_id (), mappings->get_next_hir_id (crate_num), UNKNOWN_LOCAL_DEFID); auto hir_param = HIR::FunctionParam (mapping, std::move (translated_pattern), std::move (translated_type), param.get_locus ()); function_params.push_back (hir_param); } bool terminated = false; std::unique_ptr function_body = std::unique_ptr ( ASTLoweringBlock::translate (function.get_definition ().get (), &terminated)); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, function.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id (crate_num)); mappings->insert_location (crate_num, function_body->get_mappings ().get_hirid (), function.get_locus ()); auto fn = new HIR::Function (mapping, std::move (function_name), std::move (qualifiers), std::move (generic_params), std::move (function_params), std::move (return_type), std::move (where_clause), std::move (function_body), std::move (vis), function.get_outer_attrs (), locus); mappings->insert_defid_mapping (mapping.get_defid (), translated); mappings->insert_hir_item (mapping.get_crate_num (), mapping.get_hirid (), fn); mappings->insert_location (crate_num, mapping.get_hirid (), function.get_locus ()); // add the mappings for the function params at the end for (auto ¶m : fn->get_function_params ()) { mappings->insert_hir_param (mapping.get_crate_num (), param.get_mappings ().get_hirid (), ¶m); mappings->insert_location (crate_num, mapping.get_hirid (), param.get_locus ()); } translated = fn; } void visit (AST::InherentImpl &impl_block) override { std::vector > where_clause_items; HIR::WhereClause where_clause (std::move (where_clause_items)); HIR::Visibility vis = HIR::Visibility::create_public (); std::vector > generic_params; if (impl_block.has_generics ()) { generic_params = lower_generic_params (impl_block.get_generic_params ()); for (auto &generic_param : generic_params) { switch (generic_param->get_kind ()) { case HIR::GenericParam::GenericKind::TYPE: { const HIR::TypeParam &t = static_cast (*generic_param); if (t.has_type ()) { // see https://github.com/rust-lang/rust/issues/36887 rust_error_at ( t.get_locus (), "defaults for type parameters are not allowed here"); } } break; default: break; } } } HIR::Type *impl_type = ASTLoweringType::translate (impl_block.get_type ().get ()); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, impl_block.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id (crate_num)); std::vector > impl_items; std::vector impl_item_ids; for (auto &impl_item : impl_block.get_impl_items ()) { HIR::ImplItem *lowered = ASTLowerImplItem::translate (impl_item.get (), mapping.get_hirid ()); impl_items.push_back (std::unique_ptr (lowered)); impl_item_ids.push_back (lowered->get_impl_mappings ().get_hirid ()); } HIR::ImplBlock *hir_impl_block = new HIR::ImplBlock (mapping, std::move (impl_items), std::move (generic_params), std::unique_ptr (impl_type), nullptr, where_clause, vis, impl_block.get_inner_attrs (), impl_block.get_outer_attrs (), impl_block.get_locus ()); translated = hir_impl_block; mappings->insert_defid_mapping (mapping.get_defid (), translated); mappings->insert_hir_item (mapping.get_crate_num (), mapping.get_hirid (), translated); mappings->insert_location (crate_num, mapping.get_hirid (), impl_block.get_locus ()); for (auto &impl_item_id : impl_item_ids) { mappings->insert_impl_item_mapping (impl_item_id, hir_impl_block); } } void visit (AST::Trait &trait) override { std::vector > where_clause_items; HIR::WhereClause where_clause (std::move (where_clause_items)); HIR::Visibility vis = HIR::Visibility::create_public (); std::vector > generic_params; if (trait.has_generics ()) { generic_params = lower_generic_params (trait.get_generic_params ()); for (auto &generic_param : generic_params) { switch (generic_param->get_kind ()) { case HIR::GenericParam::GenericKind::TYPE: { const HIR::TypeParam &t = static_cast (*generic_param); if (t.has_type ()) { // see https://github.com/rust-lang/rust/issues/36887 rust_error_at ( t.get_locus (), "defaults for type parameters are not allowed here"); } } break; default: break; } } } std::vector > type_param_bounds; std::vector > trait_items; for (auto &item : trait.get_trait_items ()) { HIR::TraitItem *lowered = ASTLowerTraitItem::translate (item.get ()); trait_items.push_back (std::unique_ptr (lowered)); } auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, trait.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id (crate_num)); translated = new HIR::Trait (mapping, trait.get_identifier (), trait.is_unsafe (), std::move (generic_params), std::move (type_param_bounds), where_clause, std::move (trait_items), vis, trait.get_outer_attrs (), trait.get_locus ()); mappings->insert_defid_mapping (mapping.get_defid (), translated); mappings->insert_hir_item (mapping.get_crate_num (), mapping.get_hirid (), translated); mappings->insert_location (crate_num, mapping.get_hirid (), trait.get_locus ()); } void visit (AST::TraitImpl &impl_block) override { std::vector > where_clause_items; HIR::WhereClause where_clause (std::move (where_clause_items)); HIR::Visibility vis = HIR::Visibility::create_public (); std::vector > generic_params; if (impl_block.has_generics ()) { generic_params = lower_generic_params (impl_block.get_generic_params ()); for (auto &generic_param : generic_params) { switch (generic_param->get_kind ()) { case HIR::GenericParam::GenericKind::TYPE: { const HIR::TypeParam &t = static_cast (*generic_param); if (t.has_type ()) { // see https://github.com/rust-lang/rust/issues/36887 rust_error_at ( t.get_locus (), "defaults for type parameters are not allowed here"); } } break; default: break; } } } HIR::Type *impl_type = ASTLoweringType::translate (impl_block.get_type ().get ()); HIR::TypePath *trait_ref = ASTLowerTypePath::translate (impl_block.get_trait_path ()); auto crate_num = mappings->get_current_crate (); Analysis::NodeMapping mapping (crate_num, impl_block.get_node_id (), mappings->get_next_hir_id (crate_num), mappings->get_next_localdef_id (crate_num)); std::vector > impl_items; std::vector impl_item_ids; for (auto &impl_item : impl_block.get_impl_items ()) { HIR::ImplItem *lowered = ASTLowerImplItem::translate (impl_item.get (), mapping.get_hirid ()); impl_items.push_back (std::unique_ptr (lowered)); impl_item_ids.push_back (lowered->get_impl_mappings ().get_hirid ()); } HIR::ImplBlock *hir_impl_block = new HIR::ImplBlock (mapping, std::move (impl_items), std::move (generic_params), std::unique_ptr (impl_type), std::unique_ptr (trait_ref), where_clause, vis, impl_block.get_inner_attrs (), impl_block.get_outer_attrs (), impl_block.get_locus ()); translated = hir_impl_block; mappings->insert_defid_mapping (mapping.get_defid (), translated); mappings->insert_hir_item (mapping.get_crate_num (), mapping.get_hirid (), translated); mappings->insert_location (crate_num, mapping.get_hirid (), impl_block.get_locus ()); for (auto &impl_item_id : impl_item_ids) { mappings->insert_impl_item_mapping (impl_item_id, hir_impl_block); } } private: ASTLoweringItem () : translated (nullptr) {} HIR::Item *translated; }; } // namespace HIR } // namespace Rust #endif // RUST_AST_LOWER_ITEM