pattern.cpp 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  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 AddBindingPattern(Context& context, SemIR::LocId name_loc,
  42. SemIR::NameId name_id, SemIR::TypeId type_id,
  43. SemIR::ConstantId form_id,
  44. SemIR::ExprRegionId type_region_id,
  45. SemIR::InstKind pattern_kind, bool is_template,
  46. bool is_unused) -> BindingPatternInfo {
  47. SemIR::InstKind bind_name_kind;
  48. switch (pattern_kind) {
  49. case SemIR::InstKind::FormBindingPattern:
  50. bind_name_kind = SemIR::InstKind::FormBinding;
  51. break;
  52. case SemIR::InstKind::RefBindingPattern:
  53. bind_name_kind = SemIR::InstKind::RefBinding;
  54. break;
  55. case SemIR::InstKind::SymbolicBindingPattern:
  56. bind_name_kind = SemIR::InstKind::SymbolicBinding;
  57. break;
  58. case SemIR::InstKind::ValueBindingPattern:
  59. bind_name_kind = SemIR::InstKind::ValueBinding;
  60. break;
  61. default:
  62. CARBON_FATAL("pattern_kind {0} is not a binding pattern kind",
  63. pattern_kind);
  64. }
  65. bool is_generic = pattern_kind == SemIR::SymbolicBindingPattern::Kind;
  66. SemIR::EntityName entity_name = {
  67. .name_id = name_id,
  68. .parent_scope_id = context.scope_stack().PeekNameScopeId(),
  69. .is_unused = is_unused || name_id == SemIR::NameId::Underscore};
  70. if (is_generic) {
  71. entity_name.bind_index_value =
  72. context.scope_stack().AddCompileTimeBinding().index;
  73. entity_name.is_template = is_template;
  74. } else if (pattern_kind == SemIR::InstKind::FormBindingPattern) {
  75. entity_name.form_id = form_id;
  76. }
  77. auto entity_name_id = context.entity_names().Add(entity_name);
  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 = entity_name_id,
  84. .value_id = SemIR::InstId::None}));
  85. auto pattern_type_id = GetPatternType(context, type_id);
  86. auto binding_pattern_id = AddPatternInst(
  87. context, SemIR::LocIdAndInst::UncheckedLoc(
  88. name_loc,
  89. SemIR::AnyBindingPattern{.kind = pattern_kind,
  90. .type_id = pattern_type_id,
  91. .entity_name_id = entity_name_id}));
  92. if (is_generic) {
  93. context.scope_stack().PushCompileTimeBinding(bind_id);
  94. }
  95. bool inserted =
  96. context.bind_name_map()
  97. .Insert(binding_pattern_id, {.bind_name_id = bind_id,
  98. .type_expr_region_id = type_region_id})
  99. .is_inserted();
  100. CARBON_CHECK(inserted);
  101. return {.pattern_id = binding_pattern_id, .bind_id = bind_id};
  102. }
  103. // Returns a VarStorage inst for the given `var` pattern. If the pattern
  104. // is the body of a returned var, this reuses the return parameter, and
  105. // otherwise it adds a new inst.
  106. static auto GetOrAddVarStorage(Context& context, SemIR::InstId var_pattern_id,
  107. bool is_returned_var) -> SemIR::InstId {
  108. if (is_returned_var) {
  109. if (auto return_param_id =
  110. GetReturnedVarParam(context, GetCurrentFunctionForReturn(context));
  111. return_param_id.has_value()) {
  112. return return_param_id;
  113. }
  114. }
  115. auto pattern = context.insts().GetWithLocId(var_pattern_id);
  116. return AddInstWithCleanup(
  117. context, pattern.loc_id,
  118. SemIR::VarStorage{.type_id = ExtractScrutineeType(context.sem_ir(),
  119. pattern.inst.type_id()),
  120. .pattern_id = var_pattern_id});
  121. }
  122. auto AddPatternVarStorage(Context& context, SemIR::InstBlockId pattern_block_id,
  123. bool is_returned_var) -> void {
  124. // We need to emit the VarStorage insts early, because they may be output
  125. // arguments for the initializer. However, we can't emit them when we emit
  126. // the corresponding `VarPattern`s because they're part of the pattern match,
  127. // not part of the pattern.
  128. // TODO: Find a way to do this without walking the whole pattern block.
  129. for (auto inst_id : context.inst_blocks().Get(pattern_block_id)) {
  130. if (context.insts().Is<SemIR::VarPattern>(inst_id)) {
  131. context.var_storage_map().Insert(
  132. inst_id, GetOrAddVarStorage(context, inst_id, is_returned_var));
  133. }
  134. }
  135. }
  136. auto AddParamPattern(Context& context, SemIR::LocId loc_id,
  137. SemIR::NameId name_id,
  138. SemIR::ExprRegionId type_expr_region_id,
  139. SemIR::TypeId type_id, bool is_ref) -> SemIR::InstId {
  140. const auto& binding_pattern_kind = is_ref ? SemIR::RefBindingPattern::Kind
  141. : SemIR::ValueBindingPattern::Kind;
  142. SemIR::InstId pattern_id =
  143. AddBindingPattern(context, loc_id, name_id, type_id,
  144. /*form_id=*/SemIR::ConstantId::None,
  145. type_expr_region_id, binding_pattern_kind,
  146. /*is_template=*/false, /*is_unused=*/false)
  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,
  154. SemIR::AnyParamPattern{
  155. .kind = param_pattern_kind,
  156. .type_id = context.insts().Get(pattern_id).type_id(),
  157. .subpattern_id = pattern_id,
  158. .index = context.full_pattern_stack().NextCallParamIndex()}));
  159. return pattern_id;
  160. }
  161. } // namespace Carbon::Check