| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101 |
- // 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/context.h"
- #include "toolchain/check/handle.h"
- #include "toolchain/check/inst.h"
- #include "toolchain/check/pattern.h"
- #include "toolchain/check/type.h"
- namespace Carbon::Check {
- // Handle the start of any kind of pattern list.
- static auto HandlePatternListStart(Context& context, Parse::NodeId node_id)
- -> bool {
- context.node_stack().Push(node_id);
- context.param_and_arg_refs_stack().Push();
- BeginSubpattern(context);
- return true;
- }
- auto HandleParseNode(Context& context, Parse::ImplicitParamListStartId node_id)
- -> bool {
- return HandlePatternListStart(context, node_id);
- }
- auto HandleParseNode(Context& context, Parse::TuplePatternStartId node_id)
- -> bool {
- return HandlePatternListStart(context, node_id);
- }
- auto HandleParseNode(Context& context, Parse::ExplicitParamListStartId node_id)
- -> bool {
- context.full_pattern_stack().EndImplicitParamList();
- return HandlePatternListStart(context, node_id);
- }
- // Handle the end of any kind of parameter list (tuple patterns have separate
- // logic).
- static auto HandleParamListEnd(Context& context, Parse::NodeId node_id,
- Parse::NodeKind start_kind) -> bool {
- if (context.node_stack().PeekIs(start_kind)) {
- // End the subpattern started by a trailing comma, or the opening delimiter
- // of an empty list.
- EndSubpatternAsNonExpr(context);
- }
- // Note the Start node remains on the stack, where the param list handler can
- // make use of it.
- auto refs_id = context.param_and_arg_refs_stack().EndAndPop(start_kind);
- context.node_stack().Push(node_id, refs_id);
- return true;
- }
- auto HandleParseNode(Context& context, Parse::ImplicitParamListId node_id)
- -> bool {
- return HandleParamListEnd(context, node_id,
- Parse::NodeKind::ImplicitParamListStart);
- }
- auto HandleParseNode(Context& context, Parse::ExplicitParamListId node_id)
- -> bool {
- return HandleParamListEnd(context, node_id,
- Parse::NodeKind::ExplicitParamListStart);
- }
- auto HandleParseNode(Context& context, Parse::TuplePatternId node_id) -> bool {
- if (context.node_stack().PeekIs(Parse::NodeKind::TuplePatternStart)) {
- // End the subpattern started by a trailing comma, or the opening delimiter
- // of an empty list.
- EndSubpatternAsNonExpr(context);
- }
- auto refs_id = context.param_and_arg_refs_stack().EndAndPop(
- Parse::NodeKind::TuplePatternStart);
- context.node_stack()
- .PopAndDiscardSoloNodeId<Parse::NodeKind::TuplePatternStart>();
- const auto& inst_block = context.inst_blocks().Get(refs_id);
- llvm::SmallVector<SemIR::InstId> type_inst_ids;
- type_inst_ids.reserve(inst_block.size());
- for (auto inst : inst_block) {
- auto type_id = ExtractScrutineeType(context.sem_ir(),
- context.insts().Get(inst).type_id());
- type_inst_ids.push_back(context.types().GetInstId(type_id));
- }
- auto type_id = GetPatternType(context, GetTupleType(context, type_inst_ids));
- context.node_stack().Push(
- node_id,
- AddPatternInst<SemIR::TuplePattern>(
- context, node_id, {.type_id = type_id, .elements_id = refs_id}));
- EndSubpatternAsNonExpr(context);
- return true;
- }
- auto HandleParseNode(Context& context, Parse::PatternListCommaId /*node_id*/)
- -> bool {
- context.param_and_arg_refs_stack().ApplyComma();
- BeginSubpattern(context);
- return true;
- }
- } // namespace Carbon::Check
|