9 | 9 |
preprocessGrammar (Grammar productions) =
|
10 | 10 |
let
|
11 | 11 |
productions' = map (\(term, formals, expr) -> (term, formals, preprocessExpr expr)) productions
|
12 | |
productions'' = map (\(term, formals, expr) -> (term, formals, eliminateSingleAlts expr)) productions'
|
13 | 12 |
in
|
14 | |
Grammar productions''
|
|
13 |
Grammar productions'
|
15 | 14 |
|
16 | |
preprocessExpr :: Expr -> Expr
|
17 | |
preprocessExpr (Seq exprs) = Seq (preprocessSeq exprs) where
|
|
15 |
preprocessExpr = eliminateSingleAlts . coalesceConstraints . decorateLoops
|
|
16 |
|
|
17 |
|
|
18 |
--
|
|
19 |
-- Coalesce constraints
|
|
20 |
--
|
|
21 |
|
|
22 |
coalesceConstraints (Seq exprs) = Seq (cc exprs)
|
|
23 |
coalesceConstraints (Alt exprs) = Alt (map coalesceConstraints exprs)
|
|
24 |
coalesceConstraints (Loop expr cs) = Loop (coalesceConstraints expr) cs -- cs should be empty here actually, because decorateLoops comes later
|
|
25 |
coalesceConstraints other = other
|
|
26 |
|
|
27 |
cc :: [Expr] -> [Expr]
|
|
28 |
cc [] = []
|
|
29 |
cc (Constraint c1:Constraint c2:rest) =
|
|
30 |
cc ((Constraint $ Both c1 c2):rest)
|
|
31 |
cc (other:rest) = other:(cc rest)
|
|
32 |
|
|
33 |
--
|
|
34 |
-- Copy any constraints that immediately follow a loop, into the loop itself.
|
|
35 |
--
|
|
36 |
decorateLoops :: Expr -> Expr
|
|
37 |
decorateLoops (Seq exprs) = Seq (preprocessSeq exprs) where
|
18 | 38 |
preprocessSeq [] = []
|
19 | 39 |
preprocessSeq ((Loop expr _):rest) =
|
20 | 40 |
let
|
21 | |
expr' = preprocessExpr expr
|
|
41 |
expr' = decorateLoops expr
|
22 | 42 |
(constraints, rest') = absorbConstraints rest
|
23 | 43 |
in
|
24 | 44 |
(Loop expr' constraints):(preprocessSeq rest')
|
25 | 45 |
preprocessSeq (expr:rest) =
|
26 | |
(preprocessExpr expr):(preprocessSeq rest)
|
|
46 |
(decorateLoops expr):(preprocessSeq rest)
|
27 | 47 |
absorbConstraints :: [Expr] -> ([Constraint], [Expr])
|
28 | 48 |
absorbConstraints exprs =
|
29 | 49 |
let
|
|
36 | 56 |
extractConstraint (Constraint c) = c
|
37 | 57 |
in
|
38 | 58 |
(constraints', exprs')
|
39 | |
preprocessExpr (Alt exprs) = Alt (map preprocessExpr exprs)
|
40 | |
preprocessExpr (Loop expr _) = error "Cannot preprocess Loop that is not in Seq"
|
41 | |
preprocessExpr other = other
|
|
59 |
decorateLoops (Alt exprs) = Alt (map decorateLoops exprs)
|
|
60 |
decorateLoops (Loop expr _) = error "Cannot preprocess Loop that is not in Seq"
|
|
61 |
decorateLoops other = other
|
42 | 62 |
|
|
63 |
--
|
|
64 |
-- Any Alt with only a single child can be replaced by that single child.
|
|
65 |
--
|
43 | 66 |
eliminateSingleAlts (Alt [expr]) = eliminateSingleAlts expr
|
44 | 67 |
eliminateSingleAlts (Alt exprs) = Alt $ map (eliminateSingleAlts) exprs
|
45 | 68 |
eliminateSingleAlts (Seq exprs) = Seq $ map (eliminateSingleAlts) exprs
|