// Copyright (C) 2024 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
// .
#include "expected.h"
#include "rust-ast.h"
#include "rust-diagnostics.h"
#include "rust-forever-stack.h"
#include "rust-rib.h"
#include "optional.h"
namespace Rust {
namespace Resolver2_0 {
bool
ForeverStackStore::Node::is_root () const
{
return !parent.has_value ();
}
bool
ForeverStackStore::Node::is_leaf () const
{
return children.empty ();
}
NodeId
ForeverStackStore::Node::get_id () const
{
return id;
}
ForeverStackStore::Node &
ForeverStackStore::Node::insert_child (NodeId id, tl::optional path,
Rib::Kind kind)
{
auto res = children.insert ({Link (id, path), Node (kind, id, *this)});
rust_debug ("inserting link: Link(%d [%s]): existed? %s", id,
path.has_value () ? path.value ().as_string ().c_str ()
: "",
!res.second ? "yes" : "no");
// sanity check on rib kind
// pick the value rib, since all ribs should have the same kind anyways
rust_assert (res.second || res.first->second.value_rib.kind == kind);
// verify, if we're using an existing node, our paths don't contradict
if (!res.second && path.has_value ())
{
auto other_path = res.first->first.path;
rust_assert (!other_path.has_value ()
|| other_path.value ().as_string ()
== path.value ().as_string ());
}
return res.first->second;
}
tl::optional
ForeverStackStore::Node::get_child (const Identifier &path)
{
for (auto &ent : children)
{
if (ent.first.path.has_value ()
&& ent.first.path->as_string () == path.as_string ())
return ent.second;
}
return tl::nullopt;
}
tl::optional
ForeverStackStore::Node::get_child (const Identifier &path) const
{
for (auto &ent : children)
{
if (ent.first.path.has_value ()
&& ent.first.path->as_string () == path.as_string ())
return ent.second;
}
return tl::nullopt;
}
tl::optional
ForeverStackStore::Node::get_parent ()
{
return parent;
}
tl::optional
ForeverStackStore::Node::get_parent () const
{
if (parent)
return *parent;
return tl::nullopt;
}
tl::optional
ForeverStackStore::Node::get_parent_path () const
{
if (parent.has_value ())
for (auto &ent : parent->children)
if (ent.first.id == id && ent.first.path.has_value ())
return ent.first.path.value ();
return tl::nullopt;
}
Rib &
ForeverStackStore::Node::get_rib (Namespace ns)
{
switch (ns)
{
case Namespace::Values:
return value_rib;
case Namespace::Types:
return type_rib;
case Namespace::Labels:
return label_rib;
case Namespace::Macros:
return macro_rib;
default:
rust_unreachable ();
}
}
const Rib &
ForeverStackStore::Node::get_rib (Namespace ns) const
{
switch (ns)
{
case Namespace::Values:
return value_rib;
case Namespace::Types:
return type_rib;
case Namespace::Labels:
return label_rib;
case Namespace::Macros:
return macro_rib;
default:
rust_unreachable ();
}
}
tl::expected
ForeverStackStore::Node::insert (const Identifier &name, NodeId node,
Namespace ns)
{
// So what do we do here - if the Rib has already been pushed in an earlier
// pass, we might end up in a situation where it is okay to re-add new names.
// Do we just ignore that here? Do we keep track of if the Rib is new or not?
// should our cursor have info on the current node like "is it newly pushed"?
return get_rib (ns).insert (name.as_string (),
Rib::Definition::NonShadowable (node));
}
tl::expected
ForeverStackStore::Node::insert_shadowable (const Identifier &name, NodeId node,
Namespace ns)
{
return get_rib (ns).insert (name.as_string (),
Rib::Definition::Shadowable (node));
}
tl::expected
ForeverStackStore::Node::insert_globbed (const Identifier &name, NodeId node,
Namespace ns)
{
return get_rib (ns).insert (name.as_string (),
Rib::Definition::Globbed (node));
}
void
ForeverStackStore::Node::reverse_iter (std::function lambda)
{
for (Node *tmp = this; lambda (*tmp) == KeepGoing::Yes && !tmp->is_root ();
tmp = &tmp->parent.value ())
;
}
void
ForeverStackStore::Node::reverse_iter (
std::function lambda) const
{
for (const Node *tmp = this;
lambda (*tmp) == KeepGoing::Yes && !tmp->is_root ();
tmp = &tmp->parent.value ())
;
}
void
ForeverStackStore::Node::child_iter (
std::function, Node &)>
lambda)
{
for (auto &ent : children)
{
tl::optional path;
if (ent.first.path.has_value ())
path = ent.first.path.value ();
auto keep_going = lambda (ent.first.id, path, ent.second);
if (keep_going == KeepGoing::No)
return;
}
}
void
ForeverStackStore::Node::child_iter (
std::function,
const Node &)>
lambda) const
{
for (auto &ent : children)
{
tl::optional path;
if (ent.first.path.has_value ())
path = ent.first.path.value ();
auto keep_going = lambda (ent.first.id, path, ent.second);
if (keep_going == KeepGoing::No)
return;
}
}
ForeverStackStore::Node &
ForeverStackStore::Node::find_closest_module ()
{
// get kind of value_rib
// but all ribs should share the same kind anyways
if (value_rib.kind == Rib::Kind::Module || !parent.has_value ())
return *this;
else
return parent->find_closest_module ();
}
const ForeverStackStore::Node &
ForeverStackStore::Node::find_closest_module () const
{
// get kind of value_rib
// but all ribs should share the same kind anyways
if (value_rib.kind != Rib::Kind::Module || !parent.has_value ())
return *this;
else
return parent->find_closest_module ();
}
tl::optional
ForeverStackStore::Node::dfs_node (NodeId to_find)
{
if (id == to_find)
return *this;
for (auto &child : children)
{
auto candidate = child.second.dfs_node (to_find);
if (candidate.has_value ())
return candidate;
}
return tl::nullopt;
}
tl::optional
ForeverStackStore::Node::dfs_node (NodeId to_find) const
{
if (id == to_find)
return *this;
for (auto &child : children)
{
auto candidate = child.second.dfs_node (to_find);
if (candidate.has_value ())
return candidate;
}
return tl::nullopt;
}
ForeverStackStore::Node &
ForeverStackStore::get_root ()
{
return root;
}
const ForeverStackStore::Node &
ForeverStackStore::get_root () const
{
return root;
}
tl::optional
ForeverStackStore::get_node (NodeId node_id)
{
return root.dfs_node (node_id);
}
tl::optional
ForeverStackStore::get_node (NodeId node_id) const
{
return root.dfs_node (node_id);
}
} // namespace Resolver2_0
} // namespace Rust