pattern.cpp 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #include "toolchain/check/pattern.h"
  5. #include "toolchain/check/control_flow.h"
  6. #include "toolchain/check/inst.h"
  7. #include "toolchain/check/return.h"
  8. #include "toolchain/check/type.h"
  9. namespace Carbon::Check {
  10. auto BeginSubpattern(Context& context) -> void {
  11. context.inst_block_stack().Push();
  12. context.region_stack().PushRegion(context.inst_block_stack().PeekOrAdd());
  13. }
  14. auto EndSubpatternAsExpr(Context& context, SemIR::InstId result_id)
  15. -> SemIR::ExprRegionId {
  16. if (context.region_stack().PeekRegion().size() > 1) {
  17. // End the exit block with a branch to a successor block, whose contents
  18. // will be determined later.
  19. AddInst(context,
  20. SemIR::LocIdAndInst::NoLoc<SemIR::Branch>(
  21. {.target_id = context.inst_blocks().AddPlaceholder()}));
  22. } else {
  23. // This single-block region will be inserted as a SpliceBlock, so we don't
  24. // need control flow out of it.
  25. }
  26. auto block_id = context.inst_block_stack().Pop();
  27. CARBON_CHECK(block_id == context.region_stack().PeekRegion().back());
  28. // TODO: Is it possible to validate that this region is genuinely
  29. // single-entry, single-exit?
  30. return context.sem_ir().expr_regions().Add(
  31. {.block_ids = context.region_stack().PopRegion(),
  32. .result_id = result_id});
  33. }
  34. auto EndSubpatternAsNonExpr(Context& context) -> void {
  35. auto block_id = context.inst_block_stack().Pop();
  36. CARBON_CHECK(block_id == context.region_stack().PeekRegion().back());
  37. CARBON_CHECK(context.region_stack().PeekRegion().size() == 1);
  38. CARBON_CHECK(context.inst_blocks().Get(block_id).empty());
  39. context.region_stack().PopAndDiscardRegion();
  40. }
  41. auto AddBindingEntityName(Context& context, SemIR::NameId name_id,
  42. SemIR::ConstantId form_id, bool is_unused,
  43. BindingPhase phase) -> SemIR::EntityNameId {
  44. SemIR::EntityName entity_name = {
  45. .name_id = name_id,
  46. .parent_scope_id = context.scope_stack().PeekNameScopeId(),
  47. .is_unused = is_unused || name_id == SemIR::NameId::Underscore};
  48. if (phase != BindingPhase::Runtime) {
  49. entity_name.bind_index_value =
  50. context.scope_stack().AddCompileTimeBinding().index;
  51. entity_name.is_template = phase == BindingPhase::Template;
  52. }
  53. entity_name.form_id = form_id;
  54. return context.entity_names().Add(entity_name);
  55. }
  56. auto AddBindingPattern(Context& context, SemIR::LocId name_loc,
  57. SemIR::ExprRegionId type_region_id,
  58. SemIR::AnyBindingPattern pattern) -> BindingPatternInfo {
  59. SemIR::InstKind bind_name_kind;
  60. switch (pattern.kind) {
  61. case SemIR::InstKind::FormBindingPattern:
  62. bind_name_kind = SemIR::InstKind::FormBinding;
  63. break;
  64. case SemIR::InstKind::RefBindingPattern:
  65. bind_name_kind = SemIR::InstKind::RefBinding;
  66. break;
  67. case SemIR::InstKind::SymbolicBindingPattern:
  68. bind_name_kind = SemIR::InstKind::SymbolicBinding;
  69. break;
  70. case SemIR::InstKind::ValueBindingPattern:
  71. bind_name_kind = SemIR::InstKind::ValueBinding;
  72. break;
  73. default:
  74. CARBON_FATAL("pattern_kind {0} is not a binding pattern kind",
  75. pattern.kind);
  76. }
  77. auto type_id = SemIR::ExtractScrutineeType(context.sem_ir(), pattern.type_id);
  78. auto bind_id = AddInstInNoBlock(
  79. context,
  80. SemIR::LocIdAndInst::UncheckedLoc(
  81. name_loc, SemIR::AnyBinding{.kind = bind_name_kind,
  82. .type_id = type_id,
  83. .entity_name_id = pattern.entity_name_id,
  84. .value_id = SemIR::InstId::None}));
  85. auto binding_pattern_id = AddPatternInst(
  86. context, SemIR::LocIdAndInst::UncheckedLoc(name_loc, pattern));
  87. if (pattern.kind == SemIR::SymbolicBindingPattern::Kind) {
  88. context.scope_stack().PushCompileTimeBinding(bind_id);
  89. }
  90. bool inserted =
  91. context.bind_name_map()
  92. .Insert(binding_pattern_id, {.bind_name_id = bind_id,
  93. .type_expr_region_id = type_region_id})
  94. .is_inserted();
  95. CARBON_CHECK(inserted);
  96. return {.pattern_id = binding_pattern_id, .bind_id = bind_id};
  97. }
  98. // Returns a VarStorage inst for the given `var` pattern. If the pattern
  99. // is the body of a returned var, this reuses the return parameter, and
  100. // otherwise it adds a new inst.
  101. static auto GetOrAddVarStorage(Context& context, SemIR::InstId var_pattern_id,
  102. bool is_returned_var) -> SemIR::InstId {
  103. if (is_returned_var) {
  104. if (auto return_param_id =
  105. GetReturnedVarParam(context, GetCurrentFunctionForReturn(context));
  106. return_param_id.has_value()) {
  107. return return_param_id;
  108. }
  109. }
  110. auto pattern = context.insts().GetWithLocId(var_pattern_id);
  111. return AddInstWithCleanup(
  112. context, pattern.loc_id,
  113. SemIR::VarStorage{.type_id = ExtractScrutineeType(context.sem_ir(),
  114. pattern.inst.type_id()),
  115. .pattern_id = var_pattern_id});
  116. }
  117. auto AddPatternVarStorage(Context& context, SemIR::InstBlockId pattern_block_id,
  118. bool is_returned_var) -> void {
  119. // We need to emit the VarStorage insts early, because they may be output
  120. // arguments for the initializer. However, we can't emit them when we emit
  121. // the corresponding `VarPattern`s because they're part of the pattern match,
  122. // not part of the pattern.
  123. // TODO: Find a way to do this without walking the whole pattern block.
  124. for (auto inst_id : context.inst_blocks().Get(pattern_block_id)) {
  125. if (context.insts().Is<SemIR::VarPattern>(inst_id)) {
  126. context.var_storage_map().Insert(
  127. inst_id, GetOrAddVarStorage(context, inst_id, is_returned_var));
  128. }
  129. }
  130. }
  131. auto AddParamPattern(Context& context, SemIR::LocId loc_id,
  132. SemIR::NameId name_id,
  133. SemIR::ExprRegionId type_expr_region_id,
  134. SemIR::TypeId type_id, bool is_ref) -> SemIR::InstId {
  135. const auto& binding_pattern_kind = is_ref ? SemIR::RefBindingPattern::Kind
  136. : SemIR::ValueBindingPattern::Kind;
  137. auto entity_name_id =
  138. AddBindingEntityName(context, name_id,
  139. /*form_id=*/SemIR::ConstantId::None,
  140. /*is_unused=*/false,
  141. /*phase=*/BindingPhase::Runtime);
  142. SemIR::InstId pattern_id =
  143. AddBindingPattern(context, loc_id, type_expr_region_id,
  144. {.kind = binding_pattern_kind,
  145. .type_id = GetPatternType(context, type_id),
  146. .entity_name_id = entity_name_id})
  147. .pattern_id;
  148. const auto& param_pattern_kind =
  149. is_ref ? SemIR::RefParamPattern::Kind : SemIR::ValueParamPattern::Kind;
  150. pattern_id = AddPatternInst(
  151. context,
  152. SemIR::LocIdAndInst::UncheckedLoc(
  153. loc_id, SemIR::AnyParamPattern{
  154. .kind = param_pattern_kind,
  155. .type_id = context.insts().Get(pattern_id).type_id(),
  156. .subpattern_id = pattern_id}));
  157. return pattern_id;
  158. }
  159. } // namespace Carbon::Check