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
use ast::*;
pub fn factor(rule: Rule) -> Rule {
match rule {
Rule { name, ty, expr } => Rule {
name,
ty,
expr: expr.map_top_down(|expr| {
match expr {
Expr::Choice(lhs, rhs) => match (*lhs, *rhs) {
(Expr::Seq(l1, r1), Expr::Seq(l2, r2)) => {
if l1 == l2 {
Expr::Seq(l1, Box::new(Expr::Choice(r1, r2)))
} else {
Expr::Choice(
Box::new(Expr::Seq(l1, r1)),
Box::new(Expr::Seq(l2, r2)),
)
}
}
(lhs, rhs) => Expr::Choice(Box::new(lhs), Box::new(rhs)),
},
expr => expr,
}
}),
},
}
}