-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathCodeGenerator.hs
220 lines (177 loc) · 7.55 KB
/
CodeGenerator.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
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
module CodeGenerator where
import Data.List
import Data.Maybe
import Parser
import qualified MsgTypeLexer as L
-- Generate the message type definition class M from a given message type definition
libPkg = "ial.base."
importMsg = "import " ++ libPkg ++ "Message;\n"
importTask = "import " ++ libPkg ++ "Task;\n"
importSet = "import java.util.Set;\n"
mtfBegin = "import ial.base.Message;\n\npublic class M {"
mtfEnd = "\n}"
mtfClassBegin1 = "\n\n public static class "
mtfClassBegin2 = " extends Message {"
mtfClassEnd = "\n }"
pn = "p"
mtfClassParam = "\n public final "
mtfCtorBegin = "\n\n public "
mtfCtorEnd = "\n }"
mtfCtorParam = "\n this."
-- Before the first "MsgTpyeName" it puts the "class" part
makeMsgTypeFile :: [L.Token] -> String
makeMsgTypeFile tokens = streamInitial tokens ""
streamInitial inp res =
case inp of
[] -> res ++ mtfBegin ++ mtfEnd
(L.NormalCharBlock b):inp' -> streamInitial inp' (res ++ b)
(L.MsgTypeName _):_ -> res ++ mtfBegin ++ (stream inp "" 0 "" "") ++ mtfEnd
-- paramCnt: number of next parameter
stream :: [L.Token] -> String -> Int -> String -> String -> String
stream inp res paramCnt ctorHead ctorBody =
case inp of
[] -> res ++ completeCurrentClass paramCnt ctorHead ctorBody
(L.NormalCharBlock b):inp' -> stream inp' (res ++ b) paramCnt ctorHead ctorBody
(L.MsgTypeName name):inp' -> stream inp' (res ++ completeCurrentClass paramCnt ctorHead ctorBody ++ mtfClassBegin1 ++ name ++ mtfClassBegin2) 1 (mtfCtorBegin ++ name ++ "(") ""
(L.Id pType):inp' -> stream inp' (res ++ mtfClassParam ++ pType ++ " " ++ pname ++ ";") (paramCnt+1) ctorHead' ctorBody'
where
pname = pn ++ (show paramCnt)
ctorHead' = ctorHead ++ pType ++ " " ++ pname ++ ", "
ctorBody' = ctorBody ++ mtfCtorParam ++ pname ++ " = " ++ pname ++ ";"
L.EOF:_ -> error "Unexpected token"
completeCurrentClass paramCnt ctorHead ctorBody =
if paramCnt == 0 -- no last message type definition
then ""
else ctorHead' ++ ") {" ++ ctorBody ++ mtfCtorEnd ++ mtfClassEnd
where ctorHead' =
if paramCnt == 1
then ctorHead
else take (length ctorHead - 2) ctorHead -- Remove trailing ", "
-- Generate a task class from a Task
-- Indents
indent = " "
indent2 = indent ++ indent
indent3 = indent2 ++ indent
indent4 = indent3 ++ indent
tfClassBegin = "\npublic class "
tfDestVar = indent ++ "private Set<Integer> $"
-- Guard and action methods
tfMsgParam = "(Message _m)"
tfVarMsgSrc = "int $src = _m.getSrc();\n"
-- Guard method
tfGuardMBegin = "public boolean " -- Needs no indent as takes it from source
tfGuardMName = "$Guard"
tfMSep = "\n\n"
-- Action method
tfActionMBegin = "public void " -- Needs no indent as takes it from source
tfActionMName = "$Action"
tfPname = "p"
tfMsgStart = indent2 ++ "{\n" ++ indent3 ++ "Message m_ = new M."
tfMsgSetSrc = indent3 ++ "m_.setSrc($ID);\n"
tfMsgSetDest = indent3 ++ "m_.setDest("
tfMsgEnd = indent3 ++ "send(m_);\n" ++ indent2 ++ "}\n"
tfReplyDestCode = "$src"
tfPrepareStart = indent ++ "public void prepare() {\n"
++ indent2 ++ "super.prepare();\n"
tfPrepareGetGroup = "getGroup"
tfPrepareAddIAP = "addIAP"
-- Generates Java class files from a list of Tasks (each will have destination ID sets for all tasks given)
-- Returns: [(filename, content)]
makeTaskFiles :: [Task] -> [(String,String)]
makeTaskFiles tasks =
let taskNames = map name tasks
in map (makeTaskFile taskNames) tasks
-- Generates a Java class file from a Task
-- taskNames: names of the tasks for which destination ID sets should be generated
makeTaskFile :: [String] -> Task -> (String, String)
makeTaskFile taskNames t = ((name t) ++ ".java",
(prelude t)
++ importSet
++ importTask
++ importMsg
++ tfClassBegin ++ (name t) ++ " extends Task {\n"
++ concat (map (\name -> tfDestVar ++ name ++ ";\n") taskNames) ++ "\n"
++ generatePrepareMethod taskNames (getInputMsgTypesFromTask t)
++ concat (map generateTaskElement (elements t))
++ "\n}"
)
generateTaskElement :: TaskElem -> String
generateTaskElement e =
case e of
NormalCharBlock s -> s
StmntSep -> ";"
IAP input actionElements -> let varBlock = generateVarBlock (msgT input) (params input) in
tfGuardMBegin ++ (msgT input) ++ tfGuardMName ++ tfMsgParam ++ " {\n"
++ varBlock
++ generateGuardExp (cond input) ++ "\n" ++ indent ++ "}"
++ tfMSep ++ indent
++ tfActionMBegin ++ (msgT input) ++ tfActionMName ++ tfMsgParam ++ " {\n"
++ varBlock
++ concat (map generateActionElement actionElements) ++ "\n" ++ indent ++ "}"
generateVarBlock msgType params =
indent2 ++ tfVarMsgSrc
++ generateMsgParamVars msgType params
generateMsgParamVars :: String -> [(String, String)] -> String
generateMsgParamVars msgType params = intercalate "\n" (map (generateMsgParamVar msgType) (zip params [1..])) ++ "\n"
generateMsgParamVar :: String -> ((String, String), Integer) -> String
generateMsgParamVar msgType ((t,v),n) =
indent2 ++ t ++ " " ++ v ++ " = " ++ " ((" ++ "M." ++ msgType ++ ") _m)." ++ tfPname ++ (show n) ++";"
generateGuardExp :: Maybe String -> String
generateGuardExp e = indent ++ indent ++ "return " ++ (fromMaybe "true" e) ++ ";"
generateActionElement :: ActionElem -> String
generateActionElement e =
case e of
CodeBlock s -> s
Sep -> ";"
Reply {rmsgT=msgType, paramCode=pCode} -> generateSend msgType pCode tfReplyDestCode
Send {smsgT=msgType, paramCode=pCode, toCode=destCode} -> generateSend msgType pCode destCode
generateSend :: String -> String -> String -> String
generateSend msgType paramCode destCode =
"\n" ++ tfMsgStart ++ msgType ++ "(" ++ paramCode ++ ";\n"
++ tfMsgSetSrc
++ tfMsgSetDest ++ destCode ++ ");\n"
++ tfMsgEnd
makeParamList :: [(String, String)] -> String
makeParamList ps = "(" ++ intercalate ", " (map (\(t,v) -> t ++ " " ++ v) ps) ++ ")"
generatePrepareMethod :: [String] -> [String] -> String
generatePrepareMethod taskNames msgTypes =
tfPrepareStart
++ concat (map makeDestVarInit taskNames)
++ concat (map makeAddIAP msgTypes)
++ indent ++ "}\n"
makeDestVarInit :: String -> String
makeDestVarInit name = indent2 ++ "$" ++ name ++ " = " ++ tfPrepareGetGroup ++ "(\"" ++ name ++ "\");\n"
makeAddIAP :: String -> String
makeAddIAP msgType =
let messageClass =
case msgType of
"init" -> "Message"
_ -> "M"
in indent2 ++ tfPrepareAddIAP ++ "(" ++ messageClass ++ "." ++ msgType ++ ".class, m -> " ++ msgType ++ tfGuardMName ++ "(m), m -> " ++ msgType ++ tfActionMName ++ "(m));\n"
getInputMsgTypesFromTask :: Task -> [String]
getInputMsgTypesFromTask t = catMaybes $ map getInputMsgTypeFromTaskElement (elements t)
getInputMsgTypeFromTaskElement :: TaskElem -> Maybe String
getInputMsgTypeFromTaskElement e =
case e of
IAP inp elems -> Just $ msgT inp
_ -> Nothing
-------------------------------------------------
-- Extract all message types from a list of tasks
-------------------------------------------------
getMsgTypesFromTasks :: [Task] -> [String]
getMsgTypesFromTasks ts = nub $ concat $ map getMsgTypesFromTask ts
getMsgTypesFromTask :: Task -> [String]
getMsgTypesFromTask t = nub $ concat $ map getMsgTypesFromIAP (elements t)
getMsgTypesFromIAP :: TaskElem -> [String]
getMsgTypesFromIAP e =
case e of
IAP inp elems ->
(msgT inp):(catMaybes $ map getMsgTypeFromActionElem elems)
_ -> []
getMsgTypeFromActionElem :: ActionElem -> Maybe String
getMsgTypeFromActionElem e =
case e of
Reply {rmsgT=t} -> Just t
Send {smsgT=t} -> Just t
_ -> Nothing
-- Creates Java source for a class containing all message type classes