// 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