// 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 // . #ifndef RUST_BIR_BUILDER_H #define RUST_BIR_BUILDER_H #include "rust-bir-builder-internal.h" #include "rust-bir-builder-pattern.h" #include "rust-bir-builder-expr-stmt.h" namespace Rust { namespace BIR { /** Top-level builder, which compiles a HIR function into a BIR function. */ class Builder final : public AbstractBuilder { std::vector> universal_region_bounds; public: explicit Builder (BuilderContext &ctx) : AbstractBuilder (ctx) {} Function build (HIR::Function &function) { rust_debug ("BIR::Builder::build function={%s}", function.get_function_name ().as_string ().c_str ()); auto fn_ty = lookup_type (function)->as (); handle_lifetime_params (fn_ty->get_num_lifetime_params ()); handle_lifetime_param_constraints (fn_ty->get_region_constraints ()); handle_return (fn_ty); for (auto ¶m : function.get_function_params ()) handle_param (param); handle_body (function.get_definition ()); auto region_hir_map = map_region_to_hir (function.get_generic_params (), ctx.fn_free_regions); return Function{ std::move (ctx.place_db), std::move (ctx.arguments), std::move (ctx.basic_blocks), std::move (ctx.fn_free_regions), std::move (universal_region_bounds), std::move (region_hir_map), function.get_locus (), }; } private: /** Instantiate `num_lifetime_params` free regions. */ void handle_lifetime_params (size_t num_lifetime_params) { FreeRegions regions; for (size_t i = 0; i < num_lifetime_params; i++) { regions.push_back (ctx.place_db.get_next_free_region ()); } rust_debug ("\tctx.fn_free_region={%s}", ctx.fn_free_regions.to_string ().c_str ()); ctx.fn_free_regions = regions; } void handle_lifetime_param_constraints ( const TyTy::RegionConstraints ®ion_constraints) { rust_debug ("\thandle_lifetime_param_constraints"); for (auto bound : region_constraints.region_region) { rust_assert (bound.first.is_early_bound ()); rust_assert (bound.second.is_early_bound ()); universal_region_bounds.emplace_back ( ctx.fn_free_regions[bound.first.get_index ()], ctx.fn_free_regions[bound.second.get_index ()]); auto last_bound = universal_region_bounds.back (); rust_debug ("\t\t %lu: %lu", (unsigned long) last_bound.first.value, (unsigned long) last_bound.second.value); } // TODO: handle type_region constraints } void handle_return (TyTy::FnType *fn_ty) { TyTy::BaseType *return_ty = fn_ty->get_return_type (); PlaceId return_place = ctx.place_db.add_temporary (return_ty); rust_assert (return_place == RETURN_VALUE_PLACE); // Set return place to use functions regions, not the fresh ones. ctx.place_db[return_place].regions = bind_regions (Resolver::TypeCheckContext::get () ->get_variance_analysis_ctx () .query_type_regions (fn_ty->get_return_type ()), ctx.fn_free_regions); } void handle_param (HIR::FunctionParam ¶m) { auto param_type = lookup_type (param.get_param_name ()); auto &pattern = param.get_param_name (); if (pattern.get_pattern_type () == HIR::Pattern::IDENTIFIER && !static_cast (pattern).get_is_ref ()) { // Avoid useless temporary variable for parameter to look like MIR. translated = declare_variable (pattern.get_mappings ()); ctx.arguments.push_back (translated); } else { translated = ctx.place_db.add_temporary (param_type); ctx.arguments.push_back (translated); PatternBindingBuilder (ctx, translated, tl::nullopt) .go (param.get_param_name ()); } // Set parameter place to use functions regions, not the fresh ones. ctx.place_db[translated].regions = bind_regions (Resolver::TypeCheckContext::get () ->get_variance_analysis_ctx () .query_type_regions (param_type), ctx.fn_free_regions); } void handle_body (HIR::BlockExpr &body) { translated = ExprStmtBuilder (ctx).build (body, RETURN_VALUE_PLACE); if (!ctx.get_current_bb ().is_terminated ()) { if (ctx.place_db[RETURN_VALUE_PLACE].tyty->is_unit ()) { push_assignment (RETURN_VALUE_PLACE, ctx.place_db.get_constant ( ctx.place_db[RETURN_VALUE_PLACE].tyty), body.get_end_locus ()); } auto return_location = body.has_expr () ? body.get_final_expr ().get_locus () : body.get_end_locus (); push_return (return_location); } } // Maps named lifetime parameters to their respective HIR node const std::unordered_map map_region_to_hir ( const std::vector> &generic_params, const FreeRegions ®ions) { std::unordered_map result; size_t region_index = 0; for (auto &generic_param : generic_params) { if (generic_param->get_kind () == HIR::GenericParam::GenericKind::LIFETIME) { result[regions[region_index++].value] = static_cast (generic_param.get ()); } } return result; } }; } // namespace BIR } // namespace Rust #endif // RUST_BIR_BUILDER_H