|
@@ -0,0 +1,98 @@
|
|
|
|
|
+// 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/parse/context.h"
|
|
|
|
|
+
|
|
|
|
|
+namespace Carbon::Parse {
|
|
|
|
|
+auto HandleChoiceIntroducer(Context& context) -> void {
|
|
|
|
|
+ auto state = context.PopState();
|
|
|
|
|
+
|
|
|
|
|
+ context.PushState(state, State::ChoiceDefinitionStart);
|
|
|
|
|
+ context.PushState(State::DeclNameAndParamsAsOptional, state.token);
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+auto HandleChoiceDefinitionStart(Context& context) -> void {
|
|
|
|
|
+ auto state = context.PopState();
|
|
|
|
|
+
|
|
|
|
|
+ if (!context.PositionIs(Lex::TokenKind::OpenCurlyBrace)) {
|
|
|
|
|
+ if (!state.has_error) {
|
|
|
|
|
+ CARBON_DIAGNOSTIC(ExpectedChoiceDefinition, Error,
|
|
|
|
|
+ "Choice definition expected.");
|
|
|
|
|
+ context.emitter().Emit(*context.position(), ExpectedChoiceDefinition);
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ context.AddNode(NodeKind::ChoiceDefinitionStart, *context.position(),
|
|
|
|
|
+ state.subtree_start, /*has_error=*/true);
|
|
|
|
|
+
|
|
|
|
|
+ context.AddNode(NodeKind::ChoiceDefinition, *context.position(),
|
|
|
|
|
+ state.subtree_start, /*has_error=*/true);
|
|
|
|
|
+
|
|
|
|
|
+ context.SkipPastLikelyEnd(*context.position());
|
|
|
|
|
+ return;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ context.AddNode(NodeKind::ChoiceDefinitionStart, context.Consume(),
|
|
|
|
|
+ state.subtree_start, state.has_error);
|
|
|
|
|
+
|
|
|
|
|
+ state.has_error = false;
|
|
|
|
|
+ state.state = State::ChoiceDefinitionFinish;
|
|
|
|
|
+ context.PushState(state);
|
|
|
|
|
+
|
|
|
|
|
+ if (!context.PositionIs(Lex::TokenKind::CloseCurlyBrace)) {
|
|
|
|
|
+ context.PushState(State::ChoiceAlternative);
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+auto HandleChoiceAlternative(Context& context) -> void {
|
|
|
|
|
+ auto state = context.PopState();
|
|
|
|
|
+
|
|
|
|
|
+ context.PushState(State::ChoiceAlternativeFinish);
|
|
|
|
|
+
|
|
|
|
|
+ if (!context.ConsumeAndAddLeafNodeIf(Lex::TokenKind::Identifier,
|
|
|
|
|
+ NodeKind::IdentifierName)) {
|
|
|
|
|
+ if (!state.has_error) {
|
|
|
|
|
+ CARBON_DIAGNOSTIC(ExpectedChoiceAlternativeName, Error,
|
|
|
|
|
+ "Expected choice alternative name.");
|
|
|
|
|
+ context.emitter().Emit(*context.position(),
|
|
|
|
|
+ ExpectedChoiceAlternativeName);
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ context.SkipPastLikelyEnd(*context.position());
|
|
|
|
|
+
|
|
|
|
|
+ context.ReturnErrorOnState();
|
|
|
|
|
+
|
|
|
|
|
+ return;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ if (context.PositionIs(Lex::TokenKind::OpenParen)) {
|
|
|
|
|
+ context.PushState(State::PatternListAsTuple);
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+auto HandleChoiceAlternativeFinish(Context& context) -> void {
|
|
|
|
|
+ const auto state = context.PopState();
|
|
|
|
|
+
|
|
|
|
|
+ if (state.has_error) {
|
|
|
|
|
+ context.ReturnErrorOnState();
|
|
|
|
|
+ if (!context.PositionIs(Lex::TokenKind::CloseCurlyBrace)) {
|
|
|
|
|
+ context.PushState(State::ChoiceAlternative);
|
|
|
|
|
+ }
|
|
|
|
|
+ return;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ if (context.ConsumeListToken(
|
|
|
|
|
+ NodeKind::ChoiceAlternativeListComma, Lex::TokenKind::CloseCurlyBrace,
|
|
|
|
|
+ state.has_error) == Context::ListTokenKind::Comma) {
|
|
|
|
|
+ context.PushState(State::ChoiceAlternative);
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+auto HandleChoiceDefinitionFinish(Context& context) -> void {
|
|
|
|
|
+ const auto state = context.PopState();
|
|
|
|
|
+
|
|
|
|
|
+ context.AddNode(NodeKind::ChoiceDefinition,
|
|
|
|
|
+ context.ConsumeChecked(Lex::TokenKind::CloseCurlyBrace),
|
|
|
|
|
+ state.subtree_start, state.has_error);
|
|
|
|
|
+}
|
|
|
|
|
+} // namespace Carbon::Parse
|