forked from esumii/min-caml
-
Notifications
You must be signed in to change notification settings - Fork 0
/
inline.ml
32 lines (28 loc) · 1.22 KB
/
inline.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
open KNormal
(* インライン展開する関数の最大サイズ (caml2html: inline_threshold) *)
let threshold = ref 0 (* Mainで-inlineオプションによりセットされる *)
let rec size = function
| IfEq(_, _, e1, e2) | IfLE(_, _, e1, e2)
| Let(_, e1, e2) | LetRec({ body = e1 }, e2) -> 1 + size e1 + size e2
| LetTuple(_, _, e) -> 1 + size e
| _ -> 1
let rec g env = function (* インライン展開ルーチン本体 (caml2html: inline_g) *)
| IfEq(x, y, e1, e2) -> IfEq(x, y, g env e1, g env e2)
| IfLE(x, y, e1, e2) -> IfLE(x, y, g env e1, g env e2)
| Let(xt, e1, e2) -> Let(xt, g env e1, g env e2)
| LetRec({ name = (x, t); args = yts; body = e1 }, e2) -> (* 関数定義の場合 (caml2html: inline_letrec) *)
let env = if size e1 > !threshold then env else M.add x (yts, e1) env in
LetRec({ name = (x, t); args = yts; body = g env e1}, g env e2)
| App(x, ys) when M.mem x env -> (* 関数適用の場合 (caml2html: inline_app) *)
let (zs, e) = M.find x env in
Format.eprintf "inlining %s@." x;
let env' =
List.fold_left2
(fun env' (z, t) y -> M.add z y env')
M.empty
zs
ys in
Alpha.g env' e
| LetTuple(xts, y, e) -> LetTuple(xts, y, g env e)
| e -> e
let f e = g M.empty e