-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathemit.ml
68 lines (65 loc) · 2.48 KB
/
emit.ml
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
open Syntax
let rec show_e e = match e with
| Unit -> "undefined"
| Var "String.length" -> "String.length_"
| Var s -> s
| Str s -> Printf.sprintf "\"%s\"" s
| Int s -> Printf.sprintf "%d" s
| Float f -> Printf.sprintf "%f" f
| Bool b -> Printf.sprintf "%b" b
| Fun(ss, e) ->
Printf.sprintf "function(%s){return %s;}"
(String.concat "," ss) (show_e e)
| Rec ses ->
let rec f se = match se with
| (s,e) -> s ^ ":" ^ show_e e
in
let ss = List.map f ses in
Printf.sprintf "{%s}" (String.concat "," ss)
| App(e1,e2) ->
Printf.sprintf "%s(%s)"
(show_e e1) (String.concat "," (List.map show_e e2))
| Pre(op,e1) ->
Printf.sprintf "(%s %s)" op (show_e e1)
| Bin(e1,op,e2) ->
Printf.sprintf "(%s %s %s)" (show_e e1) op (show_e e2)
| If(e1,e2,e3) ->
Printf.sprintf "(%s ? %s : %s)" (show_e e1) (show_e e2) (show_e e3)
| Let(s,e1,e2) ->
Printf.sprintf "(function(%s){return %s;}(%s))"
s (show_e e2) (show_e e1)
| LetRec(s,e1,e2) ->
Printf.sprintf "(function(){var %s=%s;return %s;}())"
s (show_e e1) (show_e e2)
| Get(e1,e2) ->
Printf.sprintf "%s[%s]" (show_e e1) (show_e e2)
| Put(e1,e2, e3) ->
Printf.sprintf "%s[%s] = %s" (show_e e1) (show_e e2) (show_e e3)
| Raise(s) -> Printf.sprintf "(function(){throw \"%s\";}())" s
| Open(s,e) -> Printf.sprintf "(function(){with(%s){return %s;}}())" s (show_e e)
| Array(_, _) | Tuple(_) | CApp(_, _) -> assert false
| Match(e1,es) ->
let es = List.map(function
| (Var "_",_,e2) ->
"default: return " ^ (show_e e2) ^ ";"
| (e1,_,e2) ->
"case " ^ (show_e e1) ^ ": return " ^ (show_e e2) ^ ";"
) es in
Printf.sprintf "(function(){switch(%s) {%s}}())"
(show_e e1) (String.concat "" es)
let rec findopen = function
| Unit | Raise _ | Int _ | Var _ | Str _ | Float _ | Bool _ -> []
| Fun(_, e1) | Pre(_,e1) | CApp(_,e1) ->
findopen e1
| App(e,es) -> (findopen e) @ List.flatten (List.map findopen es)
| Rec(ses) -> List.flatten(List.map(fun(s,e)->findopen e) ses)
| Array(e1,e2) | Bin(e1,_,e2) | Let(_,e1,e2) | LetRec(_,e1,e2) | Get(e1,e2) ->
(findopen e1) @ (findopen e2)
| Put(e1,e2, e3) | If(e1,e2,e3) ->
(findopen e1) @ (findopen e2) @ (findopen e3)
| Tuple es -> List.flatten(List.map findopen es)
| Open(s,e) -> s :: (findopen e)
let f oc ast =
Format.eprintf "generating javascript...@.";
Printf.fprintf oc "%s\n" (Syntax.read_all "../libs/lib.js");
Printf.fprintf oc "%s\n" (show_e (To_if.f ast))