// Part of the Carbon Language project, under the Apache License v2.0 with LLVM // Exceptions. See /LICENSE for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception #include "toolchain/check/pattern_match.h" #include #include #include "llvm/ADT/STLExtras.h" #include "llvm/ADT/SmallVector.h" #include "toolchain/base/kind_switch.h" #include "toolchain/check/context.h" #include "toolchain/check/convert.h" namespace Carbon::Check { // Returns a best-effort name for the given ParamPattern, suitable for use in // IR pretty-printing. // TODO: Resolve overlap with SemIR::Function::ParamPatternInfo::GetNameId template static auto GetPrettyName(Context& context, ParamPattern param_pattern) -> SemIR::NameId { if (context.insts().Is( param_pattern.subpattern_id)) { return SemIR::NameId::ReturnSlot; } if (auto binding_pattern = context.insts().TryGetAs( param_pattern.subpattern_id)) { return context.entity_names().Get(binding_pattern->entity_name_id).name_id; } return SemIR::NameId::Invalid; } namespace { // Selects between the different kinds of pattern matching. enum class MatchKind { // Caller pattern matching occurs on the caller side of a function call, and // is responsible for matching the argument expression against the portion // of the pattern above the ParamPattern insts. Caller, // Callee pattern matching occurs in the function decl block, and is // responsible for matching the function's calling-convention parameters // against the portion of the pattern below the ParamPattern insts. Callee, // TODO: Add enumerator for non-function-call pattern match. }; // The collected state of a pattern-matching operation. class MatchContext { public: struct WorkItem { SemIR::InstId pattern_id; // Invalid when processing the callee side. SemIR::InstId scrutinee_id; }; // Constructs a MatchContext. If `callee_specific_id` is valid, this pattern // match operation is part of implementing the signature of the given // specific. explicit MatchContext(MatchKind kind, SemIR::SpecificId callee_specific_id = SemIR::SpecificId::Invalid) : next_index_(0), kind_(kind), callee_specific_id_(callee_specific_id), return_slot_id_(SemIR::InstId::Invalid) {} // Adds a work item to the stack. auto AddWork(WorkItem work_item) -> void { stack_.push_back(work_item); } // Processes all work items on the stack. When performing caller pattern // matching, returns an inst block with one inst reference for each // calling-convention argument. When performing callee pattern matching, // returns an inst block with references to all the emitted BindName insts. auto DoWork(Context& context) -> SemIR::InstBlockId; auto return_slot_id() const -> SemIR::InstId { return return_slot_id_; } private: // Allocates the next unallocated RuntimeParamIndex, starting from 0. auto NextRuntimeIndex() -> SemIR::RuntimeParamIndex { auto result = next_index_; ++next_index_.index; return result; } // Emits the pattern-match insts necessary to match the pattern inst // `entry.pattern_id` against the scrutinee value `entry.scrutinee_id`, and // adds to `stack_` any work necessary to traverse into its subpatterns. This // behavior is contingent on the kind of match being performed, as indicated // by kind_`. For example, when performing a callee pattern match, this does // not emit insts for patterns on the caller side. However, it still traverses // into subpatterns if any of their descendants might emit insts. // TODO: Require that `entry.scrutinee_id` is valid if and only if insts // should be emitted, once we start emitting `Param` insts in the // `ParamPattern` case. auto EmitPatternMatch(Context& context, MatchContext::WorkItem entry) -> void; // The stack of work to be processed. llvm::SmallVector stack_; // The next index to be allocated by `NextRuntimeIndex`. SemIR::RuntimeParamIndex next_index_; // The pending results that will be returned by the current `DoWork` call. llvm::SmallVector results_; // The kind of pattern match being performed. MatchKind kind_; // The SpecificId of the function being called (if any). SemIR::SpecificId callee_specific_id_; // The return slot inst emitted by `DoWork`, if any. // TODO: Can this be added to the block returned by `DoWork`, instead? SemIR::InstId return_slot_id_; }; } // namespace auto MatchContext::DoWork(Context& context) -> SemIR::InstBlockId { results_.reserve(stack_.size()); while (!stack_.empty()) { EmitPatternMatch(context, stack_.pop_back_val()); } auto block_id = context.inst_blocks().Add(results_); results_.clear(); return block_id; } auto MatchContext::EmitPatternMatch(Context& context, MatchContext::WorkItem entry) -> void { if (entry.pattern_id == SemIR::InstId::BuiltinErrorInst) { results_.push_back(SemIR::InstId::BuiltinErrorInst); return; } DiagnosticAnnotationScope annotate_diagnostics( &context.emitter(), [&](auto& builder) { if (kind_ == MatchKind::Caller) { CARBON_DIAGNOSTIC(InCallToFunctionParam, Note, "initializing function parameter"); builder.Note(entry.pattern_id, InCallToFunctionParam); } }); auto pattern = context.insts().GetWithLocId(entry.pattern_id); CARBON_KIND_SWITCH(pattern.inst) { case SemIR::BindingPattern::Kind: case SemIR::SymbolicBindingPattern::Kind: { CARBON_CHECK(kind_ == MatchKind::Callee); auto binding_pattern = pattern.inst.As(); auto cache_entry = context.bind_name_cache().Lookup(binding_pattern.entity_name_id); // The cached bind_name should only be used once. auto bind_name_id = std::exchange(cache_entry.value(), SemIR::InstId::Invalid); auto bind_name = context.insts().GetAs(bind_name_id); CARBON_CHECK(!bind_name.value_id.is_valid()); bind_name.value_id = entry.scrutinee_id; context.ReplaceInstBeforeConstantUse(bind_name_id, bind_name); context.inst_block_stack().AddInstId(bind_name_id); results_.push_back(bind_name_id); break; } case CARBON_KIND(SemIR::AddrPattern addr_pattern): { if (kind_ == MatchKind::Callee) { // We're emitting pattern-match IR for the callee, but we're still on // the caller side of the pattern, so we traverse without emitting any // insts. AddWork({.pattern_id = addr_pattern.inner_id, .scrutinee_id = SemIR::InstId::Invalid}); break; } CARBON_CHECK(entry.scrutinee_id.is_valid()); auto scrutinee_ref_id = ConvertToValueOrRefExpr(context, entry.scrutinee_id); switch (SemIR::GetExprCategory(context.sem_ir(), scrutinee_ref_id)) { case SemIR::ExprCategory::Error: case SemIR::ExprCategory::DurableRef: case SemIR::ExprCategory::EphemeralRef: break; default: CARBON_DIAGNOSTIC(AddrSelfIsNonRef, Error, "`addr self` method cannot be invoked on a value"); context.emitter().Emit( TokenOnly(context.insts().GetLocId(entry.scrutinee_id)), AddrSelfIsNonRef); results_.push_back(SemIR::InstId::BuiltinErrorInst); return; } auto scrutinee_ref = context.insts().Get(scrutinee_ref_id); auto new_scrutinee = context.AddInst( context.insts().GetLocId(scrutinee_ref_id), {.type_id = context.GetPointerType(scrutinee_ref.type_id()), .lvalue_id = scrutinee_ref_id}); AddWork( {.pattern_id = addr_pattern.inner_id, .scrutinee_id = new_scrutinee}); break; } case CARBON_KIND(SemIR::ValueParamPattern param_pattern): { CARBON_CHECK(param_pattern.runtime_index.index < 0 || static_cast(param_pattern.runtime_index.index) == results_.size(), "Parameters out of order; expecting {0} but got {1}", results_.size(), param_pattern.runtime_index.index); switch (kind_) { case MatchKind::Caller: { CARBON_CHECK(entry.scrutinee_id.is_valid()); if (entry.scrutinee_id == SemIR::InstId::BuiltinErrorInst) { results_.push_back(SemIR::InstId::BuiltinErrorInst); } else { results_.push_back(ConvertToValueOfType( context, context.insts().GetLocId(entry.scrutinee_id), entry.scrutinee_id, SemIR::GetTypeInSpecific(context.sem_ir(), callee_specific_id_, param_pattern.type_id))); } // Do not traverse farther, because the caller side of the pattern // ends here. break; } case MatchKind::Callee: { if (param_pattern.runtime_index == SemIR::RuntimeParamIndex::Unknown) { param_pattern.runtime_index = NextRuntimeIndex(); context.ReplaceInstBeforeConstantUse(entry.pattern_id, param_pattern); } AddWork( {.pattern_id = param_pattern.subpattern_id, .scrutinee_id = context.AddInst( pattern.loc_id, {.type_id = param_pattern.type_id, .runtime_index = param_pattern.runtime_index, .pretty_name_id = GetPrettyName(context, param_pattern)})}); break; } } break; } case CARBON_KIND(SemIR::OutParamPattern param_pattern): { switch (kind_) { case MatchKind::Caller: { CARBON_CHECK(entry.scrutinee_id.is_valid()); CARBON_CHECK(context.insts().Get(entry.scrutinee_id).type_id() == SemIR::GetTypeInSpecific(context.sem_ir(), callee_specific_id_, param_pattern.type_id)); results_.push_back(entry.scrutinee_id); // Do not traverse farther, because the caller side of the pattern // ends here. break; } case MatchKind::Callee: { // TODO: Consider ways to address near-duplication with the // ValueParamPattern case. if (param_pattern.runtime_index == SemIR::RuntimeParamIndex::Unknown) { param_pattern.runtime_index = NextRuntimeIndex(); context.ReplaceInstBeforeConstantUse(entry.pattern_id, param_pattern); } AddWork( {.pattern_id = param_pattern.subpattern_id, .scrutinee_id = context.AddInst( pattern.loc_id, {.type_id = param_pattern.type_id, .runtime_index = param_pattern.runtime_index, .pretty_name_id = GetPrettyName(context, param_pattern)})}); break; } } break; } case CARBON_KIND(SemIR::ReturnSlotPattern return_slot_pattern): { CARBON_CHECK(kind_ == MatchKind::Callee); return_slot_id_ = context.AddInst( pattern.loc_id, {.type_id = return_slot_pattern.type_id, .type_inst_id = return_slot_pattern.type_inst_id, .storage_id = entry.scrutinee_id}); break; } default: { CARBON_FATAL("Inst kind not handled: {0}", pattern.inst.kind()); } } } auto CalleePatternMatch(Context& context, SemIR::InstBlockId implicit_param_patterns_id, SemIR::InstBlockId param_patterns_id, SemIR::InstId return_slot_pattern_id) -> ParameterBlocks { auto params_id = SemIR::InstBlockId::Invalid; auto implicit_params_id = SemIR::InstBlockId::Invalid; MatchContext match(MatchKind::Callee); if (implicit_param_patterns_id.is_valid()) { // We add work to the stack in reverse so that the results will be produced // in the original order. for (SemIR::InstId inst_id : llvm::reverse(context.inst_blocks().Get(implicit_param_patterns_id))) { match.AddWork( {.pattern_id = inst_id, .scrutinee_id = SemIR::InstId::Invalid}); } implicit_params_id = match.DoWork(context); } if (param_patterns_id.is_valid()) { for (SemIR::InstId inst_id : llvm::reverse(context.inst_blocks().Get(param_patterns_id))) { match.AddWork( {.pattern_id = inst_id, .scrutinee_id = SemIR::InstId::Invalid}); } params_id = match.DoWork(context); } if (return_slot_pattern_id.is_valid()) { match.AddWork({.pattern_id = return_slot_pattern_id, .scrutinee_id = SemIR::InstId::Invalid}); CARBON_CHECK(match.DoWork(context) == SemIR::InstBlockId::Empty); } return {.implicit_params_id = implicit_params_id, .params_id = params_id, .return_slot_id = match.return_slot_id()}; } auto CallerPatternMatch(Context& context, SemIR::SpecificId specific_id, SemIR::InstId self_pattern_id, SemIR::InstBlockId param_patterns_id, SemIR::InstId return_slot_pattern_id, SemIR::InstId self_arg_id, llvm::ArrayRef arg_refs, SemIR::InstId return_slot_arg_id) -> SemIR::InstBlockId { MatchContext match(MatchKind::Caller, specific_id); // Track the return storage, if present. if (return_slot_arg_id.is_valid()) { CARBON_CHECK(return_slot_pattern_id.is_valid()); match.AddWork({.pattern_id = return_slot_pattern_id, .scrutinee_id = return_slot_arg_id}); } // Check type conversions per-element. for (auto [arg_id, param_pattern_id] : llvm::reverse(llvm::zip_equal( arg_refs, context.inst_blocks().GetOrEmpty(param_patterns_id)))) { auto runtime_index = SemIR::Function::GetParamPatternInfoFromPatternId( context.sem_ir(), param_pattern_id) .inst.runtime_index; if (!runtime_index.is_valid()) { // Not a runtime parameter: we don't pass an argument. continue; } match.AddWork({.pattern_id = param_pattern_id, .scrutinee_id = arg_id}); } if (self_pattern_id.is_valid()) { match.AddWork({.pattern_id = self_pattern_id, .scrutinee_id = self_arg_id}); } return match.DoWork(context); } } // namespace Carbon::Check