aboutsummaryrefslogtreecommitdiff
path: root/gcc/rust/resolve/rust-toplevel-name-resolver-2.0.cc
diff options
context:
space:
mode:
authorPierre-Emmanuel Patry <pierre-emmanuel.patry@embecosm.com>2024-01-09 13:51:35 +0100
committerArthur Cohen <arthur.cohen@embecosm.com>2024-08-01 16:52:26 +0200
commit8c331e24575c0584be372a78bb0bbcca914d567c (patch)
tree18151cba46c42235bbb52555a6948c8d54d6e349 /gcc/rust/resolve/rust-toplevel-name-resolver-2.0.cc
parent06ce1b34fc0f1098ed45f25d52c05a6d8586c2c7 (diff)
downloadgcc-8c331e24575c0584be372a78bb0bbcca914d567c.zip
gcc-8c331e24575c0584be372a78bb0bbcca914d567c.tar.gz
gcc-8c331e24575c0584be372a78bb0bbcca914d567c.tar.bz2
gccrs: Add name resolution for on globbing use decl
This is the first part of the code required to enable globbing on use declarations. gcc/rust/ChangeLog: * resolve/rust-toplevel-name-resolver-2.0.cc (GlobbingVisitor::visit): Insert names into their namespace. (TopLevel::visit): Insert ast module. (TopLevel::handle_use_dec): Resolve use declaration. (TopLevel::handle_use_glob): Use globbing visitor. (flatten_list): Use globbing path vector. (flatten_glob): Likewise. (flatten): Likewise. (prefix_subpaths): Add a function to prefix subpath. * resolve/rust-toplevel-name-resolver-2.0.h (class GlobbingVisitor): Add globbing visitor. * util/rust-hir-map.cc (Mappings::insert_ast_module): Add function to insert module in module hashmap. (Mappings::lookup_ast_module): Add function to retrieve ast module. * util/rust-hir-map.h: Add module map and getter/setter prototypes. Signed-off-by: Pierre-Emmanuel Patry <pierre-emmanuel.patry@embecosm.com>
Diffstat (limited to 'gcc/rust/resolve/rust-toplevel-name-resolver-2.0.cc')
-rw-r--r--gcc/rust/resolve/rust-toplevel-name-resolver-2.0.cc179
1 files changed, 154 insertions, 25 deletions
diff --git a/gcc/rust/resolve/rust-toplevel-name-resolver-2.0.cc b/gcc/rust/resolve/rust-toplevel-name-resolver-2.0.cc
index 5e6f5e9..3122d41 100644
--- a/gcc/rust/resolve/rust-toplevel-name-resolver-2.0.cc
+++ b/gcc/rust/resolve/rust-toplevel-name-resolver-2.0.cc
@@ -25,6 +25,87 @@
namespace Rust {
namespace Resolver2_0 {
+void
+GlobbingVisitor::visit (AST::Module &module)
+{
+ if (module.get_visibility ().is_public ())
+ ctx.insert (module.get_name (), module.get_node_id (), Namespace::Types);
+}
+
+void
+GlobbingVisitor::visit (AST::MacroRulesDefinition &macro)
+{
+ if (macro.get_visibility ().is_public ())
+ ctx.insert (macro.get_rule_name (), macro.get_node_id (),
+ Namespace::Macros);
+}
+
+void
+GlobbingVisitor::visit (AST::Function &function)
+{
+ if (function.get_visibility ().is_public ())
+ ctx.insert (function.get_function_name (), function.get_node_id (),
+ Namespace::Values);
+}
+
+void
+GlobbingVisitor::visit (AST::StaticItem &static_item)
+{
+ if (static_item.get_visibility ().is_public ())
+ ctx.insert (static_item.get_identifier (), static_item.get_node_id (),
+ Namespace::Values);
+}
+
+void
+GlobbingVisitor::visit (AST::StructStruct &struct_item)
+{
+ if (struct_item.get_visibility ().is_public ())
+ ctx.insert (struct_item.get_identifier (), struct_item.get_node_id (),
+ Namespace::Values);
+}
+
+void
+GlobbingVisitor::visit (AST::TupleStruct &tuple_struct)
+{
+ if (tuple_struct.get_visibility ().is_public ())
+ ctx.insert (tuple_struct.get_identifier (), tuple_struct.get_node_id (),
+ Namespace::Values);
+}
+
+void
+GlobbingVisitor::visit (AST::Enum &enum_item)
+{
+ if (enum_item.get_visibility ().is_public ())
+ ctx.insert (enum_item.get_identifier (), enum_item.get_node_id (),
+ Namespace::Values);
+}
+
+void
+GlobbingVisitor::visit (AST::Union &union_item)
+{
+ if (union_item.get_visibility ().is_public ())
+ ctx.insert (union_item.get_identifier (), union_item.get_node_id (),
+ Namespace::Values);
+}
+
+void
+GlobbingVisitor::visit (AST::ConstantItem &const_item)
+{
+ if (const_item.get_visibility ().is_public ())
+ ctx.insert (const_item.get_identifier (), const_item.get_node_id (),
+ Namespace::Values);
+}
+
+void
+GlobbingVisitor::visit (AST::ExternCrate &crate)
+{}
+
+void
+GlobbingVisitor::visit (AST::UseDeclaration &use)
+{
+ // Handle cycles ?
+}
+
TopLevel::TopLevel (NameResolutionContext &resolver)
: DefaultResolver (resolver)
{}
@@ -81,6 +162,10 @@ TopLevel::visit (AST::Module &module)
ctx.scoped (Rib::Kind::Module, module.get_node_id (), sub_visitor,
module.get_name ());
+
+ if (Analysis::Mappings::get ()->lookup_ast_module (module.get_node_id ())
+ == tl::nullopt)
+ Analysis::Mappings::get ()->insert_ast_module (&module);
}
template <typename PROC_MACRO>
@@ -302,13 +387,26 @@ TopLevel::visit (AST::ConstantItem &const_item)
}
bool
-TopLevel::handle_use_dec (AST::SimplePath path)
+TopLevel::handle_use_glob (AST::SimplePath glob)
{
- // TODO: Glob imports can get shadowed by regular imports and regular items.
- // So we need to store them in a specific way in the ForeverStack - which can
- // also probably be used by labels and macros etc. Like store it as a
- // `Shadowable(NodeId)` instead of just a `NodeId`
+ auto resolved = ctx.types.resolve_path (glob.get_segments ());
+ if (!resolved.has_value ())
+ return false;
+
+ auto result = Analysis::Mappings::get ()->lookup_ast_module (*resolved);
+
+ if (!result.has_value ())
+ return false;
+
+ GlobbingVisitor gvisitor (ctx);
+ gvisitor.visit (*result.value ());
+
+ return true;
+}
+bool
+TopLevel::handle_use_dec (AST::SimplePath path)
+{
auto locus = path.get_final_segment ().get_locus ();
auto declared_name = path.get_final_segment ().as_string ();
@@ -377,14 +475,17 @@ static void
flatten_rebind (const AST::UseTreeRebind &glob,
std::vector<AST::SimplePath> &paths);
static void
-flatten_list (const AST::UseTreeList &glob,
- std::vector<AST::SimplePath> &paths);
+flatten_list (const AST::UseTreeList &glob, std::vector<AST::SimplePath> &paths,
+ std::vector<AST::SimplePath> &glob_paths,
+ NameResolutionContext &ctx);
static void
flatten_glob (const AST::UseTreeGlob &glob,
- std::vector<AST::SimplePath> &paths);
+ std::vector<AST::SimplePath> &glob_paths,
+ NameResolutionContext &ctx);
static void
-flatten (const AST::UseTree *tree, std::vector<AST::SimplePath> &paths)
+flatten (const AST::UseTree *tree, std::vector<AST::SimplePath> &paths,
+ std::vector<AST::SimplePath> &glob_paths, NameResolutionContext &ctx)
{
switch (tree->get_kind ())
{
@@ -395,13 +496,12 @@ flatten (const AST::UseTree *tree, std::vector<AST::SimplePath> &paths)
}
case AST::UseTree::List: {
auto list = static_cast<const AST::UseTreeList *> (tree);
- flatten_list (*list, paths);
+ flatten_list (*list, paths, glob_paths, ctx);
break;
}
case AST::UseTree::Glob: {
- rust_sorry_at (tree->get_locus (), "cannot resolve glob imports yet");
auto glob = static_cast<const AST::UseTreeGlob *> (tree);
- flatten_glob (*glob, paths);
+ flatten_glob (*glob, glob_paths, ctx);
break;
}
break;
@@ -432,8 +532,28 @@ flatten_rebind (const AST::UseTreeRebind &rebind,
}
}
+/** Prefix a list of subpath
+ * @param prefix A prefix for all subpath
+ * @param subs List of subpath to prefix
+ * @param size List where results should be stored
+ */
+static void
+prefix_subpaths (AST::SimplePath prefix, std::vector<AST::SimplePath> subs,
+ std::vector<AST::SimplePath> &results)
+{
+ for (auto &sub : subs)
+ {
+ auto new_path = prefix;
+ std::copy (sub.get_segments ().begin (), sub.get_segments ().end (),
+ std::back_inserter (new_path.get_segments ()));
+ results.emplace_back (new_path);
+ }
+}
+
static void
-flatten_list (const AST::UseTreeList &list, std::vector<AST::SimplePath> &paths)
+flatten_list (const AST::UseTreeList &list, std::vector<AST::SimplePath> &paths,
+ std::vector<AST::SimplePath> &glob_paths,
+ NameResolutionContext &ctx)
{
auto prefix = AST::SimplePath::create_empty ();
if (list.has_path ())
@@ -442,43 +562,52 @@ flatten_list (const AST::UseTreeList &list, std::vector<AST::SimplePath> &paths)
for (const auto &tree : list.get_trees ())
{
auto sub_paths = std::vector<AST::SimplePath> ();
- flatten (tree.get (), sub_paths);
-
- for (auto &sub_path : sub_paths)
- {
- auto new_path = prefix;
- std::copy (sub_path.get_segments ().begin (),
- sub_path.get_segments ().end (),
- std::back_inserter (new_path.get_segments ()));
+ auto sub_globs = std::vector<AST::SimplePath> ();
+ flatten (tree.get (), sub_paths, sub_globs, ctx);
- paths.emplace_back (new_path);
- }
+ prefix_subpaths (prefix, sub_paths, paths);
+ prefix_subpaths (prefix, sub_globs, glob_paths);
}
}
static void
-flatten_glob (const AST::UseTreeGlob &glob, std::vector<AST::SimplePath> &paths)
+flatten_glob (const AST::UseTreeGlob &glob, std::vector<AST::SimplePath> &paths,
+ NameResolutionContext &ctx)
{
if (glob.has_path ())
paths.emplace_back (glob.get_path ());
+
+ // (PE): Get path rib
+ auto rib = ctx.values.resolve_path (glob.get_path ().get_segments ())
+ .and_then ([&] (NodeId id) { return ctx.values.to_rib (id); });
+ if (rib.has_value ())
+ {
+ auto value = rib.value ().get_values ();
+ }
}
void
TopLevel::visit (AST::UseDeclaration &use)
{
auto paths = std::vector<AST::SimplePath> ();
+ auto glob_path = std::vector<AST::SimplePath> ();
// FIXME: How do we handle `use foo::{self}` imports? Some beforehand cleanup?
// How do we handle module imports in general? Should they get added to all
// namespaces?
const auto &tree = use.get_tree ();
- flatten (tree.get (), paths);
+ flatten (tree.get (), paths, glob_path, this->ctx);
for (auto &path : paths)
if (!handle_use_dec (path))
rust_error_at (path.get_final_segment ().get_locus (), ErrorCode::E0433,
"unresolved import %qs", path.as_string ().c_str ());
+
+ for (auto &glob : glob_path)
+ if (!handle_use_glob (glob))
+ rust_error_at (glob.get_final_segment ().get_locus (), ErrorCode::E0433,
+ "unresolved import %qs", glob.as_string ().c_str ());
}
} // namespace Resolver2_0