forked from carbon-language/carbon-lang
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhandle_var.cpp
120 lines (95 loc) · 3.9 KB
/
handle_var.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
113
114
115
116
117
118
119
120
// 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"
#include "toolchain/parse/handle.h"
namespace Carbon::Parse {
// Handles VarAs(Decl|Returned).
static auto HandleVar(Context& context, State finish_state,
Lex::TokenIndex returned_token = Lex::TokenIndex::None)
-> void {
auto state = context.PopState();
// The finished variable declaration will start at the `var` or `returned`.
context.PushState(state, finish_state);
// TODO: is there a cleaner way to give VarAfterPattern access to the `var`
// token?
state.token = *(context.position() - 1);
context.PushState(state, State::VarAfterPattern);
if (returned_token.has_value()) {
context.AddLeafNode(NodeKind::ReturnedModifier, returned_token);
}
context.PushStateForPattern(State::Pattern, /*in_var_pattern=*/true);
}
auto HandleVarAsDecl(Context& context) -> void {
HandleVar(context, State::VarFinishAsDecl);
}
auto HandleVarAsReturned(Context& context) -> void {
auto returned_token = context.Consume();
if (!context.PositionIs(Lex::TokenKind::Var)) {
CARBON_DIAGNOSTIC(ExpectedVarAfterReturned, Error,
"expected `var` after `returned`");
context.emitter().Emit(*context.position(), ExpectedVarAfterReturned);
context.AddLeafNode(NodeKind::EmptyDecl,
context.SkipPastLikelyEnd(returned_token),
/*has_error=*/true);
context.PopAndDiscardState();
return;
}
context.AddLeafNode(NodeKind::VariableIntroducer, context.Consume());
HandleVar(context, State::VarFinishAsDecl, returned_token);
}
auto HandleVarAsFor(Context& context) -> void {
auto state = context.PopState();
// The finished variable declaration will start at the `var`.
context.PushState(state, State::VarFinishAsFor);
context.PushState(State::Pattern);
}
auto HandleVarAfterPattern(Context& context) -> void {
auto state = context.PopState();
if (state.has_error) {
if (auto after_pattern =
context.FindNextOf({Lex::TokenKind::Equal, Lex::TokenKind::Semi})) {
context.SkipTo(*after_pattern);
}
}
context.AddNode(NodeKind::VariablePattern, state.token, state.has_error);
if (context.PositionIs(Lex::TokenKind::Equal)) {
context.AddLeafNode(NodeKind::VariableInitializer,
context.ConsumeChecked(Lex::TokenKind::Equal));
context.PushState(State::Expr);
}
}
auto HandleVarFinishAsDecl(Context& context) -> void {
auto state = context.PopState();
auto end_token = state.token;
if (context.PositionIs(Lex::TokenKind::Semi)) {
end_token = context.Consume();
} else {
// TODO: Disambiguate between statement and member declaration.
context.DiagnoseExpectedDeclSemi(Lex::TokenKind::Var);
state.has_error = true;
end_token = context.SkipPastLikelyEnd(state.token);
}
context.AddNode(NodeKind::VariableDecl, end_token, state.has_error);
}
auto HandleVarFinishAsFor(Context& context) -> void {
auto state = context.PopState();
context.AddNode(NodeKind::VariablePattern, state.token, state.has_error);
auto end_token = state.token;
if (context.PositionIs(Lex::TokenKind::In)) {
end_token = context.Consume();
} else if (context.PositionIs(Lex::TokenKind::Colon)) {
CARBON_DIAGNOSTIC(ExpectedInNotColon, Error,
"`:` should be replaced by `in`");
context.emitter().Emit(*context.position(), ExpectedInNotColon);
state.has_error = true;
end_token = context.Consume();
} else {
CARBON_DIAGNOSTIC(ExpectedIn, Error,
"expected `in` after loop `var` declaration");
context.emitter().Emit(*context.position(), ExpectedIn);
state.has_error = true;
}
context.AddNode(NodeKind::ForIn, end_token, state.has_error);
}
} // namespace Carbon::Parse