-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathAST.hs
144 lines (110 loc) · 5.07 KB
/
AST.hs
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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
module AST where
-- Author: Aleksander Balicki
-- This module has the definiton of the AST, pretty-printer, some functions walking over the AST
import Data.List
import Data.Function
import Text.PrettyPrint
import Control.Monad.State
import qualified Data.Map as Map
-- | State monad with 'TaggerState', 'Tagger' here is needed for fast 'Condition' equality checking, profiler was showing that this is the function with the biggest running time
type Tagger a = State TaggerState a
-- | The state of the tagger, current highest tag, and a map of conditions to tags
type TaggerState = (Int, Map.Map Condition Int)
-- | Starting 'Tagger' state
tagStart :: TaggerState
tagStart = (0, Map.empty)
data Term = TmIf Condition Term [(Condition, Term)] Term
| TmDecision NewState Utterance
| TmCase Variable [(ValueSet, Term)] Term
deriving(Eq, Show)
type EqTag = Int
data Condition = TmEquals Variable Integer EqTag
| TmAnd [Condition] EqTag
| TmOr [Condition] EqTag
| TmTrue
| TmFalse
deriving(Show, Ord)
-- | Change an 'EqTag' of a 'Condition'
changeEqTag :: Condition -> Int -> Condition
changeEqTag (TmEquals a b _) t = TmEquals a b t
changeEqTag (TmAnd cnds _) t = TmAnd cnds t
changeEqTag (TmOr cnds _) t = TmOr cnds t
changeEqTag a _ = error $ "changeEqTag " ++ show a
-- | Check if there is already a tag for the condition, if not generate one
cachedCondition :: Condition -> Tagger Condition
cachedCondition cond = do
(num, m) <- get
case Map.lookup cond m of
Just i -> return $ changeEqTag cond i
Nothing -> do
put (num+1, Map.insert cond num m)
return $ changeEqTag cond num
instance Eq Condition where
TmEquals _ _ t == TmEquals _ _ t1 = t == t1
TmAnd _ t == TmAnd _ t1 = t == t1
TmOr _ t == TmOr _ t1 = t == t1
TmTrue == TmTrue = True
TmFalse == TmFalse = True
_ == _ = False
data NewState = TmCurrent | TmState Integer
deriving(Eq, Show)
data Variable = TmVar String
deriving(Eq, Show, Ord)
type StateNumber = Integer
type ValueSet = [Integer]
type Utterance = String
type Cost = Float
type Rule = ([StateNumber], Term)
type Character = [Rule]
type Env = [(String, Integer)]
-- | Pretty printer using pretty library
pp :: Character -> String
pp = render . ppChar
ppChar :: Character -> Doc
ppChar rules = parens . nest 1 . vcat . map ppRule $ rules
ppRule :: Rule -> Doc
ppRule (statenums, term) = parens $ ppStateNums statenums <+> ppTerm term
ppStateNums :: [StateNumber] -> Doc
ppStateNums = hsep . map integer
ppTerm :: Term -> Doc
ppTerm (TmIf cond term elseifs term2) = parens $ text "IF" <+> ppCond cond $$
nest 1 (ppTerm term
$$ parens (vcat $ map ppElseif elseifs) $$ ppTerm term2)
ppTerm (TmDecision (TmState newstate) utterance) = parens $ text "DECISION" <+> integer newstate <+> doubleQuotes (text utterance)
ppTerm (TmDecision TmCurrent utterance) = parens $ text "DECISION" <+> text "_" <+> doubleQuotes (text utterance)
ppTerm (TmCase var arms term) = parens $ text "CASE" <+> ppVar var <+> parens (vcat $ map ppArm arms) <+> nest 1 (ppTerm term)
ppVar :: Variable -> Doc
ppVar (TmVar v) = parens $ text "VAR" <+> doubleQuotes (text v)
ppCond :: Condition -> Doc
ppCond (TmEquals var i _) = parens $ text "EQUALS" <+> ppVar var <+> integer i
ppCond (TmAnd conds _) = parens $ text "AND" <+> vcat (map ppCond conds)
ppCond (TmOr conds _) = parens $ text "OR" <+> vcat (map ppCond conds)
ppCond a = error $ "ppCond" ++ show a
ppArm :: (ValueSet, Term) -> Doc
ppArm (vs, t) = parens $ text "ARM" <+> parens (hcat $ map integer vs) $$ ppTerm t
ppElseif :: (Condition, Term) -> Doc
ppElseif (c, t) = parens $ text "ELSEIF" <+> ppCond c $$ nest 1 (ppTerm t)
startingStates :: Character -> [StateNumber]
startingStates = concatMap fst
vars :: Character -> [String]
vars chr = map fst (varsVals chr)
type VarUniverse = (String, [Integer])
-- | Takes out all possible variables with their possible values from the AST
varsVals :: Character -> [VarUniverse]
varsVals ch = map (foldl (\(_,ts) (b, t) -> (b,t:ts)) (error "empty group", [])) $ groupBy (on (==) fst) $ sort $ varsVals' ch where
varsVals' :: Character -> [(String, Integer)]
varsVals' rules = vunion $ map varsValsRule rules
varsValsRule (_, term) = varsValsTerm term
varsValsTerm (TmIf cnd t1 elseifs t2) = vunion $ map varsValsTerm ([t1, t2] ++ map snd elseifs) ++ map varsValsCnd (cnd : map fst elseifs)
varsValsTerm (TmDecision _ _) = []
varsValsTerm (TmCase (TmVar v) arms t1) = map (\t -> (v,t)) (concatMap fst arms) ++ vunion (map varsValsTerm $ t1 : map snd arms)
varsCnd :: Condition -> [String]
varsCnd cnd = map fst (varsValsCnd cnd)
varsValsCnd :: Condition -> [(String, Integer)]
varsValsCnd (TmEquals (TmVar v) i _) = [(v, i)]
varsValsCnd (TmAnd cnds _) = vunion $ map varsValsCnd cnds
varsValsCnd (TmOr cnds _) = vunion $ map varsValsCnd cnds
varsValsCnd TmTrue = []
varsValsCnd TmFalse = []
vunion :: Eq a => [[a]] -> [a]
vunion = foldl' union []