-
Notifications
You must be signed in to change notification settings - Fork 1.5k
/
Copy pathhandle_decl_name_and_params.cpp
112 lines (95 loc) · 4.06 KB
/
handle_decl_name_and_params.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
// 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/lex/token_index.h"
#include "toolchain/parse/context.h"
#include "toolchain/parse/handle.h"
namespace Carbon::Parse {
// Adds a leaf node for the name, and updates the state stack for parameter
// handling.
static auto HandleName(Context& context, Context::State state,
Lex::TokenIndex name_token,
NodeKind not_before_params_kind,
NodeKind not_before_params_qualifier_kind,
NodeKind before_params_kind) -> void {
switch (context.PositionKind()) {
case Lex::TokenKind::Period:
context.AddLeafNode(not_before_params_kind, name_token);
context.AddNode(not_before_params_qualifier_kind,
context.ConsumeChecked(Lex::TokenKind::Period),
state.has_error);
context.PushState(StateKind::DeclNameAndParams);
break;
case Lex::TokenKind::OpenSquareBracket:
context.AddLeafNode(before_params_kind, name_token);
state.kind = StateKind::DeclNameAndParamsAfterImplicit;
context.PushState(state);
context.PushState(StateKind::PatternListAsImplicit);
break;
case Lex::TokenKind::OpenParen:
context.AddLeafNode(before_params_kind, name_token);
state.kind = StateKind::DeclNameAndParamsAfterParams;
context.PushState(state);
context.PushState(StateKind::PatternListAsExplicit);
break;
default:
context.AddLeafNode(not_before_params_kind, name_token);
break;
}
}
auto HandleDeclNameAndParams(Context& context) -> void {
auto state = context.PopState();
if (auto identifier = context.ConsumeIf(Lex::TokenKind::Identifier)) {
HandleName(context, state, *identifier,
NodeKind::IdentifierNameNotBeforeParams,
NodeKind::IdentifierNameQualifierWithoutParams,
NodeKind::IdentifierNameBeforeParams);
return;
}
if (auto keyword = context.ConsumeIf(Lex::TokenKind::Destroy)) {
HandleName(context, state, *keyword, NodeKind::KeywordNameNotBeforeParams,
NodeKind::KeywordNameQualifierWithoutParams,
NodeKind::KeywordNameBeforeParams);
return;
}
Lex::TokenIndex token = *context.position();
if (context.tokens().GetKind(token) == Lex::TokenKind::FileEnd) {
// The end of file is an unhelpful diagnostic location. Instead, use the
// introducer token.
token = state.token;
}
if (state.token == *context.position()) {
CARBON_DIAGNOSTIC(ExpectedDeclNameAfterPeriod, Error,
"`.` should be followed by a name");
context.emitter().Emit(token, ExpectedDeclNameAfterPeriod);
} else {
CARBON_DIAGNOSTIC(ExpectedDeclName, Error,
"`{0}` introducer should be followed by a name",
Lex::TokenKind);
context.emitter().Emit(token, ExpectedDeclName,
context.tokens().GetKind(state.token));
}
context.ReturnErrorOnState();
context.AddInvalidParse(*context.position());
}
auto HandleDeclNameAndParamsAfterImplicit(Context& context) -> void {
auto state = context.PopState();
state.kind = StateKind::DeclNameAndParamsAfterParams;
context.PushState(state);
if (!context.PositionIs(Lex::TokenKind::OpenParen)) {
return;
}
context.PushState(StateKind::PatternListAsExplicit);
}
auto HandleDeclNameAndParamsAfterParams(Context& context) -> void {
auto state = context.PopState();
if (auto period = context.ConsumeIf(Lex::TokenKind::Period)) {
auto start_kind = context.tree().node_kind(NodeId(state.subtree_start));
auto node_kind = start_kind == NodeKind::IdentifierNameBeforeParams
? NodeKind::IdentifierNameQualifierWithParams
: NodeKind::KeywordNameQualifierWithParams;
context.AddNode(node_kind, *period, state.has_error);
context.PushState(StateKind::DeclNameAndParams);
}
}
} // namespace Carbon::Parse