handle_pattern_list.cpp 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  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/context.h"
  5. #include "toolchain/check/handle.h"
  6. #include "toolchain/check/inst.h"
  7. #include "toolchain/check/pattern.h"
  8. #include "toolchain/check/type.h"
  9. namespace Carbon::Check {
  10. // Handle the start of any kind of pattern list.
  11. static auto HandlePatternListStart(Context& context, Parse::NodeId node_id)
  12. -> bool {
  13. context.node_stack().Push(node_id);
  14. context.param_and_arg_refs_stack().Push();
  15. BeginSubpattern(context);
  16. return true;
  17. }
  18. auto HandleParseNode(Context& context, Parse::ImplicitParamListStartId node_id)
  19. -> bool {
  20. context.full_pattern_stack().StartImplicitParamList();
  21. return HandlePatternListStart(context, node_id);
  22. }
  23. auto HandleParseNode(Context& context, Parse::TuplePatternStartId node_id)
  24. -> bool {
  25. return HandlePatternListStart(context, node_id);
  26. }
  27. auto HandleParseNode(Context& context, Parse::ExplicitParamListStartId node_id)
  28. -> bool {
  29. context.full_pattern_stack().StartExplicitParamList();
  30. return HandlePatternListStart(context, node_id);
  31. }
  32. // Handle the end of any kind of parameter list (tuple patterns have separate
  33. // logic).
  34. static auto HandleParamListEnd(Context& context, Parse::NodeId node_id,
  35. Parse::NodeKind start_kind) -> bool {
  36. if (context.node_stack().PeekIs(start_kind)) {
  37. // End the subpattern started by a trailing comma, or the opening delimiter
  38. // of an empty list.
  39. EndEmptySubpattern(context);
  40. } else {
  41. EndSubpattern(context, context.node_stack());
  42. }
  43. // Note the Start node remains on the stack, where the param list handler can
  44. // make use of it.
  45. auto refs_id = context.param_and_arg_refs_stack().EndAndPop(start_kind);
  46. context.node_stack().Push(node_id, refs_id);
  47. return true;
  48. }
  49. auto HandleParseNode(Context& context, Parse::ImplicitParamListId node_id)
  50. -> bool {
  51. context.full_pattern_stack().EndImplicitParamList();
  52. return HandleParamListEnd(context, node_id,
  53. Parse::NodeKind::ImplicitParamListStart);
  54. }
  55. auto HandleParseNode(Context& context, Parse::ExplicitParamListId node_id)
  56. -> bool {
  57. context.full_pattern_stack().EndExplicitParamList();
  58. return HandleParamListEnd(context, node_id,
  59. Parse::NodeKind::ExplicitParamListStart);
  60. }
  61. auto HandleParseNode(Context& context, Parse::ParenPatternId node_id) -> bool {
  62. EndSubpattern(context, context.node_stack());
  63. auto pattern_id = context.node_stack().PopPattern();
  64. context.param_and_arg_refs_stack().PopAndDiscard();
  65. context.node_stack()
  66. .PopAndDiscardSoloNodeId<Parse::NodeKind::TuplePatternStart>();
  67. context.node_stack().Push(node_id, pattern_id);
  68. return true;
  69. }
  70. auto HandleParseNode(Context& context, Parse::TuplePatternId node_id) -> bool {
  71. if (context.node_stack().PeekIs(Parse::NodeKind::TuplePatternStart)) {
  72. // End the subpattern started by a trailing comma, or the opening delimiter
  73. // of an empty list.
  74. EndEmptySubpattern(context);
  75. } else {
  76. EndSubpattern(context, context.node_stack());
  77. }
  78. auto refs_id = context.param_and_arg_refs_stack().EndAndPop(
  79. Parse::NodeKind::TuplePatternStart);
  80. context.node_stack()
  81. .PopAndDiscardSoloNodeId<Parse::NodeKind::TuplePatternStart>();
  82. const auto& inst_block = context.inst_blocks().Get(refs_id);
  83. llvm::SmallVector<SemIR::InstId> type_inst_ids;
  84. type_inst_ids.reserve(inst_block.size());
  85. for (auto inst : inst_block) {
  86. auto type_id = ExtractScrutineeType(context.sem_ir(),
  87. context.insts().Get(inst).type_id());
  88. type_inst_ids.push_back(context.types().GetTypeInstId(type_id));
  89. }
  90. auto type_id = GetPatternType(context, GetTupleType(context, type_inst_ids));
  91. context.node_stack().Push(
  92. node_id,
  93. AddPatternInst<SemIR::TuplePattern>(
  94. context, node_id, {.type_id = type_id, .elements_id = refs_id}));
  95. return true;
  96. }
  97. auto HandleParseNode(Context& context, Parse::PatternListCommaId /*node_id*/)
  98. -> bool {
  99. EndSubpattern(context, context.node_stack());
  100. context.param_and_arg_refs_stack().ApplyComma();
  101. BeginSubpattern(context);
  102. return true;
  103. }
  104. } // namespace Carbon::Check