Coverage Report

Created: 2026-03-31 06:04

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/njs/src/njs_parser.c
Line
Count
Source
1
2
/*
3
 * Copyright (C) Igor Sysoev
4
 * Copyright (C) Alexander Borisov
5
 * Copyright (C) NGINX, Inc.
6
 */
7
8
9
#include <njs_main.h>
10
11
12
static njs_int_t njs_parser_scope_begin(njs_parser_t *parser, njs_scope_t type,
13
    njs_bool_t init_this);
14
static void njs_parser_scope_end(njs_parser_t *parser);
15
16
static njs_int_t njs_parser_check_error_state(njs_parser_t *parser,
17
    njs_lexer_token_t *token, njs_queue_link_t *current);
18
19
static njs_int_t njs_parser_primary_expression_test(njs_parser_t *parser,
20
    njs_lexer_token_t *token, njs_queue_link_t *current);
21
static njs_int_t njs_parser_regexp_literal(njs_parser_t *parser,
22
    njs_lexer_token_t *token, njs_queue_link_t *current);
23
static njs_int_t njs_parser_template_literal(njs_parser_t *parser,
24
    njs_lexer_token_t *token, njs_queue_link_t *current);
25
static njs_int_t njs_parser_template_literal_string(njs_parser_t *parser,
26
    njs_lexer_token_t *token, njs_queue_link_t *current);
27
static njs_int_t njs_parser_template_literal_expression(
28
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
29
static njs_int_t njs_parser_cover_parenthesized_expression(
30
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
31
static njs_int_t njs_parser_binding_identifier_pattern(njs_parser_t *parser,
32
    njs_lexer_token_t *token, njs_queue_link_t *current);
33
static njs_int_t njs_parser_cover_parenthesized_expression_after(
34
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
35
static njs_int_t njs_parser_cover_parenthesized_expression_end(
36
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
37
38
static njs_int_t njs_parser_array_literal(njs_parser_t *parser,
39
    njs_lexer_token_t *token, njs_queue_link_t *current);
40
static njs_int_t njs_parser_array_element_list(njs_parser_t *parser,
41
    njs_lexer_token_t *token, njs_queue_link_t *current);
42
static njs_int_t njs_parser_array_after(njs_parser_t *parser,
43
    njs_lexer_token_t *token, njs_queue_link_t *current);
44
static njs_int_t njs_parser_array_spread_element(njs_parser_t *parser,
45
    njs_lexer_token_t *token, njs_queue_link_t *current);
46
47
static njs_int_t njs_parser_object_literal(njs_parser_t *parser,
48
    njs_lexer_token_t *token, njs_queue_link_t *current);
49
static njs_int_t njs_parser_object_literal_after(njs_parser_t *parser,
50
    njs_lexer_token_t *token, njs_queue_link_t *current);
51
static njs_parser_node_t *njs_parser_object_value(njs_parser_t *parser,
52
    njs_parser_node_t *parent, uint32_t token_line);
53
static njs_int_t njs_parser_property_definition_list(njs_parser_t *parser,
54
    njs_lexer_token_t *token, njs_queue_link_t *current);
55
static njs_int_t njs_parser_property_definition_list_after(
56
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
57
static njs_int_t njs_parser_property_definition(njs_parser_t *parser,
58
    njs_lexer_token_t *token, njs_queue_link_t *current);
59
static njs_int_t njs_parser_property_definition_after(njs_parser_t *parser,
60
    njs_lexer_token_t *token, njs_queue_link_t *current);
61
static njs_int_t njs_parser_computed_property_name_after(njs_parser_t *parser,
62
    njs_lexer_token_t *token, njs_queue_link_t *current);
63
static njs_int_t njs_parser_computed_property_async_after(njs_parser_t *parser,
64
    njs_lexer_token_t *token, njs_queue_link_t *current);
65
static njs_int_t njs_parser_computed_property_name_handler(njs_parser_t *parser,
66
    njs_lexer_token_t *token, njs_queue_link_t *current, njs_bool_t async);
67
68
static njs_int_t njs_parser_initializer(njs_parser_t *parser,
69
    njs_lexer_token_t *token, njs_queue_link_t *current);
70
static njs_int_t njs_parser_initializer_after(njs_parser_t *parser,
71
    njs_lexer_token_t *token, njs_queue_link_t *current);
72
static njs_int_t njs_parser_initializer_assign(njs_parser_t *parser,
73
    njs_token_type_t type);
74
75
static njs_int_t njs_parser_member_expression(njs_parser_t *parser,
76
    njs_lexer_token_t *token, njs_queue_link_t *current);
77
static njs_int_t njs_parser_member_expression_next(njs_parser_t *parser,
78
    njs_lexer_token_t *token, njs_queue_link_t *current);
79
static njs_int_t njs_parser_member_expression_bracket(njs_parser_t *parser,
80
    njs_lexer_token_t *token, njs_queue_link_t *current);
81
static njs_int_t njs_parser_member_expression_new_next(njs_parser_t *parser,
82
    njs_lexer_token_t *token, njs_queue_link_t *current);
83
static njs_int_t njs_parser_super_property(njs_parser_t *parser,
84
    njs_lexer_token_t *token, njs_queue_link_t *current);
85
static njs_int_t njs_parser_member_expression_import(njs_parser_t *parser,
86
    njs_lexer_token_t *token, njs_queue_link_t *current);
87
static njs_int_t njs_parser_member_expression_new(njs_parser_t *parser,
88
    njs_lexer_token_t *token, njs_queue_link_t *current);
89
static njs_int_t njs_parser_member_expression_new_after(njs_parser_t *parser,
90
    njs_lexer_token_t *token, njs_queue_link_t *current);
91
static njs_int_t njs_parser_member_expression_new_args(njs_parser_t *parser,
92
    njs_lexer_token_t *token, njs_queue_link_t *current);
93
94
static njs_parser_node_t *njs_parser_create_call(njs_parser_t *parser,
95
    njs_parser_node_t *node, uint8_t ctor);
96
static njs_int_t njs_parser_call_arguments(njs_parser_t *parser,
97
    njs_lexer_token_t *token, njs_queue_link_t *current,
98
    njs_parser_node_t *func, njs_parser_state_func_t after);
99
100
static njs_int_t njs_parser_right_link_pop(njs_parser_t *parser);
101
static njs_parser_node_t *njs_parser_property_ref(njs_parser_node_t *node);
102
static njs_parser_node_t *njs_parser_lvalue_ref(njs_parser_node_t *node);
103
static njs_parser_node_t *njs_parser_call_receiver(njs_parser_node_t *node);
104
static njs_parser_node_t *njs_parser_optional_chain_preserve(
105
    njs_parser_t *parser, njs_parser_node_t *node, uint32_t token_line);
106
static njs_parser_node_t *njs_parser_optional_chain_source(
107
    njs_parser_node_t *node);
108
static njs_parser_node_t *njs_parser_optional_chain_receiver(
109
    njs_parser_node_t *node);
110
static njs_parser_node_t *njs_parser_optional_chain_call_this(
111
    njs_parser_node_t *node);
112
static void njs_parser_set_call_this(njs_parser_node_t *node,
113
    njs_parser_node_t *this_object);
114
static void njs_parser_set_optional_method_call_preserve(
115
    njs_parser_node_t *node, njs_parser_node_t *preserve);
116
static njs_parser_node_t *njs_parser_optional_chain_target_property(
117
    njs_parser_node_t *node);
118
static njs_parser_node_t *njs_parser_optional_chain_method_call(
119
    njs_parser_t *parser, njs_parser_node_t *node, uint32_t token_line);
120
static njs_parser_node_t *njs_parser_optional_chain_call(
121
    njs_parser_t *parser, njs_parser_node_t *node,
122
    njs_parser_node_t *fallback, uint32_t token_line);
123
static njs_int_t njs_parser_call_expression(njs_parser_t *parser,
124
    njs_lexer_token_t *token, njs_queue_link_t *current);
125
static njs_int_t njs_parser_call_expression_args(njs_parser_t *parser,
126
    njs_lexer_token_t *token, njs_queue_link_t *current);
127
static njs_int_t njs_parser_call_or_property_after(njs_parser_t *parser,
128
    njs_lexer_token_t *token, njs_queue_link_t *current,
129
    njs_parser_state_func_t after);
130
static njs_int_t njs_parser_call_expression_after(njs_parser_t *parser,
131
    njs_lexer_token_t *token, njs_queue_link_t *current);
132
static njs_int_t njs_parser_arguments(njs_parser_t *parser,
133
    njs_lexer_token_t *token, njs_queue_link_t *current);
134
static njs_int_t njs_parser_parenthesis_or_comma(njs_parser_t *parser,
135
    njs_lexer_token_t *token, njs_queue_link_t *current);
136
static njs_int_t njs_parser_argument_list(njs_parser_t *parser,
137
    njs_lexer_token_t *token, njs_queue_link_t *current);
138
static njs_int_t njs_parser_argument_list_after(njs_parser_t *parser,
139
    njs_lexer_token_t *token, njs_queue_link_t *current);
140
static njs_int_t njs_parser_optional_expression_after(njs_parser_t *parser,
141
    njs_lexer_token_t *token, njs_queue_link_t *current);
142
static njs_int_t njs_parser_optional_chain(njs_parser_t *parser,
143
    njs_lexer_token_t *token, njs_queue_link_t *current);
144
static njs_int_t njs_parser_optional_chain_after(njs_parser_t *parser,
145
    njs_lexer_token_t *token, njs_queue_link_t *current);
146
static njs_int_t njs_parser_new_expression(njs_parser_t *parser,
147
    njs_lexer_token_t *token, njs_queue_link_t *current);
148
static njs_int_t njs_parser_new_expression_after(njs_parser_t *parser,
149
    njs_lexer_token_t *token, njs_queue_link_t *current);
150
static njs_int_t njs_parser_left_hand_side_expression(njs_parser_t *parser,
151
    njs_lexer_token_t *token, njs_queue_link_t *current);
152
static njs_int_t njs_parser_left_hand_side_expression_after(
153
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
154
static njs_int_t njs_parser_left_hand_side_expression_node(
155
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
156
static njs_int_t njs_parser_left_hand_side_expression_optional(
157
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
158
static njs_int_t njs_parser_update_expression(njs_parser_t *parser,
159
    njs_lexer_token_t *token, njs_queue_link_t *current);
160
static njs_int_t njs_parser_update_expression_post(njs_parser_t *parser,
161
    njs_lexer_token_t *token, njs_queue_link_t *current);
162
static njs_int_t njs_parser_update_expression_unary(njs_parser_t *parser,
163
    njs_lexer_token_t *token, njs_queue_link_t *current);
164
165
static njs_int_t njs_parser_unary_expression(njs_parser_t *parser,
166
    njs_lexer_token_t *token, njs_queue_link_t *current);
167
static njs_int_t njs_parser_unary_expression_after(njs_parser_t *parser,
168
    njs_lexer_token_t *token, njs_queue_link_t *current);
169
static njs_int_t njs_parser_unary_expression_next(njs_parser_t *parser,
170
    njs_lexer_token_t *token, njs_queue_link_t *current);
171
static njs_int_t njs_parser_await(njs_parser_t *parser,
172
    njs_lexer_token_t *token, njs_queue_link_t *current);
173
static njs_int_t njs_parser_await_after(njs_parser_t *parser,
174
    njs_lexer_token_t *token, njs_queue_link_t *current);
175
176
static njs_int_t njs_parser_exponentiation_expression(njs_parser_t *parser,
177
    njs_lexer_token_t *token, njs_queue_link_t *current);
178
static njs_int_t njs_parser_exponentiation_expression_match(
179
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
180
181
static njs_int_t njs_parser_multiplicative_expression(njs_parser_t *parser,
182
    njs_lexer_token_t *token, njs_queue_link_t *current);
183
static njs_int_t njs_parser_multiplicative_expression_match(
184
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
185
186
static njs_int_t njs_parser_additive_expression(njs_parser_t *parser,
187
    njs_lexer_token_t *token, njs_queue_link_t *current);
188
static njs_int_t njs_parser_additive_expression_match(njs_parser_t *parser,
189
    njs_lexer_token_t *token, njs_queue_link_t *current);
190
191
static njs_int_t njs_parser_shift_expression(njs_parser_t *parser,
192
    njs_lexer_token_t *token, njs_queue_link_t *current);
193
static njs_int_t njs_parser_shift_expression_match(njs_parser_t *parser,
194
    njs_lexer_token_t *token, njs_queue_link_t *current);
195
196
static njs_int_t njs_parser_relational_expression(njs_parser_t *parser,
197
    njs_lexer_token_t *token, njs_queue_link_t *current);
198
static njs_int_t njs_parser_relational_expression_match(njs_parser_t *parser,
199
    njs_lexer_token_t *token, njs_queue_link_t *current);
200
201
static njs_int_t njs_parser_equality_expression(njs_parser_t *parser,
202
    njs_lexer_token_t *token, njs_queue_link_t *current);
203
static njs_int_t njs_parser_equality_expression_match(njs_parser_t *parser,
204
    njs_lexer_token_t *token, njs_queue_link_t *current);
205
206
static njs_int_t njs_parser_bitwise_AND_expression(njs_parser_t *parser,
207
    njs_lexer_token_t *token, njs_queue_link_t *current);
208
static njs_int_t njs_parser_bitwise_AND_expression_and(njs_parser_t *parser,
209
    njs_lexer_token_t *token, njs_queue_link_t *current);
210
static njs_int_t njs_parser_bitwise_XOR_expression(njs_parser_t *parser,
211
    njs_lexer_token_t *token, njs_queue_link_t *current);
212
static njs_int_t njs_parser_bitwise_XOR_expression_xor(njs_parser_t *parser,
213
    njs_lexer_token_t *token, njs_queue_link_t *current);
214
static njs_int_t njs_parser_bitwise_OR_expression(njs_parser_t *parser,
215
    njs_lexer_token_t *token, njs_queue_link_t *current);
216
static njs_int_t njs_parser_bitwise_OR_expression_or(njs_parser_t *parser,
217
    njs_lexer_token_t *token, njs_queue_link_t *current);
218
219
static njs_int_t njs_parser_logical_AND_expression(njs_parser_t *parser,
220
    njs_lexer_token_t *token, njs_queue_link_t *current);
221
static njs_int_t njs_parser_logical_AND_expression_and(njs_parser_t *parser,
222
    njs_lexer_token_t *token, njs_queue_link_t *current);
223
static njs_int_t njs_parser_logical_OR_expression(njs_parser_t *parser,
224
    njs_lexer_token_t *token, njs_queue_link_t *current);
225
static njs_int_t njs_parser_logical_OR_expression_or(njs_parser_t *parser,
226
    njs_lexer_token_t *token, njs_queue_link_t *current);
227
static njs_int_t njs_parser_coalesce_expression(njs_parser_t *parser,
228
    njs_lexer_token_t *token, njs_queue_link_t *current);
229
static njs_int_t njs_parser_short_circuit_expression(njs_parser_t *parser,
230
    njs_lexer_token_t *token, njs_queue_link_t *current);
231
232
static njs_int_t njs_parser_conditional_expression(njs_parser_t *parser,
233
    njs_lexer_token_t *token, njs_queue_link_t *current);
234
static njs_int_t njs_parser_conditional_question_mark(njs_parser_t *parser,
235
    njs_lexer_token_t *token, njs_queue_link_t *current);
236
static njs_int_t njs_parser_conditional_colon(njs_parser_t *parser,
237
    njs_lexer_token_t *token, njs_queue_link_t *current);
238
static njs_int_t njs_parser_conditional_colon_after(njs_parser_t *parser,
239
    njs_lexer_token_t *token, njs_queue_link_t *current);
240
241
static njs_int_t njs_parser_assignment_expression(njs_parser_t *parser,
242
    njs_lexer_token_t *token, njs_queue_link_t *current);
243
static njs_int_t njs_parser_match_arrow_expression(njs_parser_t *parser,
244
    njs_lexer_token_t *token);
245
static njs_int_t njs_parser_assignment_expression_after(njs_parser_t *parser,
246
    njs_lexer_token_t *token, njs_queue_link_t *current);
247
static njs_int_t njs_parser_assignment_operator(njs_parser_t *parser,
248
    njs_lexer_token_t *token, njs_queue_link_t *current);
249
static njs_int_t njs_parser_assignment_operator_after(njs_parser_t *parser,
250
    njs_lexer_token_t *token, njs_queue_link_t *current);
251
252
static njs_int_t njs_parser_expression(njs_parser_t *parser,
253
    njs_lexer_token_t *token, njs_queue_link_t *current);
254
static njs_int_t njs_parser_expression_comma(njs_parser_t *parser,
255
    njs_lexer_token_t *token, njs_queue_link_t *current);
256
257
static njs_int_t njs_parser_statement(njs_parser_t *parser,
258
    njs_lexer_token_t *token, njs_queue_link_t *current);
259
static njs_int_t njs_parser_statement_wo_node(njs_parser_t *parser,
260
    njs_lexer_token_t *token, njs_queue_link_t *current);
261
static njs_int_t njs_parser_statement_after(njs_parser_t *parser,
262
    njs_lexer_token_t *token, njs_queue_link_t *current);
263
static njs_int_t njs_parser_declaration(njs_parser_t *parser,
264
    njs_lexer_token_t *token, njs_queue_link_t *current);
265
static njs_int_t njs_parser_hoistable_declaration(njs_parser_t *parser,
266
    njs_lexer_token_t *token, njs_queue_link_t *current);
267
268
static njs_int_t njs_parser_block_statement(njs_parser_t *parser,
269
    njs_lexer_token_t *token, njs_queue_link_t *current);
270
static njs_int_t njs_parser_block_statement_open_brace(njs_parser_t *parser,
271
    njs_lexer_token_t *token, njs_queue_link_t *current);
272
static njs_int_t njs_parser_block_statement_close_brace(njs_parser_t *parser,
273
    njs_lexer_token_t *token, njs_queue_link_t *current);
274
static njs_int_t njs_parser_statement_list(njs_parser_t *parser,
275
    njs_lexer_token_t *token, njs_queue_link_t *current);
276
static njs_int_t njs_parser_statement_list_next(njs_parser_t *parser,
277
    njs_lexer_token_t *token, njs_queue_link_t *current);
278
static njs_int_t njs_parser_statement_list_item(njs_parser_t *parser,
279
    njs_lexer_token_t *token, njs_queue_link_t *current);
280
281
static njs_int_t njs_parser_lexical_declaration(njs_parser_t *parser,
282
    njs_lexer_token_t *token, njs_queue_link_t *current);
283
static njs_int_t njs_parser_variable_statement(njs_parser_t *parser,
284
    njs_lexer_token_t *token, njs_queue_link_t *current);
285
static njs_int_t njs_parser_variable_declaration_list(njs_parser_t *parser,
286
    njs_lexer_token_t *token, njs_queue_link_t *current);
287
static njs_int_t njs_parser_variable_declaration_list_next(
288
    njs_parser_t *parser, njs_lexer_token_t *token, njs_queue_link_t *current);
289
static njs_int_t njs_parser_variable_declaration(njs_parser_t *parser,
290
    njs_lexer_token_t *token, njs_queue_link_t *current);
291
292
static njs_int_t njs_parser_binding_pattern(njs_parser_t *parser,
293
    njs_lexer_token_t *token, njs_queue_link_t *current);
294
static njs_int_t njs_parser_object_binding_pattern(njs_parser_t *parser,
295
    njs_lexer_token_t *token, njs_queue_link_t *current);
296
static njs_int_t njs_parser_array_binding_pattern(njs_parser_t *parser,
297
    njs_lexer_token_t *token, njs_queue_link_t *current);
298
299
static njs_int_t njs_parser_expression_statement(njs_parser_t *parser,
300
    njs_lexer_token_t *token, njs_queue_link_t *current);
301
static njs_int_t njs_parser_expression_statement_after(njs_parser_t *parser,
302
    njs_lexer_token_t *token, njs_queue_link_t *current);
303
304
static njs_int_t njs_parser_if_statement(njs_parser_t *parser,
305
    njs_lexer_token_t *token, njs_queue_link_t *current);
306
static njs_int_t njs_parser_if_close_parenthesis(njs_parser_t *parser,
307
    njs_lexer_token_t *token, njs_queue_link_t *current);
308
static njs_int_t njs_parser_else_statement(njs_parser_t *parser,
309
    njs_lexer_token_t *token, njs_queue_link_t *current);
310
static njs_int_t njs_parser_else_statement_after(njs_parser_t *parser,
311
    njs_lexer_token_t *token, njs_queue_link_t *current);
312
313
static njs_int_t njs_parser_iteration_statement_do(njs_parser_t *parser,
314
    njs_lexer_token_t *token, njs_queue_link_t *current);
315
static njs_int_t njs_parser_iteration_statement_do_while(njs_parser_t *parser,
316
    njs_lexer_token_t *token, njs_queue_link_t *current);
317
static njs_int_t njs_parser_do_while_semicolon(njs_parser_t *parser,
318
    njs_lexer_token_t *token, njs_queue_link_t *current);
319
320
static njs_int_t njs_parser_iteration_statement_while(njs_parser_t *parser,
321
    njs_lexer_token_t *token, njs_queue_link_t *current);
322
static njs_int_t njs_parser_while_statement(njs_parser_t *parser,
323
    njs_lexer_token_t *token, njs_queue_link_t *current);
324
static njs_int_t njs_parser_while_after(njs_parser_t *parser,
325
    njs_lexer_token_t *token, njs_queue_link_t *current);
326
327
static njs_int_t njs_parser_iteration_statement_for(njs_parser_t *parser,
328
    njs_lexer_token_t *token, njs_queue_link_t *current);
329
static njs_int_t njs_parser_for_expression_map_continue(
330
    njs_parser_t *parser, njs_lexer_token_t *token,
331
    njs_queue_link_t *current);
332
static njs_int_t njs_parser_for_expression_map_reparse(
333
    njs_parser_t *parser, njs_lexer_token_t *token,
334
    njs_queue_link_t *current);
335
static njs_int_t njs_parser_expression_continue_op(njs_parser_t *parser,
336
    njs_lexer_token_t *token, njs_queue_link_t *current);
337
static njs_int_t njs_parser_expression_continue_assign_comma(
338
    njs_parser_t *parser, njs_lexer_token_t *token,
339
    njs_queue_link_t *current);
340
static njs_int_t njs_parser_for_in_statement_statement(njs_parser_t *parser,
341
    njs_lexer_token_t *token, njs_queue_link_t *current);
342
static njs_int_t njs_parser_iteration_statement_for_map(njs_parser_t *parser,
343
    njs_lexer_token_t *token, njs_queue_link_t *current);
344
static njs_int_t njs_parser_for_var_binding_or_var_list(njs_parser_t *parser,
345
    njs_lexer_token_t *token, njs_queue_link_t *current,
346
    njs_token_type_t token_type);
347
static njs_int_t njs_parser_for_var_in_statement(njs_parser_t *parser,
348
    njs_lexer_token_t *token, njs_queue_link_t *current);
349
static njs_int_t njs_parser_for_var_in_statement_after(njs_parser_t *parser,
350
    njs_lexer_token_t *token, njs_queue_link_t *current);
351
static njs_int_t njs_parser_for_var_in_of_expression(njs_parser_t *parser,
352
    njs_lexer_token_t *token, njs_queue_link_t *current);
353
static njs_int_t njs_parser_for_in_statement(njs_parser_t *parser,
354
    njs_lexer_token_t *token, njs_queue_link_t *current);
355
static njs_int_t njs_parser_for_in_statement_after(njs_parser_t *parser,
356
    njs_lexer_token_t *token, njs_queue_link_t *current);
357
static njs_int_t njs_parser_for_expression(njs_parser_t *parser,
358
    njs_lexer_token_t *token, njs_queue_link_t *current);
359
static njs_int_t njs_parser_for_expression_end(njs_parser_t *parser,
360
    njs_lexer_token_t *token, njs_queue_link_t *current);
361
static njs_int_t njs_parser_for_end(njs_parser_t *parser,
362
    njs_lexer_token_t *token, njs_queue_link_t *current);
363
364
static njs_int_t njs_parser_switch_statement(njs_parser_t *parser,
365
    njs_lexer_token_t *token, njs_queue_link_t *current);
366
static njs_int_t njs_parser_switch_statement_after(njs_parser_t *parser,
367
    njs_lexer_token_t *token, njs_queue_link_t *current);
368
static njs_int_t njs_parser_switch_block(njs_parser_t *parser,
369
    njs_lexer_token_t *token, njs_queue_link_t *current);
370
static njs_int_t njs_parser_switch_block_after(njs_parser_t *parser,
371
    njs_lexer_token_t *token, njs_queue_link_t *current);
372
static njs_int_t njs_parser_switch_case(njs_parser_t *parser,
373
    njs_lexer_token_t *token, njs_queue_link_t *current);
374
static njs_int_t njs_parser_switch_case_wo_def(njs_parser_t *parser,
375
    njs_lexer_token_t *token, njs_queue_link_t *current);
376
static njs_int_t njs_parser_switch_case_def(njs_parser_t *parser,
377
    njs_lexer_token_t *token, njs_queue_link_t *current,
378
    njs_bool_t with_default);
379
static njs_int_t njs_parser_switch_case_after(njs_parser_t *parser,
380
    njs_lexer_token_t *token, njs_queue_link_t *current);
381
static njs_int_t njs_parser_switch_case_block(njs_parser_t *parser,
382
    njs_lexer_token_t *token, njs_queue_link_t *current);
383
static njs_int_t njs_parser_switch_case_after_wo_def(njs_parser_t *parser,
384
    njs_lexer_token_t *token, njs_queue_link_t *current);
385
static njs_int_t njs_parser_switch_case_block_wo_def(njs_parser_t *parser,
386
    njs_lexer_token_t *token, njs_queue_link_t *current);
387
388
static njs_int_t njs_parser_continue_statement(njs_parser_t *parser,
389
    njs_lexer_token_t *token, njs_queue_link_t *current);
390
391
static njs_int_t njs_parser_break_statement(njs_parser_t *parser,
392
    njs_lexer_token_t *token, njs_queue_link_t *current);
393
394
static njs_int_t njs_parser_break_continue(njs_parser_t *parser,
395
    njs_lexer_token_t *token, njs_token_type_t type);
396
397
static njs_int_t njs_parser_return_statement(njs_parser_t *parser,
398
    njs_lexer_token_t *token, njs_queue_link_t *current);
399
static njs_int_t njs_parser_return_statement_after(njs_parser_t *parser,
400
    njs_lexer_token_t *token, njs_queue_link_t *current);
401
402
static njs_int_t njs_parser_with_statement(njs_parser_t *parser,
403
    njs_lexer_token_t *token, njs_queue_link_t *current);
404
405
static njs_int_t njs_parser_labelled_statement(njs_parser_t *parser,
406
    njs_lexer_token_t *token, njs_queue_link_t *current);
407
static njs_int_t njs_parser_labelled_statement_after(njs_parser_t *parser,
408
    njs_lexer_token_t *token, njs_queue_link_t *current);
409
410
static njs_int_t njs_parser_throw_statement(njs_parser_t *parser,
411
    njs_lexer_token_t *token, njs_queue_link_t *current);
412
static njs_int_t njs_parser_throw_statement_after(njs_parser_t *parser,
413
    njs_lexer_token_t *token, njs_queue_link_t *current);
414
415
static njs_int_t njs_parser_try_statement(njs_parser_t *parser,
416
    njs_lexer_token_t *token, njs_queue_link_t *current);
417
static njs_int_t njs_parser_catch_or_finally(njs_parser_t *parser,
418
    njs_lexer_token_t *token, njs_queue_link_t *current);
419
static njs_int_t njs_parser_catch_after(njs_parser_t *parser,
420
    njs_lexer_token_t *token, njs_queue_link_t *current);
421
static njs_int_t njs_parser_catch_parenthesis(njs_parser_t *parser,
422
    njs_lexer_token_t *token, njs_queue_link_t *current);
423
static njs_int_t njs_parser_catch_statement_open_brace(njs_parser_t *parser,
424
    njs_lexer_token_t *token, njs_queue_link_t *current);
425
static njs_int_t njs_parser_catch_finally(njs_parser_t *parser,
426
    njs_lexer_token_t *token, njs_queue_link_t *current);
427
428
static njs_int_t njs_parser_debugger_statement(njs_parser_t *parser,
429
    njs_lexer_token_t *token, njs_queue_link_t *current);
430
431
static njs_int_t njs_parser_function_declaration(njs_parser_t *parser,
432
    njs_lexer_token_t *token, njs_queue_link_t *current);
433
static njs_int_t njs_parser_function_declaration_after(njs_parser_t *parser,
434
    njs_lexer_token_t *token, njs_queue_link_t *current);
435
static njs_int_t njs_parser_function_parse(njs_parser_t *parser,
436
    njs_lexer_token_t *token, njs_queue_link_t *current);
437
static njs_int_t njs_parser_function_expression(njs_parser_t *parser,
438
    njs_lexer_token_t *token, njs_queue_link_t *current);
439
static njs_int_t njs_parser_function_expression_after(njs_parser_t *parser,
440
    njs_lexer_token_t *token, njs_queue_link_t *current);
441
static njs_int_t njs_parser_unique_formal_parameters(njs_parser_t *parser,
442
    njs_lexer_token_t *token, njs_queue_link_t *current);
443
static njs_int_t njs_parser_formal_parameters(njs_parser_t *parser,
444
    njs_lexer_token_t *token, njs_queue_link_t *current);
445
static njs_int_t njs_parser_formal_parameters_after(njs_parser_t *parser,
446
    njs_lexer_token_t *token, njs_queue_link_t *current);
447
448
static njs_int_t njs_parser_arrow_function(njs_parser_t *parser,
449
    njs_lexer_token_t *token, njs_queue_link_t *current);
450
static njs_int_t njs_parser_arrow_function_args_after(njs_parser_t *parser,
451
    njs_lexer_token_t *token, njs_queue_link_t *current);
452
static njs_int_t njs_parser_arrow_function_arrow(njs_parser_t *parser,
453
    njs_lexer_token_t *token, njs_queue_link_t *current);
454
static njs_int_t njs_parser_arrow_function_body_after(njs_parser_t *parser,
455
    njs_lexer_token_t *token, njs_queue_link_t *current);
456
457
static njs_int_t njs_parser_method_definition(njs_parser_t *parser,
458
    njs_lexer_token_t *token, njs_queue_link_t *current);
459
static njs_int_t njs_parser_get_set(njs_parser_t *parser,
460
    njs_lexer_token_t *token, njs_queue_link_t *current);
461
static njs_int_t njs_parser_get_set_after(njs_parser_t *parser,
462
    njs_lexer_token_t *token, njs_queue_link_t *current);
463
static njs_int_t njs_parser_get_after(njs_parser_t *parser,
464
    njs_lexer_token_t *token, njs_queue_link_t *current);
465
static njs_int_t njs_parser_set_after(njs_parser_t *parser,
466
    njs_lexer_token_t *token, njs_queue_link_t *current);
467
static njs_int_t njs_parser_function_lambda(njs_parser_t *parser,
468
    njs_lexer_token_t *token, njs_queue_link_t *current);
469
static njs_int_t njs_parser_function_lambda_args_after(njs_parser_t *parser,
470
    njs_lexer_token_t *token, njs_queue_link_t *current);
471
static njs_int_t njs_parser_function_lambda_body_after(njs_parser_t *parser,
472
    njs_lexer_token_t *token, njs_queue_link_t *current);
473
474
static njs_int_t njs_parser_export(njs_parser_t *parser,
475
    njs_lexer_token_t *token, njs_queue_link_t *current);
476
static njs_int_t njs_parser_export_after(njs_parser_t *parser,
477
    njs_lexer_token_t *token, njs_queue_link_t *current);
478
479
static njs_int_t njs_parser_import(njs_parser_t *parser,
480
    njs_lexer_token_t *token, njs_queue_link_t *current);
481
static njs_int_t njs_parser_export_sink(njs_parser_t *parser);
482
483
static njs_parser_node_t *njs_parser_return_set(njs_parser_t *parser,
484
    njs_parser_node_t *expr);
485
static njs_parser_node_t *njs_parser_variable_node(njs_parser_t *parser,
486
    uintptr_t atom_id, njs_variable_type_t type, njs_variable_t  **retvar);
487
488
static njs_parser_node_t *njs_parser_reference(njs_parser_t *parser,
489
    njs_lexer_token_t *token);
490
491
static njs_parser_node_t *njs_parser_argument(njs_parser_t *parser,
492
    njs_parser_node_t *expr, njs_index_t index);
493
494
static njs_int_t njs_parser_object_property(njs_parser_t *parser,
495
    njs_parser_node_t *parent, njs_parser_node_t *property,
496
    njs_parser_node_t *value, njs_bool_t proto_init);
497
static njs_int_t njs_parser_property_accessor(njs_parser_t *parser,
498
    njs_parser_node_t *parent, njs_parser_node_t *property,
499
    njs_parser_node_t *value, njs_token_type_t accessor);
500
static njs_int_t njs_parser_array_item(njs_parser_t *parser,
501
    njs_parser_node_t *array, njs_parser_node_t *value);
502
static njs_int_t njs_parser_template_string(njs_parser_t *parser,
503
    njs_lexer_token_t *token);
504
static njs_token_type_t njs_parser_escape_string_create(njs_parser_t *parser,
505
    njs_lexer_token_t *token, njs_value_t *value);
506
static njs_int_t njs_parser_escape_string_calc_length(njs_parser_t *parser,
507
    njs_lexer_token_t *token, size_t *out_size, size_t *out_length);
508
509
static void njs_parser_serialize_tree(njs_chb_t *chain,
510
    njs_parser_node_t *node, njs_int_t *ret, size_t indent);
511
static njs_int_t njs_parser_serialize_node(njs_chb_t *chain,
512
    njs_parser_node_t *node);
513
514
515
#define njs_parser_chain_top(parser)                                          \
516
272k
    ((parser)->scope->top)
517
518
519
#define njs_parser_chain_top_set(parser, node)                                \
520
1.16M
    (parser)->scope->top = node
521
522
523
njs_inline njs_int_t
524
njs_parser_not_supported(njs_parser_t *parser, njs_lexer_token_t *token)
525
6.13k
{
526
6.13k
    if (token->type != NJS_TOKEN_END) {
527
6.13k
        njs_parser_syntax_error(parser, "Token \"%V\" not supported "
528
6.13k
                                "in this version", &token->text);
529
6.13k
    } else {
530
0
        njs_parser_syntax_error(parser, "Not supported in this version");
531
0
    }
532
533
6.13k
    return NJS_DONE;
534
6.13k
}
535
536
537
njs_inline njs_int_t
538
njs_parser_reject(njs_parser_t *parser)
539
18.0k
{
540
18.0k
    njs_queue_link_t          *link;
541
18.0k
    njs_parser_stack_entry_t  *entry;
542
543
383k
    while (!njs_queue_is_empty(&parser->stack)) {
544
383k
        entry = njs_queue_link_data(njs_queue_first(&parser->stack),
545
383k
                                    njs_parser_stack_entry_t, link);
546
547
383k
        link = njs_queue_first(&parser->stack);
548
383k
        njs_queue_remove(link);
549
550
383k
        if (!entry->optional) {
551
18.0k
            njs_parser_next(parser, entry->state);
552
18.0k
            parser->target = entry->node;
553
554
18.0k
            return NJS_DECLINED;
555
18.0k
        }
556
383k
    }
557
558
0
    return njs_parser_failed(parser);
559
18.0k
}
560
561
562
njs_int_t
563
njs_parser_init(njs_vm_t *vm, njs_parser_t *parser, njs_parser_scope_t *scope,
564
    njs_str_t *file, u_char *start, u_char *end)
565
156k
{
566
156k
    njs_lexer_t  *lexer;
567
568
156k
    njs_memzero(parser, sizeof(njs_parser_t));
569
570
156k
    parser->scope = scope;
571
572
156k
    lexer = &parser->lexer0;
573
156k
    parser->lexer = lexer;
574
575
156k
    parser->use_lhs = 0;
576
577
156k
    return njs_lexer_init(vm, lexer, file, start, end);
578
156k
}
579
580
581
njs_int_t
582
njs_parser(njs_vm_t *vm, njs_parser_t *parser)
583
156k
{
584
156k
    njs_int_t          ret;
585
156k
    njs_lexer_token_t  *token;
586
587
156k
    parser->vm = vm;
588
589
156k
    njs_set_invalid(&vm->exception);
590
591
156k
    if (parser->scope == NULL) {
592
156k
        ret = njs_parser_scope_begin(parser,
593
156k
                                     parser->module ? NJS_SCOPE_FUNCTION
594
156k
                                                    : NJS_SCOPE_GLOBAL, 1);
595
156k
        if (njs_slow_path(ret != NJS_OK)) {
596
0
            return NJS_ERROR;
597
0
        }
598
599
156k
    } else {
600
0
        parser->scope->top = NULL;
601
0
        parser->node = NULL;
602
0
        parser->ret = NJS_OK;
603
0
    }
604
605
156k
    njs_queue_init(&parser->stack);
606
607
156k
    parser->target = NULL;
608
156k
    njs_parser_next(parser, njs_parser_statement_list);
609
610
156k
    ret = njs_parser_after(parser, njs_queue_first(&parser->stack),
611
156k
                           NULL, 0, njs_parser_check_error_state);
612
156k
    if (ret != NJS_OK) {
613
0
        return ret;
614
0
    }
615
616
108M
    do {
617
108M
        token = njs_lexer_token(parser->lexer, 0);
618
108M
        if (njs_slow_path(token == NULL)) {
619
0
            return NJS_ERROR;
620
0
        }
621
622
108M
        parser->ret = parser->state(parser, token,
623
108M
                                    njs_queue_first(&parser->stack));
624
625
108M
    } while (parser->ret != NJS_DONE && parser->ret != NJS_ERROR);
626
627
156k
    if (parser->ret != NJS_DONE) {
628
1.83k
        return NJS_ERROR;
629
1.83k
    }
630
631
154k
    if (njs_is_error(&vm->exception)) {
632
64.0k
        return NJS_ERROR;
633
64.0k
    }
634
635
90.7k
    if (parser->node == NULL) {
636
        /* Empty string, just semicolons or variables declarations. */
637
638
0
        parser->node = njs_parser_node_new(parser, 0);
639
0
        if (njs_slow_path(parser->node == NULL)) {
640
0
            return NJS_ERROR;
641
0
        }
642
0
    }
643
644
90.7k
    if (parser->module) {
645
0
        ret = njs_parser_export_sink(parser);
646
0
        if (ret != NJS_OK) {
647
0
            return NJS_ERROR;
648
0
        }
649
650
90.7k
    } else {
651
90.7k
        parser->node->token_type = NJS_TOKEN_END;
652
90.7k
        parser->node->token_line = parser->lexer->line;
653
654
90.7k
        njs_parser_chain_top_set(parser, parser->node);
655
90.7k
    }
656
657
90.7k
    return NJS_OK;
658
90.7k
}
659
660
661
static njs_int_t
662
njs_parser_check_error_state(njs_parser_t *parser, njs_lexer_token_t *token,
663
    njs_queue_link_t *current)
664
207
{
665
207
    return njs_parser_failed(parser);
666
207
}
667
668
669
njs_int_t
670
njs_parser_failed_state(njs_parser_t *parser, njs_lexer_token_t *token,
671
    njs_queue_link_t *current)
672
43.3k
{
673
43.3k
    if (token->type != NJS_TOKEN_END) {
674
39.8k
        njs_parser_syntax_error(parser, "Unexpected token \"%V\"",
675
39.8k
                                &token->text);
676
39.8k
    } else {
677
3.49k
        njs_parser_syntax_error(parser, "Unexpected end of input");
678
3.49k
    }
679
680
43.3k
    return NJS_DONE;
681
43.3k
}
682
683
684
static njs_int_t
685
njs_parser_scope_begin(njs_parser_t *parser, njs_scope_t type,
686
    njs_bool_t init_this)
687
548k
{
688
548k
    njs_variable_t      *var;
689
548k
    njs_parser_scope_t  *scope, *parent;
690
691
548k
    scope = njs_mp_zalloc(parser->vm->mem_pool, sizeof(njs_parser_scope_t));
692
548k
    if (njs_slow_path(scope == NULL)) {
693
0
        return NJS_ERROR;
694
0
    }
695
696
548k
    scope->type = type;
697
698
548k
    njs_rbtree_init(&scope->variables, njs_parser_scope_rbtree_compare);
699
548k
    njs_rbtree_init(&scope->labels, njs_parser_scope_rbtree_compare);
700
548k
    njs_rbtree_init(&scope->references, njs_parser_scope_rbtree_compare);
701
702
548k
    parent = parser->scope;
703
548k
    scope->parent = parent;
704
548k
    parser->scope = scope;
705
706
548k
    if (type == NJS_SCOPE_FUNCTION || type == NJS_SCOPE_GLOBAL) {
707
404k
        if (init_this) {
708
            /* Add this as first variable. */
709
319k
            var = njs_variable_add(parser, scope, NJS_ATOM_STRING_this,
710
319k
                                   NJS_VARIABLE_VAR);
711
319k
            if (njs_slow_path(var == NULL)) {
712
0
                return NJS_ERROR;
713
0
            }
714
715
319k
            var->index = njs_scope_index(type, 0, NJS_LEVEL_LOCAL,
716
319k
                                         NJS_VARIABLE_VAR);
717
319k
        }
718
404k
    }
719
720
548k
    scope->items = 1;
721
722
548k
    return NJS_OK;
723
548k
}
724
725
726
static void
727
njs_parser_scope_end(njs_parser_t *parser)
728
280k
{
729
280k
    njs_parser_scope_t  *scope, *parent;
730
731
280k
    scope = parser->scope;
732
733
280k
    parent = scope->parent;
734
280k
    parser->scope = parent;
735
280k
}
736
737
738
intptr_t
739
njs_parser_scope_rbtree_compare(njs_rbtree_node_t *node1,
740
    njs_rbtree_node_t *node2)
741
14.7M
{
742
14.7M
    njs_variable_node_t  *lex_node1, *lex_node2;
743
744
14.7M
    lex_node1 = (njs_variable_node_t *) node1;
745
14.7M
    lex_node2 = (njs_variable_node_t *) node2;
746
747
14.7M
    if (lex_node1->key < lex_node2->key) {
748
3.17M
        return -1;
749
3.17M
    }
750
751
11.5M
    if (lex_node1->key > lex_node2->key) {
752
10.4M
        return 1;
753
10.4M
    }
754
755
1.06M
    return 0;
756
11.5M
}
757
758
759
static njs_int_t
760
njs_parser_generator_expression(njs_parser_t *parser, njs_lexer_token_t *token,
761
    njs_queue_link_t *current)
762
0
{
763
0
    return njs_parser_not_supported(parser, token);
764
0
}
765
766
767
static njs_int_t
768
njs_parser_class_expression(njs_parser_t *parser, njs_lexer_token_t *token,
769
    njs_queue_link_t *current)
770
152
{
771
152
    return njs_parser_not_supported(parser, token);
772
152
}
773
774
775
static njs_int_t
776
njs_parser_async_generator_expression(njs_parser_t *parser,
777
    njs_lexer_token_t *token, njs_queue_link_t *current)
778
50
{
779
50
    return njs_parser_not_supported(parser, token);
780
50
}
781
782
783
static njs_int_t
784
njs_parser_generator_declaration(njs_parser_t *parser, njs_lexer_token_t *token,
785
    njs_queue_link_t *current)
786
0
{
787
0
    return njs_parser_not_supported(parser, token);
788
0
}
789
790
791
static njs_int_t
792
njs_parser_class_declaration(njs_parser_t *parser, njs_lexer_token_t *token,
793
    njs_queue_link_t *current)
794
0
{
795
0
    return njs_parser_not_supported(parser, token);
796
0
}
797
798
799
static njs_int_t
800
njs_parser_function_or_generator_handler(njs_parser_t *parser,
801
    njs_lexer_token_t *token, njs_queue_link_t *current, njs_bool_t is_async)
802
5.78k
{
803
5.78k
    njs_parser_node_t  *node, *cur;
804
805
5.78k
    cur = parser->node;
806
807
5.78k
    if (token->type == NJS_TOKEN_MULTIPLICATION) {
808
0
        njs_lexer_consume_token(parser->lexer, 1);
809
0
        njs_parser_next(parser, njs_parser_generator_declaration);
810
811
5.78k
    } else {
812
5.78k
        if (is_async) {
813
193
            node = njs_parser_node_new(parser,
814
193
                                       NJS_TOKEN_ASYNC_FUNCTION_DECLARATION);
815
5.59k
        } else {
816
5.59k
            node = njs_parser_node_new(parser, NJS_TOKEN_FUNCTION_DECLARATION);
817
5.59k
        }
818
819
5.78k
        if (node == NULL) {
820
0
            return NJS_ERROR;
821
0
        }
822
823
5.78k
        node->token_line = token->line;
824
5.78k
        parser->node = node;
825
826
5.78k
        njs_lexer_consume_token(parser->lexer, 1);
827
5.78k
        njs_parser_next(parser, njs_parser_function_declaration);
828
5.78k
    }
829
830
5.78k
    return njs_parser_after(parser, current, cur, 1,
831
5.78k
                            njs_parser_statement_after);
832
5.78k
}
833
834
835
static njs_int_t
836
njs_parser_function_or_generator(njs_parser_t *parser,
837
    njs_lexer_token_t *token, njs_queue_link_t *current)
838
1.23M
{
839
1.23M
    if (token->type != NJS_TOKEN_FUNCTION) {
840
1.23M
        return NJS_DECLINED;
841
1.23M
    }
842
843
5.59k
    return njs_parser_function_or_generator_handler(parser, token, current, 0);
844
1.23M
}
845
846
847
static njs_int_t
848
njs_parser_async_function_or_generator(njs_parser_t *parser,
849
    njs_lexer_token_t *token, njs_queue_link_t *current)
850
1.23M
{
851
1.23M
    if (token->type != NJS_TOKEN_ASYNC) {
852
1.22M
        return NJS_DECLINED;
853
1.22M
    }
854
855
298
    token = njs_lexer_peek_token(parser->lexer, token, 1);
856
298
    if (token == NULL) {
857
0
        return NJS_ERROR;
858
0
    }
859
860
298
    if (token->type != NJS_TOKEN_FUNCTION) {
861
105
        return NJS_DECLINED;
862
105
    }
863
864
193
    njs_lexer_consume_token(parser->lexer, 1);
865
866
193
    return njs_parser_function_or_generator_handler(parser, token, current, 1);
867
298
}
868
869
870
njs_inline njs_int_t
871
njs_parser_expect_semicolon(njs_parser_t *parser, njs_lexer_token_t *token)
872
805k
{
873
805k
    if (token->type != NJS_TOKEN_SEMICOLON) {
874
718k
        if (token->type != NJS_TOKEN_END
875
627k
            && token->type != NJS_TOKEN_CLOSE_BRACE
876
501k
            && parser->lexer->prev_type != NJS_TOKEN_LINE_END)
877
5.68k
        {
878
5.68k
            return NJS_DECLINED;
879
5.68k
        }
880
881
713k
        return NJS_OK;
882
718k
    }
883
884
87.1k
    njs_lexer_consume_token(parser->lexer, 1);
885
886
87.1k
    return NJS_OK;
887
805k
}
888
889
890
static njs_int_t
891
njs_parser_semicolon(njs_parser_t *parser, njs_lexer_token_t *token,
892
    njs_queue_link_t *current)
893
29.2k
{
894
29.2k
    if (njs_parser_expect_semicolon(parser, token) != NJS_OK) {
895
698
        return njs_parser_failed(parser);
896
698
    }
897
898
28.5k
    return njs_parser_stack_pop(parser);
899
29.2k
}
900
901
902
static njs_int_t
903
njs_parser_close_bracked(njs_parser_t *parser, njs_lexer_token_t *token,
904
    njs_queue_link_t *current)
905
0
{
906
0
    if (token->type != NJS_TOKEN_CLOSE_BRACKET) {
907
0
        return njs_parser_failed(parser);
908
0
    }
909
910
0
    njs_lexer_consume_token(parser->lexer, 1);
911
912
0
    return njs_parser_stack_pop(parser);
913
0
}
914
915
916
static njs_int_t
917
njs_parser_close_parenthesis(njs_parser_t *parser, njs_lexer_token_t *token,
918
    njs_queue_link_t *current)
919
108k
{
920
108k
    if (parser->ret != NJS_OK) {
921
3.07k
        return njs_parser_failed(parser);
922
3.07k
    }
923
924
104k
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
925
435
        return njs_parser_failed(parser);
926
435
    }
927
928
104k
    njs_lexer_consume_token(parser->lexer, 1);
929
930
104k
    return njs_parser_stack_pop(parser);
931
104k
}
932
933
934
static njs_int_t
935
njs_parser_expression_parenthesis(njs_parser_t *parser,
936
    njs_lexer_token_t *token, njs_queue_link_t *current)
937
12.8k
{
938
12.8k
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS) {
939
46
        return njs_parser_failed(parser);
940
46
    }
941
942
12.8k
    njs_lexer_consume_token(parser->lexer, 1);
943
944
12.8k
    parser->node = NULL;
945
946
12.8k
    njs_parser_next(parser, njs_parser_expression);
947
948
12.8k
    return njs_parser_after(parser, current, NULL, 0,
949
12.8k
                            njs_parser_close_parenthesis);
950
12.8k
}
951
952
953
static njs_int_t
954
njs_parser_iteration_statement_for_end(njs_parser_t *parser,
955
    njs_lexer_token_t *token, njs_queue_link_t *current)
956
19.9k
{
957
19.9k
    parser->node->token_line = (uint32_t) (uintptr_t) parser->target;
958
19.9k
    parser->target = NULL;
959
960
19.9k
    njs_parser_scope_end(parser);
961
962
19.9k
    return njs_parser_stack_pop(parser);
963
19.9k
}
964
965
966
/*
967
 * 12.2 Primary Expression.
968
 */
969
static njs_int_t
970
njs_parser_primary_expression_test(njs_parser_t *parser,
971
    njs_lexer_token_t *token, njs_queue_link_t *current)
972
2.86M
{
973
2.86M
    njs_int_t          ret;
974
2.86M
    njs_lexer_token_t  *next;
975
2.86M
    njs_parser_node_t  *node;
976
977
2.86M
    switch (token->type) {
978
    /* IdentifierReference */
979
4.46k
    case NJS_TOKEN_THIS:
980
4.82k
    case NJS_TOKEN_NULL:
981
1.32M
    case NJS_TOKEN_NAME:
982
1.32M
    case NJS_TOKEN_YIELD:
983
1.32M
    case NJS_TOKEN_AWAIT:
984
1.32M
        goto reference;
985
986
775
    case NJS_TOKEN_TRUE:
987
775
        node = njs_parser_node_new(parser, token->type);
988
775
        if (node == NULL) {
989
0
            return NJS_ERROR;
990
0
        }
991
992
775
        node->u.value = njs_value_true;
993
775
        node->token_line = token->line;
994
995
775
        parser->node = node;
996
775
        goto done;
997
998
1.22k
    case NJS_TOKEN_FALSE:
999
1.22k
        node = njs_parser_node_new(parser, token->type);
1000
1.22k
        if (node == NULL) {
1001
0
            return NJS_ERROR;
1002
0
        }
1003
1004
1.22k
        node->u.value = njs_value_false;
1005
1.22k
        node->token_line = token->line;
1006
1007
1.22k
        parser->node = node;
1008
1.22k
        goto done;
1009
1010
1.06M
    case NJS_TOKEN_NUMBER:
1011
1.06M
        node = njs_parser_node_new(parser, NJS_TOKEN_NUMBER);
1012
1.06M
        if (node == NULL) {
1013
0
            return NJS_ERROR;
1014
0
        }
1015
1016
1.06M
        njs_set_number(&node->u.value, token->number);
1017
1018
1.06M
        if (njs_number_is_integer_index(token->number)
1019
1.04M
            && token->number < 0x80000000)
1020
1.04M
        {
1021
1.04M
            node->u.value.atom_id = njs_number_atom((uint32_t) token->number);
1022
1.04M
        }
1023
1024
1.06M
        node->token_line = token->line;
1025
1026
1.06M
        parser->node = node;
1027
1.06M
        goto done;
1028
1029
5.70k
    case NJS_TOKEN_STRING:
1030
5.70k
        node = njs_parser_node_new(parser, NJS_TOKEN_STRING);
1031
5.70k
        if (node == NULL) {
1032
0
            return NJS_ERROR;
1033
0
        }
1034
1035
5.70k
        node->token_line = token->line;
1036
1037
5.70k
        ret = njs_parser_string_create(parser->vm, token, &node->u.value);
1038
5.70k
        if (ret != NJS_OK) {
1039
0
            return NJS_ERROR;
1040
0
        }
1041
1042
5.70k
        parser->node = node;
1043
5.70k
        goto done;
1044
1045
2.22k
    case NJS_TOKEN_ESCAPE_STRING:
1046
        /* Internal optimization. This is not in the specification. */
1047
1048
2.22k
        node = njs_parser_node_new(parser, NJS_TOKEN_STRING);
1049
2.22k
        if (node == NULL) {
1050
0
            return NJS_ERROR;
1051
0
        }
1052
1053
2.22k
        node->token_line = token->line;
1054
1055
2.22k
        ret = njs_parser_escape_string_create(parser, token, &node->u.value);
1056
2.22k
        if (ret != NJS_TOKEN_STRING) {
1057
683
            return NJS_ERROR;
1058
683
        }
1059
1060
1.53k
        parser->node = node;
1061
1.53k
        goto done;
1062
1063
1.77k
    case NJS_TOKEN_UNTERMINATED_STRING:
1064
        /* Internal optimization. This is not in the specification. */
1065
1066
1.77k
        njs_parser_syntax_error(parser, "Unterminated string \"%V\"",
1067
1.77k
                                &token->text);
1068
1.77k
        return NJS_ERROR;
1069
1070
    /* ArrayLiteral */
1071
41.9k
    case NJS_TOKEN_OPEN_BRACKET:
1072
41.9k
        node = njs_parser_node_new(parser, NJS_TOKEN_ARRAY);
1073
41.9k
        if (node == NULL) {
1074
0
            return NJS_ERROR;
1075
0
        }
1076
1077
41.9k
        node->token_line = token->line;
1078
41.9k
        parser->node = node;
1079
1080
41.9k
        njs_parser_next(parser, njs_parser_array_literal);
1081
41.9k
        break;
1082
1083
    /* ObjectLiteral */
1084
24.2k
    case NJS_TOKEN_OPEN_BRACE:
1085
24.2k
        node = njs_parser_node_new(parser, NJS_TOKEN_OBJECT);
1086
24.2k
        if (node == NULL) {
1087
0
            return NJS_ERROR;
1088
0
        }
1089
1090
24.2k
        node->token_line = token->line;
1091
24.2k
        parser->node = node;
1092
1093
24.2k
        njs_parser_next(parser, njs_parser_object_literal);
1094
24.2k
        break;
1095
1096
    /* FunctionExpression */
1097
151k
    case NJS_TOKEN_FUNCTION:
1098
151k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
1099
151k
        if (token == NULL) {
1100
0
            return NJS_ERROR;
1101
0
        }
1102
1103
        /* GeneratorExpression */
1104
151k
        if (token->type == NJS_TOKEN_MULTIPLICATION) {
1105
0
            njs_parser_next(parser, njs_parser_generator_expression);
1106
1107
151k
        } else {
1108
151k
            node = njs_parser_node_new(parser, NJS_TOKEN_FUNCTION_EXPRESSION);
1109
151k
            if (node == NULL) {
1110
0
                return NJS_ERROR;
1111
0
            }
1112
1113
151k
            node->token_line = token->line;
1114
151k
            parser->node = node;
1115
1116
151k
            njs_parser_next(parser, njs_parser_function_expression);
1117
151k
        }
1118
1119
151k
        break;
1120
1121
    /* ClassExpression */
1122
151k
    case NJS_TOKEN_CLASS:
1123
152
        njs_parser_next(parser, njs_parser_class_expression);
1124
152
        return NJS_OK;
1125
1126
    /* AsyncFunctionExpression */
1127
1.23k
    case NJS_TOKEN_ASYNC:
1128
1.23k
        next = njs_lexer_peek_token(parser->lexer, token, 1);
1129
1.23k
        if (next == NULL) {
1130
0
            return NJS_ERROR;
1131
0
        }
1132
1133
1.23k
        if (next->type != NJS_TOKEN_FUNCTION) {
1134
304
            goto reference;
1135
304
        }
1136
1137
929
        njs_lexer_consume_token(parser->lexer, 1);
1138
1139
929
        next = njs_lexer_peek_token(parser->lexer, next, 0);
1140
929
        if (njs_slow_path(next == NULL)) {
1141
0
            return NJS_ERROR;
1142
0
        }
1143
1144
        /* GeneratorExpression */
1145
929
        if (next->type == NJS_TOKEN_MULTIPLICATION) {
1146
50
            njs_parser_next(parser, njs_parser_async_generator_expression);
1147
1148
879
        } else {
1149
879
            node = njs_parser_node_new(parser,
1150
879
                                       NJS_TOKEN_ASYNC_FUNCTION_EXPRESSION);
1151
879
            if (node == NULL) {
1152
0
                return NJS_ERROR;
1153
0
            }
1154
1155
879
            node->token_line = next->line;
1156
879
            parser->node = node;
1157
1158
879
            njs_parser_next(parser, njs_parser_function_expression);
1159
879
        }
1160
1161
929
        break;
1162
1163
    /* RegularExpressionLiteral */
1164
24.7k
    case NJS_TOKEN_DIVISION:
1165
24.9k
    case NJS_TOKEN_DIVISION_ASSIGNMENT:
1166
24.9k
        node = njs_parser_node_new(parser, NJS_TOKEN_REGEXP);
1167
24.9k
        if (node == NULL) {
1168
0
            return NJS_ERROR;
1169
0
        }
1170
1171
24.9k
        node->token_line = token->line;
1172
24.9k
        parser->node = node;
1173
1174
24.9k
        ret = njs_parser_regexp_literal(parser, token, current);
1175
24.9k
        if (ret != NJS_OK) {
1176
2.79k
            return NJS_ERROR;
1177
2.79k
        }
1178
1179
22.2k
        goto done;
1180
1181
    /* TemplateLiteral */
1182
22.2k
    case NJS_TOKEN_GRAVE:
1183
10.6k
        node = njs_parser_node_new(parser, NJS_TOKEN_TEMPLATE_LITERAL);
1184
10.6k
        if (node == NULL) {
1185
0
            return NJS_ERROR;
1186
0
        }
1187
1188
10.6k
        node->token_line = token->line;
1189
10.6k
        parser->node = node;
1190
1191
10.6k
        njs_parser_next(parser, njs_parser_template_literal);
1192
10.6k
        return NJS_OK;
1193
1194
    /* CoverParenthesizedExpressionAndArrowParameterList */
1195
197k
    case NJS_TOKEN_OPEN_PARENTHESIS:
1196
197k
        njs_lexer_consume_token(parser->lexer, 1);
1197
1198
        /* TODO: By specification. */
1199
197k
        (void) njs_parser_cover_parenthesized_expression;
1200
1201
197k
        parser->node = NULL;
1202
1203
197k
        njs_parser_next(parser, njs_parser_expression);
1204
1205
197k
        return njs_parser_after(parser, current, NULL, 0,
1206
10.6k
                                njs_parser_close_parenthesis);
1207
1208
19.1k
    default:
1209
19.1k
        if (njs_lexer_token_is_identifier_reference(token)) {
1210
4.81k
            goto reference;
1211
4.81k
        }
1212
1213
14.3k
        return njs_parser_reject(parser);
1214
2.86M
    }
1215
1216
218k
    njs_lexer_consume_token(parser->lexer, 1);
1217
1218
218k
    return NJS_OK;
1219
1220
1.33M
reference:
1221
1222
1.33M
    node = njs_parser_reference(parser, token);
1223
1.33M
    if (node == NULL) {
1224
32
        return NJS_ERROR;
1225
32
    }
1226
1227
1.33M
    node->u.value.atom_id = token->atom_id;
1228
1229
1.33M
    node->token_line = token->line;
1230
1.33M
    parser->node = node;
1231
1232
2.42M
done:
1233
1234
2.42M
    njs_lexer_consume_token(parser->lexer, 1);
1235
1236
2.42M
    return NJS_DONE;
1237
1.33M
}
1238
1239
1240
static njs_int_t
1241
njs_parser_regexp_literal(njs_parser_t *parser, njs_lexer_token_t *token,
1242
    njs_queue_link_t *current)
1243
24.9k
{
1244
24.9k
    u_char                *p;
1245
24.9k
    njs_str_t             text;
1246
24.9k
    njs_int_t             ret;
1247
24.9k
    njs_lexer_t           *lexer;
1248
24.9k
    njs_value_t           *value, retval;
1249
24.9k
    njs_regex_flags_t     flags;
1250
24.9k
    njs_regexp_pattern_t  *pattern;
1251
1252
24.9k
    value = &parser->node->u.value;
1253
24.9k
    lexer = parser->lexer;
1254
1255
24.9k
    if (token->type == NJS_TOKEN_DIVISION_ASSIGNMENT) {
1256
255
        lexer->start--;
1257
255
    }
1258
1259
988k
    for (p = lexer->start; p < lexer->end; p++) {
1260
1261
988k
        switch (*p) {
1262
168
        case '\n':
1263
700
        case '\r':
1264
700
            goto failed;
1265
1266
3.04k
        case '[':
1267
69.5k
            while (1) {
1268
69.5k
                if (++p >= lexer->end) {
1269
2
                    goto failed;
1270
2
                }
1271
1272
69.5k
                if (*p == ']') {
1273
1.70k
                    break;
1274
1.70k
                }
1275
1276
67.8k
                switch (*p) {
1277
136
                case '\n':
1278
797
                case '\r':
1279
797
                    goto failed;
1280
1281
971
                case '\\':
1282
971
                    if (++p >= lexer->end || *p == '\n' || *p == '\r') {
1283
536
                        goto failed;
1284
536
                    }
1285
1286
435
                    break;
1287
67.8k
                }
1288
67.8k
            }
1289
1290
1.70k
            break;
1291
1292
4.82k
        case '\\':
1293
4.82k
            if (++p >= lexer->end || *p == '\n' || *p == '\r') {
1294
0
                goto failed;
1295
0
            }
1296
1297
4.82k
            break;
1298
1299
22.9k
        case '/':
1300
22.9k
            text.start = lexer->start;
1301
22.9k
            text.length = p - text.start;
1302
22.9k
            p++;
1303
22.9k
            lexer->start = p;
1304
1305
22.9k
            flags = njs_regexp_flags(&p, lexer->end);
1306
1307
22.9k
            if (njs_slow_path(flags < 0)) {
1308
287
                njs_parser_syntax_error(parser, "Invalid RegExp flags \"%*s\"",
1309
287
                                        p - lexer->start, lexer->start);
1310
1311
287
                return NJS_ERROR;
1312
287
            }
1313
1314
22.6k
            lexer->start = p;
1315
1316
22.6k
            pattern = njs_regexp_pattern_create(parser->vm, text.start,
1317
22.6k
                                                text.length, flags);
1318
1319
22.6k
            if (njs_slow_path(pattern == NULL)) {
1320
441
                retval = njs_vm_exception(parser->vm);
1321
441
                ret = njs_value_property(parser->vm, &retval,
1322
441
                                         NJS_ATOM_STRING_message, &retval);
1323
441
                if (njs_slow_path(ret != NJS_OK)) {
1324
0
                    return NJS_ERROR;
1325
0
                }
1326
1327
441
                njs_string_get(parser->vm, &retval, &text);
1328
1329
441
                njs_parser_syntax_error(parser, "%V", &text);
1330
1331
441
                return NJS_ERROR;
1332
441
            }
1333
1334
22.2k
            value->data.u.data = pattern;
1335
1336
22.2k
            return NJS_OK;
1337
988k
        }
1338
988k
    }
1339
1340
2.06k
failed:
1341
1342
2.06k
    njs_parser_syntax_error(parser, "Unterminated RegExp \"%*s\"",
1343
2.06k
                            p - (lexer->start - 1), lexer->start - 1);
1344
2.06k
    return NJS_ERROR;
1345
24.9k
}
1346
1347
1348
static njs_int_t
1349
njs_parser_template_literal(njs_parser_t *parser, njs_lexer_token_t *token,
1350
    njs_queue_link_t *current)
1351
685k
{
1352
685k
    njs_index_t        index;
1353
685k
    njs_parser_node_t  *node, *array, *template, *temp;
1354
1355
685k
    temp = njs_parser_node_new(parser, 0);
1356
685k
    if (temp == NULL) {
1357
0
        return NJS_ERROR;
1358
0
    }
1359
1360
685k
    array = njs_parser_node_new(parser, NJS_TOKEN_ARRAY);
1361
685k
    if (array == NULL) {
1362
0
        return NJS_ERROR;
1363
0
    }
1364
1365
685k
    array->token_line = token->line;
1366
1367
685k
    template = parser->node;
1368
1369
685k
    index = njs_scope_temp_index(template->scope);
1370
685k
    if (njs_slow_path(index == NJS_INDEX_ERROR)) {
1371
0
        return NJS_ERROR;
1372
0
    }
1373
1374
685k
    if (template->token_type != NJS_TOKEN_TEMPLATE_LITERAL) {
1375
674k
        node = njs_parser_argument(parser, array, index);
1376
674k
        if (node == NULL) {
1377
0
            return NJS_ERROR;
1378
0
        }
1379
1380
674k
        node->temporary = 1;
1381
1382
674k
        template->right = node;
1383
674k
        temp->right = node;
1384
1385
674k
        index = njs_scope_temp_index(template->scope);
1386
674k
        if (njs_slow_path(index == NJS_INDEX_ERROR)) {
1387
0
            return NJS_ERROR;
1388
0
        }
1389
1390
674k
    } else {
1391
10.6k
        template->left = array;
1392
10.6k
        temp->right = template;
1393
10.6k
    }
1394
1395
685k
    temp->temporary = 1;
1396
685k
    temp->left = template;
1397
685k
    temp->index = index;
1398
1399
685k
    parser->target = temp;
1400
1401
685k
    token->text.start++;
1402
685k
    token->text.length = 0;
1403
1404
685k
    njs_parser_next(parser, njs_parser_template_literal_string);
1405
1406
685k
    return NJS_OK;
1407
685k
}
1408
1409
1410
static njs_int_t
1411
njs_parser_template_literal_string(njs_parser_t *parser,
1412
    njs_lexer_token_t *token, njs_queue_link_t *current)
1413
696k
{
1414
696k
    njs_int_t          ret, ret_item;
1415
696k
    njs_parser_node_t  *template;
1416
1417
696k
    template = parser->target->left;
1418
1419
696k
    ret = njs_parser_template_string(parser, token);
1420
696k
    if (ret == NJS_ERROR) {
1421
700
        njs_parser_syntax_error(parser, "Unterminated template literal");
1422
700
        return NJS_DONE;
1423
700
    }
1424
1425
695k
    if (template->token_type != NJS_TOKEN_TEMPLATE_LITERAL) {
1426
677k
        ret_item = njs_parser_array_item(parser, template->right->left,
1427
677k
                                         parser->node);
1428
1429
677k
    } else {
1430
17.8k
        ret_item = njs_parser_array_item(parser, template->left, parser->node);
1431
17.8k
    }
1432
1433
695k
    if (ret_item != NJS_OK) {
1434
0
        return NJS_ERROR;
1435
0
    }
1436
1437
695k
    if (ret == NJS_DONE) {
1438
684k
        parser->node = template;
1439
1440
684k
        njs_parser_node_free(parser, parser->target);
1441
684k
        njs_lexer_consume_token(parser->lexer, 1);
1442
1443
684k
        return njs_parser_stack_pop(parser);
1444
684k
    }
1445
1446
11.3k
    parser->node = NULL;
1447
1448
11.3k
    njs_parser_next(parser, njs_parser_expression);
1449
1450
11.3k
    njs_lexer_consume_token(parser->lexer, 1);
1451
1452
11.3k
    return njs_parser_after(parser, current, parser->target, 0,
1453
695k
                            njs_parser_template_literal_expression);
1454
695k
}
1455
1456
1457
static njs_int_t
1458
njs_parser_template_literal_expression(njs_parser_t *parser,
1459
    njs_lexer_token_t *token, njs_queue_link_t *current)
1460
11.3k
{
1461
11.3k
    njs_int_t          ret;
1462
11.3k
    njs_parser_node_t  *template, *node, *parent;
1463
1464
11.3k
    if (parser->ret != NJS_OK) {
1465
53
        return njs_parser_failed(parser);
1466
53
    }
1467
1468
11.2k
    if (token->type != NJS_TOKEN_CLOSE_BRACE) {
1469
175
        njs_parser_syntax_error(parser, "Missing \"}\" in template expression");
1470
175
        return NJS_DONE;
1471
175
    }
1472
1473
11.0k
    parent = parser->target->right;
1474
11.0k
    template = parser->target->left;
1475
1476
11.0k
    if (template->token_type != NJS_TOKEN_TEMPLATE_LITERAL) {
1477
3.89k
        node = njs_parser_argument(parser, parser->node, parser->target->index);
1478
3.89k
        if (node == NULL) {
1479
0
            return NJS_ERROR;
1480
0
        }
1481
1482
3.89k
        parent->right = node;
1483
3.89k
        parent = node;
1484
1485
3.89k
        parser->target->index = njs_scope_temp_index(node->scope);
1486
3.89k
        if (njs_slow_path(parser->target->index == NJS_INDEX_ERROR)) {
1487
0
            return NJS_ERROR;
1488
0
        }
1489
1490
7.18k
    } else {
1491
7.18k
        ret = njs_parser_array_item(parser, template->left, parser->node);
1492
7.18k
        if (ret != NJS_OK) {
1493
0
            return NJS_ERROR;
1494
0
        }
1495
7.18k
    }
1496
1497
11.0k
    parser->target->right = parent;
1498
1499
11.0k
    parser->node = NULL;
1500
1501
11.0k
    njs_parser_next(parser, njs_parser_template_literal_string);
1502
1503
11.0k
    token->text.length = 0;
1504
11.0k
    token->text.start += 1;
1505
1506
11.0k
    return NJS_OK;
1507
11.0k
}
1508
1509
1510
static njs_int_t
1511
njs_parser_cover_parenthesized_expression(njs_parser_t *parser,
1512
    njs_lexer_token_t *token, njs_queue_link_t *current)
1513
0
{
1514
0
    switch (token->type) {
1515
0
    case NJS_TOKEN_CLOSE_PARENTHESIS:
1516
0
        (void) njs_parser_stack_pop(parser);
1517
0
        break;
1518
1519
0
    case NJS_TOKEN_ELLIPSIS:
1520
0
        njs_parser_next(parser, njs_parser_binding_identifier_pattern);
1521
0
        break;
1522
1523
0
    default:
1524
0
        parser->node = NULL;
1525
1526
0
        njs_parser_next(parser, njs_parser_expression);
1527
1528
0
        return njs_parser_after(parser, current, NULL, 0,
1529
0
                               njs_parser_cover_parenthesized_expression_after);
1530
0
    }
1531
1532
0
    njs_lexer_consume_token(parser->lexer, 1);
1533
1534
0
    return NJS_OK;
1535
0
}
1536
1537
1538
static njs_int_t
1539
njs_parser_binding_identifier_pattern(njs_parser_t *parser,
1540
    njs_lexer_token_t *token, njs_queue_link_t *current)
1541
0
{
1542
    /*
1543
     * BindingIdentifier )
1544
     * BindingPattern )
1545
     */
1546
1547
0
    switch (token->type) {
1548
1549
    /* BindingIdentifier */
1550
0
    case NJS_TOKEN_NAME:
1551
0
        njs_parser_next(parser, njs_parser_cover_parenthesized_expression_end);
1552
0
        break;
1553
1554
0
    case NJS_TOKEN_YIELD:
1555
0
        njs_parser_next(parser, njs_parser_cover_parenthesized_expression_end);
1556
0
        break;
1557
1558
0
    case NJS_TOKEN_AWAIT:
1559
0
        njs_parser_next(parser, njs_parser_cover_parenthesized_expression_end);
1560
0
        break;
1561
1562
    /* BindingPattern */
1563
0
    case NJS_TOKEN_OPEN_BRACKET:
1564
0
        njs_parser_next(parser, njs_parser_array_binding_pattern);
1565
1566
0
        njs_lexer_consume_token(parser->lexer, 1);
1567
0
        return njs_parser_after(parser, current, NULL, 0,
1568
0
                                njs_parser_cover_parenthesized_expression_end);
1569
1570
0
    case NJS_TOKEN_OPEN_BRACE:
1571
0
        njs_parser_next(parser, njs_parser_object_binding_pattern);
1572
1573
0
        njs_lexer_consume_token(parser->lexer, 1);
1574
0
        return njs_parser_after(parser, current, NULL, 0,
1575
0
                                njs_parser_cover_parenthesized_expression_end);
1576
1577
0
    default:
1578
0
        return NJS_ERROR;
1579
0
    }
1580
1581
0
    njs_lexer_consume_token(parser->lexer, 1);
1582
1583
0
    return NJS_OK;
1584
0
}
1585
1586
1587
static njs_int_t
1588
njs_parser_cover_parenthesized_expression_after(njs_parser_t *parser,
1589
    njs_lexer_token_t *token, njs_queue_link_t *current)
1590
0
{
1591
    /*
1592
     * )
1593
     * ,)
1594
     * , ... BindingIdentifier )
1595
     * , ... BindingPattern )
1596
     */
1597
1598
0
    if (token->type == NJS_TOKEN_CLOSE_PARENTHESIS) {
1599
0
        goto shift_stack;
1600
0
    }
1601
1602
0
    if (token->type != NJS_TOKEN_COMMA) {
1603
0
        return njs_parser_failed(parser);
1604
0
    }
1605
1606
0
    njs_lexer_consume_token(parser->lexer, 1);
1607
1608
0
    token = njs_lexer_token(parser->lexer, 0);
1609
0
    if (njs_slow_path(token == NULL)) {
1610
0
        return NJS_ERROR;
1611
0
    }
1612
1613
0
    if (token->type == NJS_TOKEN_CLOSE_PARENTHESIS) {
1614
0
        goto shift_stack;
1615
0
    }
1616
1617
0
    if(token->type != NJS_TOKEN_ELLIPSIS) {
1618
0
        return njs_parser_failed(parser);
1619
0
    }
1620
1621
0
    njs_lexer_consume_token(parser->lexer, 1);
1622
1623
0
    njs_parser_next(parser, njs_parser_binding_identifier_pattern);
1624
1625
0
    return NJS_OK;
1626
1627
0
shift_stack:
1628
1629
0
    njs_lexer_consume_token(parser->lexer, 1);
1630
1631
0
    return njs_parser_stack_pop(parser);
1632
0
}
1633
1634
1635
static njs_int_t
1636
njs_parser_cover_parenthesized_expression_end(njs_parser_t *parser,
1637
    njs_lexer_token_t *token, njs_queue_link_t *current)
1638
0
{
1639
0
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
1640
0
        return njs_parser_failed(parser);
1641
0
    }
1642
1643
0
    njs_lexer_consume_token(parser->lexer, 1);
1644
1645
0
    return njs_parser_stack_pop(parser);
1646
0
}
1647
1648
1649
/*
1650
 * 12.2.5 Array Initializer.
1651
 */
1652
static njs_int_t
1653
njs_parser_array_literal(njs_parser_t *parser, njs_lexer_token_t *token,
1654
    njs_queue_link_t *current)
1655
41.9k
{
1656
41.9k
    parser->target = parser->node;
1657
41.9k
    parser->node = NULL;
1658
1659
41.9k
    njs_parser_next(parser, njs_parser_array_element_list);
1660
1661
41.9k
    return NJS_OK;
1662
41.9k
}
1663
1664
1665
static njs_int_t
1666
njs_parser_array_element_list(njs_parser_t *parser, njs_lexer_token_t *token,
1667
    njs_queue_link_t *current)
1668
1.42M
{
1669
1.42M
    njs_parser_node_t  *array;
1670
1671
1.42M
    array = parser->target;
1672
1673
1.42M
    switch (token->type) {
1674
15.6k
    case NJS_TOKEN_CLOSE_BRACKET:
1675
15.6k
        njs_lexer_consume_token(parser->lexer, 1);
1676
1677
15.6k
        parser->node = array;
1678
1679
15.6k
        return njs_parser_stack_pop(parser);
1680
1681
446k
    case NJS_TOKEN_COMMA:
1682
446k
        njs_lexer_consume_token(parser->lexer, 1);
1683
1684
446k
        array->ctor = 1;
1685
446k
        array->u.length++;
1686
1687
446k
        return NJS_OK;
1688
1689
2
    case NJS_TOKEN_ELLIPSIS:
1690
#if 0
1691
        njs_lexer_consume_token(parser->lexer, 1);
1692
1693
        njs_parser_next(parser, njs_parser_assignment_expression);
1694
1695
        return njs_parser_after(parser, current, array, 0,
1696
                                njs_parser_array_spread_element);
1697
#else
1698
2
        (void) njs_parser_array_spread_element;
1699
2
        return njs_parser_failed(parser);
1700
0
#endif
1701
1702
967k
    default:
1703
967k
        break;
1704
1.42M
    }
1705
1706
967k
    njs_parser_next(parser, njs_parser_assignment_expression);
1707
1708
967k
    return njs_parser_after(parser, current, array, 0, njs_parser_array_after);
1709
1.42M
}
1710
1711
1712
static njs_int_t
1713
njs_parser_array_after(njs_parser_t *parser, njs_lexer_token_t *token,
1714
    njs_queue_link_t *current)
1715
941k
{
1716
941k
    njs_int_t  ret;
1717
1718
941k
    if (parser->ret != NJS_OK) {
1719
363
        return njs_parser_failed(parser);
1720
363
    }
1721
1722
941k
    ret = njs_parser_array_item(parser, parser->target, parser->node);
1723
941k
    if (ret != NJS_OK) {
1724
0
        return NJS_ERROR;
1725
0
    }
1726
1727
941k
    switch (token->type) {
1728
933k
    case NJS_TOKEN_COMMA:
1729
933k
        njs_lexer_consume_token(parser->lexer, 1);
1730
1731
        /* Fall through. */
1732
1733
941k
    case NJS_TOKEN_CLOSE_BRACKET:
1734
941k
        njs_parser_next(parser, njs_parser_array_element_list);
1735
941k
        break;
1736
1737
165
    default:
1738
165
        return njs_parser_failed(parser);
1739
941k
    }
1740
1741
941k
    return NJS_OK;
1742
941k
}
1743
1744
1745
static njs_int_t
1746
njs_parser_array_spread_element(njs_parser_t *parser, njs_lexer_token_t *token,
1747
    njs_queue_link_t *current)
1748
0
{
1749
0
    if (parser->ret != NJS_OK || token->type != NJS_TOKEN_CLOSE_BRACKET) {
1750
0
        return njs_parser_failed(parser);
1751
0
    }
1752
1753
0
    njs_lexer_consume_token(parser->lexer, 1);
1754
1755
0
    parser->node = parser->target;
1756
1757
0
    return njs_parser_stack_pop(parser);
1758
0
}
1759
1760
1761
/*
1762
 * 12.2.6 Object Initializer.
1763
 */
1764
static njs_int_t
1765
njs_parser_object_literal(njs_parser_t *parser, njs_lexer_token_t *token,
1766
    njs_queue_link_t *current)
1767
24.2k
{
1768
24.2k
    njs_parser_node_t *node;
1769
1770
24.2k
    node = njs_parser_node_new(parser, 0);
1771
24.2k
    if (node == NULL) {
1772
0
        return NJS_ERROR;
1773
0
    }
1774
1775
24.2k
    node->left = parser->node;
1776
1777
24.2k
    parser->node = NULL;
1778
24.2k
    parser->target = node;
1779
1780
24.2k
    njs_parser_next(parser, njs_parser_property_definition_list);
1781
1782
24.2k
    return njs_parser_after(parser, current, node, 1,
1783
24.2k
                            njs_parser_object_literal_after);
1784
24.2k
}
1785
1786
1787
static njs_int_t
1788
njs_parser_object_literal_after(njs_parser_t *parser, njs_lexer_token_t *token,
1789
    njs_queue_link_t *current)
1790
16.3k
{
1791
16.3k
    if (token->type == NJS_TOKEN_COMMA) {
1792
0
        njs_lexer_consume_token(parser->lexer, 1);
1793
1794
0
        token = njs_lexer_token(parser->lexer, 1);
1795
0
        if (token == NULL) {
1796
0
            return NJS_ERROR;
1797
0
        }
1798
0
    }
1799
1800
16.3k
    if (token->type != NJS_TOKEN_CLOSE_BRACE) {
1801
279
        return njs_parser_failed(parser);
1802
279
    }
1803
1804
16.0k
    njs_lexer_consume_token(parser->lexer, 1);
1805
1806
16.0k
    parser->node = parser->target->left;
1807
1808
16.0k
    njs_parser_node_free(parser, parser->target);
1809
16.0k
    parser->target = NULL;
1810
1811
16.0k
    return njs_parser_stack_pop(parser);
1812
16.3k
}
1813
1814
1815
static njs_int_t
1816
njs_parser_property_definition_list(njs_parser_t *parser,
1817
    njs_lexer_token_t *token, njs_queue_link_t *current)
1818
24.2k
{
1819
24.2k
    njs_parser_next(parser, njs_parser_property_definition);
1820
1821
24.2k
    return njs_parser_after(parser, current, parser->target, 1,
1822
24.2k
                            njs_parser_property_definition_list_after);
1823
24.2k
}
1824
1825
1826
static njs_int_t
1827
njs_parser_property_definition_list_after(njs_parser_t *parser,
1828
    njs_lexer_token_t *token, njs_queue_link_t *current)
1829
24.4k
{
1830
24.4k
    if (token->type != NJS_TOKEN_COMMA) {
1831
16.3k
        return njs_parser_stack_pop(parser);
1832
16.3k
    }
1833
1834
8.17k
    njs_lexer_consume_token(parser->lexer, 1);
1835
1836
8.17k
    njs_parser_next(parser, njs_parser_property_definition);
1837
1838
8.17k
    return njs_parser_after(parser, current, parser->target, 1,
1839
24.4k
                            njs_parser_property_definition_list_after);
1840
24.4k
}
1841
1842
1843
njs_inline njs_parser_node_t *
1844
njs_parser_property_name_node(njs_parser_t *parser, njs_lexer_token_t *token)
1845
16.0k
{
1846
16.0k
    njs_int_t          ret;
1847
16.0k
    njs_parser_node_t  *property;
1848
1849
16.0k
    if (token->type == NJS_TOKEN_NUMBER) {
1850
1.85k
        property = njs_parser_node_new(parser, NJS_TOKEN_NUMBER);
1851
1.85k
        if (property != NULL) {
1852
1.85k
             njs_set_number(&property->u.value, token->number);
1853
1.85k
        }
1854
1855
14.1k
    } else if (token->type == NJS_TOKEN_ESCAPE_STRING) {
1856
1.25k
        property = njs_parser_node_new(parser, NJS_TOKEN_STRING);
1857
1858
1.25k
        if (property != NULL) {
1859
1.25k
            ret = njs_parser_escape_string_create(parser, token,
1860
1.25k
                                                  &property->u.value);
1861
1.25k
            if (ret != NJS_TOKEN_STRING) {
1862
985
                return NULL;
1863
985
            }
1864
1.25k
        }
1865
1866
12.9k
    } else {
1867
12.9k
        property = njs_parser_node_string(parser->vm, token, parser);
1868
12.9k
    }
1869
1870
15.0k
    if (property != NULL) {
1871
15.0k
        property->token_line = token->line;
1872
15.0k
    }
1873
1874
15.0k
    return property;
1875
16.0k
}
1876
1877
1878
njs_inline njs_int_t
1879
njs_parser_property_name(njs_parser_t *parser, njs_queue_link_t *current,
1880
    unsigned consume)
1881
10.2k
{
1882
10.2k
    njs_lexer_token_t  *token;
1883
10.2k
    njs_parser_node_t  *property;
1884
1885
10.2k
    if (consume > 1) {
1886
10.2k
        token = njs_lexer_token(parser->lexer, 0);
1887
10.2k
        if (token == NULL) {
1888
0
            return NJS_ERROR;
1889
0
        }
1890
1891
10.2k
        property = njs_parser_property_name_node(parser, token);
1892
10.2k
        if (property == NULL) {
1893
132
            return NJS_ERROR;
1894
132
        }
1895
1896
10.1k
        parser->target->right = property;
1897
10.1k
        parser->target->index = (token->type == NJS_TOKEN_NAME);
1898
10.1k
    }
1899
1900
10.1k
    njs_lexer_consume_token(parser->lexer, consume);
1901
1902
10.1k
    parser->node = NULL;
1903
1904
10.1k
    njs_parser_next(parser, njs_parser_assignment_expression);
1905
1906
10.1k
    return njs_parser_after(parser, current, parser->target, 1,
1907
10.2k
                            njs_parser_property_definition_after);
1908
10.2k
}
1909
1910
1911
static njs_int_t
1912
njs_parser_property_definition_ident(njs_parser_t *parser,
1913
    njs_lexer_token_t *token, njs_parser_node_t *temp)
1914
9.51k
{
1915
9.51k
    temp->right = njs_parser_node_string(parser->vm, token, parser);
1916
9.51k
    if (temp->right == NULL) {
1917
0
        return NJS_ERROR;
1918
0
    }
1919
1920
9.51k
    temp->right->index = NJS_TOKEN_OPEN_BRACKET;
1921
1922
9.51k
    parser->node = njs_parser_reference(parser, token);
1923
9.51k
    if (parser->node == NULL) {
1924
0
        return NJS_ERROR;
1925
0
    }
1926
1927
9.51k
    njs_lexer_consume_token(parser->lexer, 1);
1928
1929
9.51k
    token = njs_lexer_token(parser->lexer, 0);
1930
9.51k
    if (token == NULL) {
1931
0
        return NJS_ERROR;
1932
0
    }
1933
1934
    /* CoverInitializedName */
1935
9.51k
    if (token->type == NJS_TOKEN_ASSIGNMENT) {
1936
0
        return njs_parser_not_supported(parser, token);
1937
0
    }
1938
1939
9.51k
    njs_parser_next(parser, njs_parser_property_definition_after);
1940
1941
9.51k
    return NJS_OK;
1942
9.51k
}
1943
1944
1945
static njs_int_t
1946
njs_parser_property_definition(njs_parser_t *parser, njs_lexer_token_t *token,
1947
    njs_queue_link_t *current)
1948
32.4k
{
1949
32.4k
    njs_str_t          *name;
1950
32.4k
    njs_token_type_t   accessor;
1951
32.4k
    njs_lexer_token_t  *next;
1952
32.4k
    njs_parser_node_t  *temp, *property;
1953
1954
32.4k
    temp = parser->target;
1955
1956
32.4k
    switch (token->type) {
1957
2.59k
    case NJS_TOKEN_CLOSE_BRACE:
1958
2.59k
        return njs_parser_stack_pop(parser);
1959
1960
    /* PropertyName */
1961
6.11k
    case NJS_TOKEN_STRING:
1962
6.74k
    case NJS_TOKEN_ESCAPE_STRING:
1963
8.56k
    case NJS_TOKEN_NUMBER:
1964
8.56k
        next = njs_lexer_peek_token(parser->lexer, token, 0);
1965
8.56k
        if (next == NULL) {
1966
0
            return NJS_ERROR;
1967
0
        }
1968
1969
8.56k
        if (next->type == NJS_TOKEN_COLON) {
1970
6.36k
            return njs_parser_property_name(parser, current, 2);
1971
1972
6.36k
        } else if (next->type == NJS_TOKEN_OPEN_PARENTHESIS) {
1973
2.20k
            goto method_definition;
1974
2.20k
        }
1975
1976
0
        njs_lexer_consume_token(parser->lexer, 1);
1977
1978
0
        return njs_parser_failed(parser);
1979
1980
    /* ComputedPropertyName */
1981
261
    case NJS_TOKEN_OPEN_BRACKET:
1982
261
        njs_lexer_consume_token(parser->lexer, 1);
1983
1984
261
        njs_parser_next(parser, njs_parser_assignment_expression);
1985
1986
261
        return njs_parser_after(parser, current, temp, 1,
1987
8.56k
                                njs_parser_computed_property_name_after);
1988
1989
296
    case NJS_TOKEN_ELLIPSIS:
1990
296
        return njs_parser_not_supported(parser, token);
1991
1992
3.24k
    case NJS_TOKEN_ASYNC:
1993
3.24k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
1994
3.24k
        if (token == NULL) {
1995
0
            return NJS_ERROR;
1996
0
        }
1997
1998
3.24k
        if (token->type == NJS_TOKEN_OPEN_BRACKET) {
1999
2.61k
            njs_lexer_consume_token(parser->lexer, 2);
2000
2001
2.61k
            njs_parser_next(parser, njs_parser_assignment_expression);
2002
2003
2.61k
            return njs_parser_after(parser, current, temp, 1,
2004
2.61k
                                    njs_parser_computed_property_async_after);
2005
2.61k
        }
2006
2007
625
        if (token->type == NJS_TOKEN_COLON) {
2008
131
            return njs_parser_property_name(parser, current, 2);
2009
131
        }
2010
2011
494
        if (!njs_lexer_token_is_identifier_name(token)) {
2012
132
            return njs_parser_failed(parser);
2013
132
        }
2014
2015
362
        next = njs_lexer_peek_token(parser->lexer, token, 0);
2016
362
        if (next == NULL) {
2017
0
            return NJS_ERROR;
2018
0
        }
2019
2020
362
        if (next->type == NJS_TOKEN_OPEN_PARENTHESIS) {
2021
362
            goto method_definition;
2022
362
        }
2023
2024
0
        njs_lexer_consume_token(parser->lexer, 1);
2025
2026
0
        return njs_parser_failed(parser);
2027
2028
17.4k
    default:
2029
17.4k
        if (!njs_lexer_token_is_identifier_name(token)) {
2030
408
            return njs_parser_reject(parser);
2031
408
        }
2032
2033
17.0k
        next = njs_lexer_peek_token(parser->lexer, token, 0);
2034
17.0k
        if (next == NULL) {
2035
0
            return NJS_ERROR;
2036
0
        }
2037
2038
        /* PropertyName */
2039
17.0k
        if (next->type == NJS_TOKEN_COLON) {
2040
3.78k
            return njs_parser_property_name(parser, current, 2);
2041
2042
        /* MethodDefinition */
2043
13.2k
        } else if (next->type == NJS_TOKEN_OPEN_PARENTHESIS) {
2044
953
            goto method_definition;
2045
2046
12.3k
        } else if (njs_lexer_token_is_reserved(token)) {
2047
35
            njs_lexer_consume_token(parser->lexer, 1);
2048
2049
35
            return njs_parser_failed(parser);
2050
35
        }
2051
2052
12.2k
        name = &token->text;
2053
2054
12.2k
        if (name->length == 3 && (memcmp(name->start, "get", 3) == 0
2055
1.96k
                                  || memcmp(name->start, "set", 3) == 0))
2056
2.81k
        {
2057
2.81k
            accessor = (name->start[0] == 'g') ? NJS_TOKEN_PROPERTY_GETTER
2058
2.81k
                                               : NJS_TOKEN_PROPERTY_SETTER;
2059
2060
2.81k
            temp->right = (njs_parser_node_t *) (uintptr_t) accessor;
2061
2062
2.81k
            njs_parser_next(parser, njs_parser_get_set);
2063
2064
2.81k
            return NJS_OK;
2065
2.81k
        }
2066
2067
9.47k
        return njs_parser_property_definition_ident(parser, token, temp);
2068
32.4k
    }
2069
2070
3.52k
method_definition:
2071
2072
3.52k
    property = njs_parser_property_name_node(parser, token);
2073
3.52k
    if (property == NULL) {
2074
254
        return NJS_ERROR;
2075
254
    }
2076
2077
3.26k
    temp->right = property;
2078
3.26k
    temp->right->index = NJS_TOKEN_OPEN_BRACKET;
2079
2080
3.26k
    njs_parser_next(parser, njs_parser_method_definition);
2081
2082
3.26k
    return njs_parser_after(parser, current, temp, 1,
2083
3.52k
                            njs_parser_property_definition_after);
2084
3.52k
}
2085
2086
2087
static njs_int_t
2088
njs_parser_property_definition_after(njs_parser_t *parser,
2089
    njs_lexer_token_t *token, njs_queue_link_t *current)
2090
21.5k
{
2091
21.5k
    njs_int_t          ret;
2092
21.5k
    njs_str_t          name;
2093
21.5k
    njs_bool_t         proto_init;
2094
21.5k
    njs_parser_node_t  *property, *temp;
2095
2096
21.5k
    static const njs_str_t  proto_string = njs_str("__proto__");
2097
2098
21.5k
    temp = parser->target;
2099
21.5k
    property = temp->right;
2100
2101
21.5k
    proto_init = 0;
2102
2103
21.5k
    if (property->index != NJS_TOKEN_OPEN_BRACKET
2104
9.84k
        && njs_is_string(&property->u.value))
2105
9.71k
    {
2106
9.71k
        njs_string_get(parser->vm, &property->u.value, &name);
2107
2108
9.71k
        if (njs_slow_path(njs_strstr_eq(&name, &proto_string))) {
2109
59
            if (temp->token_type == NJS_TOKEN_PROTO_INIT) {
2110
0
                njs_parser_syntax_error(parser,
2111
0
                                   "Duplicate __proto__ fields are not allowed "
2112
0
                                   "in object literals");
2113
0
                return NJS_ERROR;
2114
0
            }
2115
2116
59
            temp->token_type = NJS_TOKEN_PROTO_INIT;
2117
59
            proto_init = 1;
2118
59
        }
2119
9.71k
    }
2120
2121
21.5k
    if (property->index != 0) {
2122
11.7k
        property->index = 0;
2123
11.7k
    }
2124
2125
21.5k
    ret = njs_parser_object_property(parser, temp->left, property,
2126
21.5k
                                     parser->node, proto_init);
2127
21.5k
    if (ret != NJS_OK) {
2128
0
        return NJS_ERROR;
2129
0
    }
2130
2131
21.5k
    temp->right = NULL;
2132
2133
21.5k
    return njs_parser_stack_pop(parser);
2134
21.5k
}
2135
2136
2137
static njs_int_t
2138
njs_parser_computed_property_name_after(njs_parser_t *parser,
2139
    njs_lexer_token_t *token, njs_queue_link_t *current)
2140
261
{
2141
261
    return njs_parser_computed_property_name_handler(parser, token, current, 0);
2142
261
}
2143
2144
2145
static njs_int_t
2146
njs_parser_computed_property_async_after(njs_parser_t *parser,
2147
    njs_lexer_token_t *token, njs_queue_link_t *current)
2148
2.61k
{
2149
2.61k
    return njs_parser_computed_property_name_handler(parser, token, current, 1);
2150
2.61k
}
2151
2152
2153
static njs_int_t
2154
njs_parser_computed_property_name_handler(njs_parser_t *parser,
2155
    njs_lexer_token_t *token, njs_queue_link_t *current, njs_bool_t async)
2156
2.87k
{
2157
2.87k
    njs_token_type_t   type;
2158
2.87k
    njs_parser_node_t  *expr, *target;
2159
2160
2.87k
    if (token->type != NJS_TOKEN_CLOSE_BRACKET) {
2161
2.48k
        return njs_parser_failed(parser);
2162
2.48k
    }
2163
2164
393
    njs_lexer_consume_token(parser->lexer, 1);
2165
2166
393
    token = njs_lexer_token(parser->lexer, 0);
2167
393
    if (token == NULL) {
2168
0
        return NJS_ERROR;
2169
0
    }
2170
2171
393
    target = parser->target;
2172
2173
    /*
2174
     * For further identification.
2175
     * In njs_parser_property_definition_after() index will be reset to zero.
2176
     */
2177
393
    parser->node->index = NJS_TOKEN_OPEN_BRACKET;
2178
2179
393
    target->right = parser->node;
2180
2181
393
    if (!async && token->type == NJS_TOKEN_COLON) {
2182
0
        return njs_parser_property_name(parser, current, 1);
2183
2184
    /* MethodDefinition */
2185
393
    } else if (token->type == NJS_TOKEN_OPEN_PARENTHESIS) {
2186
261
        type = (async) ? NJS_TOKEN_ASYNC_FUNCTION : NJS_TOKEN_FUNCTION;
2187
2188
261
        expr = njs_parser_node_new(parser, type);
2189
261
        if (expr == NULL) {
2190
0
            return NJS_ERROR;
2191
0
        }
2192
2193
261
        expr->token_line = token->line;
2194
2195
261
        parser->node = expr;
2196
2197
261
        njs_lexer_consume_token(parser->lexer, 1);
2198
261
        njs_parser_next(parser, njs_parser_function_lambda);
2199
2200
261
        return njs_parser_after(parser, current, parser->target, 1,
2201
261
                                njs_parser_property_definition_after);
2202
261
    }
2203
2204
132
    return njs_parser_failed(parser);
2205
393
}
2206
2207
2208
static njs_int_t
2209
njs_parser_initializer(njs_parser_t *parser, njs_lexer_token_t *token,
2210
    njs_queue_link_t *current)
2211
22.5k
{
2212
22.5k
    njs_parser_node_t  *node;
2213
2214
22.5k
    if (token->type != NJS_TOKEN_ASSIGNMENT) {
2215
0
        return njs_parser_failed(parser);
2216
0
    }
2217
2218
22.5k
    njs_lexer_consume_token(parser->lexer, 1);
2219
2220
22.5k
    node = parser->node;
2221
2222
22.5k
    parser->node = NULL;
2223
2224
22.5k
    njs_parser_next(parser, njs_parser_assignment_expression);
2225
2226
22.5k
    return njs_parser_after(parser, current, node, 1,
2227
22.5k
                            njs_parser_initializer_after);
2228
22.5k
}
2229
2230
2231
static njs_int_t
2232
njs_parser_initializer_after(njs_parser_t *parser, njs_lexer_token_t *token,
2233
    njs_queue_link_t *current)
2234
32.0k
{
2235
32.0k
    njs_parser_node_t  *stmt;
2236
2237
32.0k
    stmt = njs_parser_node_new(parser, NJS_TOKEN_STATEMENT);
2238
32.0k
    if (stmt == NULL) {
2239
0
        return NJS_ERROR;
2240
0
    }
2241
2242
32.0k
    stmt->left = NULL;
2243
32.0k
    stmt->right = parser->target;
2244
2245
32.0k
    parser->target->right = parser->node;
2246
32.0k
    parser->node = stmt;
2247
2248
32.0k
    return njs_parser_stack_pop(parser);
2249
32.0k
}
2250
2251
2252
static njs_int_t
2253
njs_parser_initializer_assign(njs_parser_t *parser, njs_token_type_t type)
2254
32.5k
{
2255
32.5k
    njs_parser_node_t  *assign;
2256
2257
32.5k
    assign = njs_parser_node_new(parser, type);
2258
32.5k
    if (assign == NULL) {
2259
0
        return NJS_ERROR;
2260
0
    }
2261
2262
32.5k
    assign->u.operation = NJS_VMCODE_MOVE;
2263
32.5k
    assign->left = parser->node;
2264
2265
    /* assign->right in njs_parser_initializer_after. */
2266
2267
32.5k
    parser->node = assign;
2268
2269
32.5k
    return NJS_OK;
2270
32.5k
}
2271
2272
2273
static njs_int_t
2274
njs_parser_tagged_template_literal_after(njs_parser_t *parser,
2275
    njs_lexer_token_t *token, njs_queue_link_t *current)
2276
673k
{
2277
673k
    return njs_parser_stack_pop(parser);
2278
673k
}
2279
2280
/*
2281
 * 12.3 Left-Hand-Side Expressions.
2282
 */
2283
static njs_int_t
2284
njs_parser_property(njs_parser_t *parser, njs_lexer_token_t *token,
2285
    njs_queue_link_t *current)
2286
3.55M
{
2287
3.55M
    njs_parser_node_t  *node, *prop_node;
2288
2289
    /*
2290
     * [ Expression ]
2291
     * . IdentifierName
2292
     * TemplateLiteral
2293
     */
2294
2295
3.55M
    switch (token->type) {
2296
58.6k
    case NJS_TOKEN_OPEN_BRACKET:
2297
58.6k
        node = njs_parser_node_new(parser, NJS_TOKEN_PROPERTY);
2298
58.6k
        if (node == NULL) {
2299
0
            return NJS_ERROR;
2300
0
        }
2301
2302
58.6k
        node->u.operation = NJS_VMCODE_PROPERTY_GET;
2303
58.6k
        node->left = parser->node;
2304
58.6k
        node->token_line = token->line;
2305
2306
58.6k
        parser->node = NULL;
2307
2308
58.6k
        njs_lexer_consume_token(parser->lexer, 1);
2309
2310
58.6k
        njs_parser_next(parser, njs_parser_expression);
2311
2312
58.6k
        return njs_parser_after(parser, current, node, 1,
2313
58.6k
                                njs_parser_member_expression_bracket);
2314
2315
130k
    case NJS_TOKEN_DOT:
2316
130k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
2317
130k
        if (token == NULL) {
2318
0
            return NJS_ERROR;
2319
0
        }
2320
2321
130k
        if (njs_lexer_token_is_identifier_name(token)) {
2322
129k
            node = njs_parser_node_new(parser, NJS_TOKEN_PROPERTY);
2323
129k
            if (node == NULL) {
2324
0
                return NJS_ERROR;
2325
0
            }
2326
2327
129k
            node->u.operation = NJS_VMCODE_PROPERTY_ATOM_GET;
2328
129k
            node->token_line = token->line;
2329
2330
129k
            prop_node = njs_parser_node_string(parser->vm, token, parser);
2331
129k
            if (prop_node == NULL) {
2332
0
                return NJS_ERROR;
2333
0
            }
2334
2335
129k
            prop_node->token_line = token->line;
2336
2337
129k
            node->left = parser->node;
2338
129k
            node->right = prop_node;
2339
2340
129k
            parser->node = node;
2341
2342
129k
            njs_lexer_consume_token(parser->lexer, 2);
2343
129k
            return NJS_AGAIN;
2344
129k
        }
2345
2346
568
        njs_lexer_consume_token(parser->lexer, 1);
2347
2348
568
        return NJS_DECLINED;
2349
2350
674k
    case NJS_TOKEN_GRAVE:
2351
674k
        node = njs_parser_create_call(parser, parser->node, 0);
2352
674k
        if (node == NULL) {
2353
0
            return NJS_ERROR;
2354
0
        }
2355
2356
674k
        node->token_line = token->line;
2357
2358
674k
        parser->node = node;
2359
2360
674k
        njs_parser_next(parser, njs_parser_template_literal);
2361
2362
674k
        return njs_parser_after(parser, current, node, 1,
2363
674k
                                njs_parser_tagged_template_literal_after);
2364
2365
2.68M
    default:
2366
2.68M
        return NJS_DONE;
2367
3.55M
    }
2368
2369
0
    return NJS_OK;
2370
3.55M
}
2371
2372
2373
static njs_int_t
2374
njs_parser_member_expression(njs_parser_t *parser, njs_lexer_token_t *token,
2375
    njs_queue_link_t *current)
2376
2.86M
{
2377
2.86M
    njs_int_t  ret;
2378
2379
    /*
2380
     * PrimaryExpression
2381
     * MemberExpression [ Expression ]
2382
     * MemberExpression . IdentifierName
2383
     * MemberExpression TemplateLiteral
2384
     * SuperProperty
2385
     * MetaProperty
2386
     * new MemberExpression Arguments
2387
     */
2388
2389
2.86M
    switch (token->type) {
2390
    /* SuperProperty */
2391
259
    case NJS_TOKEN_SUPER:
2392
        /* TODO: By specification. */
2393
259
        (void) njs_parser_super_property;
2394
259
        return njs_parser_not_supported(parser, token);
2395
2396
    /* MetaProperty */
2397
251
    case NJS_TOKEN_IMPORT:
2398
        /* TODO: By specification. */
2399
251
        (void) njs_parser_member_expression_import;
2400
251
        return njs_parser_not_supported(parser, token);
2401
2402
5.21k
    case NJS_TOKEN_NEW:
2403
5.21k
        njs_lexer_consume_token(parser->lexer, 1);
2404
2405
5.21k
        njs_parser_next(parser, njs_parser_member_expression_new);
2406
5.21k
        break;
2407
2408
2.86M
    default:
2409
2.86M
        ret = njs_parser_primary_expression_test(parser, token, current);
2410
2411
2.86M
        if (ret != NJS_OK) {
2412
2.43M
            if (ret == NJS_DONE) {
2413
2.41M
                njs_parser_next(parser, njs_parser_member_expression_next);
2414
2.41M
                return NJS_OK;
2415
2.41M
            }
2416
2417
19.3k
            if (njs_is_error(&parser->vm->exception)) {
2418
5.28k
                return NJS_DONE;
2419
5.28k
            }
2420
2421
14.1k
            return ret;
2422
19.3k
        }
2423
2424
425k
        break;
2425
2.86M
    }
2426
2427
431k
    return njs_parser_after(parser, current, NULL, 1,
2428
2.86M
                            njs_parser_member_expression_next);
2429
2.86M
}
2430
2431
2432
static njs_int_t
2433
njs_parser_member_expression_next(njs_parser_t *parser,
2434
    njs_lexer_token_t *token, njs_queue_link_t *current)
2435
3.48M
{
2436
3.48M
    njs_int_t  ret;
2437
2438
    /*
2439
     * [ Expression ]
2440
     * . IdentifierName
2441
     * TemplateLiteral
2442
     */
2443
2444
3.48M
    ret = njs_parser_property(parser, token, current);
2445
2446
3.48M
    switch (ret) {
2447
127k
    case NJS_AGAIN:
2448
127k
        return NJS_OK;
2449
2450
2.63M
    case NJS_DONE:
2451
2.63M
        return njs_parser_stack_pop(parser);
2452
2453
530
    case NJS_DECLINED:
2454
530
        return njs_parser_failed(parser);
2455
2456
716k
    default:
2457
716k
        break;
2458
3.48M
    }
2459
2460
716k
    return njs_parser_after(parser, current, NULL, 1,
2461
3.48M
                            njs_parser_member_expression_next);
2462
3.48M
}
2463
2464
2465
static njs_int_t
2466
njs_parser_member_expression_bracket(njs_parser_t *parser,
2467
    njs_lexer_token_t *token, njs_queue_link_t *current)
2468
52.8k
{
2469
52.8k
    if (token->type != NJS_TOKEN_CLOSE_BRACKET) {
2470
2.96k
        return njs_parser_failed(parser);
2471
2.96k
    }
2472
2473
49.9k
    njs_lexer_consume_token(parser->lexer, 1);
2474
2475
49.9k
    return njs_parser_right_link_pop(parser);
2476
52.8k
}
2477
2478
2479
static njs_int_t
2480
njs_parser_member_expression_new_next(njs_parser_t *parser,
2481
    njs_lexer_token_t *token, njs_queue_link_t *current)
2482
7.26k
{
2483
7.26k
    njs_int_t  ret;
2484
2485
    /*
2486
     * PrimaryExpression
2487
     * SuperProperty
2488
     * MetaProperty
2489
     * Arguments
2490
     */
2491
2492
7.26k
    switch (token->type) {
2493
    /* SuperProperty */
2494
2.50k
    case NJS_TOKEN_SUPER:
2495
2.50k
        (void) njs_parser_super_property;
2496
2.50k
        return njs_parser_not_supported(parser, token);
2497
2498
    /* MetaProperty */
2499
1.27k
    case NJS_TOKEN_IMPORT:
2500
1.27k
        (void) njs_parser_member_expression_import;
2501
1.27k
        return njs_parser_not_supported(parser, token);
2502
2503
3.47k
    default:
2504
3.47k
        ret = njs_parser_primary_expression_test(parser, token, current);
2505
2506
3.47k
        if (ret != NJS_OK) {
2507
3.17k
            if (ret == NJS_DONE) {
2508
2.93k
                njs_parser_next(parser, njs_parser_member_expression_next);
2509
2.93k
                return NJS_OK;
2510
2.93k
            }
2511
2512
242
            return ret;
2513
3.17k
        }
2514
2515
300
        return njs_parser_after(parser, current, NULL, 1,
2516
7.26k
                                njs_parser_member_expression_next);
2517
7.26k
    }
2518
7.26k
}
2519
2520
2521
static njs_int_t
2522
njs_parser_super_property(njs_parser_t *parser, njs_lexer_token_t *token,
2523
    njs_queue_link_t *current)
2524
0
{
2525
    /*
2526
     * [ Expression ]
2527
     * . IdentifierName
2528
     */
2529
2530
0
    if (token->type == NJS_TOKEN_OPEN_BRACKET) {
2531
0
        parser->node = NULL;
2532
2533
0
        njs_parser_next(parser, njs_parser_expression);
2534
2535
0
        return njs_parser_after(parser, current, NULL, 1,
2536
0
                                njs_parser_close_bracked);
2537
0
    }
2538
2539
0
    if (token->type != NJS_TOKEN_DOT) {
2540
0
        return njs_parser_failed(parser);
2541
0
    }
2542
2543
0
    njs_lexer_consume_token(parser->lexer, 1);
2544
2545
0
    token = njs_lexer_token(parser->lexer, 0);
2546
0
    if (token == NULL) {
2547
0
        return NJS_ERROR;
2548
0
    }
2549
2550
0
    if (!njs_lexer_token_is_identifier_name(token)) {
2551
0
        return njs_parser_failed(parser);
2552
0
    }
2553
2554
0
    return njs_parser_stack_pop(parser);
2555
0
}
2556
2557
2558
static njs_int_t
2559
njs_parser_member_expression_import(njs_parser_t *parser,
2560
    njs_lexer_token_t *token, njs_queue_link_t *current)
2561
0
{
2562
    /*
2563
     * import . meta
2564
     */
2565
2566
0
    if (token->type != NJS_TOKEN_DOT) {
2567
0
        return njs_parser_failed(parser);
2568
0
    }
2569
2570
0
    njs_lexer_consume_token(parser->lexer, 1);
2571
2572
0
    token = njs_lexer_token(parser->lexer, 0);
2573
0
    if (token == NULL) {
2574
0
        return NJS_ERROR;
2575
0
    }
2576
2577
0
    if (token->type != NJS_TOKEN_META) {
2578
0
        return njs_parser_failed(parser);
2579
0
    }
2580
2581
0
    njs_lexer_consume_token(parser->lexer, 1);
2582
2583
0
    return njs_parser_stack_pop(parser);
2584
0
}
2585
2586
2587
static njs_int_t
2588
njs_parser_member_expression_new(njs_parser_t *parser,
2589
    njs_lexer_token_t *token, njs_queue_link_t *current)
2590
7.26k
{
2591
    /*
2592
     * new MemberExpression Arguments
2593
     * new . target
2594
     */
2595
2596
7.26k
    if (token->type != NJS_TOKEN_DOT) {
2597
7.26k
        njs_parser_next(parser, njs_parser_member_expression_new_next);
2598
2599
7.26k
        return njs_parser_after(parser, current, NULL, 1,
2600
7.26k
                                njs_parser_member_expression_new_after);
2601
7.26k
    }
2602
2603
    /* njs_lexer_consume_token(parser->lexer, 1); */
2604
2605
0
    token = njs_lexer_token(parser->lexer, 0);
2606
0
    if (token == NULL) {
2607
0
        return NJS_ERROR;
2608
0
    }
2609
2610
0
    if (token->type != NJS_TOKEN_TARGET) {
2611
0
        return njs_parser_failed(parser);
2612
0
    }
2613
2614
    /* njs_lexer_consume_token(parser->lexer, 1); */
2615
2616
    /* return njs_parser_stack_pop(parser); */
2617
0
    return njs_parser_not_supported(parser, token);
2618
0
}
2619
2620
2621
static njs_int_t
2622
njs_parser_member_expression_new_after(njs_parser_t *parser,
2623
    njs_lexer_token_t *token, njs_queue_link_t *current)
2624
3.45k
{
2625
3.45k
    njs_parser_node_t  *func;
2626
2627
3.45k
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS) {
2628
1.68k
        parser->node = njs_parser_create_call(parser, parser->node, 1);
2629
1.68k
        if (parser->node == NULL) {
2630
0
            return NJS_ERROR;
2631
0
        }
2632
2633
1.68k
        parser->node->token_line = token->line;
2634
2635
1.68k
        return njs_parser_stack_pop(parser);
2636
1.68k
    }
2637
2638
1.77k
    func = njs_parser_create_call(parser, parser->node, 1);
2639
1.77k
    if (func == NULL) {
2640
0
        return NJS_ERROR;
2641
0
    }
2642
2643
1.77k
    return njs_parser_call_arguments(parser, token, current, func,
2644
1.77k
                                     njs_parser_member_expression_new_args);
2645
1.77k
}
2646
2647
2648
static njs_int_t
2649
njs_parser_member_expression_new_args(njs_parser_t *parser,
2650
    njs_lexer_token_t *token, njs_queue_link_t *current)
2651
1.55k
{
2652
1.55k
    parser->node = parser->target;
2653
2654
1.55k
    return njs_parser_stack_pop(parser);
2655
1.55k
}
2656
2657
2658
static njs_parser_node_t *
2659
njs_parser_property_ref(njs_parser_node_t *node)
2660
1.52M
{
2661
1.52M
    if (node == NULL) {
2662
0
        return NULL;
2663
0
    }
2664
2665
1.52M
    if (node->token_type == NJS_TOKEN_PROPERTY) {
2666
114k
        node->token_type = NJS_TOKEN_PROPERTY_REF;
2667
114k
    }
2668
2669
1.52M
    if (node->token_type == NJS_TOKEN_PROPERTY_REF) {
2670
114k
        return node;
2671
114k
    }
2672
2673
1.40M
    return NULL;
2674
1.52M
}
2675
2676
2677
static njs_parser_node_t *
2678
njs_parser_lvalue_ref(njs_parser_node_t *node)
2679
286k
{
2680
286k
    if (node == NULL) {
2681
0
        return NULL;
2682
0
    }
2683
2684
286k
    if (node->token_type == NJS_TOKEN_NAME) {
2685
202k
        return node;
2686
202k
    }
2687
2688
83.3k
    return njs_parser_property_ref(node);
2689
286k
}
2690
2691
2692
static njs_parser_node_t *
2693
njs_parser_call_receiver(njs_parser_node_t *node)
2694
1.43M
{
2695
1.43M
    return njs_parser_property_ref(node);
2696
1.43M
}
2697
2698
2699
static njs_parser_node_t *
2700
njs_parser_optional_chain_preserve(njs_parser_t *parser,
2701
    njs_parser_node_t *node, uint32_t token_line)
2702
1.49k
{
2703
1.49k
    njs_parser_node_t  *ref;
2704
2705
1.49k
    ref = njs_parser_node_new(parser, NJS_TOKEN_OPTIONAL_PRESERVE);
2706
1.49k
    if (ref == NULL) {
2707
0
        return NULL;
2708
0
    }
2709
2710
1.49k
    ref->token_line = token_line;
2711
1.49k
    ref->u.object = node;
2712
2713
1.49k
    return ref;
2714
1.49k
}
2715
2716
2717
static void
2718
njs_parser_set_optional_chain_preserve(njs_parser_node_t *node,
2719
    njs_parser_node_t *preserve)
2720
1.49k
{
2721
1.49k
    njs_assert(node->token_type == NJS_TOKEN_OPTIONAL_CHAIN);
2722
1.49k
    njs_assert(preserve->token_type == NJS_TOKEN_OPTIONAL_PRESERVE);
2723
2724
1.49k
    node->u.object = preserve;
2725
1.49k
}
2726
2727
2728
static njs_parser_node_t *
2729
njs_parser_optional_chain_source(njs_parser_node_t *node)
2730
4
{
2731
4
    if (node != NULL && node->token_type == NJS_TOKEN_OPTIONAL_PRESERVE) {
2732
4
        return node->u.object;
2733
4
    }
2734
2735
0
    return NULL;
2736
4
}
2737
2738
2739
static njs_parser_node_t *
2740
njs_parser_create_call(njs_parser_t *parser, njs_parser_node_t *node,
2741
    uint8_t ctor)
2742
734k
{
2743
734k
    njs_parser_node_t  *func, *receiver, *this_object;
2744
2745
734k
    receiver = njs_parser_call_receiver(node);
2746
734k
    if (receiver != NULL) {
2747
30.1k
        func = njs_parser_node_new(parser, NJS_TOKEN_METHOD_CALL);
2748
30.1k
        if (func == NULL) {
2749
0
            return NULL;
2750
0
        }
2751
2752
30.1k
        func->left = receiver;
2753
30.1k
        func->ctor = ctor;
2754
2755
30.1k
        return func;
2756
30.1k
    }
2757
2758
704k
    this_object = njs_parser_optional_chain_call_this(node);
2759
2760
    /*
2761
     * NJS_TOKEN_NAME,
2762
     * NJS_TOKEN_METHOD_CALL,
2763
     * NJS_TOKEN_FUNCTION_CALL,
2764
     * NJS_TOKEN_FUNCTION_EXPRESSION,
2765
     * NJS_TOKEN_OPEN_PARENTHESIS,
2766
     * NJS_TOKEN_EVAL.
2767
     */
2768
704k
    func = njs_parser_node_new(parser, NJS_TOKEN_FUNCTION_CALL);
2769
704k
    if (func == NULL) {
2770
0
        return NULL;
2771
0
    }
2772
2773
704k
    func->left = node;
2774
704k
    func->ctor = ctor;
2775
704k
    njs_parser_set_call_this(func, this_object);
2776
2777
704k
    return func;
2778
704k
}
2779
2780
2781
static njs_int_t
2782
njs_parser_call_arguments(njs_parser_t *parser, njs_lexer_token_t *token,
2783
    njs_queue_link_t *current, njs_parser_node_t *func,
2784
    njs_parser_state_func_t after)
2785
2.66k
{
2786
2.66k
    func->token_line = token->line;
2787
2.66k
    parser->node = func;
2788
2789
2.66k
    njs_lexer_consume_token(parser->lexer, 1);
2790
2.66k
    njs_parser_next(parser, njs_parser_arguments);
2791
2792
2.66k
    return njs_parser_after(parser, current, func, 1, after);
2793
2.66k
}
2794
2795
2796
static njs_int_t
2797
njs_parser_right_link_pop(njs_parser_t *parser)
2798
313k
{
2799
313k
    parser->target->right = parser->node;
2800
313k
    parser->node = parser->target;
2801
2802
313k
    return njs_parser_stack_pop(parser);
2803
313k
}
2804
2805
2806
static njs_parser_node_t *
2807
njs_parser_optional_chain_receiver(njs_parser_node_t *node)
2808
704k
{
2809
704k
    if (njs_parser_call_receiver(node) != NULL) {
2810
0
        return node;
2811
0
    }
2812
2813
704k
    if (node != NULL
2814
704k
        && node->token_type == NJS_TOKEN_OPTIONAL_CHAIN
2815
0
        && node->right != NULL
2816
0
        && njs_parser_call_receiver(node->right) != NULL)
2817
0
    {
2818
0
        return node->right;
2819
0
    }
2820
2821
704k
    return NULL;
2822
704k
}
2823
2824
2825
static njs_vmcode_t
2826
njs_parser_optional_chain_receiver_operation(njs_parser_node_t *node)
2827
0
{
2828
0
    return node->u.operation;
2829
0
}
2830
2831
2832
static njs_parser_node_t *
2833
njs_parser_optional_chain_receiver_base(njs_parser_node_t *node)
2834
0
{
2835
0
    return node->left;
2836
0
}
2837
2838
2839
static njs_parser_node_t *
2840
njs_parser_optional_chain_receiver_key(njs_parser_node_t *node)
2841
0
{
2842
0
    return node->right;
2843
0
}
2844
2845
2846
static njs_parser_node_t *
2847
njs_parser_optional_chain_call_this(njs_parser_node_t *node)
2848
704k
{
2849
704k
    njs_parser_node_t  *receiver;
2850
2851
704k
    receiver = njs_parser_optional_chain_receiver(node);
2852
704k
    if (receiver != NULL) {
2853
0
        return njs_parser_optional_chain_receiver_base(receiver);
2854
0
    }
2855
2856
704k
    return NULL;
2857
704k
}
2858
2859
2860
static void
2861
njs_parser_set_call_this(njs_parser_node_t *node,
2862
    njs_parser_node_t *this_object)
2863
704k
{
2864
704k
    njs_assert(node->token_type == NJS_TOKEN_FUNCTION_CALL);
2865
2866
704k
    node->u.object = this_object;
2867
704k
}
2868
2869
2870
static void
2871
njs_parser_set_optional_method_call_preserve(njs_parser_node_t *node,
2872
    njs_parser_node_t *preserve)
2873
0
{
2874
0
    njs_assert(node->token_type == NJS_TOKEN_METHOD_CALL);
2875
0
    njs_assert(njs_parser_call_receiver(preserve) != NULL);
2876
2877
0
    node->u.object = preserve;
2878
0
}
2879
2880
2881
static njs_parser_node_t *
2882
njs_parser_optional_chain_target_property(njs_parser_node_t *node)
2883
12
{
2884
12
    if (node == NULL || node->token_type != NJS_TOKEN_OPTIONAL_CHAIN) {
2885
0
        return NULL;
2886
0
    }
2887
2888
12
    return njs_parser_property_ref(node->right);
2889
12
}
2890
2891
2892
static njs_parser_node_t *
2893
njs_parser_optional_chain_method_call(njs_parser_t *parser,
2894
    njs_parser_node_t *node, uint32_t token_line)
2895
4
{
2896
4
    njs_parser_node_t  *func, *prop, *src;
2897
2898
4
    src = njs_parser_optional_chain_receiver(node);
2899
4
    if (src == NULL) {
2900
4
        return NULL;
2901
4
    }
2902
2903
0
    prop = njs_parser_node_new(parser, NJS_TOKEN_PROPERTY_REF);
2904
0
    if (prop == NULL) {
2905
0
        return NULL;
2906
0
    }
2907
2908
0
    prop->u.operation = njs_parser_optional_chain_receiver_operation(src);
2909
0
    prop->token_line = token_line;
2910
0
    prop->left = njs_parser_optional_chain_receiver_base(src);
2911
0
    prop->right = njs_parser_optional_chain_receiver_key(src);
2912
2913
0
    func = njs_parser_create_call(parser, prop, 0);
2914
0
    if (func == NULL) {
2915
0
        return NULL;
2916
0
    }
2917
2918
0
    njs_parser_set_optional_method_call_preserve(func, src);
2919
2920
0
    return func;
2921
0
}
2922
2923
2924
static njs_parser_node_t *
2925
njs_parser_optional_chain_call(njs_parser_t *parser, njs_parser_node_t *node,
2926
    njs_parser_node_t *fallback, uint32_t token_line)
2927
4
{
2928
4
    njs_parser_node_t  *func;
2929
2930
4
    func = njs_parser_optional_chain_method_call(parser, node, token_line);
2931
4
    if (func != NULL) {
2932
0
        return func;
2933
0
    }
2934
2935
4
    return njs_parser_create_call(parser, fallback, 0);
2936
4
}
2937
2938
2939
static njs_int_t
2940
njs_parser_call_expression(njs_parser_t *parser, njs_lexer_token_t *token,
2941
    njs_queue_link_t *current)
2942
432
{
2943
432
    njs_int_t  ret;
2944
2945
    /*
2946
     * CoverCallExpressionAndAsyncArrowHead
2947
     * SuperCall
2948
     * ImportCall
2949
     * CallExpression Arguments
2950
     * CallExpression [ Expression ]
2951
     * CallExpression . IdentifierName
2952
     * CallExpression TemplateLiteral
2953
     */
2954
2955
432
    switch (token->type) {
2956
    /* MemberExpression or SuperCall */
2957
239
    case NJS_TOKEN_SUPER:
2958
239
        return njs_parser_not_supported(parser, token);
2959
2960
193
    case NJS_TOKEN_IMPORT:
2961
193
        return njs_parser_not_supported(parser, token);
2962
2963
0
    default:
2964
0
        break;
2965
432
    }
2966
2967
0
    njs_parser_next(parser, njs_parser_member_expression);
2968
2969
0
    ret = njs_parser_after(parser, current, NULL, 1,
2970
0
                           njs_parser_call_expression_args);
2971
0
    if (ret != NJS_OK) {
2972
0
        return NJS_ERROR;
2973
0
    }
2974
2975
0
    return njs_parser_after(parser, current, NULL, 1,
2976
0
                            njs_parser_call_expression_after);
2977
0
}
2978
2979
2980
static njs_int_t
2981
njs_parser_call_expression_args(njs_parser_t *parser, njs_lexer_token_t *token,
2982
    njs_queue_link_t *current)
2983
0
{
2984
0
    njs_parser_node_t  *func;
2985
2986
0
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS) {
2987
0
        return njs_parser_failed(parser);
2988
0
    }
2989
2990
0
    func = njs_parser_create_call(parser, parser->node, 0);
2991
0
    if (func == NULL) {
2992
0
        return NJS_ERROR;
2993
0
    }
2994
2995
0
    return njs_parser_call_arguments(parser, token, current, func,
2996
0
                                     njs_parser_left_hand_side_expression_node);
2997
0
}
2998
2999
3000
static njs_int_t
3001
njs_parser_call_or_property_after(njs_parser_t *parser,
3002
    njs_lexer_token_t *token, njs_queue_link_t *current,
3003
    njs_parser_state_func_t after)
3004
69.7k
{
3005
69.7k
    njs_int_t          ret;
3006
69.7k
    njs_parser_node_t  *func;
3007
3008
69.7k
    switch (token->type) {
3009
893
    case NJS_TOKEN_OPEN_PARENTHESIS:
3010
893
        func = njs_parser_create_call(parser, parser->node, 0);
3011
893
        if (func == NULL) {
3012
0
            return NJS_ERROR;
3013
0
        }
3014
3015
893
        ret = njs_parser_call_arguments(parser, token, current, func,
3016
893
                                        njs_parser_left_hand_side_expression_node);
3017
893
        if (ret != NJS_OK) {
3018
0
            return NJS_ERROR;
3019
0
        }
3020
3021
893
        break;
3022
3023
68.8k
    default:
3024
68.8k
        ret = njs_parser_property(parser, token, current);
3025
3026
68.8k
        switch (ret) {
3027
2.50k
        case NJS_AGAIN:
3028
2.50k
            return NJS_OK;
3029
3030
49.4k
        case NJS_DONE:
3031
49.4k
            return njs_parser_stack_pop(parser);
3032
3033
38
        case NJS_DECLINED:
3034
38
            return njs_parser_failed(parser);
3035
3036
16.8k
        default:
3037
16.8k
            break;
3038
68.8k
        }
3039
3040
16.8k
        break;
3041
69.7k
    }
3042
3043
17.7k
    return njs_parser_after(parser, current, NULL, 1, after);
3044
69.7k
}
3045
3046
3047
static njs_int_t
3048
njs_parser_call_expression_after(njs_parser_t *parser,
3049
    njs_lexer_token_t *token, njs_queue_link_t *current)
3050
0
{
3051
    /*
3052
     * Arguments
3053
     * [ Expression ]
3054
     * . IdentifierName
3055
     * TemplateLiteral
3056
     */
3057
3058
0
    return njs_parser_call_or_property_after(parser, token, current,
3059
0
                                             njs_parser_call_expression_after);
3060
0
}
3061
3062
3063
static njs_int_t
3064
njs_parser_arguments(njs_parser_t *parser, njs_lexer_token_t *token,
3065
    njs_queue_link_t *current)
3066
58.0k
{
3067
    /*
3068
     * )
3069
     * ArgumentList )
3070
     * ArgumentList , )
3071
     */
3072
3073
58.0k
    if (token->type == NJS_TOKEN_CLOSE_PARENTHESIS) {
3074
28.6k
        njs_lexer_consume_token(parser->lexer, 1);
3075
28.6k
        return njs_parser_stack_pop(parser);
3076
28.6k
    }
3077
3078
29.4k
    njs_parser_next(parser, njs_parser_argument_list);
3079
3080
29.4k
    return njs_parser_after(parser, current, NULL, 1,
3081
58.0k
                            njs_parser_parenthesis_or_comma);
3082
58.0k
}
3083
3084
3085
static njs_int_t
3086
njs_parser_parenthesis_or_comma(njs_parser_t *parser, njs_lexer_token_t *token,
3087
    njs_queue_link_t *current)
3088
23.0k
{
3089
23.0k
    if (token->type == NJS_TOKEN_CLOSE_PARENTHESIS) {
3090
22.2k
        njs_lexer_consume_token(parser->lexer, 1);
3091
22.2k
        return njs_parser_stack_pop(parser);
3092
22.2k
    }
3093
3094
727
    if (token->type != NJS_TOKEN_COMMA) {
3095
727
        return njs_parser_failed(parser);
3096
727
    }
3097
3098
0
    njs_lexer_consume_token(parser->lexer, 1);
3099
3100
0
    token = njs_lexer_token(parser->lexer, 0);
3101
0
    if (njs_slow_path(token == NULL)) {
3102
0
        return NJS_ERROR;
3103
0
    }
3104
3105
0
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
3106
0
        return njs_parser_failed(parser);
3107
0
    }
3108
3109
0
    njs_lexer_consume_token(parser->lexer, 1);
3110
3111
0
    return njs_parser_stack_pop(parser);
3112
0
}
3113
3114
3115
static njs_int_t
3116
njs_parser_argument_list(njs_parser_t *parser, njs_lexer_token_t *token,
3117
    njs_queue_link_t *current)
3118
46.8k
{
3119
    /*
3120
     * AssignmentExpression
3121
     * ... AssignmentExpression
3122
     * ArgumentList , AssignmentExpression
3123
     * ArgumentList , ... AssignmentExpression
3124
     */
3125
3126
#if 0 /* TODO. */
3127
    if (token->type == NJS_TOKEN_ELLIPSIS) {
3128
        njs_lexer_consume_token(parser->lexer, 1);
3129
    }
3130
#endif
3131
3132
46.8k
    njs_parser_next(parser, njs_parser_assignment_expression);
3133
3134
46.8k
    return njs_parser_after(parser, current, parser->node, 1,
3135
46.8k
                            njs_parser_argument_list_after);
3136
46.8k
}
3137
3138
3139
static njs_int_t
3140
njs_parser_argument_list_after(njs_parser_t *parser, njs_lexer_token_t *token,
3141
    njs_queue_link_t *current)
3142
40.4k
{
3143
40.4k
    njs_parser_node_t  *node;
3144
3145
40.4k
    node = njs_parser_node_new(parser, NJS_TOKEN_ARGUMENT);
3146
40.4k
    if (node == NULL) {
3147
0
        return NJS_ERROR;
3148
0
    }
3149
3150
40.4k
    node->index = njs_scope_temp_index(node->scope);
3151
40.4k
    if (njs_slow_path(node->index == NJS_INDEX_ERROR)) {
3152
0
        return NJS_ERROR;
3153
0
    }
3154
3155
40.4k
    node->token_line = token->line;
3156
40.4k
    node->left = parser->node;
3157
3158
40.4k
    parser->node->dest = node;
3159
40.4k
    parser->target->right = node;
3160
40.4k
    parser->node = node;
3161
3162
40.4k
    if (token->type == NJS_TOKEN_COMMA) {
3163
17.8k
        njs_lexer_consume_token(parser->lexer, 1);
3164
3165
17.8k
        token = njs_lexer_token(parser->lexer, 0);
3166
17.8k
        if (token == NULL) {
3167
0
            return NJS_ERROR;
3168
0
        }
3169
3170
17.8k
        if (token->type == NJS_TOKEN_CLOSE_PARENTHESIS) {
3171
400
            return njs_parser_stack_pop(parser);
3172
400
        }
3173
3174
17.4k
        return njs_parser_argument_list(parser, token, current);
3175
17.8k
    }
3176
3177
22.6k
    return njs_parser_stack_pop(parser);
3178
40.4k
}
3179
3180
3181
static njs_int_t
3182
njs_parser_optional_chain_wrap(njs_parser_t *parser,
3183
    njs_lexer_token_t *token, njs_queue_link_t *current)
3184
1.34k
{
3185
1.34k
    return njs_parser_right_link_pop(parser);
3186
1.34k
}
3187
3188
3189
static njs_int_t
3190
njs_parser_optional_expression_after(njs_parser_t *parser,
3191
    njs_lexer_token_t *token, njs_queue_link_t *current)
3192
44.4k
{
3193
44.4k
    njs_int_t          ret;
3194
44.4k
    njs_parser_node_t  *opt, *ref;
3195
3196
44.4k
    if (token->type != NJS_TOKEN_CONDITIONAL) {
3197
1.32k
        return njs_parser_stack_pop(parser);
3198
1.32k
    }
3199
3200
43.0k
    token = njs_lexer_peek_token(parser->lexer, token, 0);
3201
43.0k
    if (token == NULL) {
3202
0
        return NJS_ERROR;
3203
0
    }
3204
3205
43.0k
    if (token->type != NJS_TOKEN_DOT) {
3206
41.5k
        return njs_parser_stack_pop(parser);
3207
41.5k
    }
3208
3209
1.49k
    opt = njs_parser_node_new(parser, NJS_TOKEN_OPTIONAL_CHAIN);
3210
1.49k
    if (opt == NULL) {
3211
0
        return NJS_ERROR;
3212
0
    }
3213
3214
1.49k
    opt->token_line = token->line;
3215
1.49k
    opt->left = parser->node;
3216
1.49k
    opt->left->dest = opt;
3217
3218
1.49k
    ref = njs_parser_optional_chain_preserve(parser, parser->node,
3219
1.49k
                                             token->line);
3220
1.49k
    if (ref == NULL) {
3221
0
        return NJS_ERROR;
3222
0
    }
3223
3224
1.49k
    njs_parser_set_optional_chain_preserve(opt, ref);
3225
1.49k
    parser->node = ref;
3226
3227
1.49k
    njs_parser_next(parser, njs_parser_optional_chain);
3228
3229
1.49k
    ret = njs_parser_after(parser, current, opt, 1,
3230
1.49k
                           njs_parser_optional_chain_wrap);
3231
1.49k
    if (ret != NJS_OK) {
3232
0
        return NJS_ERROR;
3233
0
    }
3234
3235
1.49k
    return njs_parser_after(parser, current, NULL, 1,
3236
1.49k
                            njs_parser_optional_expression_after);
3237
1.49k
}
3238
3239
3240
static njs_int_t
3241
njs_parser_optional_chain(njs_parser_t *parser, njs_lexer_token_t *token,
3242
    njs_queue_link_t *current)
3243
1.49k
{
3244
1.49k
    njs_int_t          ret;
3245
1.49k
    njs_parser_node_t  *node, *func, *prop_node;
3246
3247
    /*
3248
     * ?. Arguments
3249
     * ?. [ Expression ]
3250
     * ?. IdentifierName
3251
     */
3252
3253
1.49k
    if (token->type != NJS_TOKEN_CONDITIONAL) {
3254
0
        return njs_parser_failed(parser);
3255
0
    }
3256
3257
1.49k
    token = njs_lexer_peek_token(parser->lexer, token, 0);
3258
1.49k
    if (token == NULL) {
3259
0
        return NJS_ERROR;
3260
0
    }
3261
3262
1.49k
    if (token->type != NJS_TOKEN_DOT) {
3263
0
        return njs_parser_failed(parser);
3264
0
    }
3265
3266
    /* Consume both '?' and '.' */
3267
1.49k
    njs_lexer_consume_token(parser->lexer, 2);
3268
3269
1.49k
    token = njs_lexer_token(parser->lexer, 0);
3270
1.49k
    if (token == NULL) {
3271
0
        return NJS_ERROR;
3272
0
    }
3273
3274
1.49k
    switch (token->type) {
3275
4
    case NJS_TOKEN_OPEN_PARENTHESIS:
3276
4
        func = njs_parser_optional_chain_call(parser,
3277
4
                                              njs_parser_optional_chain_source(
3278
4
                                                  parser->node),
3279
4
                                              parser->node, token->line);
3280
4
        if (func == NULL) {
3281
0
            return NJS_ERROR;
3282
0
        }
3283
3284
4
        ret = njs_parser_call_arguments(parser, token, current, func,
3285
4
                                        njs_parser_left_hand_side_expression_node);
3286
4
        if (ret != NJS_OK) {
3287
0
            return NJS_ERROR;
3288
0
        }
3289
3290
4
        return njs_parser_after(parser, current, NULL, 1,
3291
4
                                njs_parser_optional_chain_after);
3292
3293
0
    case NJS_TOKEN_OPEN_BRACKET:
3294
0
        node = njs_parser_node_new(parser, NJS_TOKEN_PROPERTY);
3295
0
        if (node == NULL) {
3296
0
            return NJS_ERROR;
3297
0
        }
3298
3299
0
        node->u.operation = NJS_VMCODE_PROPERTY_GET;
3300
0
        node->left = parser->node;
3301
0
        node->token_line = token->line;
3302
3303
0
        parser->node = NULL;
3304
3305
0
        njs_lexer_consume_token(parser->lexer, 1);
3306
3307
0
        njs_parser_next(parser, njs_parser_expression);
3308
3309
0
        ret = njs_parser_after(parser, current, node, 1,
3310
0
                               njs_parser_member_expression_bracket);
3311
0
        if (ret != NJS_OK) {
3312
0
            return NJS_ERROR;
3313
0
        }
3314
3315
0
        return njs_parser_after(parser, current, NULL, 1,
3316
0
                                njs_parser_optional_chain_after);
3317
3318
1.49k
    default:
3319
1.49k
        if (!njs_lexer_token_is_identifier_name(token)) {
3320
149
            return njs_parser_failed(parser);
3321
149
        }
3322
3323
1.34k
        node = njs_parser_node_new(parser, NJS_TOKEN_PROPERTY);
3324
1.34k
        if (node == NULL) {
3325
0
            return NJS_ERROR;
3326
0
        }
3327
3328
1.34k
        node->u.operation = NJS_VMCODE_PROPERTY_ATOM_GET;
3329
1.34k
        node->token_line = token->line;
3330
3331
1.34k
        prop_node = njs_parser_node_string(parser->vm, token,
3332
1.34k
                                           parser);
3333
1.34k
        if (prop_node == NULL) {
3334
0
            return NJS_ERROR;
3335
0
        }
3336
3337
1.34k
        prop_node->token_line = token->line;
3338
3339
1.34k
        node->left = parser->node;
3340
1.34k
        node->right = prop_node;
3341
3342
1.34k
        parser->node = node;
3343
3344
1.34k
        njs_lexer_consume_token(parser->lexer, 1);
3345
3346
1.34k
        njs_parser_next(parser, njs_parser_optional_chain_after);
3347
3348
1.34k
        return NJS_OK;
3349
1.49k
    }
3350
1.49k
}
3351
3352
3353
static njs_int_t
3354
njs_parser_optional_chain_after(njs_parser_t *parser, njs_lexer_token_t *token,
3355
    njs_queue_link_t *current)
3356
69.7k
{
3357
    /*
3358
     * OptionalChain Arguments
3359
     * OptionalChain [ Expression ]
3360
     * OptionalChain . IdentifierName
3361
     * OptionalChain TemplateLiteral
3362
     */
3363
3364
69.7k
    return njs_parser_call_or_property_after(parser, token, current,
3365
69.7k
                                             njs_parser_optional_chain_after);
3366
69.7k
}
3367
3368
3369
static njs_int_t
3370
njs_parser_new_expression(njs_parser_t *parser, njs_lexer_token_t *token,
3371
    njs_queue_link_t *current)
3372
4.09k
{
3373
4.09k
    if (token->type != NJS_TOKEN_NEW) {
3374
2.04k
        parser->node = NULL;
3375
3376
2.04k
        njs_parser_next(parser, njs_parser_member_expression_new);
3377
3378
2.04k
        return NJS_OK;
3379
2.04k
    }
3380
3381
2.04k
    njs_lexer_consume_token(parser->lexer, 1);
3382
3383
2.04k
    return njs_parser_after(parser, current, NULL, 1,
3384
4.09k
                            njs_parser_new_expression_after);
3385
4.09k
}
3386
3387
3388
static njs_int_t
3389
njs_parser_new_expression_after(njs_parser_t *parser, njs_lexer_token_t *token,
3390
    njs_queue_link_t *current)
3391
770
{
3392
770
    njs_parser_node_t  *func;
3393
3394
770
    if (token->type == NJS_TOKEN_OPEN_PARENTHESIS) {
3395
216
        njs_parser_next(parser, njs_parser_member_expression_new_after);
3396
216
        return NJS_OK;
3397
216
    }
3398
3399
554
    func = njs_parser_create_call(parser, parser->node, 1);
3400
554
    if (func == NULL) {
3401
0
        return NJS_ERROR;
3402
0
    }
3403
3404
554
    func->token_line = token->line;
3405
554
    parser->node = func;
3406
3407
554
    return njs_parser_stack_pop(parser);
3408
554
}
3409
3410
3411
static njs_int_t
3412
njs_parser_left_hand_side_expression(njs_parser_t *parser,
3413
    njs_lexer_token_t *token, njs_queue_link_t *current)
3414
2.87M
{
3415
    /*
3416
     * NewExpression = new MemberExpression
3417
     * CallExpression = MemberExpression Arguments
3418
     * OptionalExpression = MemberExpression OptionalChain
3419
     */
3420
3421
2.87M
    switch (token->type) {
3422
    /* NewExpression or MemberExpression */
3423
7.26k
    case NJS_TOKEN_NEW:
3424
7.26k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
3425
7.26k
        if (token == NULL) {
3426
0
            return NJS_ERROR;
3427
0
        }
3428
3429
7.26k
        if (token->type == NJS_TOKEN_NEW) {
3430
2.04k
            njs_lexer_consume_token(parser->lexer, 1);
3431
3432
2.04k
            njs_parser_next(parser, njs_parser_new_expression);
3433
3434
2.04k
            return njs_parser_after(parser, current, NULL, 1,
3435
2.04k
                                    njs_parser_left_hand_side_expression_after);
3436
2.04k
        }
3437
3438
5.21k
        break;
3439
3440
    /* CallExpression or MemberExpression */
3441
5.21k
    case NJS_TOKEN_SUPER:
3442
942
    case NJS_TOKEN_IMPORT:
3443
942
        token = njs_lexer_peek_token(parser->lexer, token, 0);
3444
942
        if (token == NULL) {
3445
0
            return NJS_ERROR;
3446
0
        }
3447
3448
942
        if (token->type == NJS_TOKEN_OPEN_PARENTHESIS) {
3449
432
            njs_parser_next(parser, njs_parser_call_expression);
3450
432
            return NJS_OK;
3451
432
        }
3452
3453
510
        break;
3454
3455
2.86M
    default:
3456
2.86M
        break;
3457
2.87M
    }
3458
3459
2.86M
    njs_parser_next(parser, njs_parser_member_expression);
3460
3461
2.86M
    return njs_parser_after(parser, current, NULL, 1,
3462
2.87M
                            njs_parser_left_hand_side_expression_after);
3463
2.87M
}
3464
3465
3466
static njs_int_t
3467
njs_parser_left_hand_side_expression_after(njs_parser_t *parser,
3468
    njs_lexer_token_t *token, njs_queue_link_t *current)
3469
2.63M
{
3470
2.63M
    njs_int_t          ret;
3471
2.63M
    njs_parser_node_t  *func;
3472
3473
2.63M
    switch (token->type) {
3474
    /* CallExpression */
3475
55.3k
    case NJS_TOKEN_OPEN_PARENTHESIS:
3476
55.3k
        func = njs_parser_create_call(parser, parser->node, 0);
3477
55.3k
        if (func == NULL) {
3478
0
            return NJS_ERROR;
3479
0
        }
3480
3481
55.3k
        func->token_line = token->line;
3482
55.3k
        parser->node = func;
3483
3484
55.3k
        njs_lexer_consume_token(parser->lexer, 1);
3485
55.3k
        njs_parser_next(parser, njs_parser_arguments);
3486
3487
55.3k
        ret = njs_parser_after(parser, current, func, 1,
3488
55.3k
                               njs_parser_left_hand_side_expression_node);
3489
55.3k
        if (ret != NJS_OK) {
3490
0
            return NJS_ERROR;
3491
0
        }
3492
3493
55.3k
        return njs_parser_after(parser, current, NULL, 1,
3494
55.3k
                                njs_parser_left_hand_side_expression_optional);
3495
3496
    /* OptionalExpression */
3497
42.9k
    case NJS_TOKEN_CONDITIONAL:
3498
42.9k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
3499
42.9k
        if (token == NULL) {
3500
0
            return NJS_ERROR;
3501
0
        }
3502
3503
42.9k
        if (token->type == NJS_TOKEN_DOT
3504
1.47k
            && parser->node != NULL
3505
1.47k
            && parser->node->ctor)
3506
0
        {
3507
0
            njs_parser_syntax_error(parser,
3508
0
                                    "Optional chaining cannot be used "
3509
0
                                    "with new");
3510
0
            return NJS_DONE;
3511
0
        }
3512
3513
42.9k
        njs_parser_next(parser, njs_parser_optional_expression_after);
3514
42.9k
        break;
3515
3516
2.53M
    default:
3517
2.53M
        return njs_parser_stack_pop(parser);
3518
2.63M
    }
3519
3520
42.9k
    return NJS_OK;
3521
2.63M
}
3522
3523
3524
static njs_int_t
3525
njs_parser_left_hand_side_expression_node(njs_parser_t *parser,
3526
    njs_lexer_token_t *token, njs_queue_link_t *current)
3527
49.3k
{
3528
49.3k
    parser->node = parser->target;
3529
3530
49.3k
    return njs_parser_stack_pop(parser);
3531
49.3k
}
3532
3533
3534
static njs_int_t
3535
njs_parser_left_hand_side_expression_optional(njs_parser_t *parser,
3536
    njs_lexer_token_t *token, njs_queue_link_t *current)
3537
48.5k
{
3538
    /* OptionalExpression */
3539
48.5k
    if (token->type == NJS_TOKEN_CONDITIONAL) {
3540
131
        njs_parser_next(parser, njs_parser_optional_expression_after);
3541
131
        return NJS_OK;
3542
131
    }
3543
3544
48.4k
    njs_parser_next(parser, njs_parser_optional_chain_after);
3545
3546
48.4k
    return NJS_OK;
3547
48.5k
}
3548
3549
3550
static njs_int_t
3551
njs_parser_expression_node(njs_parser_t *parser, njs_lexer_token_t *token,
3552
    njs_queue_link_t *current, njs_token_type_t type, njs_vmcode_t operation,
3553
    njs_parser_state_func_t after)
3554
12.9M
{
3555
12.9M
    njs_parser_node_t  *node;
3556
3557
12.9M
    if (parser->target != NULL) {
3558
108k
        parser->target->right = parser->node;
3559
108k
        parser->target->right->dest = parser->target;
3560
108k
        parser->node = parser->target;
3561
108k
    }
3562
3563
12.9M
    if (token->type != type) {
3564
12.8M
        return njs_parser_stack_pop(parser);
3565
12.8M
    }
3566
3567
110k
    node = njs_parser_node_new(parser, type);
3568
110k
    if (node == NULL) {
3569
0
        return NJS_ERROR;
3570
0
    }
3571
3572
110k
    node->token_line = token->line;
3573
110k
    node->u.operation = operation;
3574
110k
    node->left = parser->node;
3575
110k
    node->left->dest = node;
3576
3577
110k
    njs_lexer_consume_token(parser->lexer, 1);
3578
3579
110k
    return njs_parser_after(parser, current, node, 1, after);
3580
110k
}
3581
3582
3583
/*
3584
 * 12.4 Update Expressions.
3585
 */
3586
static njs_int_t
3587
njs_parser_update_expression(njs_parser_t *parser, njs_lexer_token_t *token,
3588
    njs_queue_link_t *current)
3589
2.85M
{
3590
2.85M
    njs_vmcode_t       operation;
3591
2.85M
    njs_parser_node_t  *node;
3592
3593
2.85M
    switch (token->type) {
3594
5.22k
    case NJS_TOKEN_INCREMENT:
3595
5.22k
        operation = NJS_VMCODE_INCREMENT;
3596
5.22k
        break;
3597
3598
11.0k
    case NJS_TOKEN_DECREMENT:
3599
11.0k
        operation = NJS_VMCODE_DECREMENT;
3600
11.0k
        break;
3601
3602
2.84M
    default:
3603
2.84M
        njs_parser_next(parser, njs_parser_left_hand_side_expression);
3604
3605
2.84M
        return njs_parser_after(parser, current, NULL, 1,
3606
2.85M
                                njs_parser_update_expression_post);
3607
2.85M
    }
3608
3609
16.2k
    node = njs_parser_node_new(parser, token->type);
3610
16.2k
    if (node == NULL) {
3611
0
        return NJS_ERROR;
3612
0
    }
3613
3614
16.2k
    node->token_line = token->line;
3615
16.2k
    node->u.operation = operation;
3616
3617
16.2k
    njs_lexer_consume_token(parser->lexer, 1);
3618
16.2k
    njs_parser_next(parser, njs_parser_left_hand_side_expression);
3619
3620
16.2k
    return njs_parser_after(parser, current, node, 1,
3621
16.2k
                            njs_parser_update_expression_unary);
3622
16.2k
}
3623
3624
3625
static njs_int_t
3626
njs_parser_update_expression_post(njs_parser_t *parser,
3627
    njs_lexer_token_t *token, njs_queue_link_t *current)
3628
2.60M
{
3629
2.60M
    njs_vmcode_t       operation;
3630
2.60M
    njs_token_type_t   type;
3631
2.60M
    njs_parser_node_t  *node;
3632
3633
    /* [no LineTerminator here] */
3634
3635
2.60M
    switch (token->type) {
3636
14.2k
    case NJS_TOKEN_INCREMENT:
3637
14.2k
        type = NJS_TOKEN_POST_INCREMENT;
3638
14.2k
        operation = NJS_VMCODE_POST_INCREMENT;
3639
14.2k
        break;
3640
3641
20.5k
    case NJS_TOKEN_DECREMENT:
3642
20.5k
        type = NJS_TOKEN_POST_DECREMENT;
3643
20.5k
        operation = NJS_VMCODE_POST_DECREMENT;
3644
20.5k
        break;
3645
3646
2.57M
    default:
3647
2.57M
        return njs_parser_stack_pop(parser);
3648
2.60M
    }
3649
3650
34.7k
    if (parser->lexer->prev_type == NJS_TOKEN_LINE_END) {
3651
2.79k
        return njs_parser_stack_pop(parser);
3652
2.79k
    }
3653
3654
31.9k
    if (!njs_parser_is_lvalue(parser->node)) {
3655
136
        njs_lexer_consume_token(parser->lexer, 1);
3656
3657
136
        njs_parser_ref_error(parser,
3658
136
                             "Invalid left-hand side in postfix operation");
3659
136
        return NJS_DONE;
3660
136
    }
3661
3662
31.7k
    parser->node = njs_parser_lvalue_ref(parser->node);
3663
31.7k
    if (parser->node == NULL) {
3664
0
        return NJS_ERROR;
3665
0
    }
3666
3667
31.7k
    node = njs_parser_node_new(parser, type);
3668
31.7k
    if (node == NULL) {
3669
0
        return NJS_ERROR;
3670
0
    }
3671
3672
31.7k
    node->token_line = token->line;
3673
31.7k
    node->u.operation = operation;
3674
31.7k
    node->left = parser->node;
3675
3676
31.7k
    parser->node = node;
3677
3678
31.7k
    njs_lexer_consume_token(parser->lexer, 1);
3679
3680
31.7k
    return njs_parser_stack_pop(parser);
3681
31.7k
}
3682
3683
3684
static njs_int_t
3685
njs_parser_update_expression_unary(njs_parser_t *parser,
3686
    njs_lexer_token_t *token, njs_queue_link_t *current)
3687
16.2k
{
3688
16.2k
    if (!njs_parser_is_lvalue(parser->node)) {
3689
435
        njs_parser_ref_error(parser,
3690
435
                             "Invalid left-hand side in prefix operation");
3691
435
        return NJS_DONE;
3692
435
    }
3693
3694
15.8k
    parser->node = njs_parser_lvalue_ref(parser->node);
3695
15.8k
    if (parser->node == NULL) {
3696
0
        return NJS_ERROR;
3697
0
    }
3698
3699
15.8k
    parser->target->left = parser->node;
3700
15.8k
    parser->node = parser->target;
3701
3702
15.8k
    return njs_parser_stack_pop(parser);
3703
15.8k
}
3704
3705
3706
/*
3707
 * 12.5 Unary Operators.
3708
 */
3709
static njs_int_t
3710
njs_parser_unary_expression(njs_parser_t *parser, njs_lexer_token_t *token,
3711
    njs_queue_link_t *current)
3712
3.18M
{
3713
3.18M
    njs_vmcode_t       operation;
3714
3.18M
    njs_token_type_t   type;
3715
3.18M
    njs_parser_node_t  *node;
3716
3717
3.18M
    switch (token->type) {
3718
1.73k
    case NJS_TOKEN_DELETE:
3719
1.73k
        type = NJS_TOKEN_DELETE;
3720
1.73k
        operation = NJS_VMCODE_DELETE;
3721
1.73k
        break;
3722
3723
83
    case NJS_TOKEN_VOID:
3724
83
        type = NJS_TOKEN_VOID;
3725
83
        operation = NJS_VMCODE_VOID;
3726
83
        break;
3727
3728
12.5k
    case NJS_TOKEN_TYPEOF:
3729
12.5k
        type = NJS_TOKEN_TYPEOF;
3730
12.5k
        operation = NJS_VMCODE_TYPEOF;
3731
12.5k
        break;
3732
3733
32.9k
    case NJS_TOKEN_ADDITION:
3734
32.9k
        type = NJS_TOKEN_UNARY_PLUS;
3735
32.9k
        operation = NJS_VMCODE_UNARY_PLUS;
3736
32.9k
        break;
3737
3738
20.7k
    case NJS_TOKEN_SUBTRACTION:
3739
20.7k
        type = NJS_TOKEN_UNARY_NEGATION;
3740
20.7k
        operation = NJS_VMCODE_UNARY_NEGATION;
3741
20.7k
        break;
3742
3743
21.5k
    case NJS_TOKEN_BITWISE_NOT:
3744
21.5k
        type = NJS_TOKEN_BITWISE_NOT;
3745
21.5k
        operation = NJS_VMCODE_BITWISE_NOT;
3746
21.5k
        break;
3747
3748
236k
    case NJS_TOKEN_LOGICAL_NOT:
3749
236k
        type = NJS_TOKEN_LOGICAL_NOT;
3750
236k
        operation = NJS_VMCODE_LOGICAL_NOT;
3751
236k
        break;
3752
3753
    /* AwaitExpression */
3754
658
    case NJS_TOKEN_AWAIT:
3755
658
        njs_parser_next(parser, njs_parser_await);
3756
658
        return NJS_OK;
3757
3758
2.85M
    default:
3759
2.85M
        njs_parser_next(parser, njs_parser_update_expression);
3760
3761
2.85M
        return njs_parser_after(parser, current, parser->target, 1,
3762
3.18M
                                njs_parser_unary_expression_after);
3763
3.18M
    }
3764
3765
326k
    node = njs_parser_node_new(parser, type);
3766
326k
    if (node == NULL) {
3767
0
        return NJS_ERROR;
3768
0
    }
3769
3770
326k
    node->token_line = token->line;
3771
326k
    node->u.operation = operation;
3772
3773
326k
    parser->target = node;
3774
3775
326k
    njs_lexer_consume_token(parser->lexer, 1);
3776
3777
326k
    return njs_parser_after(parser, current, node, 1,
3778
326k
                            njs_parser_unary_expression_next);
3779
326k
}
3780
3781
3782
static njs_int_t
3783
njs_parser_unary_expression_after(njs_parser_t *parser,
3784
    njs_lexer_token_t *token, njs_queue_link_t *current)
3785
2.61M
{
3786
2.61M
    if (parser->target == NULL &&
3787
2.53M
        token->type == NJS_TOKEN_EXPONENTIATION)
3788
4.57k
    {
3789
4.57k
        return njs_parser_exponentiation_expression_match(parser, token,
3790
4.57k
                                                          current);
3791
4.57k
    }
3792
3793
2.61M
    return njs_parser_stack_pop(parser);
3794
2.61M
}
3795
3796
3797
static njs_int_t
3798
njs_parser_unary_expression_next(njs_parser_t *parser,
3799
    njs_lexer_token_t *token, njs_queue_link_t *current)
3800
322k
{
3801
322k
    double             num;
3802
322k
    njs_token_type_t   type;
3803
322k
    njs_parser_node_t  *node, *prop;
3804
3805
322k
    type = parser->target->token_type;
3806
322k
    node = parser->node;
3807
3808
322k
    if (token->type == NJS_TOKEN_EXPONENTIATION) {
3809
28
        njs_parser_syntax_error(parser, "Either left-hand side or entire "
3810
28
                                "exponentiation must be parenthesized");
3811
28
        return NJS_DONE;
3812
28
    }
3813
3814
322k
    if (node->token_type == NJS_TOKEN_NUMBER) {
3815
21.6k
        if (type == NJS_TOKEN_UNARY_PLUS) {
3816
            /* Skip the unary plus of number. */
3817
4.40k
            return njs_parser_stack_pop(parser);
3818
4.40k
        }
3819
3820
17.2k
        if (type == NJS_TOKEN_UNARY_NEGATION) {
3821
            /* Optimization of common negative number. */
3822
7.71k
            num = -njs_number(&node->u.value);
3823
7.71k
            njs_set_number(&node->u.value, num);
3824
3825
7.71k
            return njs_parser_stack_pop(parser);
3826
7.71k
        }
3827
17.2k
    }
3828
3829
310k
    if (type == NJS_TOKEN_DELETE) {
3830
1.73k
        switch (node->token_type) {
3831
3832
748
        case NJS_TOKEN_PROPERTY:
3833
748
        case NJS_TOKEN_PROPERTY_REF:
3834
748
            node = njs_parser_property_ref(node);
3835
748
            if (node == NULL) {
3836
0
                return NJS_ERROR;
3837
0
            }
3838
3839
748
            node->token_type = NJS_TOKEN_PROPERTY_DELETE;
3840
748
            node->u.operation = NJS_VMCODE_PROPERTY_DELETE;
3841
3842
748
            return njs_parser_stack_pop(parser);
3843
3844
12
        case NJS_TOKEN_OPTIONAL_CHAIN:
3845
12
            prop = njs_parser_optional_chain_target_property(node);
3846
12
            if (prop != NULL) {
3847
11
                prop->token_type = NJS_TOKEN_PROPERTY_DELETE;
3848
11
                prop->u.operation = NJS_VMCODE_PROPERTY_DELETE;
3849
11
            }
3850
3851
12
            break;
3852
3853
148
        case NJS_TOKEN_NAME:
3854
148
            njs_parser_syntax_error(parser,
3855
148
                                    "Delete of an unqualified identifier");
3856
148
            return NJS_DONE;
3857
3858
826
        default:
3859
826
            break;
3860
1.73k
        }
3861
1.73k
    }
3862
3863
309k
    if (type == NJS_TOKEN_TYPEOF && node->token_type == NJS_TOKEN_NAME) {
3864
7.35k
        node->u.reference.type = NJS_TYPEOF;
3865
7.35k
    }
3866
3867
309k
    parser->target->left = parser->node;
3868
309k
    parser->target->left->dest = parser->target;
3869
309k
    parser->node = parser->target;
3870
3871
309k
    return njs_parser_stack_pop(parser);
3872
310k
}
3873
3874
3875
static njs_int_t
3876
njs_parser_await(njs_parser_t *parser, njs_lexer_token_t *token,
3877
    njs_queue_link_t *current)
3878
658
{
3879
658
    njs_parser_node_t   *node;
3880
3881
658
    if (!njs_function_scope(parser->scope)->async) {
3882
234
        njs_parser_syntax_error(parser,
3883
234
                                "await is only valid in async functions");
3884
234
        return NJS_ERROR;
3885
234
    }
3886
3887
424
    node = njs_parser_node_new(parser, NJS_TOKEN_AWAIT);
3888
424
    if (njs_slow_path(node == NULL)) {
3889
0
        return NJS_ERROR;
3890
0
    }
3891
3892
424
    node->token_line = token->line;
3893
3894
424
    njs_lexer_consume_token(parser->lexer, 1);
3895
3896
424
    parser->node = NULL;
3897
3898
424
    njs_parser_next(parser, njs_parser_unary_expression);
3899
3900
424
    return njs_parser_after(parser, current, node, 0,
3901
424
                            njs_parser_await_after);
3902
424
}
3903
3904
3905
static njs_int_t
3906
njs_parser_await_after(njs_parser_t *parser, njs_lexer_token_t *token,
3907
    njs_queue_link_t *current)
3908
421
{
3909
421
    if (parser->ret != NJS_OK) {
3910
3
        return njs_parser_failed(parser);
3911
3
    }
3912
3913
418
    return njs_parser_right_link_pop(parser);
3914
421
}
3915
3916
3917
/*
3918
 * 12.6 Exponentiation Operator.
3919
 */
3920
static njs_int_t
3921
njs_parser_exponentiation_expression(njs_parser_t *parser,
3922
    njs_lexer_token_t *token, njs_queue_link_t *current)
3923
2.86M
{
3924
2.86M
    parser->target = NULL;
3925
3926
2.86M
    if (parser->use_lhs == 0) {
3927
2.85M
        njs_parser_next(parser, njs_parser_unary_expression);
3928
3929
        /* For UpdateExpression, see njs_parser_unary_expression_after. */
3930
3931
2.85M
        return NJS_OK;
3932
2.85M
    } else {
3933
6.18k
        parser->use_lhs = 0;
3934
3935
6.18k
        return njs_parser_update_expression_post(parser, token, current);
3936
6.18k
    }
3937
2.86M
}
3938
3939
3940
static njs_int_t
3941
njs_parser_exponentiation_expression_match(njs_parser_t *parser,
3942
    njs_lexer_token_t *token, njs_queue_link_t *current)
3943
9.15k
{
3944
9.15k
    njs_parser_node_t  *node;
3945
3946
9.15k
    if (parser->target != NULL) {
3947
4.57k
        parser->target->right = parser->node;
3948
4.57k
        parser->target->right->dest = parser->target;
3949
4.57k
        parser->node = parser->target;
3950
3951
4.57k
        return njs_parser_stack_pop(parser);
3952
4.57k
    }
3953
3954
4.57k
    if (token->type != NJS_TOKEN_EXPONENTIATION) {
3955
0
        return njs_parser_stack_pop(parser);
3956
0
    }
3957
3958
4.57k
    node = njs_parser_node_new(parser, token->type);
3959
4.57k
    if (node == NULL) {
3960
0
        return NJS_ERROR;
3961
0
    }
3962
3963
4.57k
    node->token_line = token->line;
3964
4.57k
    node->u.operation = NJS_VMCODE_EXPONENTIATION;
3965
4.57k
    node->left = parser->node;
3966
4.57k
    node->left->dest = node;
3967
3968
4.57k
    njs_lexer_consume_token(parser->lexer, 1);
3969
3970
4.57k
    njs_parser_next(parser, njs_parser_exponentiation_expression);
3971
3972
4.57k
    return njs_parser_after(parser, current, node, 1,
3973
4.57k
                            njs_parser_exponentiation_expression_match);
3974
4.57k
}
3975
3976
3977
/*
3978
 * 12.7 Multiplicative Operators.
3979
 */
3980
static njs_int_t
3981
njs_parser_multiplicative_expression(njs_parser_t *parser,
3982
    njs_lexer_token_t *token, njs_queue_link_t *current)
3983
2.80M
{
3984
2.80M
    njs_parser_next(parser, njs_parser_exponentiation_expression);
3985
3986
2.80M
    return njs_parser_after(parser, current, NULL, 1,
3987
2.80M
                            njs_parser_multiplicative_expression_match);
3988
2.80M
}
3989
3990
3991
static njs_int_t
3992
njs_parser_multiplicative_expression_match(njs_parser_t *parser,
3993
    njs_lexer_token_t *token, njs_queue_link_t *current)
3994
2.61M
{
3995
2.61M
    njs_vmcode_t       operation;
3996
2.61M
    njs_parser_node_t  *node;
3997
3998
2.61M
    if (parser->target != NULL) {
3999
57.9k
        parser->target->right = parser->node;
4000
57.9k
        parser->target->right->dest = parser->target;
4001
57.9k
        parser->node = parser->target;
4002
57.9k
    }
4003
4004
2.61M
    switch (token->type) {
4005
13.4k
    case NJS_TOKEN_MULTIPLICATION:
4006
13.4k
        operation = NJS_VMCODE_MULTIPLICATION;
4007
13.4k
        break;
4008
4009
33.9k
    case NJS_TOKEN_DIVISION:
4010
33.9k
        operation = NJS_VMCODE_DIVISION;
4011
33.9k
        break;
4012
4013
11.2k
    case NJS_TOKEN_REMAINDER:
4014
11.2k
        operation = NJS_VMCODE_REMAINDER;
4015
11.2k
        break;
4016
4017
2.55M
    default:
4018
2.55M
        return njs_parser_stack_pop(parser);
4019
2.61M
    }
4020
4021
58.6k
    node = njs_parser_node_new(parser, token->type);
4022
58.6k
    if (node == NULL) {
4023
0
        return NJS_ERROR;
4024
0
    }
4025
4026
58.6k
    node->token_line = token->line;
4027
58.6k
    node->u.operation = operation;
4028
58.6k
    node->left = parser->node;
4029
58.6k
    node->left->dest = node;
4030
4031
58.6k
    njs_lexer_consume_token(parser->lexer, 1);
4032
4033
58.6k
    njs_parser_next(parser, njs_parser_exponentiation_expression);
4034
4035
58.6k
    return njs_parser_after(parser, current, node, 1,
4036
58.6k
                            njs_parser_multiplicative_expression_match);
4037
58.6k
}
4038
4039
4040
/*
4041
 * 12.8 Additive Operators.
4042
 */
4043
static njs_int_t
4044
njs_parser_additive_expression(njs_parser_t *parser, njs_lexer_token_t *token,
4045
    njs_queue_link_t *current)
4046
2.70M
{
4047
2.70M
    njs_parser_next(parser, njs_parser_multiplicative_expression);
4048
4049
2.70M
    return njs_parser_after(parser, current, NULL, 1,
4050
2.70M
                            njs_parser_additive_expression_match);
4051
2.70M
}
4052
4053
4054
static njs_int_t
4055
njs_parser_additive_expression_match(njs_parser_t *parser,
4056
    njs_lexer_token_t *token, njs_queue_link_t *current)
4057
2.55M
{
4058
2.55M
    njs_vmcode_t       operation;
4059
2.55M
    njs_parser_node_t  *node;
4060
4061
2.55M
    if (parser->target != NULL) {
4062
91.1k
        parser->target->right = parser->node;
4063
91.1k
        parser->target->right->dest = parser->target;
4064
91.1k
        parser->node = parser->target;
4065
91.1k
    }
4066
4067
2.55M
    switch (token->type) {
4068
56.3k
    case NJS_TOKEN_ADDITION:
4069
56.3k
        operation = NJS_VMCODE_ADDITION;
4070
56.3k
        break;
4071
4072
37.0k
    case NJS_TOKEN_SUBTRACTION:
4073
37.0k
        operation = NJS_VMCODE_SUBTRACTION;
4074
37.0k
        break;
4075
4076
2.46M
    default:
4077
2.46M
        return njs_parser_stack_pop(parser);
4078
2.55M
    }
4079
4080
93.3k
    node = njs_parser_node_new(parser, token->type);
4081
93.3k
    if (node == NULL) {
4082
0
        return NJS_ERROR;
4083
0
    }
4084
4085
93.3k
    node->token_line = token->line;
4086
93.3k
    node->u.operation = operation;
4087
93.3k
    node->left = parser->node;
4088
93.3k
    node->left->dest = node;
4089
4090
93.3k
    njs_lexer_consume_token(parser->lexer, 1);
4091
4092
93.3k
    njs_parser_next(parser, njs_parser_multiplicative_expression);
4093
4094
93.3k
    return njs_parser_after(parser, current, node, 1,
4095
93.3k
                            njs_parser_additive_expression_match);
4096
93.3k
}
4097
4098
4099
/*
4100
 * 12.9 Bitwise Shift Operators
4101
 */
4102
static njs_int_t
4103
njs_parser_shift_expression(njs_parser_t *parser, njs_lexer_token_t *token,
4104
    njs_queue_link_t *current)
4105
2.70M
{
4106
2.70M
    njs_parser_next(parser, njs_parser_additive_expression);
4107
4108
2.70M
    return njs_parser_after(parser, current, NULL, 1,
4109
2.70M
                            njs_parser_shift_expression_match);
4110
2.70M
}
4111
4112
4113
static njs_int_t
4114
njs_parser_shift_expression_match(njs_parser_t *parser,
4115
    njs_lexer_token_t *token, njs_queue_link_t *current)
4116
2.46M
{
4117
2.46M
    njs_vmcode_t       operation;
4118
2.46M
    njs_parser_node_t  *node;
4119
4120
2.46M
    if (parser->target != NULL) {
4121
3.82k
        parser->target->right = parser->node;
4122
3.82k
        parser->target->right->dest = parser->target;
4123
3.82k
        parser->node = parser->target;
4124
3.82k
    }
4125
4126
2.46M
    switch (token->type) {
4127
3.58k
    case NJS_TOKEN_RIGHT_SHIFT:
4128
3.58k
        operation = NJS_VMCODE_RIGHT_SHIFT;
4129
3.58k
        break;
4130
4131
403
    case NJS_TOKEN_LEFT_SHIFT:
4132
403
        operation = NJS_VMCODE_LEFT_SHIFT;
4133
403
        break;
4134
4135
249
    case NJS_TOKEN_UNSIGNED_RIGHT_SHIFT:
4136
249
        operation = NJS_VMCODE_UNSIGNED_RIGHT_SHIFT;
4137
249
        break;
4138
4139
2.45M
    default:
4140
2.45M
        return njs_parser_stack_pop(parser);
4141
2.46M
    }
4142
4143
4.24k
    node = njs_parser_node_new(parser, token->type);
4144
4.24k
    if (node == NULL) {
4145
0
        return NJS_ERROR;
4146
0
    }
4147
4148
4.24k
    node->token_line = token->line;
4149
4.24k
    node->u.operation = operation;
4150
4.24k
    node->left = parser->node;
4151
4.24k
    node->left->dest = node;
4152
4153
4.24k
    njs_lexer_consume_token(parser->lexer, 1);
4154
4155
4.24k
    njs_parser_next(parser, njs_parser_additive_expression);
4156
4157
4.24k
    return njs_parser_after(parser, current, node, 1,
4158
4.24k
                            njs_parser_shift_expression_match);
4159
4.24k
}
4160
4161
4162
/*
4163
 * 12.10 Relational Operators.
4164
 */
4165
static njs_int_t
4166
njs_parser_relational_expression(njs_parser_t *parser,
4167
    njs_lexer_token_t *token, njs_queue_link_t *current)
4168
2.67M
{
4169
2.67M
    njs_parser_next(parser, njs_parser_shift_expression);
4170
4171
2.67M
    return njs_parser_after(parser, current, NULL, 1,
4172
2.67M
                            njs_parser_relational_expression_match);
4173
2.67M
}
4174
4175
4176
static njs_int_t
4177
njs_parser_relational_expression_match(njs_parser_t *parser,
4178
    njs_lexer_token_t *token, njs_queue_link_t *current)
4179
2.45M
{
4180
2.45M
    njs_vmcode_t       operation;
4181
2.45M
    njs_parser_node_t  *node;
4182
4183
2.45M
    if (parser->target != NULL) {
4184
25.4k
        parser->target->right = parser->node;
4185
25.4k
        parser->target->right->dest = parser->target;
4186
25.4k
        parser->node = parser->target;
4187
25.4k
    }
4188
4189
2.45M
    switch (token->type) {
4190
6.89k
    case NJS_TOKEN_LESS:
4191
6.89k
        operation = NJS_VMCODE_LESS;
4192
6.89k
        break;
4193
4194
9.91k
    case NJS_TOKEN_GREATER:
4195
9.91k
        operation = NJS_VMCODE_GREATER;
4196
9.91k
        break;
4197
4198
4.11k
    case NJS_TOKEN_LESS_OR_EQUAL:
4199
4.11k
        operation = NJS_VMCODE_LESS_OR_EQUAL;
4200
4.11k
        break;
4201
4202
424
    case NJS_TOKEN_GREATER_OR_EQUAL:
4203
424
        operation = NJS_VMCODE_GREATER_OR_EQUAL;
4204
424
        break;
4205
4206
1.09k
    case NJS_TOKEN_INSTANCEOF:
4207
1.09k
        operation = NJS_VMCODE_INSTANCE_OF;
4208
1.09k
        break;
4209
4210
4.42k
    case NJS_TOKEN_IN:
4211
4.42k
        if (njs_lexer_in_fail_get(parser->lexer)) {
4212
139
            njs_parser_syntax_error(parser, "Invalid left-hand side in for-loop");
4213
139
            return NJS_ERROR;
4214
139
        }
4215
4.28k
        operation = NJS_VMCODE_PROPERTY_IN;
4216
4.28k
        break;
4217
4218
2.43M
    default:
4219
2.43M
        return njs_parser_stack_pop(parser);
4220
2.45M
    }
4221
4222
26.7k
    node = njs_parser_node_new(parser, token->type);
4223
26.7k
    if (node == NULL) {
4224
0
        return NJS_ERROR;
4225
0
    }
4226
4227
26.7k
    node->token_line = token->line;
4228
26.7k
    node->u.operation = operation;
4229
26.7k
    node->left = parser->node;
4230
26.7k
    node->left->dest = node;
4231
4232
26.7k
    njs_lexer_consume_token(parser->lexer, 1);
4233
4234
26.7k
    njs_parser_next(parser, njs_parser_shift_expression);
4235
4236
26.7k
    return njs_parser_after(parser, current, node, 1,
4237
26.7k
                            njs_parser_relational_expression_match);
4238
26.7k
}
4239
4240
4241
/*
4242
 * 12.11 Equality Operators.
4243
 */
4244
static njs_int_t
4245
njs_parser_equality_expression(njs_parser_t *parser, njs_lexer_token_t *token,
4246
    njs_queue_link_t *current)
4247
2.66M
{
4248
2.66M
    njs_parser_next(parser, njs_parser_relational_expression);
4249
4250
2.66M
    return njs_parser_after(parser, current, NULL, 1,
4251
2.66M
                            njs_parser_equality_expression_match);
4252
2.66M
}
4253
4254
4255
static njs_int_t
4256
njs_parser_equality_expression_match(njs_parser_t *parser,
4257
    njs_lexer_token_t *token, njs_queue_link_t *current)
4258
2.43M
{
4259
2.43M
    njs_vmcode_t       operation;
4260
2.43M
    njs_parser_node_t  *node;
4261
4262
2.43M
    if (parser->target != NULL) {
4263
6.98k
        parser->target->right = parser->node;
4264
6.98k
        parser->target->right->dest = parser->target;
4265
6.98k
        parser->node = parser->target;
4266
6.98k
    }
4267
4268
2.43M
    switch (token->type) {
4269
3.67k
    case NJS_TOKEN_EQUAL:
4270
3.67k
        operation = NJS_VMCODE_EQUAL;
4271
3.67k
        break;
4272
4273
258
    case NJS_TOKEN_NOT_EQUAL:
4274
258
        operation = NJS_VMCODE_NOT_EQUAL;
4275
258
        break;
4276
4277
1.23k
    case NJS_TOKEN_STRICT_EQUAL:
4278
1.23k
        operation = NJS_VMCODE_STRICT_EQUAL;
4279
1.23k
        break;
4280
4281
1.81k
    case NJS_TOKEN_STRICT_NOT_EQUAL:
4282
1.81k
        operation = NJS_VMCODE_STRICT_NOT_EQUAL;
4283
1.81k
        break;
4284
4285
2.42M
    default:
4286
2.42M
        return njs_parser_stack_pop(parser);
4287
2.43M
    }
4288
4289
6.98k
    node = njs_parser_node_new(parser, token->type);
4290
6.98k
    if (node == NULL) {
4291
0
        return NJS_ERROR;
4292
0
    }
4293
4294
6.98k
    node->token_line = token->line;
4295
6.98k
    node->u.operation = operation;
4296
6.98k
    node->left = parser->node;
4297
6.98k
    node->left->dest = node;
4298
4299
6.98k
    njs_lexer_consume_token(parser->lexer, 1);
4300
4301
6.98k
    njs_parser_next(parser, njs_parser_relational_expression);
4302
4303
6.98k
    return njs_parser_after(parser, current, node, 1,
4304
6.98k
                            njs_parser_equality_expression_match);
4305
6.98k
}
4306
4307
4308
/*
4309
 * 12.12 Binary Bitwise Operators.
4310
 */
4311
static njs_int_t
4312
njs_parser_bitwise_AND_expression(njs_parser_t *parser,
4313
    njs_lexer_token_t *token, njs_queue_link_t *current)
4314
2.65M
{
4315
2.65M
    njs_parser_next(parser, njs_parser_equality_expression);
4316
4317
2.65M
    return njs_parser_after(parser, current, NULL, 1,
4318
2.65M
                            njs_parser_bitwise_AND_expression_and);
4319
2.65M
}
4320
4321
4322
static njs_int_t
4323
njs_parser_bitwise_AND_expression_and(njs_parser_t *parser,
4324
    njs_lexer_token_t *token, njs_queue_link_t *current)
4325
2.42M
{
4326
2.42M
    njs_parser_next(parser, njs_parser_equality_expression);
4327
4328
2.42M
    return njs_parser_expression_node(parser, token, current,
4329
2.42M
                                       NJS_TOKEN_BITWISE_AND,
4330
2.42M
                                       NJS_VMCODE_BITWISE_AND,
4331
2.42M
                                       njs_parser_bitwise_AND_expression_and);
4332
2.42M
}
4333
4334
4335
static njs_int_t
4336
njs_parser_bitwise_XOR_expression(njs_parser_t *parser,
4337
    njs_lexer_token_t *token, njs_queue_link_t *current)
4338
2.64M
{
4339
2.64M
    njs_parser_next(parser, njs_parser_bitwise_AND_expression);
4340
4341
2.64M
    return njs_parser_after(parser, current, NULL, 1,
4342
2.64M
                            njs_parser_bitwise_XOR_expression_xor);
4343
2.64M
}
4344
4345
4346
static njs_int_t
4347
njs_parser_bitwise_XOR_expression_xor(njs_parser_t *parser,
4348
    njs_lexer_token_t *token, njs_queue_link_t *current)
4349
2.40M
{
4350
2.40M
    njs_parser_next(parser, njs_parser_bitwise_AND_expression);
4351
4352
2.40M
    return njs_parser_expression_node(parser, token, current,
4353
2.40M
                                       NJS_TOKEN_BITWISE_XOR,
4354
2.40M
                                       NJS_VMCODE_BITWISE_XOR,
4355
2.40M
                                       njs_parser_bitwise_XOR_expression_xor);
4356
2.40M
}
4357
4358
4359
static njs_int_t
4360
njs_parser_bitwise_OR_expression(njs_parser_t *parser,
4361
    njs_lexer_token_t *token, njs_queue_link_t *current)
4362
2.58M
{
4363
2.58M
    njs_parser_next(parser, njs_parser_bitwise_XOR_expression);
4364
4365
2.58M
    return njs_parser_after(parser, current, NULL, 1,
4366
2.58M
                            njs_parser_bitwise_OR_expression_or);
4367
2.58M
}
4368
4369
4370
static njs_int_t
4371
njs_parser_bitwise_OR_expression_or(njs_parser_t *parser,
4372
    njs_lexer_token_t *token, njs_queue_link_t *current)
4373
2.40M
{
4374
2.40M
    njs_parser_next(parser, njs_parser_bitwise_XOR_expression);
4375
4376
2.40M
    return njs_parser_expression_node(parser, token, current,
4377
2.40M
                                       NJS_TOKEN_BITWISE_OR,
4378
2.40M
                                       NJS_VMCODE_BITWISE_OR,
4379
2.40M
                                       njs_parser_bitwise_OR_expression_or);
4380
2.40M
}
4381
4382
4383
/*
4384
 * 12.13 Binary Logical Operators.
4385
 */
4386
static njs_int_t
4387
njs_parser_logical_AND_expression(njs_parser_t *parser,
4388
    njs_lexer_token_t *token, njs_queue_link_t *current)
4389
2.58M
{
4390
2.58M
    njs_parser_next(parser, njs_parser_bitwise_OR_expression);
4391
4392
2.58M
    return njs_parser_after(parser, current, NULL, 1,
4393
2.58M
                            njs_parser_logical_AND_expression_and);
4394
2.58M
}
4395
4396
4397
static njs_int_t
4398
njs_parser_logical_AND_expression_and(njs_parser_t *parser,
4399
    njs_lexer_token_t *token, njs_queue_link_t *current)
4400
2.34M
{
4401
2.34M
    njs_parser_next(parser, njs_parser_bitwise_OR_expression);
4402
4403
2.34M
    return njs_parser_expression_node(parser, token, current,
4404
2.34M
                                       NJS_TOKEN_LOGICAL_AND,
4405
2.34M
                                       NJS_VMCODE_TEST_IF_FALSE,
4406
2.34M
                                       njs_parser_logical_AND_expression_and);
4407
2.34M
}
4408
4409
4410
static njs_int_t
4411
njs_parser_logical_OR_expression(njs_parser_t *parser,
4412
    njs_lexer_token_t *token, njs_queue_link_t *current)
4413
2.58M
{
4414
2.58M
    njs_parser_next(parser, njs_parser_logical_AND_expression);
4415
4416
2.58M
    return njs_parser_after(parser, current, NULL, 1,
4417
2.58M
                            njs_parser_logical_OR_expression_or);
4418
2.58M
}
4419
4420
4421
static njs_int_t
4422
njs_parser_logical_OR_expression_or(njs_parser_t *parser,
4423
    njs_lexer_token_t *token, njs_queue_link_t *current)
4424
2.33M
{
4425
2.33M
    njs_parser_next(parser, njs_parser_logical_AND_expression);
4426
4427
2.33M
    return njs_parser_expression_node(parser, token, current,
4428
2.33M
                                       NJS_TOKEN_LOGICAL_OR,
4429
2.33M
                                       NJS_VMCODE_TEST_IF_TRUE,
4430
2.33M
                                       njs_parser_logical_OR_expression_or);
4431
2.33M
}
4432
4433
4434
static njs_int_t
4435
njs_parser_coalesce_expression(njs_parser_t *parser, njs_lexer_token_t *token,
4436
    njs_queue_link_t *current)
4437
2.33M
{
4438
2.33M
    njs_token_type_t   type;
4439
2.33M
    njs_parser_node_t  *node;
4440
4441
2.33M
    node = parser->node;
4442
4443
2.33M
    if (parser->target != NULL) {
4444
211
        parser->target->right = node;
4445
211
        parser->target->right->dest = parser->target;
4446
211
        parser->node = parser->target;
4447
211
    }
4448
4449
2.33M
    if (token->type != NJS_TOKEN_COALESCE) {
4450
2.33M
        return njs_parser_stack_pop(parser);
4451
2.33M
    }
4452
4453
483
    type = node->token_type;
4454
4455
483
    if (parser->lexer->prev_type != NJS_TOKEN_CLOSE_PARENTHESIS
4456
413
        && (type == NJS_TOKEN_LOGICAL_OR || type == NJS_TOKEN_LOGICAL_AND))
4457
81
    {
4458
81
        return njs_parser_failed(parser);
4459
81
    }
4460
4461
402
    node = njs_parser_node_new(parser, NJS_TOKEN_COALESCE);
4462
402
    if (node == NULL) {
4463
0
        return NJS_ERROR;
4464
0
    }
4465
4466
402
    node->token_line = token->line;
4467
402
    node->u.operation = NJS_VMCODE_COALESCE;
4468
402
    node->left = parser->node;
4469
402
    node->left->dest = node;
4470
4471
402
    njs_lexer_consume_token(parser->lexer, 1);
4472
402
    njs_parser_next(parser, njs_parser_bitwise_OR_expression);
4473
4474
402
    return njs_parser_after(parser, current, node, 1,
4475
402
                            njs_parser_coalesce_expression);
4476
402
}
4477
4478
4479
static njs_int_t
4480
njs_parser_short_circuit_expression(njs_parser_t *parser,
4481
    njs_lexer_token_t *token, njs_queue_link_t *current)
4482
2.58M
{
4483
2.58M
    njs_parser_next(parser, njs_parser_logical_OR_expression);
4484
4485
2.58M
    return njs_parser_after(parser, current, NULL, 1,
4486
2.58M
                            njs_parser_coalesce_expression);
4487
2.58M
}
4488
4489
4490
/*
4491
 * 12.14 Conditional Operator ( ? : ).
4492
 */
4493
static njs_int_t
4494
njs_parser_conditional_expression(njs_parser_t *parser,
4495
    njs_lexer_token_t *token, njs_queue_link_t *current)
4496
2.58M
{
4497
2.58M
    njs_parser_next(parser, njs_parser_short_circuit_expression);
4498
4499
2.58M
    return njs_parser_after(parser, current, NULL, 1,
4500
2.58M
                            njs_parser_conditional_question_mark);
4501
2.58M
}
4502
4503
4504
static njs_int_t
4505
njs_parser_conditional_question_mark(njs_parser_t *parser,
4506
    njs_lexer_token_t *token, njs_queue_link_t *current)
4507
2.33M
{
4508
2.33M
    njs_parser_node_t  *node, *cond;
4509
4510
2.33M
    if (token->type != NJS_TOKEN_CONDITIONAL) {
4511
2.29M
        return njs_parser_stack_pop(parser);
4512
2.29M
    }
4513
4514
41.5k
    cond = njs_parser_node_new(parser, NJS_TOKEN_CONDITIONAL);
4515
41.5k
    if (cond == NULL) {
4516
0
        return NJS_ERROR;
4517
0
    }
4518
4519
41.5k
    cond->token_line = token->line;
4520
41.5k
    cond->left = parser->node;
4521
4522
41.5k
    node = njs_parser_node_new(parser, NJS_TOKEN_BRANCHING);
4523
41.5k
    if (node == NULL) {
4524
0
        return NJS_ERROR;
4525
0
    }
4526
4527
41.5k
    node->token_line = token->line;
4528
41.5k
    cond->right = node;
4529
4530
41.5k
    njs_lexer_consume_token(parser->lexer, 1);
4531
4532
41.5k
    if (njs_lexer_in_stack_push(parser->lexer) != NJS_OK) {
4533
0
        return NJS_ERROR;
4534
0
    }
4535
4536
41.5k
    njs_parser_next(parser, njs_parser_assignment_expression);
4537
4538
41.5k
    return njs_parser_after(parser, current, cond, 1,
4539
41.5k
                            njs_parser_conditional_colon);
4540
41.5k
}
4541
4542
4543
static njs_int_t
4544
njs_parser_conditional_colon(njs_parser_t *parser, njs_lexer_token_t *token,
4545
    njs_queue_link_t *current)
4546
40.2k
{
4547
40.2k
    njs_parser_node_t  *node;
4548
4549
40.2k
    if (token->type != NJS_TOKEN_COLON) {
4550
137
        return njs_parser_failed(parser);
4551
137
    }
4552
4553
40.0k
    njs_lexer_in_stack_pop(parser->lexer);
4554
4555
40.0k
    njs_lexer_consume_token(parser->lexer, 1);
4556
4557
40.0k
    node = parser->target->right;
4558
4559
40.0k
    node->left = parser->node;
4560
40.0k
    node->left->dest = parser->target;
4561
4562
40.0k
    njs_parser_next(parser, njs_parser_assignment_expression);
4563
4564
40.0k
    return njs_parser_after(parser, current, parser->target, 1,
4565
40.2k
                            njs_parser_conditional_colon_after);
4566
40.2k
}
4567
4568
4569
static njs_int_t
4570
njs_parser_conditional_colon_after(njs_parser_t *parser,
4571
    njs_lexer_token_t *token, njs_queue_link_t *current)
4572
37.9k
{
4573
37.9k
    njs_parser_node_t  *node;
4574
4575
37.9k
    node = parser->target->right;
4576
4577
37.9k
    node->right = parser->node;
4578
37.9k
    node->right->dest = parser->target;
4579
4580
37.9k
    parser->node = parser->target;
4581
4582
37.9k
    return njs_parser_stack_pop(parser);
4583
37.9k
}
4584
4585
4586
/*
4587
 * 12.15 Assignment Operators.
4588
 */
4589
static njs_int_t
4590
njs_parser_assignment_expression(njs_parser_t *parser,
4591
    njs_lexer_token_t *token, njs_queue_link_t *current)
4592
2.66M
{
4593
2.66M
    njs_int_t  ret;
4594
4595
2.66M
    if (!parser->use_lhs) {
4596
2.66M
        ret = njs_parser_match_arrow_expression(parser, token);
4597
2.66M
        if (ret == NJS_OK) {
4598
85.2k
            njs_parser_next(parser, njs_parser_arrow_function);
4599
4600
85.2k
            return NJS_OK;
4601
4602
2.57M
        } else if (ret == NJS_ERROR) {
4603
0
            return NJS_ERROR;
4604
0
        }
4605
2.66M
    }
4606
4607
2.58M
    njs_parser_next(parser, njs_parser_conditional_expression);
4608
4609
2.58M
    return njs_parser_after(parser, current, NULL, 1,
4610
2.66M
                            njs_parser_assignment_expression_after);
4611
2.66M
}
4612
4613
4614
/*
4615
 * TODO: this function is a crutch.
4616
 * See NJS_TOKEN_OPEN_PARENTHESIS in njs_parser_primary_expression_test.
4617
 * and look CoverParenthesizedExpressionAndArrowParameterList in spec.
4618
 */
4619
static njs_int_t
4620
njs_parser_match_arrow_expression(njs_parser_t *parser,
4621
    njs_lexer_token_t *token)
4622
2.67M
{
4623
2.67M
    njs_bool_t  rest_parameters;
4624
4625
2.67M
    if (token->type == NJS_TOKEN_ASYNC) {
4626
2.72k
        token = njs_lexer_peek_token(parser->lexer, token, 1);
4627
2.72k
        if (token == NULL) {
4628
0
            return NJS_ERROR;
4629
0
        }
4630
2.72k
    }
4631
4632
2.67M
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS
4633
2.47M
        && !njs_lexer_token_is_binding_identifier(token))
4634
1.33M
    {
4635
1.33M
        return NJS_DECLINED;
4636
1.33M
    }
4637
4638
1.34M
    if (njs_lexer_token_is_binding_identifier(token)) {
4639
1.14M
        goto arrow;
4640
1.14M
    }
4641
4642
198k
    token = njs_lexer_peek_token(parser->lexer, token, 0);
4643
198k
    if (token == NULL) {
4644
0
        return NJS_ERROR;
4645
0
    }
4646
4647
198k
    rest_parameters = 0;
4648
4649
208k
    while (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
4650
4651
204k
        if (rest_parameters) {
4652
0
            return NJS_DECLINED;
4653
0
        }
4654
4655
204k
        if (token->type == NJS_TOKEN_ELLIPSIS) {
4656
16
            rest_parameters = 1;
4657
4658
16
            token = njs_lexer_peek_token(parser->lexer, token, 0);
4659
16
            if (token == NULL) {
4660
0
                return NJS_ERROR;
4661
0
            }
4662
16
        }
4663
4664
204k
        if (!njs_lexer_token_is_binding_identifier(token)) {
4665
194k
            return NJS_DECLINED;
4666
194k
        }
4667
4668
9.88k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
4669
9.88k
        if (token == NULL) {
4670
0
            return NJS_ERROR;
4671
0
        }
4672
4673
9.88k
        if (token->type == NJS_TOKEN_COMMA) {
4674
991
            token = njs_lexer_peek_token(parser->lexer, token, 0);
4675
991
            if (token == NULL) {
4676
0
               return NJS_ERROR;
4677
0
            }
4678
991
        }
4679
9.88k
    }
4680
4681
1.14M
arrow:
4682
4683
1.14M
    token = njs_lexer_peek_token(parser->lexer, token, 1);
4684
1.14M
    if (token == NULL) {
4685
0
        return NJS_ERROR;
4686
0
    }
4687
4688
1.14M
    if (token->type == NJS_TOKEN_LINE_END) {
4689
476k
        return NJS_DECLINED;
4690
476k
    }
4691
4692
669k
    if (token->type != NJS_TOKEN_ARROW) {
4693
584k
        return NJS_DECLINED;
4694
584k
    }
4695
4696
85.2k
    return NJS_OK;
4697
669k
}
4698
4699
4700
static njs_int_t
4701
njs_parser_assignment_expression_after(njs_parser_t *parser,
4702
    njs_lexer_token_t *token, njs_queue_link_t *current)
4703
2.33M
{
4704
2.33M
    return njs_parser_assignment_operator(parser, token, current);
4705
2.33M
}
4706
4707
4708
static njs_int_t
4709
njs_parser_assignment_operator(njs_parser_t *parser, njs_lexer_token_t *token,
4710
    njs_queue_link_t *current)
4711
2.33M
{
4712
2.33M
    njs_vmcode_t       operation;
4713
2.33M
    njs_token_type_t   type;
4714
2.33M
    njs_parser_node_t  *node;
4715
4716
2.33M
    switch (token->type) {
4717
158k
    case NJS_TOKEN_ASSIGNMENT:
4718
158k
        njs_thread_log_debug("JS: =");
4719
158k
        operation = NJS_VMCODE_MOVE;
4720
158k
        break;
4721
4722
6.85k
    case NJS_TOKEN_MULTIPLICATION_ASSIGNMENT:
4723
6.85k
        njs_thread_log_debug("JS: *=");
4724
6.85k
        operation = NJS_VMCODE_MULTIPLICATION;
4725
6.85k
        break;
4726
4727
8.80k
    case NJS_TOKEN_DIVISION_ASSIGNMENT:
4728
8.80k
        njs_thread_log_debug("JS: /=");
4729
8.80k
        operation = NJS_VMCODE_DIVISION;
4730
8.80k
        break;
4731
4732
25.9k
    case NJS_TOKEN_REMAINDER_ASSIGNMENT:
4733
25.9k
        njs_thread_log_debug("JS: %=");
4734
25.9k
        operation = NJS_VMCODE_REMAINDER;
4735
25.9k
        break;
4736
4737
8.43k
    case NJS_TOKEN_ADDITION_ASSIGNMENT:
4738
8.43k
        njs_thread_log_debug("JS: +=");
4739
8.43k
        operation = NJS_VMCODE_ADDITION;
4740
8.43k
        break;
4741
4742
5.27k
    case NJS_TOKEN_SUBTRACTION_ASSIGNMENT:
4743
5.27k
        njs_thread_log_debug("JS: -=");
4744
5.27k
        operation = NJS_VMCODE_SUBTRACTION;
4745
5.27k
        break;
4746
4747
15
    case NJS_TOKEN_LEFT_SHIFT_ASSIGNMENT:
4748
15
        njs_thread_log_debug("JS: <<=");
4749
15
        operation = NJS_VMCODE_LEFT_SHIFT;
4750
15
        break;
4751
4752
1.99k
    case NJS_TOKEN_RIGHT_SHIFT_ASSIGNMENT:
4753
1.99k
        njs_thread_log_debug("JS: >>=");
4754
1.99k
        operation = NJS_VMCODE_RIGHT_SHIFT;
4755
1.99k
        break;
4756
4757
3.59k
    case NJS_TOKEN_UNSIGNED_RIGHT_SHIFT_ASSIGNMENT:
4758
3.59k
        njs_thread_log_debug("JS: >>>=");
4759
3.59k
        operation = NJS_VMCODE_UNSIGNED_RIGHT_SHIFT;
4760
3.59k
        break;
4761
4762
6.38k
    case NJS_TOKEN_BITWISE_AND_ASSIGNMENT:
4763
6.38k
        njs_thread_log_debug("JS: &=");
4764
6.38k
        operation = NJS_VMCODE_BITWISE_AND;
4765
6.38k
        break;
4766
4767
621
    case NJS_TOKEN_BITWISE_XOR_ASSIGNMENT:
4768
621
        njs_thread_log_debug("JS: ^=");
4769
621
        operation = NJS_VMCODE_BITWISE_XOR;
4770
621
        break;
4771
4772
6.42k
    case NJS_TOKEN_BITWISE_OR_ASSIGNMENT:
4773
6.42k
        njs_thread_log_debug("JS: |=");
4774
6.42k
        operation = NJS_VMCODE_BITWISE_OR;
4775
6.42k
        break;
4776
4777
167
    case NJS_TOKEN_EXPONENTIATION_ASSIGNMENT:
4778
167
        njs_thread_log_debug("JS: **=");
4779
167
        operation = NJS_VMCODE_EXPONENTIATION;
4780
167
        break;
4781
4782
107
    case NJS_TOKEN_LOGICAL_OR_ASSIGNMENT:
4783
107
        njs_thread_log_debug("JS: ||=");
4784
107
        operation = NJS_VMCODE_TEST_IF_TRUE;
4785
107
        break;
4786
4787
0
    case NJS_TOKEN_LOGICAL_AND_ASSIGNMENT:
4788
0
        njs_thread_log_debug("JS: &&=");
4789
0
        operation = NJS_VMCODE_TEST_IF_FALSE;
4790
0
        break;
4791
4792
0
    case NJS_TOKEN_COALESCE_ASSIGNMENT:
4793
0
        njs_thread_log_debug("JS: ?\?=");
4794
0
        operation = NJS_VMCODE_COALESCE;
4795
0
        break;
4796
4797
2.10M
    default:
4798
2.10M
        return njs_parser_stack_pop(parser);
4799
2.33M
    }
4800
4801
233k
    if (!njs_parser_is_lvalue(parser->node)) {
4802
1.64k
        type = parser->node->token_type;
4803
4804
1.64k
        if (njs_parser_restricted_identifier(type)) {
4805
1.37k
            njs_parser_syntax_error(parser, "Identifier \"%s\" "
4806
1.37k
                                    "is forbidden as left-hand in assignment",
4807
1.37k
                                    (type == NJS_TOKEN_EVAL) ? "eval"
4808
1.37k
                                                             : "arguments");
4809
4810
1.37k
        } else {
4811
264
            njs_parser_ref_error(parser,
4812
264
                                 "Invalid left-hand side in assignment");
4813
264
        }
4814
4815
1.64k
        return NJS_DONE;
4816
1.64k
    }
4817
4818
231k
    parser->node = njs_parser_lvalue_ref(parser->node);
4819
231k
    if (parser->node == NULL) {
4820
0
        return NJS_ERROR;
4821
0
    }
4822
4823
231k
    node = njs_parser_node_new(parser, token->type);
4824
231k
    if (node == NULL) {
4825
0
        return NJS_ERROR;
4826
0
    }
4827
4828
231k
    node->token_line = token->line;
4829
231k
    node->u.operation = operation;
4830
231k
    node->left = parser->node;
4831
4832
231k
    njs_lexer_consume_token(parser->lexer, 1);
4833
231k
    njs_parser_next(parser, njs_parser_assignment_expression);
4834
4835
231k
    return njs_parser_after(parser, current, node, 1,
4836
231k
                            njs_parser_assignment_operator_after);
4837
231k
}
4838
4839
4840
static njs_int_t
4841
njs_parser_assignment_operator_after(njs_parser_t *parser,
4842
    njs_lexer_token_t *token, njs_queue_link_t *current)
4843
225k
{
4844
225k
    return njs_parser_right_link_pop(parser);
4845
225k
}
4846
4847
4848
/*
4849
 * 12.16 Comma Operator ( , ).
4850
 */
4851
static njs_int_t
4852
njs_parser_expression(njs_parser_t *parser, njs_lexer_token_t *token,
4853
    njs_queue_link_t *current)
4854
1.19M
{
4855
1.19M
    njs_parser_next(parser, njs_parser_assignment_expression);
4856
4857
1.19M
    return njs_parser_after(parser, current, NULL, 1,
4858
1.19M
                            njs_parser_expression_comma);
4859
1.19M
}
4860
4861
4862
static njs_int_t
4863
njs_parser_expression_comma(njs_parser_t *parser, njs_lexer_token_t *token,
4864
    njs_queue_link_t *current)
4865
1.00M
{
4866
1.00M
    njs_parser_next(parser, njs_parser_assignment_expression);
4867
4868
1.00M
    return njs_parser_expression_node(parser, token, current, NJS_TOKEN_COMMA,
4869
1.00M
                                       0,
4870
1.00M
                                       njs_parser_expression_comma);
4871
1.00M
}
4872
4873
4874
/*
4875
 * 13 Statements and Declarations.
4876
 */
4877
static njs_int_t
4878
njs_parser_statement(njs_parser_t *parser, njs_lexer_token_t *token,
4879
    njs_queue_link_t *current)
4880
1.20M
{
4881
1.20M
    njs_int_t                 ret;
4882
1.20M
    njs_queue_link_t          *lnk;
4883
1.20M
    njs_parser_stack_entry_t  *entry;
4884
4885
1.20M
    if (token->type == NJS_TOKEN_END) {
4886
94.0k
        lnk = njs_queue_next(njs_queue_first(&parser->stack));
4887
4888
94.0k
        if (lnk == njs_queue_head(&parser->stack)) {
4889
0
            return njs_parser_reject(parser);
4890
0
        }
4891
4892
94.0k
        entry = njs_queue_link_data(lnk, njs_parser_stack_entry_t, link);
4893
4894
94.0k
        if (entry->state == njs_parser_check_error_state) {
4895
90.7k
            return NJS_DONE;
4896
90.7k
        }
4897
4898
3.27k
        return njs_parser_reject(parser);
4899
94.0k
    }
4900
4901
1.11M
    switch (token->type) {
4902
86.3k
    case NJS_TOKEN_SEMICOLON:
4903
86.3k
        njs_lexer_consume_token(parser->lexer, 1);
4904
86.3k
        return njs_parser_stack_pop(parser);
4905
4906
226
    case NJS_TOKEN_EXPORT:
4907
226
        parser->line = token->line;
4908
4909
226
        njs_lexer_consume_token(parser->lexer, 1);
4910
226
        njs_parser_next(parser, njs_parser_export);
4911
4912
226
        return njs_parser_after(parser, current, parser->node, 1,
4913
0
                                njs_parser_statement_after);
4914
4915
831
    case NJS_TOKEN_IMPORT:
4916
831
        parser->line = token->line;
4917
4918
831
        njs_lexer_consume_token(parser->lexer, 1);
4919
831
        njs_parser_next(parser, njs_parser_import);
4920
4921
831
        return njs_parser_after(parser, current, parser->node, 1,
4922
0
                                njs_parser_statement_after);
4923
1.02M
    default:
4924
1.02M
        break;
4925
1.11M
    }
4926
4927
1.02M
    ret = njs_parser_statement_wo_node(parser, token, current);
4928
1.02M
    if (ret != NJS_OK) {
4929
0
        return ret;
4930
0
    }
4931
4932
1.02M
    return njs_parser_after(parser, current, parser->node, 1,
4933
1.02M
                            njs_parser_statement_after);
4934
1.02M
}
4935
4936
4937
static njs_int_t
4938
njs_parser_statement_wo_node(njs_parser_t *parser, njs_lexer_token_t *token,
4939
    njs_queue_link_t *current)
4940
1.09M
{
4941
1.09M
    switch (token->type) {
4942
60.4k
    case NJS_TOKEN_OPEN_BRACE:
4943
60.4k
        njs_parser_next(parser, njs_parser_block_statement);
4944
60.4k
        break;
4945
4946
11.3k
    case NJS_TOKEN_VAR:
4947
11.3k
        njs_lexer_consume_token(parser->lexer, 1);
4948
11.3k
        return njs_parser_variable_statement(parser, token, current);
4949
4950
2.96k
    case NJS_TOKEN_SEMICOLON:
4951
2.96k
        njs_lexer_consume_token(parser->lexer, 1);
4952
2.96k
        return njs_parser_stack_pop(parser);
4953
4954
2.74k
    case NJS_TOKEN_IF:
4955
2.74k
        njs_parser_next(parser, njs_parser_if_statement);
4956
2.74k
        break;
4957
4958
    /* BreakableStatement */
4959
2.54k
    case NJS_TOKEN_DO:
4960
2.54k
        njs_parser_next(parser, njs_parser_iteration_statement_do);
4961
2.54k
        break;
4962
4963
1.95k
    case NJS_TOKEN_WHILE:
4964
1.95k
        njs_parser_next(parser, njs_parser_iteration_statement_while);
4965
1.95k
        break;
4966
4967
32.1k
    case NJS_TOKEN_FOR:
4968
32.1k
        njs_parser_next(parser, njs_parser_iteration_statement_for);
4969
32.1k
        break;
4970
4971
8.45k
    case NJS_TOKEN_SWITCH:
4972
8.45k
        njs_parser_next(parser, njs_parser_switch_statement);
4973
8.45k
        break;
4974
4975
1.49k
    case NJS_TOKEN_CONTINUE:
4976
1.49k
        njs_parser_next(parser, njs_parser_continue_statement);
4977
1.49k
        break;
4978
4979
19.4k
    case NJS_TOKEN_BREAK:
4980
19.4k
        njs_parser_next(parser, njs_parser_break_statement);
4981
19.4k
        break;
4982
4983
13.5k
    case NJS_TOKEN_RETURN:
4984
13.5k
        njs_parser_next(parser, njs_parser_return_statement);
4985
13.5k
        break;
4986
4987
15
    case NJS_TOKEN_WITH:
4988
15
        njs_parser_next(parser, njs_parser_with_statement);
4989
15
        break;
4990
4991
24.4k
    case NJS_TOKEN_THROW:
4992
24.4k
        njs_parser_next(parser, njs_parser_throw_statement);
4993
24.4k
        break;
4994
4995
23.2k
    case NJS_TOKEN_TRY:
4996
23.2k
        njs_parser_next(parser, njs_parser_try_statement);
4997
23.2k
        break;
4998
4999
7.33k
    case NJS_TOKEN_DEBUGGER:
5000
7.33k
        njs_parser_next(parser, njs_parser_debugger_statement);
5001
7.33k
        break;
5002
5003
212
    case NJS_TOKEN_END:
5004
212
        return njs_parser_failed(parser);
5005
5006
881k
    default:
5007
881k
        if (njs_lexer_token_is_identifier_reference(token)) {
5008
649k
            token = njs_lexer_peek_token(parser->lexer, token, 0);
5009
649k
            if (token == NULL) {
5010
0
                return NJS_ERROR;
5011
0
            }
5012
5013
649k
            if (token->type == NJS_TOKEN_COLON) {
5014
35.3k
                njs_parser_next(parser, njs_parser_labelled_statement);
5015
35.3k
                return NJS_OK;
5016
35.3k
            }
5017
649k
        }
5018
5019
845k
        njs_parser_next(parser, njs_parser_expression_statement);
5020
845k
        return NJS_OK;
5021
1.09M
    }
5022
5023
197k
    parser->line = token->line;
5024
5025
197k
    njs_lexer_consume_token(parser->lexer, 1);
5026
5027
197k
    return NJS_OK;
5028
1.09M
}
5029
5030
5031
static njs_int_t
5032
njs_parser_statement_after(njs_parser_t *parser, njs_lexer_token_t *token,
5033
    njs_queue_link_t *current)
5034
903k
{
5035
903k
    njs_parser_node_t  *stmt, *last, *new_node, **child;
5036
903k
    njs_parser_node_t  *node, *top;
5037
5038
903k
    child = &parser->target;
5039
903k
    last = *child;
5040
5041
903k
    new_node = parser->node;
5042
5043
903k
    if (new_node != NULL) {
5044
903k
        if (new_node->hoist) {
5045
0
            child = &njs_parser_chain_top(parser);
5046
5047
0
            while (*child != NULL) {
5048
0
                node = *child;
5049
5050
0
                if (node->hoist) {
5051
0
                    break;
5052
0
                }
5053
5054
0
                child = &node->left;
5055
0
            }
5056
5057
0
            last = *child;
5058
0
        }
5059
5060
903k
        stmt = njs_parser_node_new(parser, NJS_TOKEN_STATEMENT);
5061
903k
        if (njs_slow_path(stmt == NULL)) {
5062
0
            return NJS_ERROR;
5063
0
        }
5064
5065
903k
        stmt->hoist = new_node->hoist;
5066
903k
        stmt->left = last;
5067
903k
        stmt->right = new_node;
5068
5069
903k
        *child = stmt;
5070
5071
903k
        top = (child != &parser->target) ? njs_parser_chain_top(parser)
5072
903k
                                         : stmt;
5073
5074
903k
        parser->node = top;
5075
5076
903k
       njs_parser_chain_top_set(parser, top);
5077
903k
    }
5078
5079
903k
    return njs_parser_stack_pop(parser);
5080
903k
}
5081
5082
5083
static njs_int_t
5084
njs_parser_declaration(njs_parser_t *parser, njs_lexer_token_t *token,
5085
    njs_queue_link_t *current)
5086
1.23M
{
5087
1.23M
    njs_int_t  ret;
5088
5089
1.23M
    ret = njs_parser_hoistable_declaration(parser, token, current);
5090
1.23M
    if (ret == NJS_OK) {
5091
5.78k
        return NJS_OK;
5092
5.78k
    }
5093
5094
1.23M
    switch (token->type) {
5095
0
    case NJS_TOKEN_CLASS:
5096
0
        njs_parser_next(parser, njs_parser_class_declaration);
5097
0
        return NJS_OK;
5098
5099
8.85k
    case NJS_TOKEN_LET:
5100
20.6k
    case NJS_TOKEN_CONST:
5101
20.6k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
5102
20.6k
        if (token == NULL) {
5103
0
            return NJS_ERROR;
5104
0
        }
5105
5106
20.6k
        switch (token->type) {
5107
152
        case NJS_TOKEN_OPEN_BRACE:
5108
152
        case NJS_TOKEN_OPEN_BRACKET:
5109
152
            njs_parser_next(parser, njs_parser_lexical_declaration);
5110
152
            break;
5111
5112
20.5k
        default:
5113
20.5k
            if (njs_lexer_token_is_reserved(token)) {
5114
140
                njs_lexer_consume_token(parser->lexer, 1);
5115
140
                njs_parser_next(parser, njs_parser_failed_state);
5116
140
                return NJS_OK;
5117
140
            }
5118
5119
20.3k
            if (njs_lexer_token_is_binding_identifier(token)) {
5120
20.3k
                njs_parser_next(parser, njs_parser_lexical_declaration);
5121
20.3k
                break;
5122
20.3k
            }
5123
5124
2
            return NJS_DECLINED;
5125
20.6k
        }
5126
5127
20.5k
        break;
5128
5129
1.20M
    default:
5130
1.20M
        return NJS_DECLINED;
5131
1.23M
    }
5132
5133
20.5k
    return njs_parser_after(parser, current, parser->node, 1,
5134
1.23M
                            njs_parser_statement_after);
5135
1.23M
}
5136
5137
5138
static njs_int_t
5139
njs_parser_hoistable_declaration(njs_parser_t *parser,
5140
    njs_lexer_token_t *token, njs_queue_link_t *current)
5141
1.23M
{
5142
1.23M
    njs_int_t  ret;
5143
5144
1.23M
    ret = njs_parser_function_or_generator(parser, token, current);
5145
1.23M
    if (ret == NJS_OK) {
5146
5.59k
        return NJS_OK;
5147
5.59k
    }
5148
5149
1.23M
    ret = njs_parser_async_function_or_generator(parser, token, current);
5150
1.23M
    if (ret == NJS_OK) {
5151
193
        return NJS_OK;
5152
193
    }
5153
5154
1.23M
    return NJS_DECLINED;
5155
1.23M
}
5156
5157
5158
/*
5159
 * 13.2 Block.
5160
 */
5161
static njs_int_t
5162
njs_parser_block_statement(njs_parser_t *parser, njs_lexer_token_t *token,
5163
    njs_queue_link_t *current)
5164
85.4k
{
5165
85.4k
    void       *target;
5166
85.4k
    njs_int_t  ret;
5167
5168
85.4k
    ret = njs_parser_scope_begin(parser, NJS_SCOPE_BLOCK, 0);
5169
85.4k
    if (ret != NJS_OK) {
5170
0
        return NJS_ERROR;
5171
0
    }
5172
5173
85.4k
    target = (void *) (uintptr_t) parser->line;
5174
85.4k
    parser->node = NULL;
5175
5176
85.4k
    if (token->type == NJS_TOKEN_CLOSE_BRACE) {
5177
7.16k
        parser->target = target;
5178
5179
7.16k
        njs_parser_next(parser, njs_parser_block_statement_close_brace);
5180
7.16k
        return NJS_OK;
5181
7.16k
    }
5182
5183
78.2k
    njs_parser_next(parser, njs_parser_statement_list);
5184
5185
78.2k
    return njs_parser_after(parser, current, target, 0,
5186
85.4k
                            njs_parser_block_statement_close_brace);
5187
85.4k
}
5188
5189
5190
static njs_int_t
5191
njs_parser_block_statement_open_brace(njs_parser_t *parser,
5192
    njs_lexer_token_t *token, njs_queue_link_t *current)
5193
25.2k
{
5194
25.2k
    if (token->type != NJS_TOKEN_OPEN_BRACE) {
5195
167
        return njs_parser_failed(parser);
5196
167
    }
5197
5198
25.0k
    parser->line = token->line;
5199
5200
25.0k
    njs_lexer_consume_token(parser->lexer, 1);
5201
5202
25.0k
    token = njs_lexer_token(parser->lexer, 0);
5203
25.0k
    if (token == NULL) {
5204
0
        return NJS_ERROR;
5205
0
    }
5206
5207
25.0k
    return njs_parser_block_statement(parser, token, current);
5208
25.0k
}
5209
5210
5211
static njs_int_t
5212
njs_parser_block_statement_close_brace(njs_parser_t *parser,
5213
    njs_lexer_token_t *token, njs_queue_link_t *current)
5214
80.9k
{
5215
80.9k
    njs_parser_node_t  *node;
5216
5217
80.9k
    if (parser->ret != NJS_OK) {
5218
937
        return njs_parser_failed(parser);
5219
937
    }
5220
5221
79.9k
    if (token->type != NJS_TOKEN_CLOSE_BRACE) {
5222
0
        return njs_parser_failed(parser);
5223
0
    }
5224
5225
79.9k
    node = njs_parser_node_new(parser, NJS_TOKEN_BLOCK);
5226
79.9k
    if (node == NULL) {
5227
0
        return NJS_ERROR;
5228
0
    }
5229
5230
79.9k
    node->token_line = (uint32_t) (uintptr_t) parser->target;
5231
79.9k
    node->left = parser->node;
5232
79.9k
    node->right = NULL;
5233
5234
79.9k
    parser->target = NULL;
5235
79.9k
    parser->node = node;
5236
5237
79.9k
    njs_parser_scope_end(parser);
5238
5239
79.9k
    njs_lexer_consume_token(parser->lexer, 1);
5240
79.9k
    return njs_parser_stack_pop(parser);
5241
79.9k
}
5242
5243
5244
static njs_int_t
5245
njs_parser_statement_list(njs_parser_t *parser, njs_lexer_token_t *token,
5246
    njs_queue_link_t *current)
5247
402k
{
5248
402k
    njs_parser_next(parser, njs_parser_statement_list_item);
5249
5250
402k
    return njs_parser_after(parser, current, NULL, 1,
5251
402k
                            njs_parser_statement_list_next);
5252
402k
}
5253
5254
5255
static njs_int_t
5256
njs_parser_statement_list_next(njs_parser_t *parser, njs_lexer_token_t *token,
5257
    njs_queue_link_t *current)
5258
1.00M
{
5259
1.00M
    if (parser->ret != NJS_OK) {
5260
11.6k
        if (token->type != NJS_TOKEN_CLOSE_BRACE) {
5261
11.5k
            parser->node = parser->target;
5262
5263
11.5k
            (void) njs_parser_stack_pop(parser);
5264
5265
11.5k
            return parser->ret;
5266
11.5k
        }
5267
5268
136
        return njs_parser_failed(parser);
5269
11.6k
    }
5270
5271
989k
    if (token->type == NJS_TOKEN_CLOSE_BRACE) {
5272
156k
        return njs_parser_stack_pop(parser);
5273
156k
    }
5274
5275
833k
    njs_parser_next(parser, njs_parser_statement_list_item);
5276
5277
833k
    return njs_parser_after(parser, current, parser->node, 0,
5278
989k
                            njs_parser_statement_list_next);
5279
989k
}
5280
5281
5282
static njs_int_t
5283
njs_parser_statement_list_item(njs_parser_t *parser, njs_lexer_token_t *token,
5284
    njs_queue_link_t *current)
5285
1.23M
{
5286
1.23M
    njs_int_t  ret;
5287
5288
1.23M
    ret = njs_parser_declaration(parser, token, current);
5289
1.23M
    if (ret == NJS_OK) {
5290
26.4k
        return NJS_OK;
5291
26.4k
    }
5292
5293
1.20M
    njs_parser_next(parser, njs_parser_statement);
5294
5295
1.20M
    return NJS_OK;
5296
1.23M
}
5297
5298
5299
/*
5300
 * 13.3.1 Let and Const Declarations
5301
 */
5302
static njs_int_t
5303
njs_parser_lexical_declaration(njs_parser_t *parser, njs_lexer_token_t *token,
5304
    njs_queue_link_t *current)
5305
20.5k
{
5306
20.5k
    parser->var_type = (token->type == NJS_TOKEN_LET) ? NJS_VARIABLE_LET
5307
20.5k
                                                      : NJS_VARIABLE_CONST;
5308
5309
20.5k
    njs_lexer_consume_token(parser->lexer, 1);
5310
5311
20.5k
    njs_parser_next(parser, njs_parser_variable_declaration_list);
5312
5313
20.5k
    return njs_parser_after(parser, current, NULL, 1, njs_parser_semicolon);
5314
20.5k
}
5315
5316
5317
/*
5318
 * 13.3.2 Variable Statement
5319
 */
5320
static njs_int_t
5321
njs_parser_variable_statement(njs_parser_t *parser, njs_lexer_token_t *token,
5322
    njs_queue_link_t *current)
5323
11.3k
{
5324
11.3k
    parser->var_type = NJS_VARIABLE_VAR;
5325
5326
11.3k
    njs_parser_next(parser, njs_parser_variable_declaration_list);
5327
5328
11.3k
    return njs_parser_after(parser, current, NULL, 1, njs_parser_semicolon);
5329
11.3k
}
5330
5331
5332
static njs_int_t
5333
njs_parser_variable_declaration_list(njs_parser_t *parser,
5334
    njs_lexer_token_t *token, njs_queue_link_t *current)
5335
32.1k
{
5336
32.1k
    njs_parser_next(parser, njs_parser_variable_declaration);
5337
5338
32.1k
    return njs_parser_after(parser, current, NULL, 1,
5339
32.1k
                            njs_parser_variable_declaration_list_next);
5340
32.1k
}
5341
5342
5343
static njs_int_t
5344
njs_parser_variable_declaration_list_next(njs_parser_t *parser,
5345
    njs_lexer_token_t *token, njs_queue_link_t *current)
5346
32.0k
{
5347
32.0k
    njs_parser_node_t  *node;
5348
5349
32.0k
    if (parser->target != NULL) {
5350
1.57k
        parser->node->left = parser->target;
5351
1.57k
    }
5352
5353
32.0k
    if (token->type != NJS_TOKEN_COMMA) {
5354
29.6k
        return njs_parser_stack_pop(parser);
5355
29.6k
    }
5356
5357
2.40k
    njs_lexer_consume_token(parser->lexer, 1);
5358
5359
2.40k
    node = parser->node;
5360
5361
2.40k
    parser->node = NULL;
5362
5363
2.40k
    njs_parser_next(parser, njs_parser_variable_declaration);
5364
5365
2.40k
    return njs_parser_after(parser, current, node, 1,
5366
32.0k
                            njs_parser_variable_declaration_list_next);
5367
32.0k
}
5368
5369
5370
static njs_int_t
5371
njs_parser_variable_declaration(njs_parser_t *parser,
5372
    njs_lexer_token_t *token, njs_queue_link_t *current)
5373
34.5k
{
5374
34.5k
    njs_int_t          ret;
5375
34.5k
    njs_variable_t     *var;
5376
34.5k
    njs_token_type_t   type;
5377
34.5k
    njs_parser_node_t  *name;
5378
5379
34.5k
    ret = njs_parser_binding_pattern(parser, token, current);
5380
34.5k
    if (ret == NJS_OK) {
5381
152
        return njs_parser_after(parser, current, NULL, 1,
5382
152
                                njs_parser_initializer);
5383
152
    }
5384
5385
34.4k
    if (!njs_lexer_token_is_binding_identifier(token)) {
5386
136
        return njs_parser_failed(parser);
5387
136
    }
5388
5389
34.3k
    if (njs_parser_restricted_identifier(token->type)) {
5390
1.44k
        njs_parser_syntax_error(parser, "Identifier \"%V\" is forbidden in"
5391
1.44k
                                " var declaration", &token->text);
5392
1.44k
        return NJS_DONE;
5393
1.44k
    }
5394
5395
32.8k
    name = njs_parser_variable_node(parser, token->atom_id, parser->var_type,
5396
32.8k
                                    &var);
5397
32.8k
    if (name == NULL) {
5398
282
        return NJS_ERROR;
5399
282
    }
5400
5401
32.5k
    if (var->self) {
5402
204
        var->type = parser->var_type;
5403
204
        var->self = 0;
5404
204
    }
5405
5406
32.5k
    name->token_line = token->line;
5407
5408
32.5k
    parser->node = name;
5409
5410
32.5k
    njs_lexer_consume_token(parser->lexer, 1);
5411
5412
32.5k
    token = njs_lexer_token(parser->lexer, 0);
5413
32.5k
    if (token == NULL) {
5414
0
        return NJS_ERROR;
5415
0
    }
5416
5417
32.5k
    switch (parser->var_type) {
5418
8.70k
    case NJS_VARIABLE_LET:
5419
8.70k
        type = NJS_TOKEN_LET;
5420
8.70k
        break;
5421
5422
13.3k
    case NJS_VARIABLE_CONST:
5423
13.3k
        type = NJS_TOKEN_CONST;
5424
13.3k
        break;
5425
5426
10.4k
    default:
5427
10.4k
        type = NJS_TOKEN_VAR;
5428
10.4k
        break;
5429
32.5k
    }
5430
5431
32.5k
    ret = njs_parser_initializer_assign(parser, type);
5432
32.5k
    if (ret != NJS_OK) {
5433
0
        return ret;
5434
0
    }
5435
5436
32.5k
    parser->node->token_line = token->line;
5437
5438
32.5k
    if (token->type == NJS_TOKEN_ASSIGNMENT) {
5439
22.5k
        njs_parser_next(parser, njs_parser_initializer);
5440
5441
22.5k
        return NJS_OK;
5442
22.5k
    }
5443
5444
10.0k
    parser->target = parser->node;
5445
10.0k
    parser->node = NULL;
5446
5447
10.0k
    njs_parser_next(parser, njs_parser_initializer_after);
5448
5449
10.0k
    return NJS_OK;
5450
32.5k
}
5451
5452
5453
/*
5454
 * 13.3.3 Destructuring Binding Patterns.
5455
 */
5456
static njs_int_t
5457
njs_parser_binding_pattern(njs_parser_t *parser, njs_lexer_token_t *token,
5458
    njs_queue_link_t *current)
5459
34.5k
{
5460
34.5k
    if (token->type == NJS_TOKEN_OPEN_BRACE) {
5461
152
        njs_parser_next(parser, njs_parser_object_binding_pattern);
5462
5463
34.4k
    } else if (token->type == NJS_TOKEN_OPEN_BRACKET) {
5464
0
        njs_parser_next(parser, njs_parser_array_binding_pattern);
5465
5466
34.4k
    } else {
5467
34.4k
        return NJS_DECLINED;
5468
34.4k
    }
5469
5470
152
    njs_lexer_consume_token(parser->lexer, 1);
5471
5472
152
    return NJS_OK;
5473
34.5k
}
5474
5475
static njs_int_t
5476
njs_parser_object_binding_pattern(njs_parser_t *parser,
5477
    njs_lexer_token_t *token, njs_queue_link_t *current)
5478
156
{
5479
156
    return njs_parser_not_supported(parser, token);
5480
156
}
5481
5482
5483
static njs_int_t
5484
njs_parser_array_binding_pattern(njs_parser_t *parser,
5485
    njs_lexer_token_t *token, njs_queue_link_t *current)
5486
0
{
5487
0
    return njs_parser_not_supported(parser, token);
5488
0
}
5489
5490
5491
/*
5492
 * 13.5 Expression Statement.
5493
 */
5494
static njs_int_t
5495
njs_parser_expression_statement(njs_parser_t *parser, njs_lexer_token_t *token,
5496
    njs_queue_link_t *current)
5497
845k
{
5498
845k
    njs_token_type_t   type;
5499
845k
    njs_lexer_token_t  *next;
5500
5501
845k
    switch (token->type) {
5502
1
    case NJS_TOKEN_FUNCTION:
5503
1
        njs_parser_syntax_error(parser, "Functions can only be declared "
5504
1
                                        "at top level or inside a block");
5505
1
        return NJS_DONE;
5506
5507
0
    case NJS_TOKEN_CLASS:
5508
0
        njs_parser_syntax_error(parser, "Class can only be declared "
5509
0
                                        "at top level or inside a block");
5510
0
        return NJS_DONE;
5511
5512
0
    case NJS_TOKEN_OPEN_BRACE:
5513
0
        return njs_parser_reject(parser);
5514
5515
105
    case NJS_TOKEN_ASYNC:
5516
105
        next = njs_lexer_peek_token(parser->lexer, token, 1);
5517
105
        if (next == NULL) {
5518
0
            return NJS_ERROR;
5519
0
        }
5520
5521
105
        if (next->type == NJS_TOKEN_FUNCTION) {
5522
0
            return njs_parser_not_supported(parser, token);
5523
0
        }
5524
5525
105
        break;
5526
5527
105
    case NJS_TOKEN_CONST:
5528
484
    case NJS_TOKEN_LET:
5529
484
        type = token->type;
5530
5531
484
        token = njs_lexer_peek_token(parser->lexer, token, 0);
5532
484
        if (token == NULL) {
5533
0
            return NJS_ERROR;
5534
0
        }
5535
5536
484
        if (token->type == NJS_TOKEN_NAME) {
5537
0
            njs_parser_syntax_error(parser, "%s declaration cannot appear "
5538
0
                                    "in a single-statement context",
5539
0
                                    (type == NJS_TOKEN_CONST ? "const"
5540
0
                                                             : "let" ));
5541
0
            return NJS_DONE;
5542
0
        }
5543
5544
484
        if (token->type == NJS_TOKEN_OPEN_BRACKET) {
5545
482
            return njs_parser_failed(parser);
5546
482
        }
5547
5548
2
        break;
5549
5550
845k
    default:
5551
845k
        break;
5552
845k
    }
5553
5554
845k
    parser->node = NULL;
5555
5556
845k
    njs_parser_next(parser, njs_parser_expression);
5557
5558
845k
    return njs_parser_after(parser, current, NULL, 1,
5559
845k
                            njs_parser_expression_statement_after);
5560
845k
}
5561
5562
5563
static njs_int_t
5564
njs_parser_expression_statement_after(njs_parser_t *parser,
5565
    njs_lexer_token_t *token, njs_queue_link_t *current)
5566
745k
{
5567
745k
    if (njs_parser_expect_semicolon(parser, token) != NJS_OK) {
5568
3.31k
        return njs_parser_failed(parser);
5569
3.31k
    }
5570
5571
742k
    return njs_parser_stack_pop(parser);
5572
745k
}
5573
5574
5575
/*
5576
 * 13.6 if Statement.
5577
 */
5578
static njs_int_t
5579
njs_parser_if_statement(njs_parser_t *parser, njs_lexer_token_t *token,
5580
    njs_queue_link_t *current)
5581
2.74k
{
5582
2.74k
    njs_int_t          ret;
5583
2.74k
    njs_parser_node_t  *node;
5584
5585
2.74k
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS) {
5586
76
        return njs_parser_failed(parser);
5587
76
    }
5588
5589
2.66k
    njs_lexer_consume_token(parser->lexer, 1);
5590
5591
2.66k
    node = njs_parser_node_new(parser, NJS_TOKEN_IF);
5592
2.66k
    if (node == NULL) {
5593
0
        return NJS_ERROR;
5594
0
    }
5595
5596
2.66k
    node->token_line = parser->line;
5597
5598
2.66k
    parser->node = NULL;
5599
5600
2.66k
    njs_parser_next(parser, njs_parser_expression);
5601
5602
2.66k
    ret = njs_parser_after(parser, current, node, 1,
5603
2.66k
                           njs_parser_if_close_parenthesis);
5604
2.66k
    if (ret != NJS_OK) {
5605
0
        return ret;
5606
0
    }
5607
5608
2.66k
    ret = njs_parser_after(parser, current, NULL, 1,
5609
2.66k
                           njs_parser_statement_wo_node);
5610
2.66k
    if (ret != NJS_OK) {
5611
0
        return ret;
5612
0
    }
5613
5614
2.66k
    return njs_parser_after(parser, current, node, 1,
5615
2.66k
                            njs_parser_else_statement);
5616
2.66k
}
5617
5618
5619
static njs_int_t
5620
njs_parser_if_close_parenthesis(njs_parser_t *parser, njs_lexer_token_t *token,
5621
    njs_queue_link_t *current)
5622
2.66k
{
5623
2.66k
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
5624
133
        return njs_parser_failed(parser);
5625
133
    }
5626
5627
2.53k
    njs_lexer_consume_token(parser->lexer, 1);
5628
5629
2.53k
    parser->target->left = parser->node;
5630
2.53k
    parser->node = NULL;
5631
5632
2.53k
    return njs_parser_stack_pop(parser);
5633
2.66k
}
5634
5635
5636
static njs_int_t
5637
njs_parser_else_statement(njs_parser_t *parser, njs_lexer_token_t *token,
5638
    njs_queue_link_t *current)
5639
2.53k
{
5640
2.53k
    njs_parser_node_t  *node;
5641
5642
2.53k
    parser->target->right = parser->node;
5643
2.53k
    parser->node = NULL;
5644
5645
2.53k
    if (token->type == NJS_TOKEN_ELSE) {
5646
1.07k
        node = njs_parser_node_new(parser, NJS_TOKEN_BRANCHING);
5647
1.07k
        if (node == NULL) {
5648
0
            return NJS_ERROR;
5649
0
        }
5650
5651
1.07k
        node->token_line = token->line;
5652
1.07k
        node->left = parser->target->right;
5653
5654
1.07k
        parser->target->right = node;
5655
5656
1.07k
        njs_lexer_consume_token(parser->lexer, 1);
5657
1.07k
        njs_parser_next(parser, njs_parser_statement_wo_node);
5658
5659
1.07k
        return njs_parser_after(parser, current, parser->target, 1,
5660
1.07k
                                njs_parser_else_statement_after);
5661
1.07k
    }
5662
5663
1.46k
    parser->node = parser->target;
5664
5665
1.46k
    return njs_parser_stack_pop(parser);
5666
2.53k
}
5667
5668
5669
static njs_int_t
5670
njs_parser_else_statement_after(njs_parser_t *parser, njs_lexer_token_t *token,
5671
    njs_queue_link_t *current)
5672
1.07k
{
5673
1.07k
    parser->target->right->right = parser->node;
5674
5675
1.07k
    parser->node = parser->target;
5676
5677
1.07k
    return njs_parser_stack_pop(parser);
5678
1.07k
}
5679
5680
5681
static njs_int_t
5682
njs_parser_iteration_statement_do(njs_parser_t *parser,
5683
    njs_lexer_token_t *token, njs_queue_link_t *current)
5684
2.54k
{
5685
2.54k
    njs_parser_node_t  *node;
5686
5687
2.54k
    node = njs_parser_node_new(parser, NJS_TOKEN_DO);
5688
2.54k
    if (node == NULL) {
5689
0
        return NJS_ERROR;
5690
0
    }
5691
5692
2.54k
    node->token_line = parser->line;
5693
5694
2.54k
    parser->node = NULL;
5695
5696
2.54k
    njs_parser_next(parser, njs_parser_statement_wo_node);
5697
5698
2.54k
    return njs_parser_after(parser, current, node, 1,
5699
2.54k
                            njs_parser_iteration_statement_do_while);
5700
2.54k
}
5701
5702
5703
static njs_int_t
5704
njs_parser_iteration_statement_do_while(njs_parser_t *parser,
5705
    njs_lexer_token_t *token, njs_queue_link_t *current)
5706
2.54k
{
5707
2.54k
    if (token->type != NJS_TOKEN_WHILE) {
5708
52
        return njs_parser_failed(parser);
5709
52
    }
5710
5711
2.49k
    parser->target->left = parser->node;
5712
5713
2.49k
    njs_lexer_consume_token(parser->lexer, 1);
5714
5715
2.49k
    njs_parser_next(parser, njs_parser_expression_parenthesis);
5716
5717
2.49k
    return njs_parser_after(parser, current, parser->target, 1,
5718
2.54k
                            njs_parser_do_while_semicolon);
5719
2.54k
}
5720
5721
5722
static njs_int_t
5723
njs_parser_do_while_semicolon(njs_parser_t *parser, njs_lexer_token_t *token,
5724
    njs_queue_link_t *current)
5725
2.49k
{
5726
2.49k
    return njs_parser_right_link_pop(parser);
5727
2.49k
}
5728
5729
5730
static njs_int_t
5731
njs_parser_iteration_statement_while(njs_parser_t *parser,
5732
    njs_lexer_token_t *token, njs_queue_link_t *current)
5733
1.95k
{
5734
1.95k
    njs_parser_node_t  *node;
5735
5736
1.95k
    node = njs_parser_node_new(parser, NJS_TOKEN_WHILE);
5737
1.95k
    if (node == NULL) {
5738
0
        return NJS_ERROR;
5739
0
    }
5740
5741
1.95k
    node->token_line = token->line;
5742
5743
1.95k
    njs_parser_next(parser, njs_parser_expression_parenthesis);
5744
5745
1.95k
    return njs_parser_after(parser, current, node, 1,
5746
1.95k
                            njs_parser_while_statement);
5747
1.95k
}
5748
5749
5750
static njs_int_t
5751
njs_parser_while_statement(njs_parser_t *parser, njs_lexer_token_t *token,
5752
    njs_queue_link_t *current)
5753
1.95k
{
5754
1.95k
    parser->target->right = parser->node;
5755
1.95k
    parser->node = NULL;
5756
5757
1.95k
    njs_parser_next(parser, njs_parser_statement_wo_node);
5758
5759
1.95k
    return njs_parser_after(parser, current, parser->target, 1,
5760
1.95k
                            njs_parser_while_after);
5761
1.95k
}
5762
5763
5764
static njs_int_t
5765
njs_parser_while_after(njs_parser_t *parser, njs_lexer_token_t *token,
5766
    njs_queue_link_t *current)
5767
1.95k
{
5768
1.95k
    parser->target->left = parser->node;
5769
1.95k
    parser->node = parser->target;
5770
5771
1.95k
    return njs_parser_stack_pop(parser);
5772
1.95k
}
5773
5774
5775
static njs_int_t
5776
njs_parser_iteration_statement_for(njs_parser_t *parser,
5777
    njs_lexer_token_t *token, njs_queue_link_t *current)
5778
32.1k
{
5779
32.1k
    njs_int_t  ret;
5780
5781
32.1k
    if (token->type == NJS_TOKEN_OPEN_PARENTHESIS) {
5782
31.9k
        njs_lexer_consume_token(parser->lexer, 1);
5783
5784
31.9k
        ret = njs_parser_scope_begin(parser, NJS_SCOPE_BLOCK, 0);
5785
31.9k
        if (njs_slow_path(ret != NJS_OK)) {
5786
0
            return ret;
5787
0
        }
5788
5789
31.9k
        njs_parser_next(parser, njs_parser_iteration_statement_for_map);
5790
5791
31.9k
        return njs_parser_after(parser, current,
5792
31.9k
                                (void *) (uintptr_t) parser->line, 1,
5793
31.9k
                                njs_parser_iteration_statement_for_end);
5794
31.9k
    }
5795
5796
185
    if (token->type == NJS_TOKEN_AWAIT) {
5797
0
        return njs_parser_not_supported(parser, token);
5798
0
    }
5799
5800
185
    return njs_parser_failed(parser);
5801
185
}
5802
5803
5804
static njs_int_t
5805
njs_parser_for_expression_map_reparse(njs_parser_t *parser,
5806
    njs_lexer_token_t *token, njs_queue_link_t *current)
5807
13.8k
{
5808
13.8k
    if (parser->ret != NJS_OK && parser->node != NULL) {
5809
1.18k
        return njs_parser_failed(parser);
5810
1.18k
    }
5811
5812
12.7k
    if (parser->node == NULL) {
5813
478
        njs_lexer_in_fail_set(parser->lexer, 1);
5814
5815
478
        njs_parser_next(parser, njs_parser_expression);
5816
5817
478
        return NJS_OK;
5818
478
    }
5819
5820
12.2k
    return njs_parser_stack_pop(parser);
5821
12.7k
}
5822
5823
5824
static njs_int_t
5825
njs_parser_for_expression_map_continue(njs_parser_t *parser,
5826
    njs_lexer_token_t *token, njs_queue_link_t *current)
5827
12.3k
{
5828
12.3k
    njs_int_t          operation;
5829
12.3k
    njs_str_t          *text;
5830
12.3k
    njs_parser_node_t  *node;
5831
5832
12.3k
    if (token->type != NJS_TOKEN_IN) {
5833
6.60k
        njs_lexer_in_fail_set(parser->lexer, 1);
5834
5835
        /* Continue parsing of expr1 in "for (expr1;[expr2];[expr3])". */
5836
5837
6.60k
        njs_parser_next(parser, njs_parser_expression_continue_op);
5838
5839
        /*
5840
         * Here we pass not a node, but a token, this is important.
5841
         * This is necessary for correct error output.
5842
         */
5843
5844
6.60k
        text = njs_mp_alloc(parser->vm->mem_pool, sizeof(njs_str_t));
5845
6.60k
        if (text == NULL) {
5846
0
            return NJS_ERROR;
5847
0
        }
5848
5849
6.60k
        *text = token->text;
5850
5851
6.60k
        return njs_parser_after(parser, current, text, 1,
5852
6.60k
                                njs_parser_for_var_in_of_expression);
5853
5854
6.60k
    } else {
5855
5856
        /* for-in */
5857
5858
5.70k
        if (!njs_parser_is_lvalue(parser->node)) {
5859
172
            text = (njs_str_t *) parser->target;
5860
5861
172
            njs_parser_ref_error(parser, "Invalid left-hand side \"%V\" "
5862
172
                                 "in for-in statement", text);
5863
5864
172
            njs_mp_free(parser->vm->mem_pool, text);
5865
5866
172
            return NJS_DONE;
5867
172
        }
5868
5869
5.52k
        parser->node = njs_parser_lvalue_ref(parser->node);
5870
5.52k
        if (parser->node == NULL) {
5871
0
            return NJS_ERROR;
5872
0
        }
5873
5874
5.52k
        operation = NJS_VMCODE_PROPERTY_IN;
5875
5876
5.52k
        node = njs_parser_node_new(parser, token->type);
5877
5.52k
        if (node == NULL) {
5878
0
            return NJS_ERROR;
5879
0
        }
5880
5881
5.52k
        node->token_line = token->line;
5882
5.52k
        node->u.operation = operation;
5883
5.52k
        node->left = parser->node;
5884
5.52k
        node->left->dest = node;
5885
5886
5.52k
        njs_lexer_consume_token(parser->lexer, 1);
5887
5888
5.52k
        njs_parser_next(parser, njs_parser_expression);
5889
5890
5.52k
        return njs_parser_after(parser, current, node, 0,
5891
5.52k
                                njs_parser_for_in_statement_statement);
5892
5.52k
    }
5893
12.3k
}
5894
5895
5896
static njs_int_t
5897
njs_parser_after_expr(njs_parser_t *parser,
5898
    njs_lexer_token_t *token, njs_queue_link_t *current)
5899
0
{
5900
0
    return njs_parser_right_link_pop(parser);
5901
0
}
5902
5903
5904
static njs_int_t
5905
njs_parser_comma_expression_comma(njs_parser_t *parser,
5906
    njs_lexer_token_t *token, njs_queue_link_t *current)
5907
4.94k
{
5908
4.94k
    njs_parser_node_t  *node;
5909
5910
4.94k
    if (parser->target != NULL) {
5911
0
        parser->target->right = parser->node;
5912
0
        parser->target->right->dest = parser->target;
5913
0
        parser->node = parser->target;
5914
0
    }
5915
5916
4.94k
    if (token->type != NJS_TOKEN_COMMA) {
5917
4.94k
        return njs_parser_stack_pop(parser);
5918
4.94k
    }
5919
5920
0
    node = njs_parser_node_new(parser, NJS_TOKEN_COMMA);
5921
0
    if (node == NULL) {
5922
0
        return NJS_ERROR;
5923
0
    }
5924
5925
0
    node->token_line = token->line;
5926
0
    node->u.operation = 0;
5927
0
    node->left = parser->node;
5928
0
    node->left->dest = node;
5929
5930
0
    njs_lexer_consume_token(parser->lexer, 1);
5931
5932
0
    njs_parser_next(parser, njs_parser_expression);
5933
5934
0
    return njs_parser_after(parser, current, node, 1, njs_parser_after_expr);
5935
0
}
5936
5937
5938
static njs_int_t
5939
njs_parser_expression_continue_op(njs_parser_t *parser,
5940
    njs_lexer_token_t *token, njs_queue_link_t *current)
5941
6.60k
{
5942
6.60k
    if (token->type == NJS_TOKEN_CONDITIONAL) {
5943
424
        njs_parser_next(parser, njs_parser_conditional_question_mark);
5944
424
        return njs_parser_after(parser, current, NULL, 0,
5945
424
                                njs_parser_expression_continue_assign_comma);
5946
6.18k
    } else {
5947
6.18k
        parser->target = NULL;
5948
5949
6.18k
        parser->use_lhs = 1;
5950
5951
6.18k
        njs_parser_next(parser, njs_parser_expression);
5952
5953
6.18k
        return njs_parser_after(parser, current, NULL, 1,
5954
6.18k
                                njs_parser_comma_expression_comma);
5955
6.18k
    }
5956
6.60k
}
5957
5958
5959
static njs_int_t
5960
njs_parser_expression_continue_assign_comma(njs_parser_t *parser,
5961
    njs_lexer_token_t *token, njs_queue_link_t *current)
5962
424
{
5963
424
    if (parser->ret != NJS_OK) {
5964
132
        return njs_parser_failed(parser);
5965
132
    }
5966
5967
292
    njs_parser_next(parser, njs_parser_assignment_expression_after);
5968
5969
292
    return njs_parser_after(parser, current, NULL, 1,
5970
424
                            njs_parser_expression_comma);
5971
424
}
5972
5973
5974
static njs_int_t
5975
njs_parser_iteration_statement_for_map(njs_parser_t *parser,
5976
    njs_lexer_token_t *token, njs_queue_link_t *current)
5977
33.1k
{
5978
33.1k
    njs_int_t         ret;
5979
33.1k
    njs_str_t         *text;
5980
33.1k
    njs_token_type_t  token_type;
5981
5982
    /*
5983
     * "var" <VariableDeclarationList> ";" <Expression>? ";" <Expression>? ")"
5984
     *     <Statement>
5985
     * "var" <ForBinding> "in" <Expression> ")" <Statement>
5986
     * "var" <ForBinding> "of" <AssignmentExpression> ")" <Statement>
5987
5988
     * <ForDeclaration> "in" <Expression> ")" <Statement>
5989
     * <ForDeclaration> "of" <AssignmentExpression> ")" <Statement>
5990
5991
     * <Expression>? ";" <Expression>? ";" <Expression>? ")" <Statement>
5992
     * <LexicalDeclaration> <Expression>? ";" <Expression>? ")" <Statement>
5993
5994
     * <LeftHandSideExpression> "in" <Expression> ")" <Statement>
5995
     * <LeftHandSideExpression> "of" <AssignmentExpression> ")" <Statement>
5996
     */
5997
5998
33.1k
    parser->node = NULL;
5999
6000
33.1k
    switch (token->type) {
6001
9.74k
    case NJS_TOKEN_SEMICOLON:
6002
9.74k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
6003
9.74k
        if (token == NULL) {
6004
0
            return NJS_ERROR;
6005
0
        }
6006
6007
9.74k
        if (token->type != NJS_TOKEN_SEMICOLON) {
6008
818
            njs_lexer_consume_token(parser->lexer, 1);
6009
6010
818
            parser->target = NULL;
6011
6012
818
            njs_parser_next(parser, njs_parser_expression);
6013
6014
818
            return njs_parser_after(parser, current, NULL, 1,
6015
818
                                    njs_parser_for_expression);
6016
818
        }
6017
6018
8.92k
        parser->node = NULL;
6019
8.92k
        parser->target = NULL;
6020
6021
8.92k
        njs_lexer_consume_token(parser->lexer, 1);
6022
8.92k
        njs_parser_next(parser, njs_parser_for_expression);
6023
6024
8.92k
        return NJS_OK;
6025
6026
8.44k
    case NJS_TOKEN_VAR:
6027
8.58k
    case NJS_TOKEN_LET:
6028
8.58k
    case NJS_TOKEN_CONST:
6029
8.58k
        token_type = token->type;
6030
6031
8.58k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
6032
8.58k
        if (token == NULL) {
6033
0
            return NJS_ERROR;
6034
0
        }
6035
6036
8.58k
        njs_lexer_consume_token(parser->lexer, 1);
6037
6038
8.58k
        ret = njs_parser_for_var_binding_or_var_list(parser, token,
6039
8.58k
                                                     current, token_type);
6040
8.58k
        if (ret != NJS_OK) {
6041
8.23k
            if (ret == NJS_DONE) {
6042
7.79k
                return NJS_OK;
6043
7.79k
            }
6044
6045
438
            return ret;
6046
8.23k
        }
6047
6048
353
        goto expression_after;
6049
6050
353
    case NJS_TOKEN_AWAIT:
6051
0
        njs_parser_next(parser, njs_parser_expression);
6052
6053
0
        goto expression_after;
6054
6055
14.8k
    default:
6056
14.8k
        ret = njs_parser_match_arrow_expression(parser, token);
6057
14.8k
        if (ret == NJS_OK) {
6058
0
            parser->target = NULL;
6059
0
            njs_parser_next(parser, njs_parser_expression);
6060
0
            goto expression_after;
6061
14.8k
        } else if (ret == NJS_ERROR) {
6062
0
            return NJS_ERROR;
6063
0
        }
6064
6065
14.8k
        parser->target = NULL;
6066
14.8k
        njs_parser_next(parser, njs_parser_left_hand_side_expression);
6067
6068
        /*
6069
         * Here we pass not a node, but a token, this is important.
6070
         * This is necessary for correct error output.
6071
         */
6072
6073
14.8k
        text = njs_mp_alloc(parser->vm->mem_pool, sizeof(njs_str_t));
6074
14.8k
        if (text == NULL) {
6075
0
            return NJS_ERROR;
6076
0
        }
6077
6078
14.8k
        *text = token->text;
6079
6080
14.8k
        ret = njs_parser_after(parser, current, text, 0,
6081
14.8k
                               njs_parser_for_expression_map_reparse);
6082
14.8k
        if (ret != NJS_OK) {
6083
0
            return NJS_ERROR;
6084
0
        }
6085
6086
14.8k
        return njs_parser_after(parser, current, text, 1,
6087
33.1k
                                njs_parser_for_expression_map_continue);
6088
33.1k
    }
6089
6090
353
expression_after:
6091
6092
    /*
6093
     * Here we pass not a node, but a token, this is important.
6094
     * This is necessary for correct error output.
6095
     */
6096
6097
353
    text = njs_mp_alloc(parser->vm->mem_pool, sizeof(njs_str_t));
6098
353
    if (text == NULL) {
6099
0
        return NJS_ERROR;
6100
0
    }
6101
6102
353
    *text = token->text;
6103
6104
353
    return njs_parser_after(parser, current, text, 1,
6105
353
                            njs_parser_for_var_in_of_expression);
6106
353
}
6107
6108
6109
static njs_int_t
6110
njs_parser_for_var_binding_or_var_list(njs_parser_t *parser,
6111
    njs_lexer_token_t *token, njs_queue_link_t *current,
6112
    njs_token_type_t token_type)
6113
8.58k
{
6114
8.58k
    njs_int_t            ret;
6115
8.58k
    njs_lexer_token_t    *next;
6116
8.58k
    njs_parser_node_t    *node, *var, *node_type, *statement;
6117
8.58k
    njs_variable_type_t  type;
6118
6119
8.58k
    switch (token_type) {
6120
137
    case NJS_TOKEN_LET:
6121
137
        type = NJS_VARIABLE_LET;
6122
137
        break;
6123
6124
8
    case NJS_TOKEN_CONST:
6125
8
        type = NJS_VARIABLE_CONST;
6126
8
        break;
6127
6128
8.44k
    default:
6129
8.44k
        type = NJS_VARIABLE_VAR;
6130
8.44k
        break;
6131
8.58k
    }
6132
6133
8.58k
    switch (token->type) {
6134
    /* BindingPattern */
6135
4
    case NJS_TOKEN_OPEN_BRACE:
6136
4
        njs_parser_next(parser, njs_parser_object_binding_pattern);
6137
4
        return NJS_DONE;
6138
6139
0
    case NJS_TOKEN_OPEN_BRACKET:
6140
0
        njs_parser_next(parser, njs_parser_array_binding_pattern);
6141
0
        return NJS_DONE;
6142
6143
8.58k
    default:
6144
8.58k
        if (njs_lexer_token_is_binding_identifier(token)) {
6145
8.14k
            if (njs_parser_restricted_identifier(token->type)) {
6146
1.23k
                njs_parser_syntax_error(parser, "Identifier \"%V\" is forbidden"
6147
1.23k
                                        " in var declaration", &token->text);
6148
1.23k
                return NJS_DONE;
6149
1.23k
            }
6150
6151
6.90k
            next = njs_lexer_peek_token(parser->lexer, token, 0);
6152
6.90k
            if (next == NULL) {
6153
0
                return NJS_ERROR;
6154
0
            }
6155
6156
6.90k
            if (next->type != NJS_TOKEN_IN) {
6157
353
                parser->var_type = type;
6158
6159
353
                njs_lexer_in_fail_set(parser->lexer, 1);
6160
6161
353
                njs_parser_next(parser, njs_parser_variable_declaration_list);
6162
353
                return NJS_OK;
6163
353
            }
6164
6165
6.55k
            statement = njs_parser_node_new(parser, NJS_TOKEN_STATEMENT);
6166
6.55k
            if (njs_slow_path(statement == NULL)) {
6167
0
                return NJS_ERROR;
6168
0
            }
6169
6170
6.55k
            node_type = njs_parser_node_new(parser, token_type);
6171
6.55k
            if (njs_slow_path(node_type == NULL)) {
6172
0
                return NJS_ERROR;
6173
0
            }
6174
6175
6.55k
            var = njs_parser_variable_node(parser, token->atom_id, type, NULL);
6176
6.55k
            if (var == NULL) {
6177
0
                return NJS_ERROR;
6178
0
            }
6179
6180
6.55k
            node_type->token_line = token->line;
6181
6.55k
            var->token_line = token->line;
6182
6183
6.55k
            statement->right = node_type;
6184
6.55k
            node_type->left = var;
6185
6.55k
            parser->node = NULL;
6186
6187
6.55k
            node = njs_parser_node_new(parser, NJS_TOKEN_IN);
6188
6.55k
            if (node == NULL) {
6189
0
                return NJS_ERROR;
6190
0
            }
6191
6192
6.55k
            node->token_line = next->line;
6193
6.55k
            node->left = statement;
6194
6195
6.55k
            njs_parser_next(parser, njs_parser_expression);
6196
6197
6.55k
            ret = njs_parser_after(parser, current, node, 1,
6198
6.55k
                                   njs_parser_for_var_in_statement);
6199
6.55k
            if (ret != NJS_OK) {
6200
0
                return NJS_ERROR;
6201
0
            }
6202
6203
6.55k
            njs_lexer_consume_token(parser->lexer, 2);
6204
6205
6.55k
            return NJS_DONE;
6206
6207
6.55k
        } else {
6208
438
            return njs_parser_failed(parser);
6209
438
        }
6210
8.58k
    }
6211
8.58k
}
6212
6213
6214
static njs_int_t
6215
njs_parser_for_var_in_statement(njs_parser_t *parser, njs_lexer_token_t *token,
6216
    njs_queue_link_t *current)
6217
6.54k
{
6218
6.54k
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
6219
141
        return njs_parser_failed(parser);
6220
141
    }
6221
6222
6.40k
    njs_lexer_consume_token(parser->lexer, 1);
6223
6224
6.40k
    parser->target->right = parser->node;
6225
6.40k
    parser->node = NULL;
6226
6227
6.40k
    njs_parser_next(parser, njs_parser_statement_wo_node);
6228
6229
6.40k
    return njs_parser_after(parser, current, parser->target, 1,
6230
6.54k
                            njs_parser_for_var_in_statement_after);
6231
6.54k
}
6232
6233
6234
static njs_int_t
6235
njs_parser_for_var_in_statement_after(njs_parser_t *parser,
6236
    njs_lexer_token_t *token, njs_queue_link_t *current)
6237
6.37k
{
6238
6.37k
    njs_parser_node_t  *foreach;
6239
6240
6.37k
    foreach = njs_parser_node_new(parser, NJS_TOKEN_FOR_IN);
6241
6.37k
    if (foreach == NULL) {
6242
0
        return NJS_ERROR;
6243
0
    }
6244
6245
6.37k
    foreach->left = parser->target;
6246
6.37k
    foreach->right = parser->node;
6247
6248
6.37k
    parser->node = foreach;
6249
6250
6.37k
    return njs_parser_stack_pop(parser);
6251
6.37k
}
6252
6253
6254
static njs_int_t
6255
njs_parser_for_var_in_of_expression(njs_parser_t *parser,
6256
    njs_lexer_token_t *token, njs_queue_link_t *current)
6257
5.58k
{
6258
5.58k
    njs_str_t          *text;
6259
5.58k
    njs_parser_node_t  *node;
6260
6261
    /*
6262
     * ";" <Expression>? ";" <Expression>? ")" <Statement>
6263
     * "in" <Expression> ")" <Statement>
6264
     * "of" <AssignmentExpression> ")" <Statement>
6265
     */
6266
6267
5.58k
    if (token->type != NJS_TOKEN_SEMICOLON &&
6268
2.31k
        token->type != NJS_TOKEN_CLOSE_PARENTHESIS &&
6269
2.31k
        parser->node != NULL && parser->node->token_type == NJS_TOKEN_IN)
6270
1.86k
    {
6271
1.86k
        node = parser->node->left;
6272
6273
1.86k
        if (!njs_parser_is_lvalue(node)) {
6274
6275
480
            text = (njs_str_t *) parser->target;
6276
6277
480
            njs_parser_ref_error(parser, "Invalid left-hand side \"%V\" "
6278
480
                                 "in for-in statement", text);
6279
6280
480
            njs_mp_free(parser->vm->mem_pool, text);
6281
6282
480
            return NJS_DONE;
6283
480
        }
6284
6285
1.38k
        node = njs_parser_lvalue_ref(node);
6286
1.38k
        if (node == NULL) {
6287
0
            return NJS_ERROR;
6288
0
        }
6289
6290
1.38k
        parser->node->left = node;
6291
6292
1.38k
        njs_parser_next(parser, njs_parser_for_in_statement);
6293
1.38k
        return NJS_OK;
6294
1.38k
    }
6295
6296
3.72k
    if (parser->target != NULL) {
6297
3.72k
        text = (njs_str_t *) parser->target;
6298
6299
3.72k
        njs_mp_free(parser->vm->mem_pool, text);
6300
3.72k
    }
6301
6302
3.72k
    switch (token->type) {
6303
3.27k
    case NJS_TOKEN_SEMICOLON:
6304
3.27k
        njs_lexer_in_fail_set(parser->lexer, 0);
6305
6306
3.27k
        token = njs_lexer_peek_token(parser->lexer, token, 0);
6307
3.27k
        if (token == NULL) {
6308
0
            return NJS_ERROR;
6309
0
        }
6310
6311
3.27k
        node = parser->node;
6312
3.27k
        parser->node = NULL;
6313
6314
3.27k
        if (token->type != NJS_TOKEN_SEMICOLON) {
6315
2.38k
            njs_lexer_consume_token(parser->lexer, 1);
6316
6317
2.38k
            njs_parser_next(parser, njs_parser_expression);
6318
6319
2.38k
            return njs_parser_after(parser, current, node, 1,
6320
2.38k
                                    njs_parser_for_expression);
6321
2.38k
        }
6322
6323
889
        parser->target = node;
6324
6325
889
        njs_lexer_consume_token(parser->lexer, 1);
6326
889
        njs_parser_next(parser, njs_parser_for_expression);
6327
6328
889
        return NJS_OK;
6329
6330
151
    case NJS_TOKEN_OF:
6331
151
        return njs_parser_not_supported(parser, token);
6332
6333
300
    default:
6334
300
        return njs_parser_failed(parser);
6335
3.72k
    }
6336
3.72k
}
6337
6338
6339
static njs_int_t
6340
njs_parser_for_in_statement(njs_parser_t *parser, njs_lexer_token_t *token,
6341
    njs_queue_link_t *current)
6342
1.38k
{
6343
1.38k
    njs_parser_node_t  *node, *forin;
6344
6345
1.38k
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
6346
1.38k
        return njs_parser_failed(parser);
6347
1.38k
    }
6348
6349
0
    njs_lexer_consume_token(parser->lexer, 1);
6350
6351
0
    node = parser->node;
6352
6353
0
    if (node->right != NULL && node->right->token_type == NJS_TOKEN_VAR) {
6354
0
        return NJS_ERROR;
6355
0
    }
6356
6357
0
    forin = njs_parser_node_new(parser, NJS_TOKEN_FOR_IN);
6358
0
    if (forin == NULL) {
6359
0
        return NJS_ERROR;
6360
0
    }
6361
6362
0
    forin->left = parser->node;
6363
6364
0
    parser->node = NULL;
6365
6366
0
    njs_parser_next(parser, njs_parser_statement_wo_node);
6367
6368
0
    return njs_parser_after(parser, current, forin, 1,
6369
0
                            njs_parser_for_in_statement_after);
6370
0
}
6371
6372
6373
static njs_int_t
6374
njs_parser_for_in_statement_statement(njs_parser_t *parser,
6375
    njs_lexer_token_t *token, njs_queue_link_t *current)
6376
5.52k
{
6377
5.52k
    njs_parser_node_t  *forin;
6378
6379
5.52k
    if (parser->ret != NJS_OK || token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
6380
502
        return njs_parser_failed(parser);
6381
502
    }
6382
6383
5.02k
    njs_lexer_consume_token(parser->lexer, 1);
6384
6385
5.02k
    parser->target->right = parser->node;
6386
6387
5.02k
    forin = njs_parser_node_new(parser, NJS_TOKEN_FOR_IN);
6388
5.02k
    if (forin == NULL) {
6389
0
        return NJS_ERROR;
6390
0
    }
6391
6392
5.02k
    forin->left = parser->target;
6393
6394
5.02k
    parser->node = NULL;
6395
6396
5.02k
    njs_parser_next(parser, njs_parser_statement_wo_node);
6397
6398
5.02k
    return njs_parser_after(parser, current, forin, 1,
6399
5.02k
                            njs_parser_for_in_statement_after);
6400
5.02k
}
6401
6402
6403
static njs_int_t
6404
njs_parser_for_in_statement_after(njs_parser_t *parser,
6405
    njs_lexer_token_t *token, njs_queue_link_t *current)
6406
4.78k
{
6407
4.78k
    return njs_parser_right_link_pop(parser);
6408
4.78k
}
6409
6410
6411
static njs_int_t
6412
njs_parser_for_expression(njs_parser_t *parser, njs_lexer_token_t *token,
6413
    njs_queue_link_t *current)
6414
12.2k
{
6415
12.2k
    njs_parser_node_t  *body, *cond, *for_node;
6416
6417
12.2k
    if (token->type != NJS_TOKEN_SEMICOLON) {
6418
558
        return njs_parser_failed(parser);
6419
558
    }
6420
6421
11.6k
    njs_lexer_consume_token(parser->lexer, 1);
6422
6423
11.6k
    for_node = njs_parser_node_new(parser, NJS_TOKEN_FOR);
6424
11.6k
    if (for_node == NULL) {
6425
0
        return NJS_ERROR;
6426
0
    }
6427
6428
11.6k
    cond = njs_parser_node_new(parser, 0);
6429
11.6k
    if (cond == NULL) {
6430
0
        return NJS_ERROR;
6431
0
    }
6432
6433
11.6k
    body = njs_parser_node_new(parser, 0);
6434
11.6k
    if (body == NULL) {
6435
0
        return NJS_ERROR;
6436
0
    }
6437
6438
11.6k
    for_node->left = parser->target;
6439
11.6k
    for_node->right = cond;
6440
6441
11.6k
    cond->left = parser->node;
6442
11.6k
    cond->right = body;
6443
6444
11.6k
    parser->node = NULL;
6445
6446
11.6k
    token = njs_lexer_token(parser->lexer, 0);
6447
11.6k
    if (token == NULL) {
6448
0
        return NJS_ERROR;
6449
0
    }
6450
6451
11.6k
    if (token->type == NJS_TOKEN_CLOSE_PARENTHESIS) {
6452
5.41k
        parser->target = for_node;
6453
6454
5.41k
        njs_parser_next(parser, njs_parser_for_expression_end);
6455
6456
5.41k
        return NJS_OK;
6457
5.41k
    }
6458
6459
6.26k
    njs_parser_next(parser, njs_parser_expression);
6460
6461
6.26k
    return njs_parser_after(parser, current, for_node, 1,
6462
11.6k
                            njs_parser_for_expression_end);
6463
11.6k
}
6464
6465
6466
static njs_int_t
6467
njs_parser_for_expression_end(njs_parser_t *parser, njs_lexer_token_t *token,
6468
    njs_queue_link_t *current)
6469
11.5k
{
6470
11.5k
    njs_parser_node_t  *body;
6471
6472
11.5k
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
6473
352
        return njs_parser_failed(parser);
6474
352
    }
6475
6476
11.1k
    njs_lexer_consume_token(parser->lexer, 1);
6477
6478
11.1k
    body = parser->target->right->right;
6479
6480
11.1k
    body->right = parser->node;
6481
6482
11.1k
    parser->node = NULL;
6483
6484
11.1k
    njs_parser_next(parser, njs_parser_statement_wo_node);
6485
6486
11.1k
    return njs_parser_after(parser, current, parser->target, 1,
6487
11.5k
                            njs_parser_for_end);
6488
11.5k
}
6489
6490
6491
static njs_int_t
6492
njs_parser_for_end(njs_parser_t *parser, njs_lexer_token_t *token,
6493
    njs_queue_link_t *current)
6494
8.75k
{
6495
8.75k
    njs_parser_node_t  *body;
6496
6497
8.75k
    body = parser->target->right->right;
6498
6499
8.75k
    body->left = parser->node;
6500
8.75k
    parser->node = parser->target;
6501
6502
8.75k
    return njs_parser_stack_pop(parser);
6503
8.75k
}
6504
6505
6506
/*
6507
 * 13.8 continue Statement.
6508
 */
6509
static njs_int_t
6510
njs_parser_continue_statement(njs_parser_t *parser, njs_lexer_token_t *token,
6511
    njs_queue_link_t *current)
6512
1.49k
{
6513
1.49k
    return njs_parser_break_continue(parser, token, NJS_TOKEN_CONTINUE);
6514
1.49k
}
6515
6516
6517
/*
6518
 * 13.9 break Statement.
6519
 */
6520
static njs_int_t
6521
njs_parser_break_statement(njs_parser_t *parser, njs_lexer_token_t *token,
6522
    njs_queue_link_t *current)
6523
19.4k
{
6524
19.4k
    return njs_parser_break_continue(parser, token, NJS_TOKEN_BREAK);
6525
19.4k
}
6526
6527
6528
static njs_int_t
6529
njs_parser_break_continue(njs_parser_t *parser, njs_lexer_token_t *token,
6530
    njs_token_type_t type)
6531
20.9k
{
6532
20.9k
    njs_int_t  ret;
6533
6534
20.9k
    parser->node = njs_parser_node_new(parser, type);
6535
20.9k
    if (parser->node == NULL) {
6536
0
        return NJS_ERROR;
6537
0
    }
6538
6539
20.9k
    parser->node->token_line = parser->line;
6540
6541
20.9k
    switch (token->type) {
6542
2.43k
    case NJS_TOKEN_SEMICOLON:
6543
2.43k
        break;
6544
6545
0
    case NJS_TOKEN_LINE_END:
6546
0
        return njs_parser_failed(parser);
6547
6548
18.5k
    default:
6549
18.5k
        if (njs_lexer_token_is_label_identifier(token)) {
6550
6.38k
            if (parser->lexer->prev_type == NJS_TOKEN_LINE_END) {
6551
5.56k
                return njs_parser_stack_pop(parser);
6552
5.56k
            }
6553
6554
819
            if (njs_label_find(parser->vm, parser->scope, token->atom_id)
6555
819
                == NULL)
6556
547
            {
6557
547
                njs_parser_syntax_error(parser, "Undefined label \"%V\"",
6558
547
                                        &token->text);
6559
547
                return NJS_DONE;
6560
547
            }
6561
6562
272
            ret = njs_name_copy(parser->vm, &parser->node->name, &token->text);
6563
272
            if (ret != NJS_OK) {
6564
0
                return NJS_ERROR;
6565
0
            }
6566
6567
272
            break;
6568
272
        }
6569
6570
12.1k
        if (token->type != NJS_TOKEN_END
6571
12.1k
            && token->type != NJS_TOKEN_CLOSE_BRACE
6572
4.24k
            && parser->lexer->prev_type != NJS_TOKEN_LINE_END)
6573
622
        {
6574
622
            return njs_parser_failed(parser);
6575
622
        }
6576
6577
11.5k
        return njs_parser_stack_pop(parser);
6578
20.9k
    }
6579
6580
2.70k
    njs_lexer_consume_token(parser->lexer, 1);
6581
6582
2.70k
    return njs_parser_stack_pop(parser);
6583
20.9k
}
6584
6585
6586
/*
6587
 * 13.10 return Statement.
6588
 */
6589
static njs_int_t
6590
njs_parser_return_statement(njs_parser_t *parser, njs_lexer_token_t *token,
6591
    njs_queue_link_t *current)
6592
13.5k
{
6593
13.5k
    njs_parser_node_t   *node;
6594
13.5k
    njs_parser_scope_t  *scope;
6595
6596
13.5k
    for (scope = parser->scope;
6597
42.6k
         scope != NULL;
6598
29.1k
         scope = scope->parent)
6599
42.6k
    {
6600
42.6k
        if (scope->type == NJS_SCOPE_FUNCTION) {
6601
13.4k
            break;
6602
13.4k
        }
6603
6604
29.2k
        if (scope->parent == NULL) {
6605
130
            njs_parser_syntax_error(parser, "Illegal return statement");
6606
130
            return NJS_ERROR;
6607
130
        }
6608
29.2k
    }
6609
6610
13.4k
    node = njs_parser_node_new(parser, NJS_TOKEN_RETURN);
6611
13.4k
    if (njs_slow_path(node == NULL)) {
6612
0
        return NJS_ERROR;
6613
0
    }
6614
6615
13.4k
    node->token_line = parser->line;
6616
6617
13.4k
    switch (token->type) {
6618
535
    case NJS_TOKEN_SEMICOLON:
6619
535
        njs_lexer_consume_token(parser->lexer, 1);
6620
535
        break;
6621
6622
0
    case NJS_TOKEN_LINE_END:
6623
0
        return njs_parser_failed(parser);
6624
6625
12.8k
    default:
6626
12.8k
        if (parser->lexer->prev_type == NJS_TOKEN_LINE_END) {
6627
3.90k
            break;
6628
3.90k
        }
6629
6630
8.97k
        parser->node = NULL;
6631
6632
8.97k
        if (token->type != NJS_TOKEN_CLOSE_BRACE) {
6633
7.32k
            njs_parser_next(parser, njs_parser_expression);
6634
6635
7.32k
            return njs_parser_after(parser, current, node, 0,
6636
7.32k
                                    njs_parser_return_statement_after);
6637
7.32k
        }
6638
13.4k
    }
6639
6640
6.09k
    parser->node = node;
6641
6642
6.09k
    return njs_parser_stack_pop(parser);
6643
13.4k
}
6644
6645
6646
static njs_int_t
6647
njs_parser_return_statement_after(njs_parser_t *parser,
6648
    njs_lexer_token_t *token, njs_queue_link_t *current)
6649
7.32k
{
6650
7.32k
    if (parser->ret != NJS_OK) {
6651
382
        njs_parser_syntax_error(parser, "Unexpected token \"%V\"",
6652
382
                                &token->text);
6653
382
        return NJS_DONE;
6654
382
    }
6655
6656
6.94k
    if (njs_parser_expect_semicolon(parser, token) != NJS_OK) {
6657
1.00k
        return njs_parser_failed(parser);
6658
1.00k
    }
6659
6660
5.93k
    return njs_parser_right_link_pop(parser);
6661
6.94k
}
6662
6663
6664
/*
6665
 * 13.11 with Statement
6666
 */
6667
static njs_int_t
6668
njs_parser_with_statement(njs_parser_t *parser, njs_lexer_token_t *token,
6669
    njs_queue_link_t *current)
6670
15
{
6671
15
    return njs_parser_not_supported(parser, token);
6672
15
}
6673
6674
6675
/*
6676
 * 13.12 switch Statement
6677
 */
6678
static njs_int_t
6679
njs_parser_switch_statement(njs_parser_t *parser, njs_lexer_token_t *token,
6680
    njs_queue_link_t *current)
6681
8.45k
{
6682
8.45k
    njs_int_t          ret;
6683
8.45k
    njs_parser_node_t  *swtch;
6684
6685
8.45k
    swtch = njs_parser_node_new(parser, NJS_TOKEN_SWITCH);
6686
8.45k
    if (swtch == NULL) {
6687
0
        return NJS_ERROR;
6688
0
    }
6689
6690
8.45k
    swtch->token_line = parser->line;
6691
6692
8.45k
    njs_parser_next(parser, njs_parser_expression_parenthesis);
6693
6694
8.45k
    ret = njs_parser_after(parser, current, swtch, 1,
6695
8.45k
                           njs_parser_switch_block);
6696
8.45k
    if (ret != NJS_OK) {
6697
0
        return ret;
6698
0
    }
6699
6700
8.45k
    return njs_parser_after(parser, current, swtch, 1,
6701
8.45k
                            njs_parser_switch_statement_after);
6702
8.45k
}
6703
6704
6705
static njs_int_t
6706
njs_parser_switch_statement_after(njs_parser_t *parser,
6707
    njs_lexer_token_t *token, njs_queue_link_t *current)
6708
4.64k
{
6709
4.64k
    parser->node = parser->target;
6710
6711
4.64k
    return njs_parser_stack_pop(parser);
6712
4.64k
}
6713
6714
6715
static njs_int_t
6716
njs_parser_switch_block(njs_parser_t *parser, njs_lexer_token_t *token,
6717
    njs_queue_link_t *current)
6718
8.34k
{
6719
8.34k
    njs_int_t  ret;
6720
6721
8.34k
    if (token->type != NJS_TOKEN_OPEN_BRACE) {
6722
322
        return njs_parser_failed(parser);
6723
322
    }
6724
6725
8.02k
    njs_lexer_consume_token(parser->lexer, 1);
6726
6727
8.02k
    parser->target->left = parser->node;
6728
6729
8.02k
    ret = njs_parser_scope_begin(parser, NJS_SCOPE_BLOCK, 0);
6730
8.02k
    if (ret != NJS_OK) {
6731
0
        return NJS_ERROR;
6732
0
    }
6733
6734
8.02k
    njs_parser_next(parser, njs_parser_switch_case);
6735
6736
8.02k
    return njs_parser_after(parser, current, NULL, 1,
6737
8.02k
                            njs_parser_switch_block_after);
6738
8.02k
}
6739
6740
static njs_int_t
6741
njs_parser_switch_block_after(njs_parser_t *parser, njs_lexer_token_t *token,
6742
    njs_queue_link_t *current)
6743
4.64k
{
6744
4.64k
    njs_parser_scope_end(parser);
6745
6746
4.64k
    return njs_parser_stack_pop(parser);
6747
4.64k
}
6748
6749
6750
static njs_int_t
6751
njs_parser_switch_case(njs_parser_t *parser, njs_lexer_token_t *token,
6752
    njs_queue_link_t *current)
6753
15.5k
{
6754
15.5k
    return njs_parser_switch_case_def(parser, token, current, 1);
6755
15.5k
}
6756
6757
6758
static njs_int_t
6759
njs_parser_switch_case_wo_def(njs_parser_t *parser, njs_lexer_token_t *token,
6760
    njs_queue_link_t *current)
6761
2.90k
{
6762
2.90k
    return njs_parser_switch_case_def(parser, token, current, 0);
6763
2.90k
}
6764
6765
6766
static njs_int_t
6767
njs_parser_switch_case_def(njs_parser_t *parser, njs_lexer_token_t *token,
6768
    njs_queue_link_t *current, njs_bool_t with_default)
6769
18.4k
{
6770
18.4k
    njs_parser_node_t  *node, *branch;
6771
6772
18.4k
    node = njs_parser_node_new(parser, 0);
6773
18.4k
    if (node == NULL) {
6774
0
        return NJS_ERROR;
6775
0
    }
6776
6777
18.4k
    parser->node = NULL;
6778
6779
18.4k
    switch (token->type) {
6780
9.88k
    case NJS_TOKEN_CASE:
6781
9.88k
        branch = njs_parser_node_new(parser, 0);
6782
9.88k
        if (branch == NULL) {
6783
0
            return NJS_ERROR;
6784
0
        }
6785
6786
9.88k
        branch->token_line = token->line;
6787
9.88k
        branch->right = node;
6788
6789
9.88k
        njs_parser_next(parser, njs_parser_expression);
6790
6791
9.88k
        njs_lexer_consume_token(parser->lexer, 1);
6792
6793
9.88k
        if (parser->target->token_type == NJS_TOKEN_SWITCH) {
6794
7.00k
            parser->target->right = branch;
6795
6796
7.00k
        } else {
6797
2.88k
            parser->target->left = branch;
6798
2.88k
        }
6799
6800
9.88k
        if (with_default) {
6801
9.09k
            return njs_parser_after(parser, current, branch, 1,
6802
9.09k
                                    njs_parser_switch_case_after);
6803
6804
9.09k
        } else {
6805
797
            return njs_parser_after(parser, current, branch, 1,
6806
797
                                    njs_parser_switch_case_after_wo_def);
6807
797
        }
6808
6809
3.87k
    case NJS_TOKEN_DEFAULT:
6810
3.87k
        if (!with_default) {
6811
1.03k
            njs_parser_syntax_error(parser, "More than one default clause "
6812
1.03k
                                            "in switch statement");
6813
1.03k
            return NJS_DONE;
6814
1.03k
        }
6815
6816
2.84k
        if (parser->target->token_type == NJS_TOKEN_SWITCH) {
6817
603
            parser->target->right = node;
6818
6819
2.24k
        } else {
6820
2.24k
            parser->target->left = node;
6821
2.24k
        }
6822
6823
2.84k
        node->token_line = token->line;
6824
2.84k
        node->token_type = NJS_TOKEN_DEFAULT;
6825
6826
2.84k
        parser->target = node;
6827
6828
2.84k
        njs_lexer_consume_token(parser->lexer, 1);
6829
2.84k
        njs_parser_next(parser, njs_parser_switch_case_after_wo_def);
6830
6831
2.84k
        break;
6832
6833
4.64k
    case NJS_TOKEN_CLOSE_BRACE:
6834
4.64k
        njs_lexer_consume_token(parser->lexer, 1);
6835
4.64k
        return njs_parser_stack_pop(parser);
6836
6837
66
    default:
6838
66
        return njs_parser_failed(parser);
6839
18.4k
    }
6840
6841
2.84k
    return NJS_OK;
6842
18.4k
}
6843
6844
6845
static njs_int_t
6846
njs_parser_switch_case_after(njs_parser_t *parser, njs_lexer_token_t *token,
6847
    njs_queue_link_t *current)
6848
8.42k
{
6849
8.42k
    if (token->type != NJS_TOKEN_COLON) {
6850
203
        return njs_parser_failed(parser);
6851
203
    }
6852
6853
8.21k
    njs_lexer_consume_token(parser->lexer, 1);
6854
6855
8.21k
    parser->target->right->left = parser->node;
6856
8.21k
    parser->node = NULL;
6857
6858
8.21k
    token = njs_lexer_token(parser->lexer, 0);
6859
8.21k
    if (token == NULL) {
6860
0
        return NJS_ERROR;
6861
0
    }
6862
6863
8.21k
    switch (token->type) {
6864
442
    case NJS_TOKEN_CASE:
6865
1.46k
    case NJS_TOKEN_DEFAULT:
6866
2.63k
    case NJS_TOKEN_CLOSE_BRACE:
6867
2.63k
        njs_parser_next(parser, njs_parser_switch_case_block);
6868
2.63k
        return NJS_OK;
6869
6870
5.58k
    default:
6871
5.58k
        njs_parser_next(parser, njs_parser_statement_list);
6872
5.58k
        break;
6873
8.21k
    }
6874
6875
5.58k
    return njs_parser_after(parser, current, parser->target, 1,
6876
8.21k
                            njs_parser_switch_case_block);
6877
8.21k
}
6878
6879
6880
static njs_int_t
6881
njs_parser_switch_case_block(njs_parser_t *parser, njs_lexer_token_t *token,
6882
    njs_queue_link_t *current)
6883
8.21k
{
6884
8.21k
    parser->target->right->right = parser->node;
6885
6886
8.21k
    if (parser->ret != NJS_OK && parser->target->scope != parser->scope) {
6887
675
        return njs_parser_failed(parser);
6888
675
    }
6889
6890
7.54k
    njs_parser_next(parser, njs_parser_switch_case);
6891
6892
7.54k
    return NJS_OK;
6893
8.21k
}
6894
6895
6896
static njs_int_t
6897
njs_parser_switch_case_after_wo_def(njs_parser_t *parser,
6898
    njs_lexer_token_t *token, njs_queue_link_t *current)
6899
3.63k
{
6900
3.63k
    if (token->type != NJS_TOKEN_COLON) {
6901
405
        return njs_parser_failed(parser);
6902
405
    }
6903
6904
3.22k
    njs_lexer_consume_token(parser->lexer, 1);
6905
6906
3.22k
    if (parser->target->right != NULL) {
6907
649
        parser->target->right->left = parser->node;
6908
649
    }
6909
6910
3.22k
    parser->node = NULL;
6911
6912
3.22k
    token = njs_lexer_token(parser->lexer, 0);
6913
3.22k
    if (token == NULL) {
6914
0
        return NJS_ERROR;
6915
0
    }
6916
6917
3.22k
    switch (token->type) {
6918
783
    case NJS_TOKEN_CASE:
6919
1.80k
    case NJS_TOKEN_DEFAULT:
6920
2.02k
    case NJS_TOKEN_CLOSE_BRACE:
6921
2.02k
        njs_parser_next(parser, njs_parser_switch_case_block_wo_def);
6922
2.02k
        return NJS_OK;
6923
6924
1.19k
    default:
6925
1.19k
        njs_parser_next(parser, njs_parser_statement_list);
6926
1.19k
        break;
6927
3.22k
    }
6928
6929
1.19k
    return njs_parser_after(parser, current, parser->target, 1,
6930
3.22k
                            njs_parser_switch_case_block_wo_def);
6931
3.22k
}
6932
6933
6934
static njs_int_t
6935
njs_parser_switch_case_block_wo_def(njs_parser_t *parser,
6936
    njs_lexer_token_t *token, njs_queue_link_t *current)
6937
3.17k
{
6938
3.17k
    if (parser->target->right != NULL) {
6939
602
        parser->target->right->right = parser->node;
6940
6941
2.57k
    } else {
6942
2.57k
        parser->target->right = parser->node;
6943
2.57k
    }
6944
6945
3.17k
    if (parser->ret != NJS_OK && parser->target->scope != parser->scope) {
6946
273
        return njs_parser_failed(parser);
6947
273
    }
6948
6949
2.90k
    njs_parser_next(parser, njs_parser_switch_case_wo_def);
6950
6951
2.90k
    return NJS_OK;
6952
3.17k
}
6953
6954
6955
/*
6956
 * 13.13 Labelled Statements
6957
 */
6958
static njs_int_t
6959
njs_parser_labelled_statement(njs_parser_t *parser, njs_lexer_token_t *token,
6960
    njs_queue_link_t *current)
6961
35.3k
{
6962
35.3k
    uintptr_t       atom_id;
6963
35.3k
    njs_variable_t  *label;
6964
6965
35.3k
    atom_id = token->atom_id;
6966
6967
35.3k
    label = njs_label_find(parser->vm, parser->scope, atom_id);
6968
35.3k
    if (label != NULL) {
6969
277
        njs_parser_syntax_error(parser, "Label \"%V\" "
6970
277
                                "has already been declared", &token->text);
6971
277
        return NJS_DONE;
6972
277
    }
6973
6974
35.0k
    label = njs_label_add(parser->vm, parser->scope, atom_id);
6975
35.0k
    if (label == NULL) {
6976
0
        return NJS_ERROR;
6977
0
    }
6978
6979
35.0k
    njs_lexer_consume_token(parser->lexer, 2);
6980
6981
35.0k
    token = njs_lexer_token(parser->lexer, 0);
6982
35.0k
    if (token == NULL) {
6983
0
        return NJS_ERROR;
6984
0
    }
6985
6986
35.0k
    parser->node = NULL;
6987
6988
35.0k
    if (token->type == NJS_TOKEN_FUNCTION) {
6989
148
        njs_syntax_error(parser->vm, "In strict mode code, functions can only "
6990
148
                                "be declared at top level or inside a block.");
6991
148
        return NJS_DONE;
6992
6993
34.9k
    } else {
6994
34.9k
        njs_parser_next(parser, njs_parser_statement_wo_node);
6995
34.9k
    }
6996
6997
34.9k
    return njs_parser_after(parser, current, (void *) atom_id, 1,
6998
35.0k
                            njs_parser_labelled_statement_after);
6999
35.0k
}
7000
7001
7002
static njs_int_t
7003
njs_parser_labelled_statement_after(njs_parser_t *parser,
7004
    njs_lexer_token_t *token, njs_queue_link_t *current)
7005
30.8k
{
7006
30.8k
    njs_int_t          ret;
7007
30.8k
    njs_str_t          str;
7008
30.8k
    uintptr_t          atom_id;
7009
30.8k
    njs_parser_node_t  *node;
7010
7011
30.8k
    node = parser->node;
7012
30.8k
    if (node == NULL) {
7013
1.26k
        node = njs_parser_node_new(parser, NJS_TOKEN_BLOCK);
7014
1.26k
        if (node == NULL) {
7015
0
            return NJS_ERROR;
7016
0
        }
7017
7018
1.26k
        node->token_line = token->line;
7019
7020
1.26k
        parser->node = node;
7021
1.26k
    }
7022
7023
30.8k
    atom_id = (uint32_t) (uintptr_t) parser->target;
7024
7025
30.8k
    njs_atom_string_get(parser->vm, atom_id, &str);
7026
7027
30.8k
    ret = njs_name_copy(parser->vm, &parser->node->name, &str);
7028
30.8k
    if (ret != NJS_OK) {
7029
0
        return NJS_ERROR;
7030
0
    }
7031
7032
30.8k
    ret = njs_label_remove(parser->vm, parser->scope, atom_id);
7033
30.8k
    if (ret != NJS_OK) {
7034
0
        return NJS_ERROR;
7035
0
    }
7036
7037
30.8k
    return njs_parser_stack_pop(parser);
7038
30.8k
}
7039
7040
7041
/*
7042
 * 13.14 throw Statement
7043
 */
7044
static njs_int_t
7045
njs_parser_throw_statement(njs_parser_t *parser, njs_lexer_token_t *token,
7046
    njs_queue_link_t *current)
7047
24.4k
{
7048
24.4k
    njs_parser_node_t  *node;
7049
7050
24.4k
    node = njs_parser_node_new(parser, NJS_TOKEN_THROW);
7051
24.4k
    if (node == NULL) {
7052
0
        return NJS_ERROR;
7053
0
    }
7054
7055
24.4k
    node->token_line = parser->line;
7056
7057
24.4k
    if (parser->lexer->prev_type == NJS_TOKEN_LINE_END) {
7058
294
        njs_parser_syntax_error(parser, "Illegal newline after throw");
7059
294
        return NJS_DONE;
7060
294
    }
7061
7062
24.1k
    parser->node = NULL;
7063
7064
24.1k
    njs_parser_next(parser, njs_parser_expression);
7065
7066
24.1k
    return njs_parser_after(parser, current, node, 1,
7067
24.4k
                            njs_parser_throw_statement_after);
7068
24.4k
}
7069
7070
7071
static njs_int_t
7072
njs_parser_throw_statement_after(njs_parser_t *parser,
7073
    njs_lexer_token_t *token, njs_queue_link_t *current)
7074
24.1k
{
7075
24.1k
    if (parser->ret != NJS_OK) {
7076
0
        parser->node = parser->target;
7077
0
        return njs_parser_reject(parser);
7078
0
    }
7079
7080
24.1k
    if (njs_parser_expect_semicolon(parser, token) != NJS_OK) {
7081
658
        return njs_parser_failed(parser);
7082
658
    }
7083
7084
23.5k
    return njs_parser_right_link_pop(parser);
7085
24.1k
}
7086
7087
7088
/*
7089
 * 13.15 try Statement.
7090
 */
7091
static njs_int_t
7092
njs_parser_try_statement(njs_parser_t *parser, njs_lexer_token_t *token,
7093
    njs_queue_link_t *current)
7094
23.2k
{
7095
23.2k
    njs_parser_node_t  *try;
7096
7097
23.2k
    try = njs_parser_node_new(parser, NJS_TOKEN_TRY);
7098
23.2k
    if (try == NULL) {
7099
0
        return NJS_ERROR;
7100
0
    }
7101
7102
23.2k
    try->token_line = parser->line;
7103
7104
23.2k
    parser->node = NULL;
7105
7106
23.2k
    njs_parser_next(parser, njs_parser_block_statement_open_brace);
7107
7108
23.2k
    return njs_parser_after(parser, current, try, 1,
7109
23.2k
                            njs_parser_catch_or_finally);
7110
23.2k
}
7111
7112
7113
static njs_int_t
7114
njs_parser_catch_or_finally(njs_parser_t *parser, njs_lexer_token_t *token,
7115
    njs_queue_link_t *current)
7116
21.2k
{
7117
21.2k
    njs_int_t          ret;
7118
21.2k
    njs_parser_node_t  *catch, *node, *try;
7119
7120
21.2k
    try = parser->target;
7121
7122
21.2k
    try->left = parser->node;
7123
7124
21.2k
    if (token->type == NJS_TOKEN_FINALLY) {
7125
1.73k
        node = njs_parser_node_new(parser, NJS_TOKEN_FINALLY);
7126
1.73k
        if (node == NULL) {
7127
0
            return NJS_ERROR;
7128
0
        }
7129
7130
1.73k
        node->token_line = token->line;
7131
7132
1.73k
        if (try->right != NULL) {
7133
0
            node->left = try->right;
7134
0
        }
7135
7136
1.73k
        try->right = node;
7137
1.73k
        parser->node = NULL;
7138
7139
1.73k
        njs_lexer_consume_token(parser->lexer, 1);
7140
1.73k
        njs_parser_next(parser, njs_parser_block_statement_open_brace);
7141
7142
1.73k
        return njs_parser_after(parser, current, try, 0,
7143
1.73k
                                njs_parser_catch_finally);
7144
7145
19.4k
    } else if (token->type != NJS_TOKEN_CATCH) {
7146
370
        njs_parser_syntax_error(parser, "Missing catch or finally after try");
7147
370
        return NJS_DONE;
7148
370
    }
7149
7150
19.1k
    catch = njs_parser_node_new(parser, NJS_TOKEN_CATCH);
7151
19.1k
    if (catch == NULL) {
7152
0
        return NJS_ERROR;
7153
0
    }
7154
7155
19.1k
    catch->token_line = token->line;
7156
7157
19.1k
    njs_lexer_consume_token(parser->lexer, 1);
7158
7159
19.1k
    token = njs_lexer_token(parser->lexer, 0);
7160
19.1k
    if (token == NULL) {
7161
0
        return NJS_ERROR;
7162
0
    }
7163
7164
19.1k
    ret = njs_parser_scope_begin(parser, NJS_SCOPE_BLOCK, 0);
7165
19.1k
    if (ret != NJS_OK) {
7166
0
        return NJS_ERROR;
7167
0
    }
7168
7169
19.1k
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS) {
7170
112
        parser->node = NULL;
7171
7172
112
        njs_parser_next(parser, njs_parser_block_statement_open_brace);
7173
7174
112
        try->right = catch;
7175
7176
        /* TODO: it is necessary to change the generator. */
7177
7178
112
        return njs_parser_not_supported(parser, token);
7179
7180
        /*
7181
         *  return njs_parser_after(parser, current, parser->target, 0,
7182
         *                          njs_parser_catch_after);
7183
         */
7184
112
    }
7185
7186
19.0k
    njs_lexer_consume_token(parser->lexer, 1);
7187
7188
19.0k
    token = njs_lexer_token(parser->lexer, 0);
7189
19.0k
    if (token == NULL) {
7190
0
        return NJS_ERROR;
7191
0
    }
7192
7193
19.0k
    try->right = catch;
7194
7195
19.0k
    if (njs_lexer_token_is_binding_identifier(token)) {
7196
16.2k
        node = njs_parser_variable_node(parser, token->atom_id,
7197
16.2k
                                        NJS_VARIABLE_CATCH, NULL);
7198
16.2k
        if (node == NULL) {
7199
0
            return NJS_ERROR;
7200
0
        }
7201
7202
16.2k
        node->token_line = token->line;
7203
7204
16.2k
        catch->left = node;
7205
7206
16.2k
        njs_lexer_consume_token(parser->lexer, 1);
7207
16.2k
        njs_parser_next(parser, njs_parser_catch_parenthesis);
7208
16.2k
        return NJS_OK;
7209
16.2k
    }
7210
7211
2.72k
    if (token->type != NJS_TOKEN_OPEN_BRACE) {
7212
2.59k
        return njs_parser_failed(parser);
7213
2.59k
    }
7214
7215
    /*
7216
     * njs_parser_next(parser, njs_parser_block_statement_open_brace);
7217
     *
7218
     * return njs_parser_after(parser, current, try, 1,
7219
     *                       njs_parser_catch_parenthesis);
7220
     */
7221
7222
131
    return njs_parser_not_supported(parser, token);
7223
2.72k
}
7224
7225
7226
static njs_int_t
7227
njs_parser_catch_after(njs_parser_t *parser, njs_lexer_token_t *token,
7228
    njs_queue_link_t *current)
7229
15.5k
{
7230
15.5k
    njs_parser_node_t  *node;
7231
7232
15.5k
    parser->target->right->right = parser->node;
7233
7234
15.5k
    if (token->type == NJS_TOKEN_FINALLY) {
7235
253
        node = njs_parser_node_new(parser, NJS_TOKEN_FINALLY);
7236
253
        if (node == NULL) {
7237
0
            return NJS_ERROR;
7238
0
        }
7239
7240
253
        node->token_line = token->line;
7241
7242
253
        if (parser->target->right != NULL) {
7243
253
            node->left = parser->target->right;
7244
253
        }
7245
7246
253
        parser->target->right = node;
7247
253
        parser->node = NULL;
7248
7249
253
        njs_lexer_consume_token(parser->lexer, 1);
7250
253
        njs_parser_next(parser, njs_parser_block_statement_open_brace);
7251
7252
253
        return njs_parser_after(parser, current, parser->target, 1,
7253
253
                                njs_parser_catch_finally);
7254
253
    }
7255
7256
15.3k
    parser->node = parser->target;
7257
7258
15.3k
    return njs_parser_stack_pop(parser);
7259
15.5k
}
7260
7261
7262
static njs_int_t
7263
njs_parser_catch_parenthesis(njs_parser_t *parser, njs_lexer_token_t *token,
7264
    njs_queue_link_t *current)
7265
16.2k
{
7266
16.2k
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
7267
228
        return njs_parser_failed(parser);
7268
228
    }
7269
7270
16.0k
    njs_lexer_consume_token(parser->lexer, 1);
7271
7272
16.0k
    parser->target->right->right = parser->node;
7273
16.0k
    parser->node = NULL;
7274
7275
16.0k
    njs_parser_next(parser, njs_parser_catch_statement_open_brace);
7276
7277
16.0k
    return njs_parser_after(parser, current, parser->target, 1,
7278
16.2k
                            njs_parser_catch_after);
7279
16.2k
}
7280
7281
7282
static njs_int_t
7283
njs_parser_catch_statement_open_brace(njs_parser_t *parser,
7284
    njs_lexer_token_t *token, njs_queue_link_t *current)
7285
16.0k
{
7286
16.0k
    void  *target;
7287
7288
16.0k
    if (token->type != NJS_TOKEN_OPEN_BRACE) {
7289
157
        return njs_parser_failed(parser);
7290
157
    }
7291
7292
15.8k
    parser->line = token->line;
7293
7294
15.8k
    njs_lexer_consume_token(parser->lexer, 1);
7295
7296
15.8k
    token = njs_lexer_token(parser->lexer, 0);
7297
15.8k
    if (token == NULL) {
7298
0
        return NJS_ERROR;
7299
0
    }
7300
7301
15.8k
    target = (void *) (uintptr_t) parser->line;
7302
15.8k
    parser->node = NULL;
7303
7304
15.8k
    if (token->type == NJS_TOKEN_CLOSE_BRACE) {
7305
15.1k
        parser->target = target;
7306
7307
15.1k
        njs_parser_next(parser, njs_parser_block_statement_close_brace);
7308
15.1k
        return NJS_OK;
7309
15.1k
    }
7310
7311
772
    njs_parser_next(parser, njs_parser_statement_list);
7312
7313
772
    return njs_parser_after(parser, current, target, 0,
7314
15.8k
                            njs_parser_block_statement_close_brace);
7315
15.8k
}
7316
7317
7318
static njs_int_t
7319
njs_parser_catch_finally(njs_parser_t *parser, njs_lexer_token_t *token,
7320
    njs_queue_link_t *current)
7321
1.98k
{
7322
1.98k
    if (parser->ret != NJS_OK) {
7323
0
        return njs_parser_failed(parser);
7324
0
    }
7325
7326
1.98k
    parser->target->right->right = parser->node;
7327
1.98k
    parser->node = parser->target;
7328
7329
1.98k
    return njs_parser_stack_pop(parser);
7330
1.98k
}
7331
7332
7333
static njs_int_t
7334
njs_parser_debugger_statement(njs_parser_t *parser, njs_lexer_token_t *token,
7335
    njs_queue_link_t *current)
7336
7.33k
{
7337
7.33k
    parser->node = njs_parser_node_new(parser, NJS_TOKEN_DEBUGGER);
7338
7.33k
    if (parser->node == NULL) {
7339
0
        return NJS_ERROR;
7340
0
    }
7341
7342
7.33k
    parser->node->token_line = parser->line;
7343
7344
7.33k
    if (token->type != NJS_TOKEN_SEMICOLON
7345
514
        && token->type != NJS_TOKEN_END)
7346
514
    {
7347
514
        return njs_parser_failed(parser);
7348
514
    }
7349
7350
6.82k
    njs_lexer_consume_token(parser->lexer, 1);
7351
7352
6.82k
    return njs_parser_stack_pop(parser);
7353
7.33k
}
7354
7355
7356
/*
7357
 * 14.1 Function Definitions.
7358
 */
7359
static njs_int_t
7360
njs_parser_function_declaration(njs_parser_t *parser, njs_lexer_token_t *token,
7361
    njs_queue_link_t *current)
7362
5.78k
{
7363
5.78k
    njs_int_t          ret;
7364
5.78k
    uintptr_t          atom_id;
7365
5.78k
    njs_bool_t         async;
7366
5.78k
    njs_variable_t     *var;
7367
5.78k
    njs_parser_node_t  *node;
7368
7369
5.78k
    if (!njs_lexer_token_is_binding_identifier(token)) {
7370
83
        return njs_parser_failed(parser);
7371
83
    }
7372
7373
5.70k
    if (njs_parser_restricted_identifier(token->type)) {
7374
21
        njs_parser_syntax_error(parser, "Identifier \"%V\" is forbidden"
7375
21
                                " in function declaration", &token->text);
7376
21
        return NJS_DONE;
7377
21
    }
7378
7379
5.67k
    node = parser->node;
7380
5.67k
    atom_id = token->atom_id;
7381
7382
5.67k
    njs_lexer_consume_token(parser->lexer, 1);
7383
7384
5.67k
    token = njs_lexer_token(parser->lexer, 0);
7385
5.67k
    if (token == NULL) {
7386
0
        return NJS_ERROR;
7387
0
    }
7388
7389
5.67k
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS) {
7390
415
        return njs_parser_failed(parser);
7391
415
    }
7392
7393
5.26k
    njs_lexer_consume_token(parser->lexer, 1);
7394
7395
5.26k
    var = njs_variable_function_add(parser, parser->scope, atom_id);
7396
5.26k
    if (var == NULL) {
7397
60
        return NJS_ERROR;
7398
60
    }
7399
7400
5.20k
    node->u.value.data.u.lambda = njs_variable_lambda(var);
7401
7402
5.20k
    node->left = (njs_parser_node_t *) atom_id;
7403
7404
5.20k
    parser->node = node;
7405
7406
5.20k
    ret = njs_parser_scope_begin(parser, NJS_SCOPE_FUNCTION, 1);
7407
5.20k
    if (ret != NJS_OK) {
7408
0
        return NJS_ERROR;
7409
0
    }
7410
7411
5.20k
    async = (node->token_type == NJS_TOKEN_ASYNC_FUNCTION_DECLARATION);
7412
5.20k
    parser->scope->async = async;
7413
7414
5.20k
    njs_parser_next(parser, njs_parser_function_parse);
7415
7416
5.20k
    return njs_parser_after(parser, current, node, 1,
7417
5.20k
                            njs_parser_function_declaration_after);
7418
5.20k
}
7419
7420
7421
static njs_int_t
7422
njs_parser_function_declaration_after(njs_parser_t *parser,
7423
    njs_lexer_token_t *token, njs_queue_link_t *current)
7424
5.00k
{
7425
5.00k
    njs_int_t  ret;
7426
5.00k
    uintptr_t  atom_id;
7427
7428
5.00k
    atom_id = (uint32_t) (uintptr_t) parser->node->left;
7429
7430
5.00k
    parser->node->left = NULL;
7431
7432
5.00k
    njs_value_null_set(&parser->node->u.value);
7433
7434
5.00k
    ret = njs_parser_variable_reference(parser, parser->scope, parser->node,
7435
5.00k
                                        atom_id, NJS_DECLARATION);
7436
5.00k
    if (ret != NJS_OK) {
7437
0
        return NJS_ERROR;
7438
0
    }
7439
7440
5.00k
    return njs_parser_stack_pop(parser);
7441
5.00k
}
7442
7443
7444
static njs_int_t
7445
njs_parser_function_parse(njs_parser_t *parser, njs_lexer_token_t *token,
7446
    njs_queue_link_t *current)
7447
157k
{
7448
157k
    parser->target = parser->node;
7449
157k
    parser->node = NULL;
7450
7451
157k
    njs_parser_next(parser, njs_parser_formal_parameters);
7452
7453
157k
    return njs_parser_after(parser, current, parser->target, 1,
7454
157k
                            njs_parser_function_lambda_args_after);
7455
157k
}
7456
7457
7458
static njs_int_t
7459
njs_parser_function_expression(njs_parser_t *parser, njs_lexer_token_t *token,
7460
    njs_queue_link_t *current)
7461
152k
{
7462
152k
    njs_int_t              ret;
7463
152k
    uintptr_t              atom_id;
7464
152k
    njs_bool_t             async;
7465
152k
    njs_variable_t         *var;
7466
152k
    njs_function_lambda_t  *lambda;
7467
7468
152k
    ret = njs_parser_scope_begin(parser, NJS_SCOPE_FUNCTION, 1);
7469
152k
    if (ret != NJS_OK) {
7470
0
        return NJS_ERROR;
7471
0
    }
7472
7473
152k
    async = (parser->node->token_type == NJS_TOKEN_ASYNC_FUNCTION_EXPRESSION);
7474
152k
    parser->scope->async = async;
7475
7476
152k
    var = NULL;
7477
7478
152k
    if (njs_lexer_token_is_binding_identifier(token)) {
7479
1.38k
        atom_id = token->atom_id;
7480
7481
1.38k
        njs_lexer_consume_token(parser->lexer, 1);
7482
7483
1.38k
        token = njs_lexer_token(parser->lexer, 0);
7484
1.38k
        if (token == NULL) {
7485
0
            return NJS_ERROR;
7486
0
        }
7487
7488
150k
    } else {
7489
150k
        atom_id = NJS_ATOM_STRING_empty;
7490
150k
    }
7491
7492
152k
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS) {
7493
138
        return njs_parser_failed(parser);
7494
138
    }
7495
7496
151k
    njs_lexer_consume_token(parser->lexer, 1);
7497
7498
151k
    parser->node->left = njs_parser_node_new(parser, NJS_TOKEN_NAME);
7499
151k
    if (parser->node->left == NULL) {
7500
0
        return NJS_ERROR;
7501
0
    }
7502
7503
151k
    var = njs_variable_scope_add(parser, parser->scope, parser->scope,
7504
151k
                                 atom_id, NJS_VARIABLE_FUNCTION, 1);
7505
151k
    if (var == NULL) {
7506
0
        return NJS_ERROR;
7507
0
    }
7508
7509
151k
    var->self = 1;
7510
7511
151k
    ret = njs_parser_variable_reference(parser, parser->scope,
7512
151k
                                        parser->node->left, atom_id,
7513
151k
                                        NJS_DECLARATION);
7514
151k
    if (ret != NJS_OK) {
7515
0
        return NJS_ERROR;
7516
0
    }
7517
7518
151k
    lambda = njs_function_lambda_alloc(parser->vm, !async);
7519
151k
    if (lambda == NULL) {
7520
0
        return NJS_ERROR;
7521
0
    }
7522
7523
151k
    parser->node->u.value.data.u.lambda = lambda;
7524
7525
151k
    njs_parser_next(parser, njs_parser_function_parse);
7526
7527
151k
    return njs_parser_after(parser, current, var, 1,
7528
151k
                            njs_parser_function_expression_after);
7529
151k
}
7530
7531
7532
static njs_int_t
7533
njs_parser_function_expression_after(njs_parser_t *parser,
7534
    njs_lexer_token_t *token, njs_queue_link_t *current)
7535
89.2k
{
7536
89.2k
    njs_variable_t  *var;
7537
7538
89.2k
    var = (njs_variable_t *) parser->target;
7539
7540
89.2k
    if (var->self) {
7541
88.8k
        var->init = 1;
7542
88.8k
        var->type = NJS_VARIABLE_CONST;
7543
88.8k
    }
7544
7545
89.2k
    var->index = njs_scope_index(var->scope->type, var->scope->items,
7546
89.2k
                                 NJS_LEVEL_LOCAL, var->type);
7547
89.2k
    var->scope->items++;
7548
7549
89.2k
    if (var->self) {
7550
88.8k
        parser->node->u.value.data.u.lambda->self = var->index;
7551
88.8k
    }
7552
7553
89.2k
    return njs_parser_stack_pop(parser);
7554
89.2k
}
7555
7556
7557
static njs_int_t
7558
njs_parser_unique_formal_parameters(njs_parser_t *parser,
7559
    njs_lexer_token_t *token, njs_queue_link_t *current)
7560
5.24k
{
7561
5.24k
    parser->node = NULL;
7562
7563
5.24k
    njs_parser_next(parser, njs_parser_formal_parameters);
7564
7565
5.24k
    return NJS_OK;
7566
5.24k
}
7567
7568
7569
static njs_int_t
7570
njs_parser_formal_parameters(njs_parser_t *parser, njs_lexer_token_t *token,
7571
    njs_queue_link_t *current)
7572
166k
{
7573
166k
    njs_variable_t         *arg;
7574
166k
    njs_rbtree_node_t      *rb_node;
7575
166k
    njs_variable_node_t    var_node;
7576
166k
    njs_function_lambda_t  *lambda;
7577
7578
166k
    lambda = parser->target->u.value.data.u.lambda;
7579
7580
166k
    switch (token->type) {
7581
    /* BindingRestElement */
7582
142
    case NJS_TOKEN_ELLIPSIS:
7583
142
        if (lambda->rest_parameters != 0) {
7584
0
            return njs_parser_failed(parser);
7585
0
        }
7586
7587
142
        njs_lexer_consume_token(parser->lexer, 1);
7588
7589
142
        token = njs_lexer_token(parser->lexer, 0);
7590
142
        if (token == NULL) {
7591
0
            return NJS_ERROR;
7592
0
        }
7593
7594
142
        if (!njs_lexer_token_is_binding_identifier(token)) {
7595
0
            return njs_parser_failed(parser);
7596
0
        }
7597
7598
142
        lambda->rest_parameters = 1;
7599
7600
142
        return NJS_OK;
7601
7602
    /* BindingElement */
7603
192
    case NJS_TOKEN_OPEN_BRACE:
7604
192
        return njs_parser_not_supported(parser, token);
7605
7606
151
    case NJS_TOKEN_OPEN_BRACKET:
7607
151
        return njs_parser_not_supported(parser, token);
7608
7609
166k
    default:
7610
        /* SingleNameBinding */
7611
166k
        if (njs_lexer_token_is_binding_identifier(token)) {
7612
8.22k
            var_node.key = token->atom_id;
7613
7614
8.22k
            rb_node = njs_rbtree_find(&parser->scope->variables,
7615
8.22k
                                      &var_node.node);
7616
8.22k
            if (rb_node != NULL) {
7617
398
                arg = ((njs_variable_node_t *) rb_node)->variable;
7618
7619
398
                if (!arg->self) {
7620
1
                    njs_parser_syntax_error(parser,
7621
1
                                            "Duplicate parameter names");
7622
1
                    return NJS_DONE;
7623
1
                }
7624
7625
397
                arg->self = 0;
7626
7627
7.82k
            } else {
7628
7.82k
                arg = njs_variable_add(parser, parser->scope,
7629
7.82k
                                       token->atom_id, NJS_VARIABLE_VAR);
7630
7.82k
            }
7631
7632
8.22k
            if (arg == NULL) {
7633
0
                return NJS_ERROR;
7634
0
            }
7635
7636
8.22k
            arg->argument = 1;
7637
7638
8.22k
            lambda->nargs++;
7639
7640
            /* Crutch for temporary storage. */
7641
8.22k
            parser->node = (njs_parser_node_t *) arg;
7642
7643
8.22k
            njs_lexer_consume_token(parser->lexer, 1);
7644
7645
8.22k
            njs_parser_next(parser, njs_parser_formal_parameters_after);
7646
8.22k
            break;
7647
8.22k
        }
7648
7649
157k
        return njs_parser_stack_pop(parser);
7650
166k
    }
7651
7652
8.22k
    return NJS_OK;
7653
166k
}
7654
7655
7656
static njs_int_t
7657
njs_parser_formal_parameters_after(njs_parser_t *parser,
7658
    njs_lexer_token_t *token, njs_queue_link_t *current)
7659
8.22k
{
7660
8.22k
    njs_function_lambda_t  *lambda;
7661
7662
8.22k
    if (token->type != NJS_TOKEN_COMMA) {
7663
6.73k
        return njs_parser_stack_pop(parser);
7664
6.73k
    }
7665
7666
1.48k
    lambda = parser->target->u.value.data.u.lambda;
7667
7668
1.48k
    if (lambda->rest_parameters) {
7669
0
        njs_parser_syntax_error(parser, "Rest parameter must be "
7670
0
                                        "last formal parameter");
7671
0
        return NJS_DONE;
7672
0
    }
7673
7674
1.48k
    njs_lexer_consume_token(parser->lexer, 1);
7675
7676
1.48k
    njs_parser_next(parser, njs_parser_formal_parameters);
7677
7678
1.48k
    return NJS_OK;
7679
1.48k
}
7680
7681
7682
/*
7683
 * 14.2 Arrow Function Definitions
7684
 *
7685
 * TODO: implement according to specification.
7686
 */
7687
static njs_int_t
7688
njs_parser_arrow_function(njs_parser_t *parser, njs_lexer_token_t *token,
7689
    njs_queue_link_t *current)
7690
85.2k
{
7691
85.2k
    njs_int_t              ret;
7692
85.2k
    njs_bool_t             async;
7693
85.2k
    njs_variable_t         *arg, *var;
7694
85.2k
    njs_parser_node_t      *node, *name;
7695
85.2k
    njs_function_lambda_t  *lambda;
7696
7697
85.2k
    if (token->type == NJS_TOKEN_ASYNC) {
7698
1.66k
        njs_lexer_consume_token(parser->lexer, 1);
7699
7700
1.66k
        token = njs_lexer_token(parser->lexer, 0);
7701
1.66k
        if (token == NULL) {
7702
0
            return NJS_ERROR;
7703
0
        }
7704
7705
1.66k
        async = 1;
7706
1.66k
        node = njs_parser_node_new(parser, NJS_TOKEN_ASYNC_FUNCTION_EXPRESSION);
7707
7708
83.6k
    } else {
7709
83.6k
        async = 0;
7710
83.6k
        node = njs_parser_node_new(parser, NJS_TOKEN_FUNCTION_EXPRESSION);
7711
83.6k
    }
7712
7713
85.2k
    if (node == NULL) {
7714
0
        return NJS_ERROR;
7715
0
    }
7716
7717
85.2k
    node->token_line = token->line;
7718
85.2k
    parser->node = node;
7719
7720
85.2k
    ret = njs_parser_scope_begin(parser, NJS_SCOPE_FUNCTION, 0);
7721
85.2k
    if (ret != NJS_OK) {
7722
0
        return NJS_ERROR;
7723
0
    }
7724
7725
85.2k
    parser->scope->async = async;
7726
7727
85.2k
    name = njs_parser_node_new(parser, NJS_TOKEN_NAME);
7728
85.2k
    if (name == NULL) {
7729
0
        return NJS_ERROR;
7730
0
    }
7731
7732
85.2k
    node->left = name;
7733
7734
85.2k
    var = njs_variable_scope_add(parser, parser->scope, parser->scope,
7735
85.2k
                                 NJS_ATOM_STRING_empty, NJS_VARIABLE_FUNCTION,
7736
85.2k
                                 1);
7737
85.2k
    if (var == NULL) {
7738
0
        return NJS_ERROR;
7739
0
    }
7740
7741
85.2k
    ret = njs_parser_variable_reference(parser, parser->scope, node->left,
7742
85.2k
                                        NJS_ATOM_STRING_empty, NJS_DECLARATION);
7743
85.2k
    if (ret != NJS_OK) {
7744
0
        return NJS_ERROR;
7745
0
    }
7746
7747
85.2k
    node->left->u.reference.variable = var;
7748
7749
85.2k
    lambda = njs_function_lambda_alloc(parser->vm, 0);
7750
85.2k
    if (lambda == NULL) {
7751
0
        return NJS_ERROR;
7752
0
    }
7753
7754
85.2k
    node->u.value.data.u.lambda = lambda;
7755
7756
85.2k
    parser->scope->arrow_function = 1;
7757
7758
85.2k
    if (token->type == NJS_TOKEN_OPEN_PARENTHESIS) {
7759
2.65k
        njs_lexer_consume_token(parser->lexer, 1);
7760
7761
2.65k
        parser->node = NULL;
7762
2.65k
        parser->target = node;
7763
7764
2.65k
        njs_parser_next(parser, njs_parser_formal_parameters);
7765
7766
2.65k
        return njs_parser_after(parser, current, node, 1,
7767
2.65k
                                njs_parser_arrow_function_args_after);
7768
7769
82.6k
    } else if (njs_lexer_token_is_binding_identifier(token)) {
7770
82.6k
        arg = njs_variable_add(parser, parser->scope, token->atom_id,
7771
82.6k
                               NJS_VARIABLE_VAR);
7772
82.6k
        if (arg == NULL) {
7773
0
            return NJS_ERROR;
7774
0
        }
7775
7776
82.6k
        arg->argument = 1;
7777
7778
82.6k
        var->index = njs_scope_index(parser->scope->type, parser->scope->items,
7779
82.6k
                                     NJS_LEVEL_LOCAL, NJS_VARIABLE_VAR);
7780
82.6k
        parser->scope->items++;
7781
7782
82.6k
        lambda->self = var->index;
7783
82.6k
        lambda->nargs++;
7784
7785
82.6k
        njs_lexer_consume_token(parser->lexer, 1);
7786
7787
82.6k
        parser->target = node;
7788
7789
82.6k
        njs_parser_next(parser, njs_parser_arrow_function_arrow);
7790
7791
82.6k
    } else {
7792
0
        return njs_parser_failed(parser);
7793
0
    }
7794
7795
82.6k
    return NJS_OK;
7796
85.2k
}
7797
7798
7799
static njs_int_t
7800
njs_parser_arrow_function_args_after(njs_parser_t *parser,
7801
    njs_lexer_token_t *token, njs_queue_link_t *current)
7802
2.65k
{
7803
2.65k
    njs_variable_t  *var, **vv;
7804
7805
2.65k
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
7806
295
        return njs_parser_failed(parser);
7807
295
    }
7808
7809
2.36k
    njs_lexer_consume_token(parser->lexer, 1);
7810
7811
2.36k
    vv = &parser->target->left->u.reference.variable;
7812
7813
2.36k
    var = *vv;
7814
2.36k
    *vv = NULL;
7815
7816
2.36k
    var->index = njs_scope_index(var->scope->type, var->scope->items,
7817
2.36k
                                 NJS_LEVEL_LOCAL, NJS_VARIABLE_VAR);
7818
2.36k
    var->scope->items++;
7819
7820
2.36k
    parser->target->u.value.data.u.lambda->self = var->index;
7821
7822
2.36k
    njs_parser_next(parser, njs_parser_arrow_function_arrow);
7823
7824
2.36k
    return NJS_OK;
7825
2.65k
}
7826
7827
7828
static njs_int_t
7829
njs_parser_arrow_function_arrow(njs_parser_t *parser,
7830
    njs_lexer_token_t *token, njs_queue_link_t *current)
7831
84.9k
{
7832
84.9k
    if (token->type != NJS_TOKEN_ARROW) {
7833
0
        return njs_parser_failed(parser);
7834
0
    }
7835
7836
84.9k
    njs_lexer_consume_token(parser->lexer, 1);
7837
7838
84.9k
    token = njs_lexer_token(parser->lexer, 0);
7839
84.9k
    if (token == NULL) {
7840
0
        return NJS_ERROR;
7841
0
    }
7842
7843
84.9k
    if (token->type == NJS_TOKEN_OPEN_BRACE) {
7844
2.59k
        njs_lexer_consume_token(parser->lexer, 1);
7845
7846
2.59k
        token = njs_lexer_token(parser->lexer, 0);
7847
2.59k
        if (token == NULL) {
7848
0
            return NJS_ERROR;
7849
0
        }
7850
7851
2.59k
        if (token->type == NJS_TOKEN_CLOSE_BRACE) {
7852
152
            parser->node = NULL;
7853
7854
152
            njs_parser_next(parser, njs_parser_function_lambda_body_after);
7855
7856
152
            return NJS_OK;
7857
152
        }
7858
7859
2.44k
        parser->node = NULL;
7860
7861
2.44k
        njs_parser_next(parser, njs_parser_statement_list);
7862
7863
2.44k
        return njs_parser_after(parser, current, parser->target, 1,
7864
2.59k
                                njs_parser_function_lambda_body_after);
7865
2.59k
    }
7866
7867
82.3k
    parser->node = NULL;
7868
7869
82.3k
    njs_parser_next(parser, njs_parser_assignment_expression);
7870
7871
82.3k
    return njs_parser_after(parser, current, parser->target, 1,
7872
84.9k
                            njs_parser_arrow_function_body_after);
7873
84.9k
}
7874
7875
7876
static njs_int_t
7877
njs_parser_arrow_function_body_after(njs_parser_t *parser,
7878
    njs_lexer_token_t *token, njs_queue_link_t *current)
7879
79.0k
{
7880
79.0k
    njs_parser_node_t  *body;
7881
7882
79.0k
    body = njs_parser_return_set(parser, parser->node);
7883
79.0k
    if (body == NULL) {
7884
0
        return NJS_ERROR;
7885
0
    }
7886
7887
79.0k
    parser->target->right = body;
7888
79.0k
    parser->node = parser->target;
7889
7890
79.0k
    njs_parser_scope_end(parser);
7891
7892
79.0k
    return njs_parser_stack_pop(parser);
7893
79.0k
}
7894
7895
7896
/*
7897
 * 14.3 Method Definitions.
7898
 */
7899
static njs_int_t
7900
njs_parser_method_definition(njs_parser_t *parser, njs_lexer_token_t *token,
7901
    njs_queue_link_t *current)
7902
3.26k
{
7903
3.26k
    njs_token_type_t   type;
7904
3.26k
    njs_lexer_token_t  *next;
7905
3.26k
    njs_parser_node_t  *expr;
7906
7907
3.26k
    type = NJS_TOKEN_FUNCTION;
7908
7909
3.26k
    if (token->type == NJS_TOKEN_ASYNC) {
7910
362
        njs_lexer_consume_token(parser->lexer, 1);
7911
7912
362
        token = njs_lexer_token(parser->lexer, 0);
7913
362
        if (token == NULL) {
7914
0
            return NJS_ERROR;
7915
0
        }
7916
7917
362
        type = NJS_TOKEN_ASYNC_FUNCTION;
7918
362
    }
7919
7920
3.26k
    switch (token->type) {
7921
    /* PropertyName */
7922
129
    case NJS_TOKEN_STRING:
7923
261
    case NJS_TOKEN_ESCAPE_STRING:
7924
1.95k
    case NJS_TOKEN_NUMBER:
7925
1.95k
        break;
7926
7927
1.31k
    default:
7928
1.31k
        if (njs_lexer_token_is_identifier_name(token)) {
7929
1.31k
            break;
7930
1.31k
        }
7931
7932
0
        return njs_parser_failed(parser);
7933
3.26k
    }
7934
7935
3.26k
    njs_lexer_consume_token(parser->lexer, 1);
7936
7937
3.26k
    next = njs_lexer_token(parser->lexer, 0);
7938
3.26k
    if (next == NULL) {
7939
0
        return NJS_ERROR;
7940
0
    }
7941
7942
3.26k
    if (next->type != NJS_TOKEN_OPEN_PARENTHESIS) {
7943
0
        return njs_parser_failed(parser);
7944
0
    }
7945
7946
3.26k
    expr = njs_parser_node_new(parser, type);
7947
3.26k
    if (expr == NULL) {
7948
0
        return NJS_ERROR;
7949
0
    }
7950
7951
3.26k
    expr->token_line = next->line;
7952
7953
3.26k
    parser->node = expr;
7954
7955
3.26k
    njs_lexer_consume_token(parser->lexer, 1);
7956
3.26k
    njs_parser_next(parser, njs_parser_function_lambda);
7957
7958
3.26k
    return NJS_OK;
7959
3.26k
}
7960
7961
7962
static njs_int_t
7963
njs_parser_get_set(njs_parser_t *parser, njs_lexer_token_t *token,
7964
    njs_queue_link_t *current)
7965
2.81k
{
7966
2.81k
    njs_token_type_t   accessor;
7967
2.81k
    njs_lexer_token_t  *name;
7968
2.81k
    njs_parser_node_t  *property, *expression, *temp;
7969
7970
2.81k
    temp = parser->target;
7971
2.81k
    accessor = (njs_token_type_t) (uintptr_t) temp->right;
7972
7973
2.81k
    name = token;
7974
7975
2.81k
    token = njs_lexer_peek_token(parser->lexer, token, 0);
7976
2.81k
    if (token == NULL) {
7977
0
        return NJS_ERROR;
7978
0
    }
7979
7980
2.81k
    switch (token->type) {
7981
    /* IdentifierReference */
7982
787
    case NJS_TOKEN_NAME:
7983
1.00k
    case NJS_TOKEN_STRING:
7984
1.63k
    case NJS_TOKEN_ESCAPE_STRING:
7985
1.67k
    case NJS_TOKEN_NUMBER:
7986
1.67k
        break;
7987
7988
515
    case NJS_TOKEN_OPEN_BRACKET:
7989
515
        njs_lexer_consume_token(parser->lexer, 2);
7990
7991
515
        njs_parser_next(parser, njs_parser_assignment_expression);
7992
7993
515
        return njs_parser_after(parser, current, temp, 1,
7994
1.63k
                                njs_parser_get_set_after);
7995
7996
626
    default:
7997
626
        if (njs_lexer_token_is_identifier_name(token)) {
7998
586
            break;
7999
586
        }
8000
8001
40
        return njs_parser_property_definition_ident(parser, name, temp);
8002
2.81k
    }
8003
8004
2.25k
    property = njs_parser_property_name_node(parser, token);
8005
2.25k
    if (property == NULL) {
8006
599
        return NJS_ERROR;
8007
599
    }
8008
8009
1.65k
    njs_lexer_consume_token(parser->lexer, 2);
8010
8011
1.65k
    token = njs_lexer_token(parser->lexer, 0);
8012
1.65k
    if (token == NULL) {
8013
0
        return NJS_ERROR;
8014
0
    }
8015
8016
1.65k
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS) {
8017
249
        return njs_parser_failed(parser);
8018
249
    }
8019
8020
1.41k
    expression = njs_parser_node_new(parser, NJS_TOKEN_FUNCTION);
8021
1.41k
    if (expression == NULL) {
8022
0
        return NJS_ERROR;
8023
0
    }
8024
8025
1.41k
    expression->token_line = token->line;
8026
8027
1.41k
    temp->right = property;
8028
8029
1.41k
    parser->node = expression;
8030
8031
1.41k
    njs_lexer_consume_token(parser->lexer, 1);
8032
1.41k
    njs_parser_next(parser, njs_parser_function_lambda);
8033
8034
1.41k
    if (accessor == NJS_TOKEN_PROPERTY_GETTER) {
8035
623
        return njs_parser_after(parser, current, temp, 1, njs_parser_get_after);
8036
623
    }
8037
8038
787
    return njs_parser_after(parser, current, temp, 1, njs_parser_set_after);
8039
1.41k
}
8040
8041
8042
static njs_int_t
8043
njs_parser_get_set_after(njs_parser_t *parser, njs_lexer_token_t *token,
8044
    njs_queue_link_t *current)
8045
515
{
8046
515
    njs_token_type_t   accessor;
8047
515
    njs_parser_node_t  *expression, *temp;
8048
8049
515
    if (token->type != NJS_TOKEN_CLOSE_BRACKET) {
8050
0
        return njs_parser_failed(parser);
8051
0
    }
8052
8053
515
    njs_lexer_consume_token(parser->lexer, 1);
8054
8055
515
    token = njs_lexer_token(parser->lexer, 0);
8056
515
    if (token == NULL) {
8057
0
        return NJS_ERROR;
8058
0
    }
8059
8060
515
    if (token->type != NJS_TOKEN_OPEN_PARENTHESIS) {
8061
206
        return njs_parser_failed(parser);
8062
206
    }
8063
8064
309
    expression = njs_parser_node_new(parser, NJS_TOKEN_FUNCTION);
8065
309
    if (expression == NULL) {
8066
0
        return NJS_ERROR;
8067
0
    }
8068
8069
309
    expression->token_line = token->line;
8070
8071
309
    temp = parser->target;
8072
8073
309
    accessor = (njs_token_type_t) (uintptr_t) temp->right;
8074
309
    temp->right = parser->node;
8075
8076
309
    parser->node = expression;
8077
8078
309
    njs_lexer_consume_token(parser->lexer, 1);
8079
309
    njs_parser_next(parser, njs_parser_function_lambda);
8080
8081
309
    if (accessor == NJS_TOKEN_PROPERTY_GETTER) {
8082
151
        return njs_parser_after(parser, current, temp, 1, njs_parser_get_after);
8083
151
    }
8084
8085
158
    return njs_parser_after(parser, current, temp, 1, njs_parser_set_after);
8086
309
}
8087
8088
8089
static njs_int_t
8090
njs_parser_get_after(njs_parser_t *parser, njs_lexer_token_t *token,
8091
    njs_queue_link_t *current)
8092
170
{
8093
170
    njs_int_t              ret;
8094
170
    njs_parser_node_t      *expression, *temp;
8095
170
    njs_function_lambda_t  *lambda;
8096
8097
170
    temp = parser->target;
8098
8099
170
    expression = parser->node;
8100
170
    lambda = expression->u.value.data.u.lambda;
8101
8102
170
    if (lambda->nargs != 0) {
8103
133
        njs_parser_syntax_error(parser,
8104
133
                                "Getter must not have any formal parameters");
8105
133
        return NJS_DONE;
8106
133
    }
8107
8108
37
    ret = njs_parser_property_accessor(parser, temp->left, temp->right,
8109
37
                                       expression, NJS_TOKEN_PROPERTY_GETTER);
8110
37
    if (ret != NJS_OK) {
8111
0
        return NJS_ERROR;
8112
0
    }
8113
8114
37
    parser->node = temp->left;
8115
37
    parser->target = NULL;
8116
8117
37
    return njs_parser_stack_pop(parser);
8118
37
}
8119
8120
8121
static njs_int_t
8122
njs_parser_set_after(njs_parser_t *parser, njs_lexer_token_t *token,
8123
    njs_queue_link_t *current)
8124
260
{
8125
260
    njs_int_t              ret;
8126
260
    njs_parser_node_t      *expression, *temp;
8127
260
    njs_function_lambda_t  *lambda;
8128
8129
260
    temp = parser->target;
8130
8131
260
    expression = parser->node;
8132
260
    lambda = expression->u.value.data.u.lambda;
8133
8134
260
    if (lambda->nargs != 1) {
8135
0
        njs_parser_syntax_error(parser,
8136
0
                               "Setter must have exactly one formal parameter");
8137
0
        return NJS_DONE;
8138
0
    }
8139
8140
260
    ret = njs_parser_property_accessor(parser, temp->left, temp->right,
8141
260
                                       expression, NJS_TOKEN_PROPERTY_SETTER);
8142
260
    if (ret != NJS_OK) {
8143
0
        return NJS_ERROR;
8144
0
    }
8145
8146
260
    parser->node = temp->left;
8147
260
    parser->target = NULL;
8148
8149
260
    return njs_parser_stack_pop(parser);
8150
260
}
8151
8152
8153
static njs_int_t
8154
njs_parser_function_lambda(njs_parser_t *parser,
8155
    njs_lexer_token_t *token, njs_queue_link_t *current)
8156
5.24k
{
8157
5.24k
    njs_int_t              ret;
8158
5.24k
    njs_parser_node_t      *expr;
8159
5.24k
    njs_function_lambda_t  *lambda;
8160
8161
5.24k
    lambda = njs_function_lambda_alloc(parser->vm, 0);
8162
5.24k
    if (lambda == NULL) {
8163
0
        return NJS_ERROR;
8164
0
    }
8165
8166
5.24k
    expr = parser->node;
8167
5.24k
    expr->u.value.data.u.lambda = lambda;
8168
8169
5.24k
    ret = njs_parser_scope_begin(parser, NJS_SCOPE_FUNCTION, 1);
8170
5.24k
    if (ret != NJS_OK) {
8171
0
        return NJS_ERROR;
8172
0
    }
8173
8174
5.24k
    parser->scope->async =
8175
5.24k
                        (parser->node->token_type == NJS_TOKEN_ASYNC_FUNCTION);
8176
8177
5.24k
    parser->node = NULL;
8178
5.24k
    parser->target = expr;
8179
8180
5.24k
    njs_parser_next(parser, njs_parser_unique_formal_parameters);
8181
8182
5.24k
    return njs_parser_after(parser, current, expr, 1,
8183
5.24k
                            njs_parser_function_lambda_args_after);
8184
5.24k
}
8185
8186
8187
static njs_int_t
8188
njs_parser_function_lambda_args_after(njs_parser_t *parser,
8189
    njs_lexer_token_t *token, njs_queue_link_t *current)
8190
162k
{
8191
162k
    if (token->type != NJS_TOKEN_CLOSE_PARENTHESIS) {
8192
2.87k
        return njs_parser_failed(parser);
8193
2.87k
    }
8194
8195
159k
    njs_lexer_consume_token(parser->lexer, 1);
8196
8197
159k
    token = njs_lexer_token(parser->lexer, 0);
8198
159k
    if (token == NULL) {
8199
0
        return NJS_ERROR;
8200
0
    }
8201
8202
159k
    if (token->type != NJS_TOKEN_OPEN_BRACE) {
8203
137
        return njs_parser_failed(parser);
8204
137
    }
8205
8206
159k
    njs_lexer_consume_token(parser->lexer, 1);
8207
8208
159k
    token = njs_lexer_token(parser->lexer, 0);
8209
159k
    if (token == NULL) {
8210
0
        return NJS_ERROR;
8211
0
    }
8212
8213
159k
    if (token->type == NJS_TOKEN_CLOSE_BRACE) {
8214
1.17k
        parser->node = NULL;
8215
8216
1.17k
        njs_parser_next(parser, njs_parser_function_lambda_body_after);
8217
8218
1.17k
        return NJS_OK;
8219
1.17k
    }
8220
8221
157k
    parser->node = NULL;
8222
8223
157k
    njs_parser_next(parser, njs_parser_statement_list);
8224
8225
157k
    return njs_parser_after(parser, current, parser->target, 1,
8226
159k
                            njs_parser_function_lambda_body_after);
8227
159k
}
8228
8229
8230
static njs_int_t
8231
njs_parser_function_lambda_body_after(njs_parser_t *parser,
8232
    njs_lexer_token_t *token, njs_queue_link_t *current)
8233
104k
{
8234
104k
    njs_parser_node_t *body, *last, *parent;
8235
8236
104k
    if (token->type != NJS_TOKEN_CLOSE_BRACE) {
8237
7.15k
        return njs_parser_failed(parser);
8238
7.15k
    }
8239
8240
97.0k
    parent = parser->target;
8241
8242
97.0k
    last = NULL;
8243
97.0k
    body = njs_parser_chain_top(parser);
8244
8245
97.0k
    if (body != NULL) {
8246
        /* Take the last function body statement. */
8247
95.7k
        last = body->right;
8248
8249
95.7k
        if (last == NULL) {
8250
            /*
8251
             * The last statement is terminated by semicolon.
8252
             * Take the last statement itself.
8253
             */
8254
0
            last = body->left;
8255
0
        }
8256
95.7k
    }
8257
8258
97.0k
    if (last == NULL || last->token_type != NJS_TOKEN_RETURN) {
8259
        /*
8260
         * There is no function body or the last function body
8261
         * body statement is not "return" statement.
8262
         */
8263
96.1k
        body = njs_parser_return_set(parser, NULL);
8264
96.1k
        if (body == NULL) {
8265
0
            return NJS_ERROR;
8266
0
        }
8267
8268
96.1k
        body->right->token_line = token->line;
8269
96.1k
    }
8270
8271
97.0k
    parent->right = body;
8272
97.0k
    parser->node = parent;
8273
8274
97.0k
    njs_parser_scope_end(parser);
8275
8276
97.0k
    njs_lexer_consume_token(parser->lexer, 1);
8277
8278
97.0k
    return njs_parser_stack_pop(parser);
8279
97.0k
}
8280
8281
8282
static njs_int_t
8283
njs_parser_export(njs_parser_t *parser, njs_lexer_token_t *token,
8284
    njs_queue_link_t *current)
8285
226
{
8286
226
    njs_parser_node_t  *node;
8287
226
    njs_lexer_token_t  *peek;
8288
8289
226
    static const njs_str_t  as_string = njs_str("as");
8290
8291
226
    if (!parser->module) {
8292
226
        njs_parser_syntax_error(parser, "Illegal export statement");
8293
226
        return NJS_DONE;
8294
226
    }
8295
8296
0
    if (token->type != NJS_TOKEN_DEFAULT) {
8297
8298
0
        if (token->type != NJS_TOKEN_OPEN_BRACE) {
8299
0
            njs_parser_syntax_error(parser,
8300
0
                                    "Non-default export is not supported");
8301
0
            return NJS_DONE;
8302
0
        }
8303
8304
        /*
8305
         * 'export {'
8306
         *    supported only: export {identifier as default};
8307
         */
8308
8309
0
        njs_lexer_consume_token(parser->lexer, 1);
8310
8311
0
        token = njs_lexer_token(parser->lexer, 0);
8312
0
        if (njs_slow_path(token == NULL)) {
8313
0
            return NJS_ERROR;
8314
0
        }
8315
8316
0
        if (token->type != NJS_TOKEN_NAME) {
8317
0
            njs_parser_syntax_error(parser, "Identifier expected");
8318
0
            return NJS_DONE;
8319
0
        }
8320
8321
0
        peek = njs_lexer_peek_token(parser->lexer, token, 0);
8322
0
        if (njs_slow_path(peek == NULL)) {
8323
0
            return NJS_ERROR;
8324
0
        }
8325
8326
0
        if (peek->type != NJS_TOKEN_NAME ||
8327
0
            !njs_strstr_eq(&peek->text, &as_string))
8328
0
        {
8329
0
            njs_parser_syntax_error(parser, "'as' expected");
8330
0
            return NJS_DONE;
8331
0
        }
8332
8333
0
        peek = njs_lexer_peek_token(parser->lexer, peek, 0);
8334
0
        if (njs_slow_path(peek == NULL)) {
8335
0
            return NJS_ERROR;
8336
0
        }
8337
8338
0
        if (peek->type != NJS_TOKEN_DEFAULT) {
8339
0
            njs_parser_syntax_error(parser,
8340
0
                                    "Non-default export is not supported");
8341
0
            return NJS_DONE;
8342
0
        }
8343
8344
0
        peek = njs_lexer_peek_token(parser->lexer, peek, 0);
8345
0
        if (njs_slow_path(peek == NULL)) {
8346
0
            return NJS_ERROR;
8347
0
        }
8348
8349
0
        if (peek->type != NJS_TOKEN_CLOSE_BRACE) {
8350
0
            njs_parser_syntax_error(parser, "Close brace is expected");
8351
0
            return NJS_DONE;
8352
0
        }
8353
8354
0
        node = njs_parser_node_new(parser, NJS_TOKEN_EXPORT);
8355
0
        if (node == NULL) {
8356
0
            return NJS_ERROR;
8357
0
        }
8358
8359
0
        node->token_line = parser->line;
8360
0
        node->right = njs_parser_reference(parser, token);
8361
0
        if (node->right == NULL) {
8362
0
            return NJS_ERROR;
8363
0
        }
8364
8365
0
        parser->node = node;
8366
8367
0
        njs_lexer_consume_token(parser->lexer, 4);
8368
8369
0
        return njs_parser_stack_pop(parser);
8370
0
    }
8371
8372
0
    njs_lexer_consume_token(parser->lexer, 1);
8373
8374
0
    node = njs_parser_node_new(parser, NJS_TOKEN_EXPORT);
8375
0
    if (node == NULL) {
8376
0
        return NJS_ERROR;
8377
0
    }
8378
8379
0
    node->token_line = parser->line;
8380
0
    parser->node = node;
8381
8382
0
    njs_parser_next(parser, njs_parser_assignment_expression);
8383
8384
0
    return njs_parser_after(parser, current, node, 1, njs_parser_export_after);
8385
0
}
8386
8387
8388
static njs_int_t
8389
njs_parser_export_after(njs_parser_t *parser, njs_lexer_token_t *token,
8390
    njs_queue_link_t *current)
8391
0
{
8392
0
    if (njs_parser_expect_semicolon(parser, token) != NJS_OK) {
8393
0
        return njs_parser_failed(parser);
8394
0
    }
8395
8396
0
    return njs_parser_right_link_pop(parser);
8397
0
}
8398
8399
8400
static njs_mod_t *
8401
njs_parser_module(njs_parser_t *parser, njs_str_t *name)
8402
0
{
8403
0
    njs_vm_t   *vm;
8404
0
    njs_mod_t  *module;
8405
8406
0
    vm = parser->vm;
8407
8408
0
    if (name->length == 0) {
8409
0
        njs_parser_ref_error(parser, "Cannot load module \"%V\"", name);
8410
0
        return NULL;
8411
0
    }
8412
8413
0
    module = njs_module_find(vm, name, 1);
8414
0
    if (module != NULL) {
8415
0
        goto done;
8416
0
    }
8417
8418
0
    if (vm->module_loader == NULL) {
8419
0
        njs_parser_ref_error(parser, "Module loader callback is not provided");
8420
0
        return NULL;
8421
0
    }
8422
8423
0
    module = vm->module_loader(vm, vm->module_loader_opaque, name);
8424
0
    if (module == NULL) {
8425
0
        if (!njs_is_valid(&vm->exception)) {
8426
0
            njs_parser_ref_error(parser, "Cannot load module \"%V\"", name);
8427
0
        }
8428
8429
0
        return NULL;
8430
0
    }
8431
8432
0
done:
8433
8434
0
    return module;
8435
0
}
8436
8437
8438
static njs_int_t
8439
njs_parser_import(njs_parser_t *parser, njs_lexer_token_t *token,
8440
    njs_queue_link_t *current)
8441
831
{
8442
831
    njs_variable_t     *var;
8443
831
    njs_parser_node_t  *name, *import;
8444
8445
831
    if (parser->scope->parent != NULL) {
8446
169
        njs_parser_syntax_error(parser, "Illegal import statement");
8447
169
        return NJS_DONE;
8448
169
    }
8449
8450
662
    if (token->type == NJS_TOKEN_MULTIPLICATION
8451
662
        || token->type == NJS_TOKEN_OPEN_BRACE
8452
476
        || token->type == NJS_TOKEN_STRING)
8453
272
    {
8454
272
        njs_parser_syntax_error(parser, "Non-default import is not supported");
8455
272
        return NJS_DONE;
8456
272
    }
8457
8458
390
    if (token->type != NJS_TOKEN_NAME) {
8459
376
        return njs_parser_failed(parser);
8460
376
     }
8461
8462
14
    name = njs_parser_variable_node(parser, token->atom_id, NJS_VARIABLE_LET,
8463
14
                                    &var);
8464
14
    if (name == NULL) {
8465
0
        return NJS_ERROR;
8466
0
    }
8467
8468
14
    var->init = 1;
8469
8470
14
    name->token_line = token->line;
8471
8472
14
    njs_lexer_consume_token(parser->lexer, 1);
8473
8474
14
    token = njs_lexer_token(parser->lexer, 0);
8475
14
    if (token == NULL) {
8476
0
        return NJS_ERROR;
8477
0
    }
8478
8479
14
    if (token->type != NJS_TOKEN_FROM) {
8480
14
        return njs_parser_failed(parser);
8481
14
    }
8482
8483
0
    njs_lexer_consume_token(parser->lexer, 1);
8484
8485
0
    token = njs_lexer_token(parser->lexer, 0);
8486
0
    if (token == NULL) {
8487
0
        return NJS_ERROR;
8488
0
    }
8489
8490
0
    if (token->type != NJS_TOKEN_STRING) {
8491
0
        return njs_parser_failed(parser);
8492
0
    }
8493
8494
0
    import = njs_parser_node_new(parser, NJS_TOKEN_IMPORT);
8495
0
    if (import == NULL) {
8496
0
        return NJS_ERROR;
8497
0
    }
8498
8499
0
    import->hoist = 1;
8500
0
    import->token_line = parser->line;
8501
0
    import->left = name;
8502
8503
0
    import->u.module = njs_parser_module(parser, &token->text);
8504
0
    if (njs_slow_path(import->u.module == NULL)) {
8505
0
        return NJS_ERROR;
8506
0
    }
8507
8508
0
    njs_lexer_consume_token(parser->lexer, 1);
8509
8510
0
    token = njs_lexer_token(parser->lexer, 0);
8511
0
    if (token == NULL) {
8512
0
        return NJS_ERROR;
8513
0
    }
8514
8515
0
    if (njs_parser_expect_semicolon(parser, token) != NJS_OK) {
8516
0
        return njs_parser_failed(parser);
8517
0
    }
8518
8519
0
    parser->node = import;
8520
8521
0
    return njs_parser_stack_pop(parser);
8522
0
}
8523
8524
8525
static njs_int_t
8526
njs_parser_export_sink(njs_parser_t *parser)
8527
0
{
8528
0
    njs_uint_t         n;
8529
0
    njs_parser_node_t  *node, *prev;
8530
8531
0
    n = 0;
8532
8533
0
    for (node = njs_parser_chain_top(parser);
8534
0
         node != NULL;
8535
0
         node = node->left)
8536
0
    {
8537
0
        if (node->right != NULL
8538
0
            && node->right->token_type == NJS_TOKEN_EXPORT)
8539
0
        {
8540
0
            n++;
8541
0
        }
8542
0
    }
8543
8544
0
    if (n != 1) {
8545
0
        njs_parser_syntax_error(parser,
8546
0
             (n == 0) ? "export statement is required"
8547
0
                      : "Identifier \"default\" has already been declared");
8548
0
        return NJS_ERROR;
8549
0
    }
8550
8551
0
    node = njs_parser_chain_top(parser);
8552
8553
0
    if (node->right && node->right->token_type == NJS_TOKEN_EXPORT) {
8554
0
        return NJS_OK;
8555
0
    }
8556
8557
0
    prev = njs_parser_chain_top(parser);
8558
8559
0
    while (prev->left != NULL) {
8560
0
        node = prev->left;
8561
8562
0
        if (node->right != NULL
8563
0
            && node->right->token_type == NJS_TOKEN_EXPORT)
8564
0
        {
8565
0
            prev->left = node->left;
8566
0
            break;
8567
0
        }
8568
8569
0
        prev = prev->left;
8570
0
    }
8571
8572
0
    node->left = njs_parser_chain_top(parser);
8573
0
    njs_parser_chain_top_set(parser, node);
8574
8575
0
    return NJS_OK;
8576
0
}
8577
8578
8579
static njs_parser_node_t *
8580
njs_parser_return_set(njs_parser_t *parser, njs_parser_node_t *expr)
8581
175k
{
8582
175k
    njs_parser_node_t  *stmt, *node;
8583
8584
175k
    node = njs_parser_node_new(parser, NJS_TOKEN_RETURN);
8585
175k
    if (njs_slow_path(node == NULL)) {
8586
0
        return NULL;
8587
0
    }
8588
8589
175k
    if (expr != NULL) {
8590
79.0k
        node->token_line = expr->token_line;
8591
79.0k
    }
8592
8593
175k
    node->right = expr;
8594
8595
175k
    stmt = njs_parser_node_new(parser, NJS_TOKEN_STATEMENT);
8596
175k
    if (njs_slow_path(stmt == NULL)) {
8597
0
        return NULL;
8598
0
    }
8599
8600
175k
    stmt->left = njs_parser_chain_top(parser);
8601
175k
    stmt->right = node;
8602
8603
175k
    njs_parser_chain_top_set(parser, stmt);
8604
8605
175k
    return stmt;
8606
175k
}
8607
8608
8609
static njs_parser_node_t *
8610
njs_parser_variable_node(njs_parser_t *parser, uintptr_t atom_id,
8611
    njs_variable_type_t type, njs_variable_t  **retvar)
8612
55.7k
{
8613
55.7k
    njs_int_t          ret;
8614
55.7k
    njs_variable_t     *var;
8615
55.7k
    njs_parser_node_t  *node;
8616
8617
55.7k
    var = njs_variable_add(parser, parser->scope, atom_id, type);
8618
55.7k
    if (njs_slow_path(var == NULL)) {
8619
282
        return NULL;
8620
282
    }
8621
8622
55.4k
    if (retvar != NULL) {
8623
32.5k
        *retvar = var;
8624
32.5k
    }
8625
8626
55.4k
    node = njs_parser_node_new(parser, NJS_TOKEN_NAME);
8627
55.4k
    if (njs_slow_path(node == NULL)) {
8628
0
        return NULL;
8629
0
    }
8630
8631
55.4k
    ret = njs_parser_variable_reference(parser, parser->scope, node, atom_id,
8632
55.4k
                                        NJS_DECLARATION);
8633
55.4k
    if (njs_slow_path(ret != NJS_OK)) {
8634
0
        return NULL;
8635
0
    }
8636
8637
55.4k
    return node;
8638
55.4k
}
8639
8640
8641
static njs_parser_node_t *
8642
njs_parser_reference(njs_parser_t *parser, njs_lexer_token_t *token)
8643
1.33M
{
8644
1.33M
    njs_int_t           ret;
8645
1.33M
    njs_index_t         index;
8646
1.33M
    njs_variable_t      *var;
8647
1.33M
    njs_parser_node_t   *node;
8648
1.33M
    njs_parser_scope_t  *scope;
8649
8650
1.33M
    node = njs_parser_node_new(parser, token->type);
8651
1.33M
    if (njs_slow_path(node == NULL)) {
8652
0
        return NULL;
8653
0
    }
8654
8655
1.33M
    switch (token->type) {
8656
8657
361
    case NJS_TOKEN_NULL:
8658
361
        njs_thread_log_debug("JS: null");
8659
361
        break;
8660
8661
4.46k
    case NJS_TOKEN_THIS:
8662
4.46k
        njs_thread_log_debug("JS: this");
8663
8664
4.46k
        scope = njs_function_scope(parser->scope);
8665
4.46k
        if (njs_slow_path(scope == NULL)) {
8666
0
            njs_parser_syntax_error(parser,
8667
0
                                    "function or global scope not found");
8668
0
            return NULL;
8669
0
        }
8670
8671
4.46k
        if (parser->vm->options.module) {
8672
0
            token->atom_id = NJS_ATOM_STRING_undefined;
8673
8674
4.46k
        } else if (!scope->arrow_function) {
8675
4.09k
            index = njs_scope_index(scope->type, 0, NJS_LEVEL_LOCAL,
8676
4.09k
                                    NJS_VARIABLE_VAR);
8677
8678
4.09k
            var = njs_variable_scope_add(parser, scope, scope, token->atom_id,
8679
4.09k
                                         NJS_VARIABLE_VAR, index);
8680
4.09k
            if (njs_slow_path(var == NULL)) {
8681
0
                return NULL;
8682
0
            }
8683
4.09k
        }
8684
8685
4.46k
        node->token_type = NJS_TOKEN_THIS;
8686
4.46k
        node->token_line = token->line;
8687
8688
4.46k
        ret = njs_parser_variable_reference(parser, parser->scope, node,
8689
4.46k
                                            token->atom_id, NJS_REFERENCE);
8690
4.46k
        if (njs_slow_path(ret != NJS_OK)) {
8691
0
            return NULL;
8692
0
        }
8693
8694
4.46k
        break;
8695
8696
4.46k
    case NJS_TOKEN_ARGUMENTS:
8697
2.60k
        njs_thread_log_debug("JS: arguments");
8698
8699
2.60k
        scope = njs_function_scope(parser->scope);
8700
8701
4.26k
        while (scope->arrow_function) {
8702
1.66k
            scope = njs_function_scope(scope->parent);
8703
1.66k
        }
8704
8705
2.60k
        if (scope->parent == NULL) {
8706
32
            njs_parser_syntax_error(parser, "\"%V\" object in global scope",
8707
32
                                    &token->text);
8708
32
            return NULL;
8709
32
        }
8710
8711
2.57k
        node->token_line = token->line;
8712
8713
2.57k
        ret = njs_parser_variable_reference(parser, parser->scope, node,
8714
2.57k
                                            token->atom_id, NJS_REFERENCE);
8715
2.57k
        if (njs_slow_path(ret != NJS_OK)) {
8716
0
            return NULL;
8717
0
        }
8718
8719
2.57k
        var = njs_variable_add(parser, scope, token->atom_id,
8720
2.57k
                               NJS_VARIABLE_VAR);
8721
2.57k
        if (njs_slow_path(var == NULL)) {
8722
0
            return NULL;
8723
0
        }
8724
8725
2.57k
        var->arguments_object = 1;
8726
8727
2.57k
        break;
8728
8729
1.33M
    default:
8730
1.33M
        if (token->type == NJS_TOKEN_EVAL
8731
1.33M
            || njs_lexer_token_is_identifier_reference(token))
8732
1.33M
        {
8733
1.33M
            njs_thread_log_debug("JS: %V", name);
8734
8735
1.33M
            if (token->type != NJS_TOKEN_EVAL) {
8736
1.33M
                node->token_type = NJS_TOKEN_NAME;
8737
1.33M
            }
8738
8739
1.33M
            node->token_line = token->line;
8740
8741
1.33M
            ret = njs_parser_variable_reference(parser, parser->scope, node,
8742
1.33M
                                               token->atom_id, NJS_REFERENCE);
8743
1.33M
            if (njs_slow_path(ret != NJS_OK)) {
8744
0
                return NULL;
8745
0
            }
8746
8747
1.33M
            break;
8748
1.33M
        }
8749
8750
0
        (void) njs_parser_unexpected_token(parser->vm, parser, &token->text,
8751
0
                                           token->type);
8752
0
        return NULL;
8753
1.33M
    }
8754
8755
1.33M
    return node;
8756
1.33M
}
8757
8758
8759
static njs_parser_node_t *
8760
njs_parser_object_value(njs_parser_t *parser, njs_parser_node_t *parent,
8761
    uint32_t token_line)
8762
1.66M
{
8763
1.66M
    njs_parser_node_t  *object;
8764
8765
1.66M
    njs_assert(parent->token_type == NJS_TOKEN_OBJECT
8766
1.66M
               || parent->token_type == NJS_TOKEN_ARRAY);
8767
8768
1.66M
    object = njs_parser_node_new(parser, NJS_TOKEN_OBJECT_VALUE);
8769
1.66M
    if (njs_slow_path(object == NULL)) {
8770
0
        return NULL;
8771
0
    }
8772
8773
1.66M
    object->token_line = token_line;
8774
1.66M
    object->u.object = parent;
8775
8776
1.66M
    return object;
8777
1.66M
}
8778
8779
8780
static njs_parser_node_t *
8781
njs_parser_argument(njs_parser_t *parser, njs_parser_node_t *expr,
8782
    njs_index_t index)
8783
678k
{
8784
678k
    njs_parser_node_t  *node;
8785
8786
678k
    node = njs_parser_node_new(parser, NJS_TOKEN_ARGUMENT);
8787
678k
    if (njs_slow_path(node == NULL)) {
8788
0
        return NULL;
8789
0
    }
8790
8791
678k
    node->token_line = expr->token_line;
8792
678k
    node->index = index;
8793
8794
678k
    node->left = expr;
8795
678k
    expr->dest = node;
8796
8797
678k
    return node;
8798
678k
}
8799
8800
8801
static njs_int_t
8802
njs_parser_object_property(njs_parser_t *parser, njs_parser_node_t *parent,
8803
    njs_parser_node_t *property, njs_parser_node_t *value,
8804
    njs_bool_t proto_init)
8805
1.66M
{
8806
1.66M
    njs_token_type_t   type;
8807
1.66M
    njs_parser_node_t  *stmt, *assign, *object, *propref;
8808
8809
1.66M
    object = njs_parser_object_value(parser, parent, value->token_line);
8810
1.66M
    if (njs_slow_path(object == NULL)) {
8811
0
        return NJS_TOKEN_ERROR;
8812
0
    }
8813
8814
1.66M
    type = proto_init ? NJS_TOKEN_PROTO_INIT : NJS_TOKEN_PROPERTY_INIT;
8815
8816
1.66M
    propref = njs_parser_node_new(parser, type);
8817
1.66M
    if (njs_slow_path(propref == NULL)) {
8818
0
        return NJS_ERROR;
8819
0
    }
8820
8821
1.66M
    propref->token_line = value->token_line;
8822
1.66M
    propref->left = object;
8823
1.66M
    propref->right = property;
8824
8825
1.66M
    assign = njs_parser_node_new(parser, NJS_TOKEN_ASSIGNMENT);
8826
1.66M
    if (njs_slow_path(assign == NULL)) {
8827
0
        return NJS_ERROR;
8828
0
    }
8829
8830
1.66M
    assign->token_line = value->token_line;
8831
1.66M
    assign->u.operation = NJS_VMCODE_MOVE;
8832
1.66M
    assign->left = propref;
8833
1.66M
    assign->right = value;
8834
8835
1.66M
    stmt = njs_parser_node_new(parser, NJS_TOKEN_STATEMENT);
8836
1.66M
    if (njs_slow_path(stmt == NULL)) {
8837
0
        return NJS_ERROR;
8838
0
    }
8839
8840
1.66M
    stmt->right = assign;
8841
1.66M
    stmt->left = parent->left;
8842
1.66M
    parent->left = stmt;
8843
8844
1.66M
    return NJS_OK;
8845
1.66M
}
8846
8847
8848
static njs_int_t
8849
njs_parser_property_accessor(njs_parser_t *parser, njs_parser_node_t *parent,
8850
    njs_parser_node_t *property, njs_parser_node_t *value,
8851
    njs_token_type_t accessor)
8852
297
{
8853
297
    njs_parser_node_t  *node, *stmt, *object, *propref;
8854
8855
297
    object = njs_parser_object_value(parser, parent, value->token_line);
8856
297
    if (njs_slow_path(object == NULL)) {
8857
0
        return NJS_TOKEN_ERROR;
8858
0
    }
8859
8860
297
    propref = njs_parser_node_new(parser, 0);
8861
297
    if (njs_slow_path(propref == NULL)) {
8862
0
        return NJS_ERROR;
8863
0
    }
8864
8865
297
    propref->left = object;
8866
297
    propref->right = property;
8867
8868
297
    node = njs_parser_node_new(parser, accessor);
8869
297
    if (njs_slow_path(node == NULL)) {
8870
0
        return NJS_ERROR;
8871
0
    }
8872
8873
297
    node->token_line = value->token_line;
8874
297
    node->left = propref;
8875
297
    node->right = value;
8876
8877
297
    stmt = njs_parser_node_new(parser, NJS_TOKEN_STATEMENT);
8878
297
    if (njs_slow_path(stmt == NULL)) {
8879
0
        return NJS_ERROR;
8880
0
    }
8881
8882
297
    stmt->right = node;
8883
297
    stmt->left = parent->left;
8884
297
    parent->left = stmt;
8885
8886
297
    return NJS_OK;
8887
297
}
8888
8889
8890
static njs_int_t
8891
njs_parser_array_item(njs_parser_t *parser, njs_parser_node_t *array,
8892
    njs_parser_node_t *value)
8893
1.64M
{
8894
1.64M
    njs_int_t          ret;
8895
1.64M
    njs_parser_node_t  *number;
8896
8897
1.64M
    number = njs_parser_node_new(parser, NJS_TOKEN_NUMBER);
8898
1.64M
    if (njs_slow_path(number == NULL)) {
8899
0
        return NJS_ERROR;
8900
0
    }
8901
8902
1.64M
    njs_set_number(&number->u.value, array->u.length);
8903
8904
1.64M
    number->token_line = value->token_line;
8905
8906
1.64M
    ret = njs_parser_object_property(parser, array, number, value, 0);
8907
1.64M
    if (njs_slow_path(ret != NJS_OK)) {
8908
0
        return NJS_ERROR;
8909
0
    }
8910
8911
1.64M
    array->ctor = 0;
8912
1.64M
    array->u.length++;
8913
8914
1.64M
    return NJS_OK;
8915
1.64M
}
8916
8917
8918
static njs_int_t
8919
njs_parser_template_string(njs_parser_t *parser, njs_lexer_token_t *token)
8920
696k
{
8921
696k
    u_char             *p, c;
8922
696k
    njs_int_t          ret;
8923
696k
    njs_str_t          *text;
8924
696k
    njs_bool_t         escape;
8925
696k
    njs_lexer_t        *lexer;
8926
696k
    njs_parser_node_t  *node;
8927
8928
696k
    lexer = parser->lexer;
8929
696k
    text = &token->text;
8930
8931
696k
    escape = 0;
8932
696k
    p = text->start;
8933
8934
696k
    if (p == NULL) {
8935
0
        return NJS_ERROR;
8936
0
    }
8937
8938
4.20M
    while (p < lexer->end) {
8939
8940
4.20M
        c = *p++;
8941
8942
4.20M
        switch (c) {
8943
34.5k
        case '\\':
8944
34.5k
            if (p == lexer->end) {
8945
0
                return NJS_ERROR;
8946
0
            }
8947
8948
34.5k
            p++;
8949
34.5k
            escape = 1;
8950
8951
34.5k
            continue;
8952
8953
684k
        case '`':
8954
684k
            text->length = p - text->start - 1;
8955
684k
            goto done;
8956
8957
28.6k
        case '$':
8958
28.6k
            if (p < lexer->end && *p == '{') {
8959
11.3k
                p++;
8960
11.3k
                text->length = p - text->start - 2;
8961
8962
11.3k
                ret = njs_lexer_in_stack_push(lexer);
8963
11.3k
                if (njs_slow_path(ret != NJS_OK)) {
8964
0
                    return NJS_ERROR;
8965
0
                }
8966
8967
11.3k
                goto done;
8968
11.3k
            }
8969
8970
17.3k
            break;
8971
8972
252k
        case '\n':
8973
252k
            parser->lexer->line++;
8974
252k
            break;
8975
4.20M
        }
8976
4.20M
    }
8977
8978
558
    return NJS_ERROR;
8979
8980
695k
done:
8981
8982
695k
    node = njs_parser_node_new(parser, NJS_TOKEN_STRING);
8983
695k
    if (njs_slow_path(node == NULL)) {
8984
0
        return NJS_ERROR;
8985
0
    }
8986
8987
695k
    node->token_line = token->line;
8988
8989
695k
    if (escape) {
8990
12.2k
        ret = njs_parser_escape_string_create(parser, token, &node->u.value);
8991
12.2k
        if (njs_slow_path(ret != NJS_TOKEN_STRING)) {
8992
142
            return NJS_ERROR;
8993
142
        }
8994
8995
683k
    } else {
8996
683k
        ret = njs_parser_string_create(parser->vm, token, &node->u.value);
8997
683k
        if (njs_slow_path(ret != NJS_OK)) {
8998
0
            return NJS_ERROR;
8999
0
        }
9000
683k
    }
9001
9002
695k
    lexer->start = p;
9003
695k
    parser->node = node;
9004
9005
695k
    return c == '`' ? NJS_DONE : NJS_OK;
9006
695k
}
9007
9008
9009
njs_int_t
9010
njs_parser_string_create(njs_vm_t *vm, njs_lexer_token_t *token,
9011
    njs_value_t *value)
9012
842k
{
9013
842k
    size_t     length;
9014
842k
    njs_int_t  ret;
9015
842k
    njs_str_t  dst;
9016
9017
842k
    length = njs_decode_utf8_length(&token->text, &dst.length);
9018
842k
    dst.start = njs_string_alloc(vm, value, dst.length, length);
9019
842k
    if (njs_slow_path(dst.start == NULL)) {
9020
0
        return NJS_ERROR;
9021
0
    }
9022
9023
842k
    njs_decode_utf8(&dst, &token->text);
9024
9025
842k
    if (length > NJS_STRING_MAP_STRIDE && dst.length != length) {
9026
8.17k
        njs_string_utf8_offset_map_init(value->string.data->start,
9027
8.17k
                                        dst.length);
9028
8.17k
    }
9029
9030
842k
    ret = njs_atom_atomize_key(vm, value);
9031
842k
    if (njs_slow_path(ret != NJS_OK)) {
9032
0
        return NJS_TOKEN_ERROR;
9033
0
    }
9034
9035
842k
    return NJS_OK;
9036
842k
}
9037
9038
9039
static njs_token_type_t
9040
njs_parser_escape_string_create(njs_parser_t *parser, njs_lexer_token_t *token,
9041
    njs_value_t *value)
9042
15.7k
{
9043
15.7k
    u_char                c, *start, *dst;
9044
15.7k
    size_t                size, length, hex_length;
9045
15.7k
    uint64_t              cp, cp_pair;
9046
15.7k
    njs_int_t             ret;
9047
15.7k
    njs_str_t             *string;
9048
15.7k
    const u_char          *src, *end, *hex_end;
9049
15.7k
    njs_unicode_decode_t  ctx;
9050
9051
15.7k
    ret = njs_parser_escape_string_calc_length(parser, token, &size, &length);
9052
15.7k
    if (njs_slow_path(ret != NJS_OK)) {
9053
547
        return NJS_TOKEN_ILLEGAL;
9054
547
    }
9055
9056
15.2k
    start = njs_string_alloc(parser->vm, value, size, length);
9057
15.2k
    if (njs_slow_path(start == NULL)) {
9058
0
        return NJS_TOKEN_ERROR;
9059
0
    }
9060
9061
15.2k
    dst = start;
9062
15.2k
    cp_pair = 0;
9063
9064
15.2k
    string = &token->text;
9065
15.2k
    src = string->start;
9066
15.2k
    end = src + string->length;
9067
9068
1.75M
    while (src < end) {
9069
1.73M
        c = *src++;
9070
9071
1.73M
        if (c == '\\') {
9072
            /*
9073
             * Testing "src == end" is not required here
9074
             * since this has been already tested by lexer.
9075
             */
9076
9077
38.8k
            c = *src++;
9078
9079
38.8k
            switch (c) {
9080
2.71k
            case 'u':
9081
                /*
9082
                 * A character after "u" can be safely tested here
9083
                 * because there is always a closing quote at the
9084
                 * end of string: ...\u".
9085
                 */
9086
9087
2.71k
                if (*src != '{') {
9088
2.67k
                    hex_length = 4;
9089
2.67k
                    goto hex_length;
9090
2.67k
                }
9091
9092
42
                src++;
9093
42
                hex_length = 0;
9094
42
                hex_end = end;
9095
9096
42
                goto hex;
9097
9098
13
            case 'x':
9099
13
                hex_length = 2;
9100
13
                goto hex_length;
9101
9102
529
            case '0':
9103
529
                c = '\0';
9104
529
                break;
9105
9106
306
            case '1':
9107
679
            case '2':
9108
773
            case '3':
9109
1.02k
            case '4':
9110
1.42k
            case '5':
9111
1.87k
            case '6':
9112
2.17k
            case '7':
9113
2.17k
                if (parser->node != NULL) {
9114
1.77k
                    switch (parser->node->token_type) {
9115
670
                    case NJS_TOKEN_METHOD_CALL:
9116
1.42k
                    case NJS_TOKEN_FUNCTION_CALL:
9117
1.55k
                    case NJS_TOKEN_FUNCTION_EXPRESSION:
9118
1.63k
                    case NJS_TOKEN_EVAL:
9119
1.63k
                        goto next_char;
9120
9121
144
                    default:
9122
144
                        break;
9123
1.77k
                    }
9124
1.77k
                }
9125
9126
539
                njs_parser_syntax_error(parser,
9127
539
                                        "Octal escape sequences can't be used "
9128
539
                                        "in untagged template literals "
9129
539
                                        "or in strict mode code");
9130
9131
539
                return NJS_TOKEN_ILLEGAL;
9132
9133
741
            case '8':
9134
966
            case '9':
9135
966
                if (parser->node != NULL) {
9136
289
                    switch (parser->node->token_type) {
9137
135
                    case NJS_TOKEN_METHOD_CALL:
9138
218
                    case NJS_TOKEN_FUNCTION_CALL:
9139
218
                    case NJS_TOKEN_FUNCTION_EXPRESSION:
9140
284
                    case NJS_TOKEN_EVAL:
9141
284
                        goto next_char;
9142
9143
5
                    default:
9144
5
                        break;
9145
289
                    }
9146
289
                }
9147
9148
682
                njs_parser_syntax_error(parser,
9149
682
                                        "The escapes \\8 and \\9 can't be used "
9150
682
                                        "in untagged template literals "
9151
682
                                        "or in strict mode code");
9152
9153
682
                return NJS_TOKEN_ILLEGAL;
9154
9155
1.91k
next_char:
9156
9157
1.91k
                break;
9158
9159
843
            case 'b':
9160
843
                c = '\b';
9161
843
                break;
9162
9163
189
            case 'f':
9164
189
                c = '\f';
9165
189
                break;
9166
9167
765
            case 'n':
9168
765
                c = '\n';
9169
765
                break;
9170
9171
478
            case 'r':
9172
478
                c = '\r';
9173
478
                break;
9174
9175
56
            case 't':
9176
56
                c = '\t';
9177
56
                break;
9178
9179
519
            case 'v':
9180
519
                c = '\v';
9181
519
                break;
9182
9183
545
            case '\r':
9184
                /*
9185
                 * A character after "\r" can be safely tested here
9186
                 * because there is always a closing quote at the
9187
                 * end of string: ...\\r".
9188
                 */
9189
9190
545
                if (*src == '\n') {
9191
199
                    src++;
9192
199
                }
9193
9194
545
                continue;
9195
9196
6.00k
            case '\n':
9197
6.00k
                continue;
9198
9199
23.0k
            default:
9200
23.0k
                if (c >= 0x80) {
9201
1.56k
                    goto utf8_copy;
9202
1.56k
                }
9203
9204
21.4k
                break;
9205
38.8k
            }
9206
38.8k
        }
9207
9208
1.72M
        if (c < 0x80) {
9209
1.22M
            *dst++ = c;
9210
9211
1.22M
            continue;
9212
1.22M
        }
9213
9214
500k
    utf8_copy:
9215
9216
500k
        src--;
9217
9218
500k
        njs_utf8_decode_init(&ctx);
9219
9220
500k
        cp = njs_utf8_decode(&ctx, &src, end);
9221
500k
        if (cp > NJS_UNICODE_MAX_CODEPOINT) {
9222
474k
            cp = NJS_UNICODE_REPLACEMENT;
9223
474k
        }
9224
9225
500k
        dst = njs_utf8_encode(dst, cp);
9226
9227
500k
        continue;
9228
9229
2.68k
    hex_length:
9230
9231
2.68k
        hex_end = src + hex_length;
9232
9233
2.72k
    hex:
9234
2.72k
        cp = njs_number_hex_parse(&src, hex_end, 0);
9235
9236
        /* Skip '}' character. */
9237
9238
2.72k
        if (hex_length == 0) {
9239
42
            src++;
9240
42
        }
9241
9242
2.72k
        if (cp_pair != 0) {
9243
504
            if (njs_fast_path(njs_surrogate_trailing(cp))) {
9244
81
                cp = njs_surrogate_pair(cp_pair, cp);
9245
9246
423
            } else if (njs_slow_path(njs_surrogate_leading(cp))) {
9247
134
                cp = NJS_UNICODE_REPLACEMENT;
9248
9249
134
                dst = njs_utf8_encode(dst, (uint32_t) cp);
9250
9251
289
            } else {
9252
289
                dst = njs_utf8_encode(dst, NJS_UNICODE_REPLACEMENT);
9253
289
            }
9254
9255
504
            cp_pair = 0;
9256
9257
2.22k
        } else if (njs_surrogate_any(cp)) {
9258
1.75k
            if (cp <= 0xdbff && src[0] == '\\' && src[1] == 'u') {
9259
504
                cp_pair = cp;
9260
504
                continue;
9261
504
            }
9262
9263
1.24k
            cp = NJS_UNICODE_REPLACEMENT;
9264
1.24k
        }
9265
9266
2.22k
        dst = njs_utf8_encode(dst, (uint32_t) cp);
9267
2.22k
        if (njs_slow_path(dst == NULL)) {
9268
42
            njs_parser_syntax_error(parser, "Invalid Unicode code point \"%V\"",
9269
42
                                    &token->text);
9270
9271
42
            return NJS_TOKEN_ILLEGAL;
9272
42
        }
9273
2.22k
    }
9274
9275
13.9k
    if (length > NJS_STRING_MAP_STRIDE && length != size) {
9276
4.92k
        njs_string_utf8_offset_map_init(start, size);
9277
4.92k
    }
9278
9279
13.9k
    ret = njs_atom_atomize_key(parser->vm, value);
9280
13.9k
    if (njs_slow_path(ret != NJS_OK)) {
9281
0
        return NJS_TOKEN_ERROR;
9282
0
    }
9283
9284
13.9k
    return NJS_TOKEN_STRING;
9285
13.9k
}
9286
9287
9288
static njs_int_t
9289
njs_parser_escape_string_calc_length(njs_parser_t *parser,
9290
    njs_lexer_token_t *token, size_t *out_size, size_t *out_length)
9291
15.7k
{
9292
15.7k
    size_t                size, length, hex_length;
9293
15.7k
    uint64_t              cp, cp_pair;
9294
15.7k
    njs_str_t             *string;
9295
15.7k
    const u_char          *ptr, *src, *end, *hex_end;
9296
15.7k
    njs_unicode_decode_t  ctx;
9297
9298
15.7k
    size = 0;
9299
15.7k
    length = 0;
9300
15.7k
    cp_pair = 0;
9301
9302
15.7k
    string = &token->text;
9303
15.7k
    src = string->start;
9304
15.7k
    end = src + string->length;
9305
9306
1.77M
    while (src < end) {
9307
9308
1.76M
        if (*src == '\\') {
9309
39.5k
            src++;
9310
9311
39.5k
            switch (*src) {
9312
3.27k
            case 'u':
9313
3.27k
                src++;
9314
9315
3.27k
                if (*src != '{') {
9316
2.82k
                    hex_length = 4;
9317
2.82k
                    goto hex_length;
9318
2.82k
                }
9319
9320
448
                src++;
9321
448
                hex_length = 0;
9322
448
                hex_end = end;
9323
9324
448
                goto hex;
9325
9326
13
            case 'x':
9327
13
                src++;
9328
13
                hex_length = 2;
9329
13
                goto hex_length;
9330
9331
545
            case '\r':
9332
545
                src++;
9333
9334
545
                if (*src == '\n') {
9335
199
                    src++;
9336
199
                }
9337
9338
545
                continue;
9339
9340
6.00k
            case '\n':
9341
6.00k
                src++;
9342
6.00k
                continue;
9343
9344
29.7k
            default:
9345
29.7k
                break;
9346
39.5k
            }
9347
39.5k
        }
9348
9349
1.75M
        if (*src >= 0x80) {
9350
503k
            njs_utf8_decode_init(&ctx);
9351
9352
503k
            cp = njs_utf8_decode(&ctx, &src, end);
9353
503k
            if (cp > NJS_UNICODE_MAX_CODEPOINT) {
9354
474k
                cp = NJS_UNICODE_REPLACEMENT;
9355
474k
            }
9356
9357
503k
            size += njs_utf8_size(cp);
9358
503k
            length++;
9359
9360
503k
            continue;
9361
503k
        }
9362
9363
1.25M
        src++;
9364
1.25M
        size++;
9365
1.25M
        length++;
9366
9367
1.25M
        continue;
9368
9369
2.84k
    hex_length:
9370
9371
2.84k
        hex_end = src + hex_length;
9372
9373
2.84k
        if (njs_slow_path(hex_end > end)) {
9374
0
            goto invalid;
9375
0
        }
9376
9377
3.29k
    hex:
9378
9379
3.29k
        ptr = src;
9380
3.29k
        cp = njs_number_hex_parse(&src, hex_end, 0);
9381
9382
3.29k
        if (hex_length != 0) {
9383
2.84k
            if (src != hex_end) {
9384
141
                goto invalid;
9385
141
            }
9386
9387
2.84k
        } else {
9388
448
            if (src == ptr || (src - ptr) > 6) {
9389
361
                goto invalid;
9390
361
            }
9391
9392
87
            if (src == end || *src++ != '}') {
9393
45
                goto invalid;
9394
45
            }
9395
87
        }
9396
9397
2.74k
        if (cp_pair != 0) {
9398
504
            if (njs_fast_path(njs_surrogate_trailing(cp))) {
9399
81
                cp = njs_surrogate_pair(cp_pair, cp);
9400
9401
423
            } else if (njs_slow_path(njs_surrogate_leading(cp))) {
9402
134
                cp = NJS_UNICODE_REPLACEMENT;
9403
9404
134
                size += njs_utf8_size(cp);
9405
134
                length++;
9406
9407
289
            } else {
9408
289
                size += njs_utf8_size(NJS_UNICODE_REPLACEMENT);
9409
289
                length++;
9410
289
            }
9411
9412
504
            cp_pair = 0;
9413
9414
2.23k
        } else if (njs_surrogate_any(cp)) {
9415
1.76k
            if (cp <= 0xdbff && src[0] == '\\' && src[1] == 'u') {
9416
504
                cp_pair = cp;
9417
504
                continue;
9418
504
            }
9419
9420
1.26k
            cp = NJS_UNICODE_REPLACEMENT;
9421
1.26k
        }
9422
9423
2.23k
        size += njs_utf8_size(cp);
9424
2.23k
        length++;
9425
2.23k
    }
9426
9427
15.2k
    *out_size = size;
9428
15.2k
    *out_length = length;
9429
9430
15.2k
    return NJS_OK;
9431
9432
547
invalid:
9433
9434
547
    njs_parser_syntax_error(parser, "Invalid Unicode code point \"%V\"",
9435
547
                            &token->text);
9436
547
    return NJS_ERROR;
9437
15.7k
}
9438
9439
9440
njs_bool_t
9441
njs_parser_has_side_effect(njs_parser_node_t *node)
9442
6.83M
{
9443
6.83M
    njs_bool_t  side_effect;
9444
9445
6.83M
    if (node == NULL) {
9446
3.78M
        return 0;
9447
3.78M
    }
9448
9449
3.05M
    if (node->token_type >= NJS_TOKEN_ASSIGNMENT
9450
3.04M
        && node->token_type <= NJS_TOKEN_LAST_ASSIGNMENT)
9451
93.4k
    {
9452
93.4k
        return 1;
9453
93.4k
    }
9454
9455
2.96M
    if (node->token_type == NJS_TOKEN_FUNCTION_CALL
9456
2.95M
        || node->token_type == NJS_TOKEN_METHOD_CALL)
9457
3.72k
    {
9458
3.72k
        return 1;
9459
3.72k
    }
9460
9461
2.95M
    side_effect = njs_parser_has_side_effect(node->left);
9462
9463
2.95M
    if (njs_fast_path(!side_effect)) {
9464
2.04M
        return njs_parser_has_side_effect(node->right);
9465
2.04M
    }
9466
9467
911k
    return side_effect;
9468
2.95M
}
9469
9470
9471
njs_int_t
9472
njs_parser_variable_reference(njs_parser_t *parser, njs_parser_scope_t *scope,
9473
    njs_parser_node_t *node, uintptr_t atom_id, njs_reference_type_t type)
9474
1.63M
{
9475
1.63M
    njs_rbtree_node_t         *rb_node;
9476
1.63M
    njs_variable_reference_t  *vr;
9477
1.63M
    njs_parser_rbtree_node_t  parse_node, *rb_parse_node;
9478
9479
1.63M
    vr = &node->u.reference;
9480
9481
1.63M
    vr->atom_id = atom_id;
9482
1.63M
    vr->type = type;
9483
9484
1.63M
    parse_node.key = atom_id;
9485
9486
1.63M
    rb_node = njs_rbtree_find(&scope->references, &parse_node.node);
9487
1.63M
    if (rb_node != NULL) {
9488
477k
        return NJS_OK;
9489
477k
    }
9490
9491
1.15M
    rb_parse_node = njs_mp_alloc(parser->vm->mem_pool,
9492
1.15M
                                 sizeof(njs_parser_rbtree_node_t));
9493
1.15M
    if (njs_slow_path(rb_parse_node == NULL)) {
9494
0
        return NJS_ERROR;
9495
0
    }
9496
9497
1.15M
    rb_parse_node->key = atom_id;
9498
1.15M
    rb_parse_node->index = NJS_INDEX_NONE;
9499
9500
1.15M
    njs_rbtree_insert(&scope->references, &rb_parse_node->node);
9501
9502
1.15M
    return NJS_OK;
9503
1.15M
}
9504
9505
9506
njs_token_type_t
9507
njs_parser_unexpected_token(njs_vm_t *vm, njs_parser_t *parser,
9508
    njs_str_t *name, njs_token_type_t type)
9509
0
{
9510
0
    if (type != NJS_TOKEN_END) {
9511
0
        njs_parser_syntax_error(parser, "Unexpected token \"%V\"", name);
9512
9513
0
    } else {
9514
0
        njs_parser_syntax_error(parser, "Unexpected end of input");
9515
0
    }
9516
9517
0
    return NJS_DONE;
9518
0
}
9519
9520
9521
static void
9522
njs_parser_error(njs_vm_t *vm, njs_object_type_t type, njs_str_t *file,
9523
    uint32_t line, const char *fmt, va_list args)
9524
69.5k
{
9525
69.5k
    u_char       msg[NJS_MAX_ERROR_STR];
9526
69.5k
    u_char       *p, *end;
9527
69.5k
    njs_value_t  error;
9528
9529
69.5k
    if (njs_slow_path(vm->top_frame == NULL)) {
9530
0
        njs_vm_runtime_init(vm);
9531
0
    }
9532
9533
69.5k
    p = msg;
9534
69.5k
    end = msg + NJS_MAX_ERROR_STR;
9535
9536
69.5k
    p = njs_vsprintf(p, end, fmt, args);
9537
9538
69.5k
    njs_error_new(vm, &error, njs_vm_proto(vm, type), msg, p - msg);
9539
9540
69.5k
    if (!vm->options.quiet && file->length != 0) {
9541
69.5k
        njs_error_stack_set(vm, &error, file, line);
9542
69.5k
    }
9543
9544
69.5k
    njs_vm_throw(vm, &error);
9545
69.5k
}
9546
9547
9548
void
9549
njs_parser_lexer_error(njs_parser_t *parser, njs_object_type_t type,
9550
    const char *fmt, ...)
9551
67.3k
{
9552
67.3k
    va_list  args;
9553
9554
67.3k
    if (njs_is_error(&parser->vm->exception)) {
9555
1.77k
        return;
9556
1.77k
    }
9557
9558
67.3k
    va_start(args, fmt);
9559
65.6k
    njs_parser_error(parser->vm, type, &parser->lexer->file,
9560
65.6k
                     parser->lexer->line, fmt, args);
9561
65.6k
    va_end(args);
9562
65.6k
}
9563
9564
9565
void
9566
njs_parser_node_error(njs_vm_t *vm, njs_object_type_t type,
9567
    njs_parser_node_t *node, njs_str_t *file, const char *fmt, ...)
9568
3.91k
{
9569
3.91k
    va_list  args;
9570
9571
3.91k
    va_start(args, fmt);
9572
3.91k
    njs_parser_error(vm, type, file, node->token_line, fmt, args);
9573
3.91k
    va_end(args);
9574
3.91k
}
9575
9576
9577
njs_int_t
9578
njs_parser_traverse(njs_vm_t *vm, njs_parser_node_t *root, void *ctx,
9579
    njs_parser_traverse_cb_t cb)
9580
9.18k
{
9581
9.18k
    njs_int_t          ret;
9582
9.18k
    njs_arr_t          *stack;
9583
9.18k
    njs_parser_node_t  *node, **ref;
9584
9585
9.18k
    if (root == NULL) {
9586
4.75k
        return NJS_OK;
9587
4.75k
    }
9588
9589
4.43k
    stack = njs_arr_create(vm->mem_pool, 8, sizeof(njs_parser_node_t *));
9590
4.43k
    if (njs_slow_path(stack == NULL)) {
9591
0
        return NJS_ERROR;
9592
9593
0
    }
9594
9595
4.43k
    ref = njs_arr_add(stack);
9596
4.43k
    if (njs_slow_path(ref == NULL)) {
9597
0
        goto failed;
9598
0
    }
9599
9600
4.43k
    *ref = root;
9601
9602
32.7k
    while (1) {
9603
32.7k
        if (njs_arr_is_empty(stack)) {
9604
4.43k
            break;
9605
4.43k
        }
9606
9607
28.3k
        ref = njs_arr_remove_last(stack);
9608
28.3k
        node = *ref;
9609
9610
28.3k
        ret = cb(vm, node, ctx);
9611
28.3k
        if (njs_slow_path(ret != NJS_OK)) {
9612
0
            goto failed;
9613
0
        }
9614
9615
28.3k
        if (node->left != NULL) {
9616
10.3k
            ref = njs_arr_add(stack);
9617
10.3k
            if (njs_slow_path(ref == NULL)) {
9618
0
                goto failed;
9619
0
            }
9620
9621
10.3k
            *ref = node->left;
9622
10.3k
        }
9623
9624
28.3k
        if (node->right != NULL) {
9625
13.5k
            ref = njs_arr_add(stack);
9626
13.5k
            if (njs_slow_path(ref == NULL)) {
9627
0
                goto failed;
9628
0
            }
9629
9630
13.5k
            *ref = node->right;
9631
13.5k
        }
9632
28.3k
    }
9633
9634
4.43k
    njs_arr_destroy(stack);
9635
9636
4.43k
    return NJS_OK;
9637
9638
0
failed:
9639
9640
0
    njs_arr_destroy(stack);
9641
9642
0
    return NJS_ERROR;
9643
4.43k
}
9644
9645
9646
njs_int_t
9647
njs_parser_serialize_ast(njs_parser_node_t *node, njs_chb_t *chain)
9648
0
{
9649
0
    njs_int_t  ret;
9650
9651
0
    ret = NJS_OK;
9652
9653
0
    njs_parser_serialize_tree(chain, node, &ret, 0);
9654
0
    njs_chb_append_literal(chain, "\n");
9655
9656
0
    return ret;
9657
0
}
9658
9659
9660
njs_inline void
9661
njs_parser_serialize_indent(njs_chb_t *chain, size_t indent)
9662
0
{
9663
0
    size_t  i;
9664
9665
0
    for (i = 0; i < indent; i++) {
9666
0
        njs_chb_append_literal(chain, "  ");
9667
0
    }
9668
0
}
9669
9670
9671
static void
9672
njs_parser_serialize_tree(njs_chb_t *chain, njs_parser_node_t *node,
9673
    njs_int_t *ret, size_t indent)
9674
0
{
9675
0
    njs_str_t  str;
9676
9677
0
    njs_chb_append_literal(chain, "{\"name\": \"");
9678
9679
0
    *ret |= njs_parser_serialize_node(chain, node);
9680
9681
0
    njs_chb_append_literal(chain, "\",\n");
9682
0
    njs_parser_serialize_indent(chain, indent);
9683
0
    njs_chb_sprintf(chain, 32, " \"line\": %d", node->token_line);
9684
9685
0
    switch (node->token_type) {
9686
0
    case NJS_TOKEN_NUMBER:
9687
0
    case NJS_TOKEN_STRING:
9688
0
    case NJS_TOKEN_NAME:
9689
0
    case NJS_TOKEN_FUNCTION_CALL:
9690
0
        njs_chb_append_literal(chain, ",\n");
9691
0
        njs_parser_serialize_indent(chain, indent);
9692
0
        njs_chb_sprintf(chain, 32, " \"index\": \"%p\"", node->index);
9693
9694
0
        switch (node->token_type) {
9695
0
        case NJS_TOKEN_NUMBER:
9696
0
        case NJS_TOKEN_STRING:
9697
0
            njs_chb_append_literal(chain, ",\n");
9698
0
            njs_parser_serialize_indent(chain, indent);
9699
9700
0
            if (node->token_type == NJS_TOKEN_NUMBER) {
9701
0
                njs_chb_sprintf(chain, 32, " \"value\": %f",
9702
0
                                njs_number(&node->u.value));
9703
9704
0
            } else {
9705
0
                njs_string_get_unsafe(&node->u.value, &str);
9706
0
                njs_chb_append_literal(chain, " \"value\": \"");
9707
0
                njs_chb_append_str(chain, &str);
9708
0
                njs_chb_append_literal(chain, "\"");
9709
0
            }
9710
9711
0
            break;
9712
9713
0
        default:
9714
0
            break;
9715
0
        }
9716
9717
0
        break;
9718
9719
0
    default:
9720
0
        break;
9721
0
    }
9722
9723
0
    if (node->left != NULL) {
9724
0
        njs_chb_append_literal(chain, ",\n");
9725
0
        njs_parser_serialize_indent(chain, indent);
9726
0
        njs_chb_append_literal(chain, " \"left\": ");
9727
9728
0
        njs_parser_serialize_tree(chain, node->left, ret, indent + 1);
9729
0
    }
9730
9731
0
    if (node->right != NULL) {
9732
0
        njs_chb_append_literal(chain, ",\n");
9733
0
        njs_parser_serialize_indent(chain, indent);
9734
0
        njs_chb_append_literal(chain, " \"right\": ");
9735
9736
0
        njs_parser_serialize_tree(chain, node->right, ret, indent + 1);
9737
0
    }
9738
9739
0
    njs_chb_append_literal(chain, "}");
9740
0
}
9741
9742
9743
static njs_int_t
9744
njs_parser_serialize_node(njs_chb_t *chain, njs_parser_node_t *node)
9745
0
{
9746
0
    const char  *name;
9747
9748
0
#define njs_token_serialize(token)                                          \
9749
0
    case token:                                                             \
9750
0
        name = &njs_stringify(token)[njs_length("NJS_TOKEN_")];             \
9751
0
        njs_chb_append(chain, name, njs_strlen(name));                      \
9752
0
        break
9753
9754
0
    switch (node->token_type) {
9755
0
    njs_token_serialize(NJS_TOKEN_END);
9756
    /* FIXME: NJS_TOKEN_ILLEGAL should not be present in AST */
9757
0
    njs_token_serialize(NJS_TOKEN_ILLEGAL);
9758
0
    njs_token_serialize(NJS_TOKEN_COMMA);
9759
0
    njs_token_serialize(NJS_TOKEN_CONDITIONAL);
9760
0
    njs_token_serialize(NJS_TOKEN_ASSIGNMENT);
9761
0
    njs_token_serialize(NJS_TOKEN_ADDITION_ASSIGNMENT);
9762
0
    njs_token_serialize(NJS_TOKEN_SUBTRACTION_ASSIGNMENT);
9763
0
    njs_token_serialize(NJS_TOKEN_MULTIPLICATION_ASSIGNMENT);
9764
0
    njs_token_serialize(NJS_TOKEN_EXPONENTIATION_ASSIGNMENT);
9765
0
    njs_token_serialize(NJS_TOKEN_DIVISION_ASSIGNMENT);
9766
0
    njs_token_serialize(NJS_TOKEN_REMAINDER_ASSIGNMENT);
9767
0
    njs_token_serialize(NJS_TOKEN_LEFT_SHIFT_ASSIGNMENT);
9768
0
    njs_token_serialize(NJS_TOKEN_RIGHT_SHIFT_ASSIGNMENT);
9769
0
    njs_token_serialize(NJS_TOKEN_UNSIGNED_RIGHT_SHIFT_ASSIGNMENT);
9770
0
    njs_token_serialize(NJS_TOKEN_BITWISE_OR_ASSIGNMENT);
9771
0
    njs_token_serialize(NJS_TOKEN_BITWISE_XOR_ASSIGNMENT);
9772
0
    njs_token_serialize(NJS_TOKEN_BITWISE_AND_ASSIGNMENT);
9773
0
    njs_token_serialize(NJS_TOKEN_LOGICAL_OR_ASSIGNMENT);
9774
0
    njs_token_serialize(NJS_TOKEN_LOGICAL_AND_ASSIGNMENT);
9775
0
    njs_token_serialize(NJS_TOKEN_COALESCE_ASSIGNMENT);
9776
0
    njs_token_serialize(NJS_TOKEN_EQUAL);
9777
0
    njs_token_serialize(NJS_TOKEN_NOT_EQUAL);
9778
0
    njs_token_serialize(NJS_TOKEN_STRICT_EQUAL);
9779
0
    njs_token_serialize(NJS_TOKEN_STRICT_NOT_EQUAL);
9780
0
    njs_token_serialize(NJS_TOKEN_ADDITION);
9781
0
    njs_token_serialize(NJS_TOKEN_UNARY_PLUS);
9782
0
    njs_token_serialize(NJS_TOKEN_INCREMENT);
9783
0
    njs_token_serialize(NJS_TOKEN_POST_INCREMENT);
9784
0
    njs_token_serialize(NJS_TOKEN_SUBTRACTION);
9785
0
    njs_token_serialize(NJS_TOKEN_UNARY_NEGATION);
9786
0
    njs_token_serialize(NJS_TOKEN_DECREMENT);
9787
0
    njs_token_serialize(NJS_TOKEN_POST_DECREMENT);
9788
0
    njs_token_serialize(NJS_TOKEN_MULTIPLICATION);
9789
0
    njs_token_serialize(NJS_TOKEN_EXPONENTIATION);
9790
0
    njs_token_serialize(NJS_TOKEN_DIVISION);
9791
0
    njs_token_serialize(NJS_TOKEN_REMAINDER);
9792
0
    njs_token_serialize(NJS_TOKEN_LESS);
9793
0
    njs_token_serialize(NJS_TOKEN_LESS_OR_EQUAL);
9794
0
    njs_token_serialize(NJS_TOKEN_LEFT_SHIFT);
9795
0
    njs_token_serialize(NJS_TOKEN_GREATER);
9796
0
    njs_token_serialize(NJS_TOKEN_GREATER_OR_EQUAL);
9797
0
    njs_token_serialize(NJS_TOKEN_RIGHT_SHIFT);
9798
0
    njs_token_serialize(NJS_TOKEN_UNSIGNED_RIGHT_SHIFT);
9799
0
    njs_token_serialize(NJS_TOKEN_BITWISE_OR);
9800
0
    njs_token_serialize(NJS_TOKEN_LOGICAL_OR);
9801
0
    njs_token_serialize(NJS_TOKEN_BITWISE_XOR);
9802
0
    njs_token_serialize(NJS_TOKEN_BITWISE_AND);
9803
0
    njs_token_serialize(NJS_TOKEN_LOGICAL_AND);
9804
0
    njs_token_serialize(NJS_TOKEN_BITWISE_NOT);
9805
0
    njs_token_serialize(NJS_TOKEN_LOGICAL_NOT);
9806
0
    njs_token_serialize(NJS_TOKEN_COALESCE);
9807
0
    njs_token_serialize(NJS_TOKEN_OPTIONAL_CHAIN);
9808
0
    njs_token_serialize(NJS_TOKEN_IN);
9809
0
    njs_token_serialize(NJS_TOKEN_OF);
9810
0
    njs_token_serialize(NJS_TOKEN_INSTANCEOF);
9811
0
    njs_token_serialize(NJS_TOKEN_TYPEOF);
9812
0
    njs_token_serialize(NJS_TOKEN_VOID);
9813
0
    njs_token_serialize(NJS_TOKEN_NEW);
9814
0
    njs_token_serialize(NJS_TOKEN_DELETE);
9815
0
    njs_token_serialize(NJS_TOKEN_YIELD);
9816
9817
0
    njs_token_serialize(NJS_TOKEN_NULL);
9818
0
    njs_token_serialize(NJS_TOKEN_NUMBER);
9819
0
    njs_token_serialize(NJS_TOKEN_TRUE);
9820
0
    njs_token_serialize(NJS_TOKEN_FALSE);
9821
0
    njs_token_serialize(NJS_TOKEN_STRING);
9822
0
    njs_token_serialize(NJS_TOKEN_TEMPLATE_LITERAL);
9823
0
    njs_token_serialize(NJS_TOKEN_NAME);
9824
0
    njs_token_serialize(NJS_TOKEN_OBJECT);
9825
0
    njs_token_serialize(NJS_TOKEN_OBJECT_VALUE);
9826
0
    njs_token_serialize(NJS_TOKEN_OPTIONAL_PRESERVE);
9827
0
    njs_token_serialize(NJS_TOKEN_ARRAY);
9828
0
    njs_token_serialize(NJS_TOKEN_REGEXP);
9829
9830
0
    njs_token_serialize(NJS_TOKEN_PROPERTY);
9831
0
    njs_token_serialize(NJS_TOKEN_PROPERTY_REF);
9832
0
    njs_token_serialize(NJS_TOKEN_PROPERTY_INIT);
9833
0
    njs_token_serialize(NJS_TOKEN_PROPERTY_DELETE);
9834
0
    njs_token_serialize(NJS_TOKEN_PROPERTY_GETTER);
9835
0
    njs_token_serialize(NJS_TOKEN_PROPERTY_SETTER);
9836
9837
0
    njs_token_serialize(NJS_TOKEN_PROTO_INIT);
9838
9839
0
    njs_token_serialize(NJS_TOKEN_FUNCTION);
9840
0
    njs_token_serialize(NJS_TOKEN_ASYNC_FUNCTION);
9841
0
    njs_token_serialize(NJS_TOKEN_FUNCTION_DECLARATION);
9842
0
    njs_token_serialize(NJS_TOKEN_ASYNC_FUNCTION_DECLARATION);
9843
0
    njs_token_serialize(NJS_TOKEN_FUNCTION_EXPRESSION);
9844
0
    njs_token_serialize(NJS_TOKEN_ASYNC_FUNCTION_EXPRESSION);
9845
0
    njs_token_serialize(NJS_TOKEN_FUNCTION_CALL);
9846
0
    njs_token_serialize(NJS_TOKEN_METHOD_CALL);
9847
9848
0
    njs_token_serialize(NJS_TOKEN_ARGUMENT);
9849
0
    njs_token_serialize(NJS_TOKEN_RETURN);
9850
0
    njs_token_serialize(NJS_TOKEN_STATEMENT);
9851
0
    njs_token_serialize(NJS_TOKEN_BLOCK);
9852
0
    njs_token_serialize(NJS_TOKEN_VAR);
9853
0
    njs_token_serialize(NJS_TOKEN_LET);
9854
0
    njs_token_serialize(NJS_TOKEN_CONST);
9855
0
    njs_token_serialize(NJS_TOKEN_IF);
9856
0
    njs_token_serialize(NJS_TOKEN_ELSE);
9857
0
    njs_token_serialize(NJS_TOKEN_BRANCHING);
9858
0
    njs_token_serialize(NJS_TOKEN_WHILE);
9859
0
    njs_token_serialize(NJS_TOKEN_DO);
9860
0
    njs_token_serialize(NJS_TOKEN_FOR);
9861
0
    njs_token_serialize(NJS_TOKEN_FOR_IN);
9862
0
    njs_token_serialize(NJS_TOKEN_BREAK);
9863
0
    njs_token_serialize(NJS_TOKEN_CONTINUE);
9864
0
    njs_token_serialize(NJS_TOKEN_SWITCH);
9865
0
    njs_token_serialize(NJS_TOKEN_CASE);
9866
0
    njs_token_serialize(NJS_TOKEN_DEFAULT);
9867
0
    njs_token_serialize(NJS_TOKEN_WITH);
9868
0
    njs_token_serialize(NJS_TOKEN_TRY);
9869
0
    njs_token_serialize(NJS_TOKEN_CATCH);
9870
0
    njs_token_serialize(NJS_TOKEN_FINALLY);
9871
0
    njs_token_serialize(NJS_TOKEN_THROW);
9872
0
    njs_token_serialize(NJS_TOKEN_THIS);
9873
0
    njs_token_serialize(NJS_TOKEN_ARGUMENTS);
9874
0
    njs_token_serialize(NJS_TOKEN_EVAL);
9875
0
    njs_token_serialize(NJS_TOKEN_IMPORT);
9876
0
    njs_token_serialize(NJS_TOKEN_EXPORT);
9877
0
    njs_token_serialize(NJS_TOKEN_DEBUGGER);
9878
9879
#if 0
9880
9881
    njs_token_serialize(NJS_TOKEN_TARGET);
9882
    njs_token_serialize(NJS_TOKEN_META);
9883
    njs_token_serialize(NJS_TOKEN_ASYNC);
9884
    njs_token_serialize(NJS_TOKEN_AWAIT);
9885
    njs_token_serialize(NJS_TOKEN_ENUM);
9886
9887
    njs_token_serialize(NJS_TOKEN_CLASS);
9888
    njs_token_serialize(NJS_TOKEN_EXTENDS);
9889
    njs_token_serialize(NJS_TOKEN_IMPLEMENTS);
9890
    njs_token_serialize(NJS_TOKEN_INTERFACE);
9891
    njs_token_serialize(NJS_TOKEN_PACKAGE);
9892
    njs_token_serialize(NJS_TOKEN_PRIVATE);
9893
    njs_token_serialize(NJS_TOKEN_PROTECTED);
9894
    njs_token_serialize(NJS_TOKEN_PUBLIC);
9895
    njs_token_serialize(NJS_TOKEN_STATIC);
9896
    njs_token_serialize(NJS_TOKEN_SUPER);
9897
9898
#endif
9899
9900
0
    default:
9901
0
        njs_chb_sprintf(chain, 32, "#UNDEF(%d)", (int) node->token_type);
9902
0
        return NJS_DECLINED;
9903
0
    }
9904
9905
0
    return NJS_OK;
9906
0
}