// Copyright (C) 2020-2021 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_RESOLVE_EXPR_H
#define RUST_AST_RESOLVE_EXPR_H
#include "rust-ast-resolve-base.h"
#include "rust-ast-full.h"
#include "rust-ast-resolve-struct-expr-field.h"
#include "rust-ast-verify-assignee.h"
#include "rust-ast-resolve-type.h"
namespace Rust {
namespace Resolver {
class ResolvePath : public ResolverBase
{
using Rust::Resolver::ResolverBase::visit;
public:
static void go (AST::PathInExpression *expr, NodeId parent)
{
ResolvePath resolver (parent);
resolver.resolve_path (expr);
}
static void go (AST::QualifiedPathInExpression *expr, NodeId parent)
{
ResolvePath resolver (parent);
resolver.resolve_path (expr);
}
private:
ResolvePath (NodeId parent) : ResolverBase (parent) {}
void resolve_path (AST::PathInExpression *expr);
void resolve_path (AST::QualifiedPathInExpression *expr);
void resolve_segments (CanonicalPath prefix, size_t offs,
std::vector &segs,
NodeId expr_node_id, Location expr_locus);
};
class ResolveExpr : public ResolverBase
{
using Rust::Resolver::ResolverBase::visit;
public:
static void go (AST::Expr *expr, NodeId parent)
{
ResolveExpr resolver (parent);
expr->accept_vis (resolver);
};
void visit (AST::TupleIndexExpr &expr) override
{
ResolveExpr::go (expr.get_tuple_expr ().get (), expr.get_node_id ());
}
void visit (AST::TupleExpr &expr) override
{
if (expr.is_unit ())
return;
for (auto &elem : expr.get_tuple_elems ())
ResolveExpr::go (elem.get (), expr.get_node_id ());
}
void visit (AST::PathInExpression &expr) override
{
ResolvePath::go (&expr, parent);
}
void visit (AST::QualifiedPathInExpression &expr) override
{
ResolvePath::go (&expr, parent);
}
void visit (AST::ReturnExpr &expr) override
{
if (expr.has_returned_expr ())
ResolveExpr::go (expr.get_returned_expr ().get (), expr.get_node_id ());
}
void visit (AST::CallExpr &expr) override
{
ResolveExpr::go (expr.get_function_expr ().get (), expr.get_node_id ());
expr.iterate_params ([&] (AST::Expr *p) mutable -> bool {
ResolveExpr::go (p, expr.get_node_id ());
return true;
});
}
void visit (AST::MethodCallExpr &expr) override
{
ResolveExpr::go (expr.get_receiver_expr ().get (), expr.get_node_id ());
if (expr.get_method_name ().has_generic_args ())
{
AST::GenericArgs &args = expr.get_method_name ().get_generic_args ();
ResolveTypeToCanonicalPath::type_resolve_generic_args (args);
}
expr.iterate_params ([&] (AST::Expr *p) mutable -> bool {
ResolveExpr::go (p, expr.get_node_id ());
return true;
});
}
void visit (AST::AssignmentExpr &expr) override
{
ResolveExpr::go (expr.get_left_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_right_expr ().get (), expr.get_node_id ());
// need to verify the assignee
VerifyAsignee::go (expr.get_left_expr ().get (), expr.get_node_id ());
}
void visit (AST::IdentifierExpr &expr) override
{
if (resolver->get_name_scope ().lookup (
CanonicalPath::new_seg (expr.get_node_id (), expr.as_string ()),
&resolved_node))
{
resolver->insert_resolved_name (expr.get_node_id (), resolved_node);
resolver->insert_new_definition (expr.get_node_id (),
Definition{expr.get_node_id (),
parent});
}
else if (resolver->get_type_scope ().lookup (
CanonicalPath::new_seg (expr.get_node_id (), expr.as_string ()),
&resolved_node))
{
resolver->insert_resolved_type (expr.get_node_id (), resolved_node);
resolver->insert_new_definition (expr.get_node_id (),
Definition{expr.get_node_id (),
parent});
}
else
{
rust_error_at (expr.get_locus (), "failed to find name: %s",
expr.as_string ().c_str ());
}
}
void visit (AST::ArithmeticOrLogicalExpr &expr) override
{
ResolveExpr::go (expr.get_left_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_right_expr ().get (), expr.get_node_id ());
}
void visit (AST::CompoundAssignmentExpr &expr) override
{
ResolveExpr::go (expr.get_left_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_right_expr ().get (), expr.get_node_id ());
// need to verify the assignee
VerifyAsignee::go (expr.get_left_expr ().get (), expr.get_node_id ());
}
void visit (AST::ComparisonExpr &expr) override
{
ResolveExpr::go (expr.get_left_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_right_expr ().get (), expr.get_node_id ());
}
void visit (AST::LazyBooleanExpr &expr) override
{
ResolveExpr::go (expr.get_left_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_right_expr ().get (), expr.get_node_id ());
}
void visit (AST::NegationExpr &expr) override
{
ResolveExpr::go (expr.get_negated_expr ().get (), expr.get_node_id ());
}
void visit (AST::TypeCastExpr &expr) override
{
ResolveType::go (expr.get_type_to_cast_to ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_casted_expr ().get (), expr.get_node_id ());
}
void visit (AST::IfExpr &expr) override
{
ResolveExpr::go (expr.get_condition_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_if_block ().get (), expr.get_node_id ());
}
void visit (AST::IfExprConseqElse &expr) override
{
ResolveExpr::go (expr.get_condition_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_if_block ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_else_block ().get (), expr.get_node_id ());
}
void visit (AST::IfExprConseqIf &expr) override
{
ResolveExpr::go (expr.get_condition_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_if_block ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_conseq_if_expr ().get (), expr.get_node_id ());
}
void visit (AST::BlockExpr &expr) override;
void visit (AST::UnsafeBlockExpr &expr) override
{
expr.get_block_expr ()->accept_vis (*this);
}
void visit (AST::ArrayElemsValues &elems) override
{
elems.iterate ([&] (AST::Expr *elem) mutable -> bool {
ResolveExpr::go (elem, elems.get_node_id ());
return true;
});
}
void visit (AST::ArrayExpr &expr) override
{
expr.get_array_elems ()->accept_vis (*this);
}
void visit (AST::ArrayIndexExpr &expr) override
{
ResolveExpr::go (expr.get_array_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_index_expr ().get (), expr.get_node_id ());
}
void visit (AST::ArrayElemsCopied &elems) override
{
ResolveExpr::go (elems.get_num_copies ().get (), elems.get_node_id ());
ResolveExpr::go (elems.get_elem_to_copy ().get (), elems.get_node_id ());
}
// this this an empty struct constructor like 'S {}'
void visit (AST::StructExprStruct &struct_expr) override
{
ResolveExpr::go (&struct_expr.get_struct_name (),
struct_expr.get_node_id ());
}
// this this a struct constructor with fields
void visit (AST::StructExprStructFields &struct_expr) override
{
ResolveExpr::go (&struct_expr.get_struct_name (),
struct_expr.get_node_id ());
if (struct_expr.has_struct_base ())
{
AST::StructBase &base = struct_expr.get_struct_base ();
ResolveExpr::go (base.get_base_struct ().get (),
struct_expr.get_node_id ());
}
auto const &struct_fields = struct_expr.get_fields ();
for (auto &struct_field : struct_fields)
{
ResolveStructExprField::go (struct_field.get (),
struct_expr.get_node_id ());
}
}
void visit (AST::GroupedExpr &expr) override
{
ResolveExpr::go (expr.get_expr_in_parens ().get (), expr.get_node_id ());
}
void visit (AST::FieldAccessExpr &expr) override
{
ResolveExpr::go (expr.get_receiver_expr ().get (), expr.get_node_id ());
}
void visit (AST::LoopExpr &expr) override
{
if (expr.has_loop_label ())
{
auto label = expr.get_loop_label ();
if (label.get_lifetime ().get_lifetime_type ()
!= AST::Lifetime::LifetimeType::NAMED)
{
rust_error_at (label.get_locus (),
"Labels must be a named lifetime value");
return;
}
auto label_name = label.get_lifetime ().get_lifetime_name ();
auto label_lifetime_node_id = label.get_lifetime ().get_node_id ();
resolver->get_label_scope ().insert (
CanonicalPath::new_seg (expr.get_node_id (), label_name),
label_lifetime_node_id, label.get_locus (), false,
[&] (const CanonicalPath &, NodeId, Location locus) -> void {
rust_error_at (label.get_locus (),
"label redefined multiple times");
rust_error_at (locus, "was defined here");
});
resolver->insert_new_definition (label_lifetime_node_id,
Definition{label_lifetime_node_id,
label.get_node_id ()});
}
ResolveExpr::go (expr.get_loop_block ().get (), expr.get_node_id ());
}
void visit (AST::BreakExpr &expr) override
{
if (expr.has_label ())
{
auto label = expr.get_label ();
if (label.get_lifetime_type () != AST::Lifetime::LifetimeType::NAMED)
{
rust_error_at (label.get_locus (),
"Labels must be a named lifetime value");
return;
}
NodeId resolved_node = UNKNOWN_NODEID;
if (!resolver->get_label_scope ().lookup (
CanonicalPath::new_seg (label.get_node_id (),
label.get_lifetime_name ()),
&resolved_node))
{
rust_error_at (expr.get_label ().get_locus (),
"failed to resolve label");
return;
}
resolver->insert_resolved_label (label.get_node_id (), resolved_node);
}
if (expr.has_break_expr ())
ResolveExpr::go (expr.get_break_expr ().get (), expr.get_node_id ());
}
void visit (AST::WhileLoopExpr &expr) override
{
if (expr.has_loop_label ())
{
auto label = expr.get_loop_label ();
if (label.get_lifetime ().get_lifetime_type ()
!= AST::Lifetime::LifetimeType::NAMED)
{
rust_error_at (label.get_locus (),
"Labels must be a named lifetime value");
return;
}
auto label_name = label.get_lifetime ().get_lifetime_name ();
auto label_lifetime_node_id = label.get_lifetime ().get_node_id ();
resolver->get_label_scope ().insert (
CanonicalPath::new_seg (label.get_node_id (), label_name),
label_lifetime_node_id, label.get_locus (), false,
[&] (const CanonicalPath &, NodeId, Location locus) -> void {
rust_error_at (label.get_locus (),
"label redefined multiple times");
rust_error_at (locus, "was defined here");
});
resolver->insert_new_definition (label_lifetime_node_id,
Definition{label_lifetime_node_id,
label.get_node_id ()});
}
ResolveExpr::go (expr.get_predicate_expr ().get (), expr.get_node_id ());
ResolveExpr::go (expr.get_loop_block ().get (), expr.get_node_id ());
}
void visit (AST::ContinueExpr &expr) override
{
if (expr.has_label ())
{
auto label = expr.get_label ();
if (label.get_lifetime_type () != AST::Lifetime::LifetimeType::NAMED)
{
rust_error_at (label.get_locus (),
"Labels must be a named lifetime value");
return;
}
NodeId resolved_node = UNKNOWN_NODEID;
if (!resolver->get_label_scope ().lookup (
CanonicalPath::new_seg (label.get_node_id (),
label.get_lifetime_name ()),
&resolved_node))
{
rust_error_at (expr.get_label ().get_locus (),
"failed to resolve label");
return;
}
resolver->insert_resolved_label (label.get_node_id (), resolved_node);
}
}
void visit (AST::BorrowExpr &expr) override
{
ResolveExpr::go (expr.get_borrowed_expr ().get (), expr.get_node_id ());
}
void visit (AST::DereferenceExpr &expr) override
{
ResolveExpr::go (expr.get_dereferenced_expr ().get (), expr.get_node_id ());
}
private:
ResolveExpr (NodeId parent) : ResolverBase (parent) {}
};
} // namespace Resolver
} // namespace Rust
#endif // RUST_AST_RESOLVE_EXPR_H