| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124 |
- // 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 {
- // Handles an unrecognized declaration, adding an error node.
- static auto HandleUnrecognizedDecl(Context& context) -> void {
- CARBON_DIAGNOSTIC(UnrecognizedDecl, Error,
- "Unrecognized declaration introducer.");
- context.emitter().Emit(*context.position(), UnrecognizedDecl);
- auto cursor = *context.position();
- auto semi = context.SkipPastLikelyEnd(cursor);
- // Locate the EmptyDecl at the semi when found, but use the
- // original cursor location for an error when not.
- context.AddLeafNode(NodeKind::EmptyDecl, semi ? *semi : cursor,
- /*has_error=*/true);
- }
- auto HandleDeclScopeLoop(Context& context) -> void {
- // This maintains the current state unless we're at the end of the scope.
- auto position_kind = context.PositionKind();
- switch (position_kind) {
- case Lex::TokenKind::CloseCurlyBrace:
- case Lex::TokenKind::EndOfFile: {
- // This is the end of the scope, so the loop state ends.
- context.PopAndDiscardState();
- return;
- }
- // `import`, `library`, and `package` manage their packaging state.
- case Lex::TokenKind::Import: {
- context.PushState(State::Import);
- return;
- }
- case Lex::TokenKind::Library: {
- context.PushState(State::Library);
- return;
- }
- case Lex::TokenKind::Package: {
- context.PushState(State::Package);
- return;
- }
- default: {
- break;
- }
- }
- // Because a non-packaging keyword was encountered, packaging is complete.
- // Misplaced packaging keywords may lead to this being re-triggered.
- if (context.packaging_state() !=
- Context::PackagingState::AfterNonPackagingDecl) {
- if (!context.first_non_packaging_token().is_valid()) {
- context.set_first_non_packaging_token(*context.position());
- }
- context.set_packaging_state(Context::PackagingState::AfterNonPackagingDecl);
- }
- // Remaining keywords are only valid after imports are complete, and so all
- // result in a `set_packaging_state` call. Note, this may not always be
- // necessary but is probably cheaper than validating.
- switch (position_kind) {
- case Lex::TokenKind::Abstract:
- case Lex::TokenKind::Base: {
- if (context.PositionIs(Lex::TokenKind::Class, Lookahead::NextToken)) {
- context.PushState(State::TypeAfterIntroducerAsClass);
- auto modifier_token = context.Consume();
- auto class_token = context.Consume();
- context.AddLeafNode(NodeKind::ClassIntroducer, class_token);
- context.AddLeafNode(position_kind == Lex::TokenKind::Abstract
- ? NodeKind::AbstractModifier
- : NodeKind::BaseModifier,
- modifier_token);
- return;
- }
- break;
- }
- case Lex::TokenKind::Class: {
- context.PushState(State::TypeAfterIntroducerAsClass);
- context.AddLeafNode(NodeKind::ClassIntroducer, context.Consume());
- return;
- }
- case Lex::TokenKind::Constraint: {
- context.PushState(State::TypeAfterIntroducerAsNamedConstraint);
- context.AddLeafNode(NodeKind::NamedConstraintIntroducer,
- context.Consume());
- return;
- }
- case Lex::TokenKind::Fn: {
- context.PushState(State::FunctionIntroducer);
- return;
- }
- case Lex::TokenKind::Interface: {
- context.PushState(State::TypeAfterIntroducerAsInterface);
- context.AddLeafNode(NodeKind::InterfaceIntroducer, context.Consume());
- return;
- }
- case Lex::TokenKind::Namespace: {
- context.PushState(State::Namespace);
- return;
- }
- case Lex::TokenKind::Semi: {
- context.AddLeafNode(NodeKind::EmptyDecl, context.Consume());
- return;
- }
- case Lex::TokenKind::Var: {
- context.PushState(State::VarAsDecl);
- return;
- }
- case Lex::TokenKind::Let: {
- context.PushState(State::Let);
- return;
- }
- default: {
- break;
- }
- }
- HandleUnrecognizedDecl(context);
- }
- } // namespace Carbon::Parse
|