forked from rescript-lang/rescript
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathjs_pass_flatten.ml
79 lines (71 loc) · 2.95 KB
/
js_pass_flatten.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
69
70
71
72
73
74
75
76
77
78
79
(* OCamlScript compiler
* Copyright (C) 2015-2016 Bloomberg Finance L.P.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, with linking exception;
* either version 2.1 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*)
(* Author: Hongbo Zhang *)
module E = J_helper.Exp
module S = J_helper.Stmt
let flatten_map =
object(self)
inherit Js_map.map as super
method! statement x =
match x.statement_desc with
| Exp ({expression_desc = Seq _; _} as v) ->
(S.block ( List.rev_map (self#statement) (Js_analyzer.rev_flatten_seq v )))
| Exp ({expression_desc = Cond(a,b,c); _} ) ->
(S.if_ a ([ self#statement (S.exp b) ])
~else_:([self#statement (S.exp c)]))
(* CHECK? Trick semantics difference *)
(* super#statement (S.if_ a ([ (\* self#statement *\) (S.exp b) ]) *)
(* ~else_:([self#statement (S.exp c)]) *)
(* ) *)
| Exp ({expression_desc = Bin(Eq, a, ({expression_desc = Seq _; _ } as v)); _} )
->
let block = Js_analyzer.rev_flatten_seq v in
begin match block with
| {statement_desc = Exp last_one ; _} :: rest_rev
->
S.block (Ext_list.rev_map_append (self#statement) rest_rev
[self#statement @@ S.exp (E.assign a last_one)])
(* super#statement *)
(* (S.block (List.rev_append rest_rev [S.exp (E.assign a last_one)])) *)
| _ ->
assert false
end
| Return ( {return_value = {expression_desc = Cond (a,b,c); _}; })
->
S.if_ a [S.return b] ~else_:[S.return c]
| Return ({return_value = {expression_desc = Seq _; _} as v}) ->
let block = Js_analyzer.rev_flatten_seq v in
begin match block with
| {statement_desc = Exp last_one ; _} :: rest_rev
-> super#statement (S.block (List.rev_append rest_rev [S.return last_one]))
| _ -> assert false
end
| Block [x]
->
self#statement x
| _ -> super#statement x
method! block b =
match b with
| {statement_desc = Block bs } :: rest ->
self#block ( bs @ rest)
| x::rest
->
self#statement x :: self#block rest
| [] -> []
end
let program ( x : J.program) = flatten_map # program x