| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221 |
- // 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/diagnostics/format_providers.h"
- #include "toolchain/parse/context.h"
- #include "toolchain/parse/handle.h"
- namespace Carbon::Parse {
- auto HandleBraceExpr(Context& context) -> void {
- auto state = context.PopState();
- context.PushState(state, StateKind::BraceExprFinishAsUnknown);
- CARBON_CHECK(context.ConsumeAndAddLeafNodeIf(Lex::TokenKind::OpenCurlyBrace,
- NodeKind::Placeholder));
- if (!context.PositionIs(Lex::TokenKind::CloseCurlyBrace)) {
- context.PushState(StateKind::BraceExprParamAsUnknown);
- }
- }
- // Prints a diagnostic for brace expression syntax errors.
- static auto HandleBraceExprParamError(Context& context, Context::State state,
- StateKind param_finish_state_kind)
- -> void {
- Diagnostics::IntAsSelect mode(0);
- switch (param_finish_state_kind) {
- case StateKind::BraceExprParamFinishAsType:
- mode.value = 0;
- break;
- case StateKind::BraceExprParamFinishAsValue:
- mode.value = 1;
- break;
- case StateKind::BraceExprParamFinishAsUnknown:
- mode.value = 2;
- break;
- default:
- CARBON_FATAL("Unexpected state: {0}", param_finish_state_kind);
- }
- CARBON_DIAGNOSTIC(
- ExpectedStructLiteralField, Error,
- "expected {0:=0:`.field: field_type`|"
- "=1:`.field = value`|=2:`.field: field_type` or `.field = value`}",
- Diagnostics::IntAsSelect);
- context.emitter().Emit(*context.position(), ExpectedStructLiteralField, mode);
- state.has_error = true;
- context.PushState(state, param_finish_state_kind);
- }
- // Handles BraceExprParamAs(Type|Value|Unknown).
- static auto HandleBraceExprParam(Context& context,
- StateKind after_designator_kind,
- StateKind param_finish_kind) -> void {
- auto state = context.PopState();
- if (!context.PositionIs(Lex::TokenKind::Period)) {
- HandleBraceExprParamError(context, state, param_finish_kind);
- return;
- }
- context.PushState(state, after_designator_kind);
- context.PushState(StateKind::PeriodAsStruct);
- }
- auto HandleBraceExprParamAsType(Context& context) -> void {
- HandleBraceExprParam(context, StateKind::BraceExprParamAfterDesignatorAsType,
- StateKind::BraceExprParamFinishAsType);
- }
- auto HandleBraceExprParamAsValue(Context& context) -> void {
- HandleBraceExprParam(context, StateKind::BraceExprParamAfterDesignatorAsValue,
- StateKind::BraceExprParamFinishAsValue);
- }
- auto HandleBraceExprParamAsUnknown(Context& context) -> void {
- HandleBraceExprParam(context,
- StateKind::BraceExprParamAfterDesignatorAsUnknown,
- StateKind::BraceExprParamFinishAsUnknown);
- }
- // Handles BraceExprParamAfterDesignatorAs(Type|Value|Unknown).
- static auto HandleBraceExprParamAfterDesignator(Context& context,
- StateKind param_finish_kind)
- -> void {
- auto state = context.PopState();
- if (state.has_error) {
- auto recovery_pos = context.FindNextOf(
- {Lex::TokenKind::Equal, Lex::TokenKind::Colon, Lex::TokenKind::Comma});
- if (!recovery_pos ||
- context.tokens().GetKind(*recovery_pos) == Lex::TokenKind::Comma) {
- context.PushState(state, param_finish_kind);
- return;
- }
- context.SkipTo(*recovery_pos);
- }
- // Work out the kind of this element.
- bool is_type;
- if (context.PositionIs(Lex::TokenKind::Colon)) {
- is_type = true;
- } else if (context.PositionIs(Lex::TokenKind::Equal)) {
- is_type = false;
- } else {
- HandleBraceExprParamError(context, state, param_finish_kind);
- return;
- }
- // If we're changing from unknown, update the related finish states.
- if (param_finish_kind == StateKind::BraceExprParamFinishAsUnknown) {
- auto finish_state = context.PopState();
- CARBON_CHECK(finish_state.kind == StateKind::BraceExprFinishAsUnknown);
- if (is_type) {
- finish_state.kind = StateKind::BraceExprFinishAsType;
- param_finish_kind = StateKind::BraceExprParamFinishAsType;
- } else {
- finish_state.kind = StateKind::BraceExprFinishAsValue;
- param_finish_kind = StateKind::BraceExprParamFinishAsValue;
- }
- context.PushState(finish_state);
- }
- auto want_param_finish_kind = is_type
- ? StateKind::BraceExprParamFinishAsType
- : StateKind::BraceExprParamFinishAsValue;
- if (param_finish_kind != want_param_finish_kind) {
- HandleBraceExprParamError(context, state, param_finish_kind);
- return;
- }
- // Struct type fields and value fields use the same grammar except
- // that one has a `:` separator and the other has an `=` separator.
- state.token = context.Consume();
- context.PushState(state, param_finish_kind);
- if (param_finish_kind == StateKind::BraceExprParamFinishAsValue &&
- context.PositionIs(Lex::TokenKind::Ref)) {
- context.PushState(StateKind::RefTagFinishAsRegular);
- context.ConsumeChecked(Lex::TokenKind::Ref);
- }
- context.PushState(StateKind::Expr);
- }
- auto HandleBraceExprParamAfterDesignatorAsType(Context& context) -> void {
- HandleBraceExprParamAfterDesignator(context,
- StateKind::BraceExprParamFinishAsType);
- }
- auto HandleBraceExprParamAfterDesignatorAsValue(Context& context) -> void {
- HandleBraceExprParamAfterDesignator(context,
- StateKind::BraceExprParamFinishAsValue);
- }
- auto HandleBraceExprParamAfterDesignatorAsUnknown(Context& context) -> void {
- HandleBraceExprParamAfterDesignator(context,
- StateKind::BraceExprParamFinishAsUnknown);
- }
- // Handles BraceExprParamFinishAs(Type|Value|Unknown).
- static auto HandleBraceExprParamFinish(Context& context, NodeKind field_kind,
- NodeKind comma_kind,
- StateKind param_state) -> void {
- auto state = context.PopState();
- if (state.has_error) {
- context.AddLeafNode(NodeKind::InvalidParse, state.token,
- /*has_error=*/true);
- context.ReturnErrorOnState();
- } else {
- context.AddNode(field_kind, state.token, /*has_error=*/false);
- }
- if (context.ConsumeListToken(comma_kind, Lex::TokenKind::CloseCurlyBrace,
- state.has_error) ==
- Context::ListTokenKind::Comma) {
- context.PushState(param_state);
- }
- }
- auto HandleBraceExprParamFinishAsType(Context& context) -> void {
- HandleBraceExprParamFinish(context, NodeKind::StructTypeLiteralField,
- NodeKind::StructTypeLiteralComma,
- StateKind::BraceExprParamAsType);
- }
- auto HandleBraceExprParamFinishAsValue(Context& context) -> void {
- HandleBraceExprParamFinish(context, NodeKind::StructLiteralField,
- NodeKind::StructLiteralComma,
- StateKind::BraceExprParamAsValue);
- }
- auto HandleBraceExprParamFinishAsUnknown(Context& context) -> void {
- HandleBraceExprParamFinish(context, NodeKind::InvalidParse,
- NodeKind::InvalidParse,
- StateKind::BraceExprParamAsUnknown);
- }
- // Handles BraceExprFinishAs(Type|Value|Unknown).
- static auto HandleBraceExprFinish(Context& context, NodeKind start_kind,
- NodeKind end_kind) -> void {
- auto state = context.PopState();
- context.ReplacePlaceholderNode(state.subtree_start, start_kind, state.token);
- context.AddNode(end_kind, context.Consume(), state.has_error);
- }
- auto HandleBraceExprFinishAsType(Context& context) -> void {
- HandleBraceExprFinish(context, NodeKind::StructTypeLiteralStart,
- NodeKind::StructTypeLiteral);
- }
- auto HandleBraceExprFinishAsValue(Context& context) -> void {
- HandleBraceExprFinish(context, NodeKind::StructLiteralStart,
- NodeKind::StructLiteral);
- }
- auto HandleBraceExprFinishAsUnknown(Context& context) -> void {
- HandleBraceExprFinishAsValue(context);
- }
- } // namespace Carbon::Parse
|