1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
#[cfg(test)]
mod tests;
use crate::syntax::{
ast::{node::Break, Keyword, Punctuator},
lexer::TokenKind,
parser::{
cursor::{Cursor, SemicolonResult},
expression::LabelIdentifier,
AllowAwait, AllowYield, ParseError, TokenParser,
},
};
use boa_interner::Interner;
use boa_profiler::Profiler;
use std::io::Read;
#[derive(Debug, Clone, Copy)]
pub(super) struct BreakStatement {
allow_yield: AllowYield,
allow_await: AllowAwait,
}
impl BreakStatement {
pub(super) fn new<Y, A>(allow_yield: Y, allow_await: A) -> Self
where
Y: Into<AllowYield>,
A: Into<AllowAwait>,
{
Self {
allow_yield: allow_yield.into(),
allow_await: allow_await.into(),
}
}
}
impl<R> TokenParser<R> for BreakStatement
where
R: Read,
{
type Output = Break;
fn parse(
self,
cursor: &mut Cursor<R>,
interner: &mut Interner,
) -> Result<Self::Output, ParseError> {
let _timer = Profiler::global().start_event("BreakStatement", "Parsing");
cursor.expect((Keyword::Break, false), "break statement", interner)?;
let label = if let SemicolonResult::Found(tok) = cursor.peek_semicolon(interner)? {
match tok {
Some(tok) if tok.kind() == &TokenKind::Punctuator(Punctuator::Semicolon) => {
let _next = cursor.next(interner)?;
}
_ => {}
}
None
} else {
let label =
LabelIdentifier::new(self.allow_yield, self.allow_await).parse(cursor, interner)?;
cursor.expect_semicolon("break statement", interner)?;
Some(label)
};
Ok(Break::new(label))
}
}