diff options
author | Owen Avery <powerboat9.gamer@gmail.com> | 2023-10-22 21:44:01 -0400 |
---|---|---|
committer | Arthur Cohen <arthur.cohen@embecosm.com> | 2024-01-16 19:09:35 +0100 |
commit | da87ef4d692b348be41c082eabb37f21af73f392 (patch) | |
tree | ef6a74c39041b5191987608a3ba9a27faf9ad9be /gcc/rust/backend/rust-compile-expr.cc | |
parent | 3a874d936b2fcbb1c77b146d8b1d651e52eb618f (diff) | |
download | gcc-da87ef4d692b348be41c082eabb37f21af73f392.zip gcc-da87ef4d692b348be41c082eabb37f21af73f392.tar.gz gcc-da87ef4d692b348be41c082eabb37f21af73f392.tar.bz2 |
gccrs: Compile pattern match statements into conditional statements
gcc/rust/ChangeLog:
* backend/rust-compile-expr.cc
(patterns_mergeable): Remove.
(struct PatternMerge): Remove.
(sort_tuple_patterns): Remove.
(simplify_tuple_match): Remove.
(CompileExpr::visit): Modify compilation of MatchExpr.
* backend/rust-compile-pattern.cc
(CompilePatternCaseLabelExpr::visit): Replace with...
(CompilePatternCheckExpr::visit): ...this.
* backend/rust-compile-pattern.h
(class CompilePatternCaseLabelExpr): Replace with...
(class CompilePatternCheckExpr): ...this.
* backend/rust-compile-type.cc
(TyTyResolveCompile::get_implicit_enumeral_node_type):
Make enumeral type equivalent to isize.
Signed-off-by: Owen Avery <powerboat9.gamer@gmail.com>
Diffstat (limited to 'gcc/rust/backend/rust-compile-expr.cc')
-rw-r--r-- | gcc/rust/backend/rust-compile-expr.cc | 494 |
1 files changed, 36 insertions, 458 deletions
diff --git a/gcc/rust/backend/rust-compile-expr.cc b/gcc/rust/backend/rust-compile-expr.cc index acec57d..30863e4 100644 --- a/gcc/rust/backend/rust-compile-expr.cc +++ b/gcc/rust/backend/rust-compile-expr.cc @@ -929,322 +929,6 @@ CompileExpr::visit (HIR::AssignmentExpr &expr) ctx->add_statement (assignment); } -// Helper for sort_tuple_patterns. -// Determine whether Patterns a and b are really the same pattern. -// FIXME: This is a nasty hack to avoid properly implementing a comparison -// for Patterns, which we really probably do want at some point. -static bool -patterns_mergeable (HIR::Pattern *a, HIR::Pattern *b) -{ - if (!a || !b) - return false; - - HIR::Pattern::PatternType pat_type = a->get_pattern_type (); - if (b->get_pattern_type () != pat_type) - return false; - - switch (pat_type) - { - case HIR::Pattern::PatternType::PATH: { - // FIXME: this is far too naive - HIR::PathPattern &aref = *static_cast<HIR::PathPattern *> (a); - HIR::PathPattern &bref = *static_cast<HIR::PathPattern *> (b); - if (aref.get_num_segments () != bref.get_num_segments ()) - return false; - - const auto &asegs = aref.get_segments (); - const auto &bsegs = bref.get_segments (); - for (size_t i = 0; i < asegs.size (); i++) - { - if (asegs[i].as_string () != bsegs[i].as_string ()) - return false; - } - return true; - } - break; - case HIR::Pattern::PatternType::LITERAL: { - HIR::LiteralPattern &aref = *static_cast<HIR::LiteralPattern *> (a); - HIR::LiteralPattern &bref = *static_cast<HIR::LiteralPattern *> (b); - return aref.get_literal ().is_equal (bref.get_literal ()); - } - break; - case HIR::Pattern::PatternType::IDENTIFIER: { - // TODO - } - break; - case HIR::Pattern::PatternType::WILDCARD: - return true; - break; - - // TODO - - default:; - } - return false; -} - -// A little container for rearranging the patterns and cases in a match -// expression while simplifying. -struct PatternMerge -{ - std::unique_ptr<HIR::MatchCase> wildcard; - std::vector<std::unique_ptr<HIR::Pattern>> heads; - std::vector<std::vector<HIR::MatchCase>> cases; -}; - -// Helper for simplify_tuple_match. -// For each tuple pattern in a given match, pull out the first elt of the -// tuple and construct a new MatchCase with the remaining tuple elts as the -// pattern. Return a mapping from each _unique_ first tuple element to a -// vec of cases for a new match. -// -// FIXME: This used to be a std::map<Pattern, Vec<MatchCase>>, but it doesn't -// actually work like we want - the Pattern includes an HIR ID, which is unique -// per Pattern object. This means we don't have a good means for comparing -// Patterns. It would probably be best to actually implement a means of -// properly comparing patterns, and then use an actual map. -// -static struct PatternMerge -sort_tuple_patterns (HIR::MatchExpr &expr) -{ - rust_assert (expr.get_scrutinee_expr ()->get_expression_type () - == HIR::Expr::ExprType::Tuple); - - struct PatternMerge result; - result.wildcard = nullptr; - result.heads = std::vector<std::unique_ptr<HIR::Pattern>> (); - result.cases = std::vector<std::vector<HIR::MatchCase>> (); - - for (auto &match_case : expr.get_match_cases ()) - { - HIR::MatchArm &case_arm = match_case.get_arm (); - - // FIXME: Note we are only dealing with the first pattern in the arm. - // The patterns vector in the arm might hold many patterns, which are the - // patterns separated by the '|' token. Rustc abstracts these as "Or" - // patterns, and part of its simplification process is to get rid of them. - // We should get rid of the ORs too, maybe here or earlier than here? - auto pat = case_arm.get_patterns ()[0]->clone_pattern (); - - // Record wildcards so we can add them in inner matches. - if (pat->get_pattern_type () == HIR::Pattern::PatternType::WILDCARD) - { - // The *whole* pattern is a wild card (_). - result.wildcard - = std::unique_ptr<HIR::MatchCase> (new HIR::MatchCase (match_case)); - continue; - } - - rust_assert (pat->get_pattern_type () - == HIR::Pattern::PatternType::TUPLE); - - auto ref = *static_cast<HIR::TuplePattern *> (pat.get ()); - - rust_assert (ref.has_tuple_pattern_items ()); - - auto items - = HIR::TuplePattern (ref).get_items ()->clone_tuple_pattern_items (); - if (items->get_item_type () - == HIR::TuplePatternItems::TuplePatternItemType::MULTIPLE) - { - auto items_ref - = *static_cast<HIR::TuplePatternItemsMultiple *> (items.get ()); - - // Pop the first pattern out - auto patterns = std::vector<std::unique_ptr<HIR::Pattern>> (); - auto first = items_ref.get_patterns ()[0]->clone_pattern (); - for (auto p = items_ref.get_patterns ().begin () + 1; - p != items_ref.get_patterns ().end (); p++) - { - patterns.push_back ((*p)->clone_pattern ()); - } - - // if there is only one pattern left, don't make a tuple out of it - std::unique_ptr<HIR::Pattern> result_pattern; - if (patterns.size () == 1) - { - result_pattern = std::move (patterns[0]); - } - else - { - auto new_items = std::unique_ptr<HIR::TuplePatternItems> ( - new HIR::TuplePatternItemsMultiple (std::move (patterns))); - - // Construct a TuplePattern from the rest of the patterns - result_pattern = std::unique_ptr<HIR::Pattern> ( - new HIR::TuplePattern (ref.get_mappings (), - std::move (new_items), - ref.get_locus ())); - } - - // I don't know why we need to make foo separately here but - // using the { new_tuple } syntax in new_arm constructor does not - // compile. - auto foo = std::vector<std::unique_ptr<HIR::Pattern>> (); - foo.emplace_back (std::move (result_pattern)); - HIR::MatchArm new_arm (std::move (foo), UNDEF_LOCATION, nullptr, - AST::AttrVec ()); - - HIR::MatchCase new_case (match_case.get_mappings (), new_arm, - match_case.get_expr ()->clone_expr ()); - - bool pushed = false; - for (size_t i = 0; i < result.heads.size (); i++) - { - if (patterns_mergeable (result.heads[i].get (), first.get ())) - { - result.cases[i].push_back (new_case); - pushed = true; - } - } - - if (!pushed) - { - result.heads.push_back (std::move (first)); - result.cases.push_back ({new_case}); - } - } - else /* TuplePatternItemType::RANGED */ - { - // FIXME - rust_unreachable (); - } - } - - return result; -} - -// Helper for CompileExpr::visit (HIR::MatchExpr). -// Given a MatchExpr where the scrutinee is some kind of tuple, build an -// equivalent match where only one element of the tuple is examined at a time. -// This resulting match can then be lowered to a SWITCH_EXPR tree directly. -// -// The approach is as follows: -// 1. Split the scrutinee and each pattern into the first (head) and the -// rest (tail). -// 2. Build a mapping of unique pattern heads to the cases (tail and expr) -// that shared that pattern head in the original match. -// (This is the job of sort_tuple_patterns ()). -// 3. For each unique pattern head, build a new MatchCase where the pattern -// is the unique head, and the expression is a new match where: -// - The scrutinee is the tail of the original scrutinee -// - The cases are are those built by the mapping in step 2, i.e. the -// tails of the patterns and the corresponing expressions from the -// original match expression. -// 4. Do this recursively for each inner match, until there is nothing more -// to simplify. -// 5. Build the resulting match which scrutinizes the head of the original -// scrutinee, using the cases built in step 3. -static HIR::MatchExpr -simplify_tuple_match (HIR::MatchExpr &expr) -{ - if (expr.get_scrutinee_expr ()->get_expression_type () - != HIR::Expr::ExprType::Tuple) - return expr; - - auto ref = *static_cast<HIR::TupleExpr *> (expr.get_scrutinee_expr ().get ()); - - auto &tail = ref.get_tuple_elems (); - rust_assert (tail.size () > 1); - - auto head = std::move (tail[0]); - tail.erase (tail.begin (), tail.begin () + 1); - - // e.g. - // match (tupA, tupB, tupC) { - // (a1, b1, c1) => { blk1 }, - // (a2, b2, c2) => { blk2 }, - // (a1, b3, c3) => { blk3 }, - // } - // tail = (tupB, tupC) - // head = tupA - - // Make sure the tail is only a tuple if it consists of at least 2 elements. - std::unique_ptr<HIR::Expr> remaining; - if (tail.size () == 1) - remaining = std::move (tail[0]); - else - remaining = std::unique_ptr<HIR::Expr> ( - new HIR::TupleExpr (ref.get_mappings (), std::move (tail), - AST::AttrVec (), ref.get_outer_attrs (), - ref.get_locus ())); - - // e.g. - // a1 -> [(b1, c1) => { blk1 }, - // (b3, c3) => { blk3 }] - // a2 -> [(b2, c2) => { blk2 }] - struct PatternMerge map = sort_tuple_patterns (expr); - - std::vector<HIR::MatchCase> cases; - // Construct the inner match for each unique first elt of the tuple - // patterns - for (size_t i = 0; i < map.heads.size (); i++) - { - auto inner_match_cases = map.cases[i]; - - // If there is a wildcard at the outer match level, then need to - // propegate the wildcard case into *every* inner match. - // FIXME: It is probably not correct to add this unconditionally, what if - // we have a pattern like (a, _, c)? Then there is already a wildcard in - // the inner matches, and having two will cause two 'default:' blocks - // which is an error. - if (map.wildcard != nullptr) - { - inner_match_cases.push_back (*(map.wildcard.get ())); - } - - // match (tupB, tupC) { - // (b1, c1) => { blk1 }, - // (b3, c3) => { blk3 } - // } - HIR::MatchExpr inner_match (expr.get_mappings (), - remaining->clone_expr (), inner_match_cases, - AST::AttrVec (), expr.get_outer_attrs (), - expr.get_locus ()); - - inner_match = simplify_tuple_match (inner_match); - - auto outer_arm_pat = std::vector<std::unique_ptr<HIR::Pattern>> (); - outer_arm_pat.emplace_back (map.heads[i]->clone_pattern ()); - - HIR::MatchArm outer_arm (std::move (outer_arm_pat), expr.get_locus ()); - - // Need to move the inner match to the heap and put it in a unique_ptr to - // build the actual match case of the outer expression - // auto inner_expr = std::unique_ptr<HIR::Expr> (new HIR::MatchExpr - // (inner_match)); - auto inner_expr = inner_match.clone_expr (); - - // a1 => match (tupB, tupC) { ... } - HIR::MatchCase outer_case (expr.get_mappings (), outer_arm, - std::move (inner_expr)); - - cases.push_back (outer_case); - } - - // If there was a wildcard, make sure to include it at the outer match level - // too. - if (map.wildcard != nullptr) - { - cases.push_back (*(map.wildcard.get ())); - } - - // match tupA { - // a1 => match (tupB, tupC) { - // (b1, c1) => { blk1 }, - // (b3, c3) => { blk3 } - // } - // a2 => match (tupB, tupC) { - // (b2, c2) => { blk2 } - // } - // } - HIR::MatchExpr outer_match (expr.get_mappings (), std::move (head), cases, - AST::AttrVec (), expr.get_outer_attrs (), - expr.get_locus ()); - - return outer_match; -} - // Helper for CompileExpr::visit (HIR::MatchExpr). // Check that the scrutinee of EXPR is a valid kind of expression to match on. // Return the TypeKind of the scrutinee if it is valid, or TyTy::TypeKind::ERROR @@ -1347,111 +1031,6 @@ CompileExpr::visit (HIR::MatchExpr &expr) expr.get_locus ()); ctx->add_statement (assignment); - tree match_scrutinee_expr_qualifier_expr; - if (TyTy::is_primitive_type_kind (scrutinee_kind)) - { - match_scrutinee_expr_qualifier_expr = match_scrutinee_expr; - } - else if (scrutinee_kind == TyTy::TypeKind::ADT) - { - // need to access qualifier the field, if we use QUAL_UNION_TYPE this - // would be DECL_QUALIFIER i think. For now this will just access the - // first record field and its respective qualifier because it will always - // be set because this is all a big special union - tree scrutinee_first_record_expr = Backend::struct_field_expression ( - match_scrutinee_expr, 0, expr.get_scrutinee_expr ()->get_locus ()); - match_scrutinee_expr_qualifier_expr = Backend::struct_field_expression ( - scrutinee_first_record_expr, 0, - expr.get_scrutinee_expr ()->get_locus ()); - } - else if (scrutinee_kind == TyTy::TypeKind::REF) - { - tree indirect - = indirect_expression (match_scrutinee_expr, expr.get_locus ()); - match_scrutinee_expr_qualifier_expr = indirect; - } - else if (scrutinee_kind == TyTy::TypeKind::TUPLE) - { - // match on tuple becomes a series of nested switches, with one level - // for each element of the tuple from left to right. - auto exprtype = expr.get_scrutinee_expr ()->get_expression_type (); - switch (exprtype) - { - case HIR::Expr::ExprType::Tuple: { - // Build an equivalent expression which is nicer to lower. - HIR::MatchExpr outer_match = simplify_tuple_match (expr); - - // We've rearranged the match into something that lowers better - // to GENERIC trees. - // For actually doing the lowering we need to compile the match - // we've just made. But we're half-way through compiling the - // original one. - // ... - // For now, let's just replace the original with the rearranged one - // we just made, and compile that instead. What could go wrong? :) - // - // FIXME: What about when we decide a temporary is needed above? - // We might have already pushed a statement for it that - // we no longer need. Probably need to rearrange the order - // of these steps. - expr = outer_match; - - scrutinee_kind = check_match_scrutinee (expr, ctx); - if (scrutinee_kind == TyTy::TypeKind::ERROR) - { - translated = error_mark_node; - return; - } - - // Now compile the scrutinee of the simplified match. - // FIXME: this part is duplicated from above. - match_scrutinee_expr - = CompileExpr::Compile (expr.get_scrutinee_expr ().get (), ctx); - - if (TyTy::is_primitive_type_kind (scrutinee_kind)) - { - match_scrutinee_expr_qualifier_expr = match_scrutinee_expr; - } - else if (scrutinee_kind == TyTy::TypeKind::ADT) - { - // need to access qualifier the field, if we use QUAL_UNION_TYPE - // this would be DECL_QUALIFIER i think. For now this will just - // access the first record field and its respective qualifier - // because it will always be set because this is all a big - // special union - tree scrutinee_first_record_expr - = Backend::struct_field_expression ( - match_scrutinee_expr, 0, - expr.get_scrutinee_expr ()->get_locus ()); - match_scrutinee_expr_qualifier_expr - = Backend::struct_field_expression ( - scrutinee_first_record_expr, 0, - expr.get_scrutinee_expr ()->get_locus ()); - } - else - { - // FIXME: There are other cases, but it better not be a Tuple - rust_unreachable (); - } - } - break; - - case HIR::Expr::ExprType::Path: { - // FIXME - rust_unreachable (); - } - break; - - default: - rust_unreachable (); - } - } - else - { - // FIXME: match on other types of expressions not yet implemented. - rust_unreachable (); - } - // setup the end label so the cases can exit properly tree fndecl = fnctx.fndecl; location_t end_label_locus = expr.get_locus (); // FIXME @@ -1461,58 +1040,57 @@ CompileExpr::visit (HIR::MatchExpr &expr) tree end_label_decl_statement = Backend::label_definition_statement (end_label); - // setup the switch-body-block - location_t start_location = UNKNOWN_LOCATION; // FIXME - location_t end_location = UNKNOWN_LOCATION; // FIXME - tree switch_body_block = Backend::block (fndecl, enclosing_scope, {}, - start_location, end_location); - ctx->push_block (switch_body_block); - for (auto &kase : expr.get_match_cases ()) { // for now lets just get single pattern's working HIR::MatchArm &kase_arm = kase.get_arm (); rust_assert (kase_arm.get_patterns ().size () > 0); - // generate implicit label - location_t arm_locus = kase_arm.get_locus (); - tree case_label - = Backend::label (fndecl, "" /* empty creates an artificial label */, - arm_locus); - - // setup the bindings for the block for (auto &kase_pattern : kase_arm.get_patterns ()) { - tree switch_kase_expr - = CompilePatternCaseLabelExpr::Compile (kase_pattern.get (), - case_label, ctx); - ctx->add_statement (switch_kase_expr); + // setup the match-arm-body-block + location_t start_location = UNKNOWN_LOCATION; // FIXME + location_t end_location = UNKNOWN_LOCATION; // FIXME + tree arm_body_block = Backend::block (fndecl, enclosing_scope, {}, + start_location, end_location); + + ctx->push_block (arm_body_block); + // setup the bindings for the block CompilePatternBindings::Compile (kase_pattern.get (), match_scrutinee_expr, ctx); - } - // compile the expr and setup the assignment if required when tmp != NULL - tree kase_expr_tree = CompileExpr::Compile (kase.get_expr ().get (), ctx); - tree result_reference = Backend::var_expression (tmp, arm_locus); - tree assignment - = Backend::assignment_statement (result_reference, kase_expr_tree, - arm_locus); - ctx->add_statement (assignment); - - // go to end label - tree goto_end_label - = build1_loc (arm_locus, GOTO_EXPR, void_type_node, end_label); - ctx->add_statement (goto_end_label); + // compile the expr and setup the assignment if required when tmp != + // NULL + location_t arm_locus = kase_arm.get_locus (); + tree kase_expr_tree + = CompileExpr::Compile (kase.get_expr ().get (), ctx); + tree result_reference = Backend::var_expression (tmp, arm_locus); + tree assignment + = Backend::assignment_statement (result_reference, kase_expr_tree, + arm_locus); + ctx->add_statement (assignment); + + // go to end label + tree goto_end_label + = build1_loc (arm_locus, GOTO_EXPR, void_type_node, end_label); + ctx->add_statement (goto_end_label); + + ctx->pop_block (); + + tree check_expr + = CompilePatternCheckExpr::Compile (kase_pattern.get (), + match_scrutinee_expr, ctx); + + tree check_stmt + = Backend::if_statement (NULL_TREE, check_expr, arm_body_block, + NULL_TREE, kase_pattern->get_locus ()); + + ctx->add_statement (check_stmt); + } } // setup the switch expression - tree match_body = ctx->pop_block (); - tree match_expr_stmt - = build2_loc (expr.get_locus (), SWITCH_EXPR, - TREE_TYPE (match_scrutinee_expr_qualifier_expr), - match_scrutinee_expr_qualifier_expr, match_body); - ctx->add_statement (match_expr_stmt); ctx->add_statement (end_label_decl_statement); translated = Backend::var_expression (tmp, expr.get_locus ()); |