// 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-ast-builder.h" #include "rust-ast-builder-type.h" #include "rust-ast.h" #include "rust-common.h" #include "rust-expr.h" #include "rust-keyword-values.h" #include "rust-path.h" #include "rust-item.h" #include "rust-path.h" #include "rust-pattern.h" #include "rust-system.h" #include "rust-token.h" #include namespace Rust { namespace AST { std::unique_ptr Builder::statementify (std::unique_ptr &&value, bool semicolon_followed) const { return std::make_unique (std::move (value), loc, semicolon_followed); } std::unique_ptr Builder::literal_string (std::string &&content) const { return std::unique_ptr ( new AST::LiteralExpr (std::move (content), Literal::LitType::STRING, PrimitiveCoreType::CORETYPE_STR, {}, loc)); } std::unique_ptr Builder::literal_bool (bool b) const { auto str = b ? Values::Keywords::TRUE_LITERAL : Values::Keywords::FALSE_LITERAL; return std::unique_ptr ( new AST::LiteralExpr (std::move (str), Literal::LitType::BOOL, PrimitiveCoreType::CORETYPE_BOOL, {}, loc)); } std::unique_ptr Builder::call (std::unique_ptr &&path, std::vector> &&args) const { return std::unique_ptr ( new CallExpr (std::move (path), std::move (args), {}, loc)); } std::unique_ptr Builder::call (std::unique_ptr &&path, std::unique_ptr &&arg) const { auto args = std::vector> (); args.emplace_back (std::move (arg)); return call (std::move (path), std::move (args)); } std::unique_ptr Builder::array (std::vector> &&members) const { auto elts = std::make_unique (std::move (members), loc); return std::unique_ptr (new ArrayExpr (std::move (elts), {}, {}, loc)); } std::unique_ptr Builder::qualified_path_in_expression (std::unique_ptr &&type, TypePath trait, PathExprSegment segment) const { auto segments = {segment}; return qualified_path_in_expression (std::move (type), trait, segments); } std::unique_ptr Builder::qualified_path_in_expression ( std::unique_ptr &&type, TypePath trait, std::vector &&segments) const { auto qual_type = QualifiedPathType (std::move (type), loc, trait); return std::unique_ptr ( new QualifiedPathInExpression (qual_type, std::move (segments), {}, loc)); } std::unique_ptr Builder::identifier (std::string name) const { return std::unique_ptr (new IdentifierExpr (name, {}, loc)); } std::unique_ptr Builder::identifier_pattern (std::string name, bool mut) const { return std::unique_ptr ( new IdentifierPattern (name, loc, false, mut)); } std::unique_ptr Builder::tuple_idx (std::string receiver, int idx) const { return std::unique_ptr ( new TupleIndexExpr (identifier (receiver), idx, {}, loc)); } std::unique_ptr Builder::tuple (std::vector> &&values) const { return std::unique_ptr ( new TupleExpr (std::move (values), {}, {}, loc)); } std::unique_ptr Builder::self_ref_param (bool mutability) const { return std::make_unique (Lifetime::error (), mutability, loc); } std::unique_ptr Builder::function_param (std::unique_ptr &&pattern, std::unique_ptr &&type) const { return std::unique_ptr ( new FunctionParam (std::move (pattern), std::move (type), {}, loc)); } FunctionQualifiers Builder::fn_qualifiers () const { return FunctionQualifiers (loc, Async::No, Const::No, Unsafety::Normal); } std::unique_ptr Builder::function (std::string function_name, std::vector> params, std::unique_ptr return_type, std::unique_ptr block, std::vector> generic_params, FunctionQualifiers qualifiers, WhereClause where_clause, Visibility visibility) const { return std::unique_ptr ( new Function (function_name, qualifiers, std::move (generic_params), std::move (params), std::move (return_type), where_clause, std::move (block), visibility, {}, loc)); } PathExprSegment Builder::path_segment (std::string seg) const { return PathExprSegment (PathIdentSegment (seg, loc), loc); } std::unique_ptr Builder::type_path_segment (std::string seg) const { return std::unique_ptr ( new TypePathSegment (seg, false, loc)); } std::unique_ptr Builder::type_path_segment (LangItem::Kind lang_item) const { return std::unique_ptr ( new TypePathSegment (lang_item, loc)); } std::unique_ptr Builder::type_path_segment_generic (std::string seg, GenericArgs args) const { return std::unique_ptr ( new TypePathSegmentGeneric (PathIdentSegment (seg, loc), false, args, loc)); } std::unique_ptr Builder::type_path_segment_generic (LangItem::Kind lang_item, GenericArgs args) const { return std::unique_ptr ( new TypePathSegmentGeneric (lang_item, args, loc)); } std::unique_ptr Builder::single_type_path (std::string type) const { auto segments = std::vector> (); segments.emplace_back (type_path_segment (type)); return std::unique_ptr (new TypePath (std::move (segments), loc)); } std::unique_ptr Builder::single_type_path (LangItem::Kind lang_item) const { return std::unique_ptr (new TypePath (lang_item, {}, loc)); } std::unique_ptr Builder::single_generic_type_path (std::string type, GenericArgs args) const { auto segments = std::vector> (); segments.emplace_back (type_path_segment_generic (type, args)); return std::unique_ptr (new TypePath (std::move (segments), loc)); } std::unique_ptr Builder::single_generic_type_path (LangItem::Kind lang_item, GenericArgs args) const { auto segments = std::vector> (); segments.emplace_back (type_path_segment_generic (lang_item, args)); return std::unique_ptr (new TypePath (std::move (segments), loc)); } TypePath Builder::type_path (std::vector> &&segments, bool opening_scope) const { return TypePath (std::move (segments), loc, opening_scope); } TypePath Builder::type_path (std::vector &&segments, bool opening_scope) const { auto type_segments = std::vector> (); for (auto &&segment : segments) type_segments.emplace_back (type_path_segment (segment)); return TypePath (std::move (type_segments), loc, opening_scope); } TypePath Builder::type_path (std::unique_ptr &&segment) const { auto segments = std::vector> (); segments.emplace_back (std::move (segment)); return type_path (std::move (segments)); } TypePath Builder::type_path (std::string type) const { return type_path (type_path_segment (type)); } TypePath Builder::type_path (LangItem::Kind lang_item) const { return type_path (type_path_segment (lang_item)); } std::unique_ptr Builder::reference_type (std::unique_ptr &&inner_type, bool mutability) const { return std::make_unique (mutability, std::move (inner_type), loc); } PathInExpression Builder::path_in_expression (std::vector &&segments, bool opening_scope) const { auto path_segments = std::vector (); for (auto &seg : segments) path_segments.emplace_back (path_segment (seg)); return PathInExpression (std::move (path_segments), {}, loc, opening_scope); } PathInExpression Builder::path_in_expression (LangItem::Kind lang_item) const { return PathInExpression (lang_item, {}, loc); } PathInExpression Builder::variant_path (const std::string &enum_path, const std::string &variant) const { return PathInExpression ({path_segment (enum_path), path_segment (variant)}, {}, loc, false); } std::unique_ptr Builder::block (tl::optional> &&stmt, std::unique_ptr &&tail_expr) const { auto stmts = std::vector> (); if (stmt) stmts.emplace_back (std::move (*stmt)); return block (std::move (stmts), std::move (tail_expr)); } std::unique_ptr Builder::block () const { auto stmts = std::vector> (); return block (std::move (stmts)); } std::unique_ptr Builder::block (std::vector> &&stmts, std::unique_ptr &&tail_expr) const { return std::unique_ptr ( new BlockExpr (std::move (stmts), std::move (tail_expr), {}, {}, LoopLabel::error (), loc, loc)); } std::unique_ptr Builder::return_expr (std::unique_ptr &&to_return) { return std::unique_ptr ( new ReturnExpr (std::move (to_return), {}, loc)); } std::unique_ptr Builder::let (std::unique_ptr &&pattern, std::unique_ptr &&type, std::unique_ptr &&init) const { return std::unique_ptr (new LetStmt (std::move (pattern), std::move (init), std::move (type), {}, loc)); } std::unique_ptr Builder::ref (std::unique_ptr &&of, bool mut) const { auto mutability = mut ? Mutability::Mut : Mutability::Imm; return std::unique_ptr ( new BorrowExpr (std::move (of), mutability, /* raw */ false, /* is double */ false, {}, loc)); } std::unique_ptr Builder::deref (std::unique_ptr &&of) const { return std::unique_ptr (new DereferenceExpr (std::move (of), {}, loc)); } std::unique_ptr Builder::comparison_expr (std::unique_ptr &&lhs, std::unique_ptr &&rhs, ComparisonOperator op) const { return std::make_unique (std::move (lhs), std::move (rhs), op, loc); } std::unique_ptr Builder::boolean_operation (std::unique_ptr &&lhs, std::unique_ptr &&rhs, LazyBooleanOperator op) const { return std::make_unique (std::move (lhs), std::move (rhs), op, loc); } std::unique_ptr Builder::struct_struct (std::string struct_name, std::vector> &&generics, std::vector &&fields) { auto is_unit = fields.empty (); return std::unique_ptr ( new StructStruct (std::move (fields), struct_name, std::move (generics), WhereClause::create_empty (), is_unit, Visibility::create_private (), {}, loc)); } std::unique_ptr Builder::struct_expr_struct (std::string struct_name) const { return std::unique_ptr ( new StructExprStruct (path_in_expression ({struct_name}), {}, {}, loc)); } std::unique_ptr Builder::struct_expr ( std::string struct_name, std::vector> &&fields) const { return struct_expr (path_in_expression ({struct_name}), std::move (fields)); } std::unique_ptr Builder::struct_expr ( PathInExpression struct_name, std::vector> &&fields) const { return std::unique_ptr ( new StructExprStructFields (struct_name, std::move (fields), loc)); } std::unique_ptr Builder::struct_expr_field (std::string field_name, std::unique_ptr &&value) const { return std::unique_ptr ( new StructExprFieldIdentifierValue (field_name, std::move (value), loc)); } std::unique_ptr Builder::field_access (std::unique_ptr &&instance, std::string field) const { return std::unique_ptr ( new FieldAccessExpr (std::move (instance), field, {}, loc)); } std::unique_ptr Builder::wildcard () const { return std::unique_ptr (new WildcardPattern (loc)); } std::unique_ptr Builder::ref_pattern (std::unique_ptr &&inner) const { return std::make_unique (std::move (inner), false, false, loc); } std::unique_ptr Builder::lang_item_path (LangItem::Kind kind) const { return std::unique_ptr (new PathInExpression (kind, {}, loc)); } std::unique_ptr Builder::match (std::unique_ptr &&scrutinee, std::vector &&cases) { return std::unique_ptr ( new MatchExpr (std::move (scrutinee), std::move (cases), {}, {}, loc)); } MatchArm Builder::match_arm (std::unique_ptr &&pattern) { auto patterns = std::vector> (); patterns.emplace_back (std::move (pattern)); return MatchArm (std::move (patterns), loc); } MatchCase Builder::match_case (std::unique_ptr &&pattern, std::unique_ptr &&expr) { return MatchCase (match_arm (std::move (pattern)), std::move (expr)); } std::unique_ptr Builder::loop (std::vector> &&stmts) { auto block_expr = block (std::move (stmts), nullptr); return std::unique_ptr (new LoopExpr (std::move (block_expr), loc)); } std::unique_ptr Builder::trait_bound (TypePath bound) { return std::make_unique (bound, loc); } std::unique_ptr Builder::trait_impl (TypePath trait_path, std::unique_ptr target, std::vector> trait_items, std::vector> generics, WhereClause where_clause, Visibility visibility) const { return std::unique_ptr ( new TraitImpl (trait_path, /* unsafe */ false, /* exclam */ false, std::move (trait_items), std::move (generics), std::move (target), where_clause, visibility, {}, {}, loc)); } std::unique_ptr Builder::generic_type_param ( std::string type_representation, std::vector> &&bounds, std::unique_ptr &&type) { return std::make_unique (type_representation, loc, std::move (bounds), std::move (type), std::vector ()); } std::unique_ptr Builder::new_type (Type &type) { Type *t = ASTTypeBuilder::build (type); return std::unique_ptr (t); } std::unique_ptr Builder::new_lifetime_param (LifetimeParam ¶m) { Lifetime l = new_lifetime (param.get_lifetime ()); std::vector lifetime_bounds; for (auto b : param.get_lifetime_bounds ()) { Lifetime bl = new_lifetime (b); lifetime_bounds.push_back (bl); } auto p = new LifetimeParam (l, std::move (lifetime_bounds), param.get_outer_attrs (), param.get_locus ()); return std::unique_ptr (p); } std::unique_ptr Builder::new_type_param ( TypeParam ¶m, std::vector> extra_bounds) { location_t locus = param.get_locus (); AST::AttrVec outer_attrs = param.get_outer_attrs (); Identifier type_representation = param.get_type_representation (); std::vector> type_param_bounds; std::unique_ptr type = nullptr; if (param.has_type ()) type = new_type (param.get_type ()); for (auto &&extra_bound : extra_bounds) type_param_bounds.emplace_back (std::move (extra_bound)); for (const auto &b : param.get_type_param_bounds ()) { switch (b->get_bound_type ()) { case TypeParamBound::TypeParamBoundType::TRAIT: { const TraitBound &tb = (const TraitBound &) *b.get (); const TypePath &path = tb.get_type_path (); std::vector for_lifetimes; for (const auto &lifetime : tb.get_for_lifetimes ()) { std::vector lifetime_bounds; for (const auto &b : lifetime.get_lifetime_bounds ()) { Lifetime bl = new_lifetime (b); lifetime_bounds.push_back (std::move (bl)); } Lifetime nl = new_lifetime (lifetime.get_lifetime ()); LifetimeParam p (std::move (nl), std::move (lifetime_bounds), {}, lifetime.get_locus ()); for_lifetimes.push_back (std::move (p)); } std::vector> segments; for (auto &seg : path.get_segments ()) { switch (seg->get_type ()) { case TypePathSegment::REG: { const TypePathSegment &segment = (const TypePathSegment &) (*seg.get ()); TypePathSegment *s = new TypePathSegment ( segment.get_ident_segment (), segment.get_separating_scope_resolution (), segment.get_locus ()); std::unique_ptr sg (s); segments.push_back (std::move (sg)); } break; case TypePathSegment::GENERIC: { TypePathSegmentGeneric &generic = (TypePathSegmentGeneric &) (*seg.get ()); GenericArgs args = new_generic_args (generic.get_generic_args ()); TypePathSegmentGeneric *s = new TypePathSegmentGeneric ( generic.get_ident_segment (), false, std::move (args), generic.get_locus ()); std::unique_ptr sg (s); segments.push_back (std::move (sg)); } break; case TypePathSegment::FUNCTION: { rust_unreachable (); // TODO // const TypePathSegmentFunction &fn // = (const TypePathSegmentFunction &) (*seg.get ()); } break; } } TypePath p (std::move (segments), path.get_locus (), path.has_opening_scope_resolution_op ()); TraitBound *b = new TraitBound (std::move (p), tb.get_locus (), tb.is_in_parens (), tb.has_opening_question_mark (), std::move (for_lifetimes)); std::unique_ptr bound (b); type_param_bounds.push_back (std::move (bound)); } break; case TypeParamBound::TypeParamBoundType::LIFETIME: { const Lifetime &l = (const Lifetime &) *b.get (); auto bl = new Lifetime (l.get_lifetime_type (), l.get_lifetime_name (), l.get_locus ()); std::unique_ptr bound (bl); type_param_bounds.push_back (std::move (bound)); } break; } } auto type_param = new TypeParam (type_representation, locus, std::move (type_param_bounds), std::move (type), std::move (outer_attrs)); return std::unique_ptr (type_param); } Lifetime Builder::new_lifetime (const Lifetime &lifetime) { return Lifetime (lifetime.get_lifetime_type (), lifetime.get_lifetime_name (), lifetime.get_locus ()); } GenericArgs Builder::new_generic_args (GenericArgs &args) { std::vector lifetime_args; std::vector generic_args; std::vector binding_args; location_t locus = args.get_locus (); for (const auto &lifetime : args.get_lifetime_args ()) { Lifetime l = new_lifetime (lifetime); lifetime_args.push_back (std::move (l)); } for (auto &binding : args.get_binding_args ()) { Type &t = *binding.get_type_ptr ().get (); std::unique_ptr ty = new_type (t); GenericArgsBinding b (binding.get_identifier (), std::move (ty), binding.get_locus ()); binding_args.push_back (std::move (b)); } for (auto &arg : args.get_generic_args ()) { switch (arg.get_kind ()) { case GenericArg::Kind::Type: { std::unique_ptr ty = new_type (arg.get_type ()); GenericArg arg = GenericArg::create_type (std::move (ty)); } break; default: // FIXME rust_unreachable (); break; } } return GenericArgs (std::move (lifetime_args), std::move (generic_args), std::move (binding_args), locus); } } // namespace AST } // namespace Rust