1use super::item::VarFlags;
2use crate::{PResult, Parser, parser::SeqSep};
3use smallvec::SmallVec;
4use solar_ast::{token::*, *};
5use solar_data_structures::BumpExt;
6use solar_interface::{Ident, Span, kw, sym};
7
8impl<'sess, 'ast> Parser<'sess, 'ast> {
9 #[instrument(level = "trace", skip_all)]
11 pub fn parse_stmt(&mut self) -> PResult<'sess, Stmt<'ast>> {
12 self.with_recursion_limit("statement", |this| {
13 let docs = this.parse_doc_comments();
14 this.parse_spanned(Self::parse_stmt_kind).map(|(span, kind)| Stmt { docs, kind, span })
15 })
16 }
17
18 pub fn parse_stmt_boxed(&mut self) -> PResult<'sess, Box<'ast, Stmt<'ast>>> {
20 self.parse_stmt().map(|stmt| self.alloc(stmt))
21 }
22
23 fn parse_stmt_kind(&mut self) -> PResult<'sess, StmtKind<'ast>> {
25 let mut semi = true;
26 let kind = if self.eat_keyword(kw::If) {
27 semi = false;
28 self.parse_stmt_if()
29 } else if self.eat_keyword(kw::While) {
30 semi = false;
31 self.parse_stmt_while()
32 } else if self.eat_keyword(kw::Do) {
33 self.parse_stmt_do_while()
34 } else if self.eat_keyword(kw::For) {
35 semi = false;
36 self.parse_stmt_for()
37 } else if self.eat_keyword(kw::Unchecked) {
38 semi = false;
39 self.parse_block().map(StmtKind::UncheckedBlock)
40 } else if self.check(TokenKind::OpenDelim(Delimiter::Brace)) {
41 semi = false;
42 self.parse_block().map(StmtKind::Block)
43 } else if self.eat_keyword(kw::Continue) {
44 Ok(StmtKind::Continue)
45 } else if self.eat_keyword(kw::Break) {
46 Ok(StmtKind::Break)
47 } else if self.eat_keyword(kw::Return) {
48 let expr = if self.check(TokenKind::Semi) { None } else { Some(self.parse_expr()?) };
49 Ok(StmtKind::Return(expr))
50 } else if self.eat_keyword(kw::Throw) {
51 let msg = "`throw` statements have been removed; use `revert`, `require`, or `assert` instead";
52 Err(self.dcx().err(msg).span(self.prev_token.span))
53 } else if self.eat_keyword(kw::Try) {
54 semi = false;
55 self.parse_stmt_try().map(|stmt| StmtKind::Try(self.alloc(stmt)))
56 } else if self.eat_keyword(kw::Assembly) {
57 semi = false;
58 self.parse_stmt_assembly().map(StmtKind::Assembly)
59 } else if self.eat_keyword(kw::Emit) {
60 self.parse_path_call().map(|(path, params)| StmtKind::Emit(path, params))
61 } else if self.check_keyword(kw::Revert) && self.look_ahead(1).is_ident() {
62 self.bump(); self.parse_path_call().map(|(path, params)| StmtKind::Revert(path, params))
64 } else if self.check_keyword(sym::underscore) && self.look_ahead(1).kind == TokenKind::Semi
65 {
66 self.bump(); Ok(StmtKind::Placeholder)
68 } else {
69 self.parse_simple_stmt_kind()
70 };
71 if semi && kind.is_ok() {
72 self.expect_semi()?;
73 }
74 kind
75 }
76
77 pub(super) fn parse_block(&mut self) -> PResult<'sess, Block<'ast>> {
79 let lo = self.token.span;
80 self.parse_delim_seq(Delimiter::Brace, SeqSep::none(), true, Self::parse_stmt)
81 .map(|stmts| Block { span: lo.to(self.prev_token.span), stmts })
82 }
83
84 fn parse_stmt_if(&mut self) -> PResult<'sess, StmtKind<'ast>> {
86 self.expect(TokenKind::OpenDelim(Delimiter::Parenthesis))?;
87 let expr = self.parse_expr()?;
88 self.expect(TokenKind::CloseDelim(Delimiter::Parenthesis))?;
89 let true_stmt = self.parse_stmt()?;
90 let else_stmt =
91 if self.eat_keyword(kw::Else) { Some(self.parse_stmt_boxed()?) } else { None };
92 Ok(StmtKind::If(expr, self.alloc(true_stmt), else_stmt))
93 }
94
95 fn parse_stmt_while(&mut self) -> PResult<'sess, StmtKind<'ast>> {
97 self.expect(TokenKind::OpenDelim(Delimiter::Parenthesis))?;
98 let expr = self.parse_expr()?;
99 self.expect(TokenKind::CloseDelim(Delimiter::Parenthesis))?;
100 let stmt = self.parse_stmt()?;
101 Ok(StmtKind::While(expr, self.alloc(stmt)))
102 }
103
104 fn parse_stmt_do_while(&mut self) -> PResult<'sess, StmtKind<'ast>> {
106 let stmt = self.parse_stmt()?;
107 let stmt = self.alloc(stmt);
108 self.expect_keyword(kw::While)?;
109 self.expect(TokenKind::OpenDelim(Delimiter::Parenthesis))?;
110 let expr = self.parse_expr()?;
111 self.expect(TokenKind::CloseDelim(Delimiter::Parenthesis))?;
112 Ok(StmtKind::DoWhile(stmt, expr))
113 }
114
115 fn parse_stmt_for(&mut self) -> PResult<'sess, StmtKind<'ast>> {
117 self.expect(TokenKind::OpenDelim(Delimiter::Parenthesis))?;
118
119 let init = if self.check(TokenKind::Semi) { None } else { Some(self.parse_simple_stmt()?) };
120 self.expect(TokenKind::Semi)?;
121
122 let cond = if self.check(TokenKind::Semi) { None } else { Some(self.parse_expr()?) };
123 self.expect_semi()?;
124
125 let next = if self.check_noexpect(TokenKind::CloseDelim(Delimiter::Parenthesis)) {
126 None
127 } else {
128 Some(self.parse_expr()?)
129 };
130 self.expect(TokenKind::CloseDelim(Delimiter::Parenthesis))?;
131 let body = self.parse_stmt_boxed()?;
132 Ok(StmtKind::For { init: init.map(|init| self.alloc(init)), cond, next, body })
133 }
134
135 fn parse_stmt_try(&mut self) -> PResult<'sess, StmtTry<'ast>> {
137 let expr = self.parse_expr()?;
138 let mut clauses = SmallVec::<[_; 4]>::new();
139
140 let mut lo = self.token.span;
141 let returns = if self.eat_keyword(kw::Returns) {
142 self.parse_parameter_list(false, VarFlags::FUNCTION)?
143 } else {
144 Default::default()
145 };
146 let block = self.parse_block()?;
147 let span = lo.to(self.prev_token.span);
148 clauses.push(TryCatchClause { name: None, args: returns, block, span });
149
150 lo = self.token.span;
151 self.expect_keyword(kw::Catch)?;
152 loop {
153 let name = self.parse_ident_opt()?;
154 let args = if self.check(TokenKind::OpenDelim(Delimiter::Parenthesis)) {
155 self.parse_parameter_list(false, VarFlags::FUNCTION)?
156 } else {
157 Default::default()
158 };
159 let block = self.parse_block()?;
160 let span = lo.to(self.prev_token.span);
161 clauses.push(TryCatchClause { name, args, block, span });
162 lo = self.token.span;
163 if !self.eat_keyword(kw::Catch) {
164 break;
165 }
166 }
167
168 let clauses = self.alloc_smallvec(clauses);
169 Ok(StmtTry { expr, clauses })
170 }
171
172 fn parse_stmt_assembly(&mut self) -> PResult<'sess, StmtAssembly<'ast>> {
174 let dialect = self.parse_str_lit_opt();
175 let flags = if self.check(TokenKind::OpenDelim(Delimiter::Parenthesis)) {
176 self.parse_paren_comma_seq(false, Self::parse_str_lit)?
177 } else {
178 Default::default()
179 };
180 let block = self.parse_yul_block()?;
181 Ok(StmtAssembly { dialect, flags, block })
182 }
183
184 fn parse_simple_stmt(&mut self) -> PResult<'sess, Stmt<'ast>> {
186 let docs = self.parse_doc_comments();
187 self.parse_spanned(Self::parse_simple_stmt_kind).map(|(span, kind)| Stmt {
188 docs,
189 kind,
190 span,
191 })
192 }
193
194 fn parse_simple_stmt_kind(&mut self) -> PResult<'sess, StmtKind<'ast>> {
198 let lo = self.token.span;
199 if self.eat(TokenKind::OpenDelim(Delimiter::Parenthesis)) {
200 let mut empty_components = 0usize;
201 while self.eat(TokenKind::Comma) {
202 empty_components += 1;
203 }
204
205 let (statement_type, iap) = self.try_parse_iap()?;
206 match statement_type {
207 LookAheadInfo::VariableDeclaration => {
208 let mut variables = smallvec_repeat_none(empty_components);
209 let ty = iap.into_ty(self);
210 variables
211 .push(Some(self.parse_variable_definition_with(VarFlags::FUNCTION, ty)?));
212 self.parse_optional_items_seq_required(
213 Delimiter::Parenthesis,
214 &mut variables,
215 |this| this.parse_variable_definition(VarFlags::FUNCTION),
216 )?;
217 self.expect(TokenKind::Eq)?;
218 let expr = self.parse_expr()?;
219 Ok(StmtKind::DeclMulti(self.alloc_smallvec(variables), expr))
220 }
221 LookAheadInfo::Expression => {
222 let mut components = smallvec_repeat_none(empty_components);
223 let expr = iap.into_expr(self);
224 components.push(Some(self.parse_expr_with(expr)?));
225 self.parse_optional_items_seq_required(
226 Delimiter::Parenthesis,
227 &mut components,
228 Self::parse_expr,
229 )?;
230 let partially_parsed = Expr {
231 span: lo.to(self.prev_token.span),
232 kind: ExprKind::Tuple(self.alloc_smallvec(components)),
233 };
234 self.parse_expr_with(Some(self.alloc(partially_parsed))).map(StmtKind::Expr)
235 }
236 LookAheadInfo::IndexAccessStructure => unreachable!(),
237 }
238 } else {
239 let (statement_type, iap) = self.try_parse_iap()?;
240 match statement_type {
241 LookAheadInfo::VariableDeclaration => {
242 let ty = iap.into_ty(self);
243 self.parse_variable_definition_with(VarFlags::VAR, ty)
244 .map(|var| StmtKind::DeclSingle(self.alloc(var)))
245 }
246 LookAheadInfo::Expression => {
247 let expr = iap.into_expr(self);
248 self.parse_expr_with(expr).map(StmtKind::Expr)
249 }
250 LookAheadInfo::IndexAccessStructure => unreachable!(),
251 }
252 }
253 }
254
255 pub(super) fn parse_optional_items_seq<T>(
258 &mut self,
259 delim: Delimiter,
260 mut f: impl FnMut(&mut Self) -> PResult<'sess, T>,
261 ) -> PResult<'sess, Box<'ast, [Option<T>]>> {
262 self.expect(TokenKind::OpenDelim(delim))?;
263 let mut out = SmallVec::<[_; 8]>::new();
264 while self.eat(TokenKind::Comma) {
265 out.push(None);
266 }
267 if !self.check(TokenKind::CloseDelim(delim)) {
268 out.push(Some(f(self)?));
269 }
270 self.parse_optional_items_seq_required(delim, &mut out, f)
271 .map(|()| self.alloc_smallvec(out))
272 }
273
274 fn parse_optional_items_seq_required<T>(
275 &mut self,
276 delim: Delimiter,
277 out: &mut SmallVec<[Option<T>; 8]>,
278 mut f: impl FnMut(&mut Self) -> PResult<'sess, T>,
279 ) -> PResult<'sess, ()> {
280 let close = TokenKind::CloseDelim(delim);
281 while !self.eat(close) {
282 self.expect(TokenKind::Comma)?;
283 if self.check(TokenKind::Comma) || self.check(close) {
284 out.push(None);
285 } else {
286 out.push(Some(f(self)?));
287 }
288 }
289 Ok(())
290 }
291
292 fn parse_path_call(&mut self) -> PResult<'sess, (AstPath<'ast>, CallArgs<'ast>)> {
294 let path = self.parse_path()?;
295 let params = self.parse_call_args()?;
296 Ok((path, params))
297 }
298
299 fn try_parse_iap(&mut self) -> PResult<'sess, (LookAheadInfo, IndexAccessedPath<'ast>)> {
301 if let ty @ (LookAheadInfo::VariableDeclaration | LookAheadInfo::Expression) =
303 self.peek_statement_type()
304 {
305 return Ok((ty, IndexAccessedPath::default()));
306 }
307
308 let iap = self.parse_iap()?;
309 let ty = if self.token.is_non_reserved_ident(self.in_yul)
310 || self.token.is_location_specifier()
311 {
312 LookAheadInfo::VariableDeclaration
314 } else {
315 LookAheadInfo::Expression
316 };
317 Ok((ty, iap))
318 }
319
320 fn peek_statement_type(&mut self) -> LookAheadInfo {
321 if self.token.is_keyword_any(&[kw::Mapping, kw::Function]) {
323 return LookAheadInfo::VariableDeclaration;
324 }
325
326 if self.check_nr_ident() || self.check_elementary_type() {
327 let next = self.look_ahead(1);
328 if self.token.is_elementary_type() && next.is_ident_where(|id| id.name == kw::Payable) {
329 return LookAheadInfo::VariableDeclaration;
330 }
331 if next.is_non_reserved_ident(self.in_yul)
332 || next.is_location_specifier()
333 || next.is_mutability_specifier()
335 || next.is_visibility_specifier()
336 {
337 return LookAheadInfo::VariableDeclaration;
338 }
339 if matches!(next.kind, TokenKind::OpenDelim(Delimiter::Bracket) | TokenKind::Dot) {
340 return LookAheadInfo::IndexAccessStructure;
341 }
342 }
343 LookAheadInfo::Expression
344 }
345
346 fn parse_iap(&mut self) -> PResult<'sess, IndexAccessedPath<'ast>> {
347 let mut path = SmallVec::<[_; 4]>::new();
349 if self.check_nr_ident() {
350 path.push(IapKind::Member(self.parse_ident()?));
351 while self.eat(TokenKind::Dot) {
352 let id = self.ident_or_err(true)?;
353 if id.name != kw::Address && id.is_reserved(self.in_yul) {
354 self.expected_ident_found_err().emit();
355 }
356 self.bump(); path.push(IapKind::Member(id));
358 }
359 } else if self.check_elementary_type() {
360 let (span, kind) = self.parse_spanned(Self::parse_elementary_type)?;
361 path.push(IapKind::MemberTy(span, kind));
362 } else {
363 return self.unexpected();
364 }
365 let n_idents = path.len();
366
367 while self.check(TokenKind::OpenDelim(Delimiter::Bracket)) {
368 let (span, kind) = self.parse_spanned(Self::parse_expr_index_kind)?;
369 path.push(IapKind::Index(span, kind));
370 }
371
372 Ok(IndexAccessedPath { path, n_idents })
373 }
374}
375
376#[derive(Debug)]
377enum LookAheadInfo {
378 IndexAccessStructure,
380 VariableDeclaration,
381 Expression,
382}
383
384#[derive(Debug)]
385enum IapKind<'ast> {
386 Index(Span, IndexKind<'ast>),
388 Member(Ident),
390 MemberTy(Span, ElementaryType),
392}
393
394#[derive(Debug, Default)]
395struct IndexAccessedPath<'ast> {
396 path: SmallVec<[IapKind<'ast>; 4]>,
397 n_idents: usize,
399}
400
401impl<'ast> IndexAccessedPath<'ast> {
402 fn into_ty(self, parser: &mut Parser<'_, 'ast>) -> Option<Type<'ast>> {
403 let mut path = self.path.into_iter();
405 let first = path.next()?;
406
407 let mut ty = if let IapKind::MemberTy(span, kind) = first {
408 debug_assert_eq!(self.n_idents, 1);
409 Type { span, kind: TypeKind::Elementary(kind) }
410 } else {
411 debug_assert!(self.n_idents >= 1);
412 let first = std::iter::once(&first);
413 let path = first
414 .chain(path.as_slice())
415 .map(|x| match x {
416 IapKind::Member(id) => *id,
417 kind => unreachable!("{kind:?}"),
418 })
419 .take(self.n_idents);
420 let path = PathSlice::from_mut_slice(parser.arena.alloc_from_iter(path));
421 Type { span: path.span(), kind: TypeKind::Custom(path) }
422 };
423
424 for index in path.skip(self.n_idents - 1) {
425 let IapKind::Index(span, kind) = index else { panic!("parsed too much") };
426 let size = match kind {
427 IndexKind::Index(expr) => expr,
428 IndexKind::Range(l, r) => {
429 let msg = "expected array length, got range expression";
430 parser.dcx().err(msg).span(span).emit();
431 l.or(r)
432 }
433 };
434 let span = ty.span.to(span);
435 ty =
436 Type { span, kind: TypeKind::Array(parser.alloc(TypeArray { element: ty, size })) };
437 }
438
439 Some(ty)
440 }
441
442 fn into_expr(self, parser: &mut Parser<'_, 'ast>) -> Option<Box<'ast, Expr<'ast>>> {
443 let mut path = self.path.into_iter();
445
446 let mut expr = parser.alloc(match path.next()? {
447 IapKind::Member(ident) => Expr::from_ident(ident),
448 IapKind::MemberTy(span, kind) => {
449 Expr { span, kind: ExprKind::Type(Type { span, kind: TypeKind::Elementary(kind) }) }
450 }
451 IapKind::Index(..) => panic!("should not happen"),
452 });
453 for index in path {
454 expr = parser.alloc(match index {
455 IapKind::Member(ident) => {
456 Expr { span: expr.span.to(ident.span), kind: ExprKind::Member(expr, ident) }
457 }
458 IapKind::MemberTy(..) => panic!("should not happen"),
459 IapKind::Index(span, kind) => {
460 Expr { span: expr.span.to(span), kind: ExprKind::Index(expr, kind) }
461 }
462 });
463 }
464 Some(expr)
465 }
466}
467
468fn smallvec_repeat_none<T>(n: usize) -> SmallVec<[Option<T>; 8]> {
470 let mut v = SmallVec::with_capacity(n);
471 v.extend(std::iter::repeat_with(|| None).take(n));
472 v
473}
474
475#[cfg(test)]
476mod tests {
477 use super::*;
478 use solar_interface::{Result, Session, source_map::FileName};
479
480 #[test]
481 fn optional_items_seq() {
482 fn check(tests: &[(&str, &[Option<&str>])]) {
483 solar_interface::enter(|| -> Result {
484 let sess = Session::builder().with_test_emitter().build();
485 for (i, &(s, results)) in tests.iter().enumerate() {
486 let name = i.to_string();
487 let arena = Arena::new();
488 let mut parser =
489 Parser::from_source_code(&sess, &arena, FileName::Custom(name), s)?;
490
491 let list = parser
492 .parse_optional_items_seq(Delimiter::Parenthesis, Parser::parse_ident)
493 .map_err(|e| e.emit())
494 .unwrap_or_else(|_| panic!("src: {s:?}"));
495 sess.dcx.has_errors().unwrap();
496 let formatted: Vec<_> =
497 list.iter().map(|o| o.as_ref().map(|i| i.as_str())).collect();
498 assert_eq!(formatted.as_slice(), results, "{s:?}");
499 }
500 Ok(())
501 })
502 .unwrap();
503 }
504
505 check(&[
506 ("()", &[]),
507 ("(a)", &[Some("a")]),
508 ("(a,)", &[Some("a"), None]),
510 ("(,b)", &[None, Some("b")]),
511 ("(a,b)", &[Some("a"), Some("b")]),
512 ("(a,b,)", &[Some("a"), Some("b"), None]),
513 ("(a,,)", &[Some("a"), None, None]),
515 ("(a,b,)", &[Some("a"), Some("b"), None]),
516 ("(a,b,c)", &[Some("a"), Some("b"), Some("c")]),
517 ("(,b,c)", &[None, Some("b"), Some("c")]),
518 ("(,,c)", &[None, None, Some("c")]),
519 ("(a,,c)", &[Some("a"), None, Some("c")]),
520 ]);
521 }
522}