// Copyright (C) 2020-2025 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 "rust-rib.h"
#include "rust-name-resolution-context.h"
namespace Rust {
namespace Resolver2_0 {
Rib::Definition::Definition (NodeId id, bool shadowable)
: ids ({id}), shadowable (shadowable)
{}
bool
Rib::Definition::is_ambiguous () const
{
return shadowable && ids.size () > 1;
}
std::string
Rib::Definition::to_string () const
{
std::stringstream out;
out << (shadowable ? "(S)" : "(NS)") << "[";
std::string sep;
for (auto id : ids)
{
out << sep << id;
sep = ",";
}
out << "]";
return out.str ();
}
Rib::Definition
Rib::Definition::Shadowable (NodeId id)
{
return Definition (id, true);
}
Rib::Definition
Rib::Definition::NonShadowable (NodeId id)
{
return Definition (id, false);
}
DuplicateNameError::DuplicateNameError (std::string name, NodeId existing)
: name (name), existing (existing)
{}
Rib::Rib (Kind kind) : kind (kind) {}
Rib::Rib (Kind kind, std::string identifier, NodeId id)
: Rib (kind, {{identifier, id}})
{}
Rib::Rib (Kind kind, std::unordered_map to_insert)
: kind (kind)
{
for (auto &value : to_insert)
values.insert ({value.first, Definition::NonShadowable (value.second)});
}
tl::expected
Rib::insert (std::string name, Definition def)
{
auto it = values.find (name);
if (it == values.end ())
{
/* No old value */
values[name] = def;
}
else if (it->second.shadowable && def.shadowable)
{ /* Both shadowable */
auto ¤t = values[name];
for (auto id : def.ids)
{
if (std::find (current.ids.cbegin (), current.ids.cend (), id)
== current.ids.cend ())
{
current.ids.push_back (id);
}
}
}
else if (it->second.shadowable)
{ /* Only old shadowable : replace value */
values[name] = def;
}
else /* Neither are shadowable */
{
return tl::make_unexpected (
DuplicateNameError (name, it->second.ids.back ()));
}
return def.ids.back ();
}
tl::optional
Rib::get (const std::string &name)
{
auto it = values.find (name);
if (it == values.end ())
return tl::nullopt;
return it->second;
}
const std::unordered_map &
Rib::get_values () const
{
return values;
}
} // namespace Resolver2_0
} // namespace Rust