util.ml 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  1. open Printf
  2. open Lexing
  3. open Types
  4. (* Logging functions *)
  5. let prt_line = prerr_endline
  6. let prt_node node = prt_line (Stringify.node2str node)
  7. let log_line verbosity line =
  8. if args.verbose >= verbosity then prt_line line
  9. let log_node verbosity node =
  10. if args.verbose >= verbosity then prt_node node
  11. let dbg_line = log_line verbosity_debug
  12. let dbg_node = log_node verbosity_debug
  13. (* Variable generation *)
  14. let var_counter = ref 0
  15. let fresh_var prefix =
  16. var_counter := !var_counter + 1;
  17. prefix ^ "$" ^ string_of_int !var_counter
  18. (* Constants are marked by a double $$ for recognition during constant
  19. * propagation *)
  20. let fresh_const prefix = fresh_var (prefix ^ "$")
  21. let loc_from_lexpos pstart pend =
  22. let (fname, ystart, yend, xstart, xend) = (
  23. pstart.pos_fname,
  24. pstart.pos_lnum,
  25. pend.pos_lnum,
  26. (pstart.pos_cnum - pstart.pos_bol + 1),
  27. (pend.pos_cnum - pend.pos_bol)
  28. ) in
  29. if ystart = yend && xend < xstart then
  30. (fname, ystart, yend, xstart, xstart)
  31. else
  32. (fname, ystart, yend, xstart, xend)
  33. let rec flatten_blocks lst =
  34. let flatten = flatten_blocks in
  35. let rec trav = function
  36. | Block body ->
  37. Block (flatten body)
  38. | FunDef (export, ret_type, name, params, body, ann) ->
  39. FunDef (export, ret_type, name, flatten params, trav body, ann)
  40. | If (cond, body, ann) ->
  41. If (cond, trav body, ann)
  42. | IfElse (cond, tbody, fbody, ann) ->
  43. IfElse (cond, trav tbody, trav fbody, ann)
  44. | While (cond, body, ann) ->
  45. While (cond, trav body, ann)
  46. | DoWhile (cond, body, ann) ->
  47. DoWhile (cond, trav body, ann)
  48. | For (counter, start, stop, step, body, ann) ->
  49. For (counter, start, stop, step, trav body, ann)
  50. | VarDecs decs ->
  51. VarDecs (flatten decs)
  52. | LocalFuns decs ->
  53. LocalFuns (flatten decs)
  54. | node -> node
  55. in
  56. match lst with
  57. | [] -> []
  58. | Block nodes :: tl -> flatten nodes @ (flatten tl)
  59. | DummyNode :: tl -> flatten tl
  60. | hd :: tl -> trav hd :: (flatten tl)
  61. (* Default tree transformation
  62. * (node -> node) -> node -> node *)
  63. let transform_children trav node =
  64. let trav_all nodes = List.map trav nodes in
  65. match node with
  66. | Program (decls, ann) ->
  67. Program (flatten_blocks (trav_all decls), ann)
  68. | FunDec (ret_type, name, params, ann) ->
  69. FunDec (ret_type, name, trav_all params, ann)
  70. | FunDef (export, ret_type, name, params, body, ann) ->
  71. FunDef (export, ret_type, name, trav_all params, trav body, ann)
  72. | GlobalDec (ctype, name, ann) ->
  73. GlobalDec (ctype, name, ann)
  74. | GlobalDef (export, ctype, name, Some init, ann) ->
  75. GlobalDef (export, ctype, name, Some (trav init), ann)
  76. | VarDec (ctype, name, Some init, ann) ->
  77. VarDec (ctype, name, Some (trav init), ann)
  78. | Assign (name, None, value, ann) ->
  79. Assign (name, None, trav value, ann)
  80. | Assign (name, Some dims, value, ann) ->
  81. Assign (name, Some (trav_all dims), trav value, ann)
  82. | VarLet (dec, None, value, ann) ->
  83. VarLet (dec, None, trav value, ann)
  84. | VarLet (dec, Some dims, value, ann) ->
  85. VarLet (dec, Some (trav_all dims), trav value, ann)
  86. | Return (value, ann) ->
  87. Return (trav value, ann)
  88. | If (cond, body, ann) ->
  89. If (trav cond, trav body, ann)
  90. | IfElse (cond, true_body, false_body, ann) ->
  91. IfElse (trav cond, trav true_body, trav false_body, ann)
  92. | While (cond, body, ann) ->
  93. While (trav cond, trav body, ann)
  94. | DoWhile (cond, body, ann) ->
  95. DoWhile (trav cond, trav body, ann)
  96. | For (counter, start, stop, step, body, ann) ->
  97. For (counter, trav start, trav stop, trav step, trav body, ann)
  98. | Allocate (dec, dims, ann) ->
  99. Allocate (dec, trav_all dims, ann)
  100. | Expr value ->
  101. Expr (trav value)
  102. | Block (body) ->
  103. Block (trav_all body)
  104. | Monop (op, value, ann) ->
  105. Monop (op, trav value, ann)
  106. | Binop (op, left, right, ann) ->
  107. Binop (op, trav left, trav right, ann)
  108. | Cond (cond, true_expr, false_expr, ann) ->
  109. Cond (trav cond, trav true_expr, trav false_expr, ann)
  110. | TypeCast (ctype, value, ann) ->
  111. TypeCast (ctype, trav value, ann)
  112. | FunCall (name, args, ann) ->
  113. FunCall (name, trav_all args, ann)
  114. | Arg value ->
  115. Arg (trav value)
  116. | ArrayInit (value, dims) ->
  117. ArrayInit (trav value, dims)
  118. | ArrayScalar value ->
  119. ArrayScalar (trav value)
  120. | Var (dec, Some dims, ann) ->
  121. Var (dec, Some (trav_all dims), ann)
  122. | VarUse (dec, Some dims, ann) ->
  123. VarUse (dec, Some (trav_all dims), ann)
  124. | FunUse (dec, params, ann) ->
  125. FunUse (dec, trav_all params, ann)
  126. | VarDecs decs ->
  127. VarDecs (trav_all decs)
  128. | LocalFuns funs ->
  129. LocalFuns (trav_all funs)
  130. | _ -> node
  131. let annotate a = function
  132. | Program (decls, ann) ->
  133. Program (decls, a :: ann)
  134. | FunDec (ret_type, name, params, ann) ->
  135. FunDec (ret_type, name, params, a :: ann)
  136. | FunDef (export, ret_type, name, params, body, ann) ->
  137. FunDef (export, ret_type, name, params, body, a :: ann)
  138. | GlobalDec (ctype, name, ann) ->
  139. GlobalDec (ctype, name, a :: ann)
  140. | GlobalDef (export, ctype, name, init, ann) ->
  141. GlobalDef (export, ctype, name, init, a :: ann)
  142. | VarDec (ctype, name, init, ann) ->
  143. VarDec (ctype, name, init, a :: ann)
  144. | Assign (name, dims, value, ann) ->
  145. Assign (name, dims, value, a :: ann)
  146. | VarLet (dec, dims, value, ann) ->
  147. VarLet (dec, dims, value, a :: ann)
  148. | Return (value, ann) ->
  149. Return (value, a :: ann)
  150. | If (cond, body, ann) ->
  151. If (cond, body, a :: ann)
  152. | IfElse (cond, true_body, false_body, ann) ->
  153. IfElse (cond, true_body, false_body, a :: ann)
  154. | While (cond, body, ann) ->
  155. While (cond, body, a :: ann)
  156. | DoWhile (cond, body, ann) ->
  157. DoWhile (cond, body, a :: ann)
  158. | For (counter, start, stop, step, body, ann) ->
  159. For (counter, start, stop, step, body, a :: ann)
  160. | Allocate (dec, dims, ann) ->
  161. Allocate (dec, dims, a :: ann)
  162. | Monop (op, value, ann) ->
  163. Monop (op, value, a :: ann)
  164. | Binop (op, left, right, ann) ->
  165. Binop (op, left, right, a :: ann)
  166. | Cond (cond, true_expr, false_expr, ann) ->
  167. Cond (cond, true_expr, false_expr, a :: ann)
  168. | TypeCast (ctype, value, ann) ->
  169. TypeCast (ctype, value, a :: ann)
  170. | FunCall (name, args, ann) ->
  171. FunCall (name, args, a :: ann)
  172. | Arg value ->
  173. Arg (value)
  174. | Var (dec, dims, ann) ->
  175. Var (dec, dims, a :: ann)
  176. | VarUse (dec, dims, ann) ->
  177. VarUse (dec, dims, a :: ann)
  178. | FunUse (dec, params, ann) ->
  179. FunUse (dec, params, a :: ann)
  180. | Const (BoolVal value, ann) ->
  181. Const (BoolVal value, a :: ann)
  182. | Const (IntVal value, ann) ->
  183. Const (IntVal value, a :: ann)
  184. | Const (FloatVal value, ann) ->
  185. Const (FloatVal value, a :: ann)
  186. | ArrayConst (value, ann) ->
  187. ArrayConst (value, a :: ann)
  188. | Param (ctype, name, ann) ->
  189. Param (ctype, name, a :: ann)
  190. | Dim (name, ann) ->
  191. Dim (name, a :: ann)
  192. | _ -> raise InvalidNode
  193. let rec annof = function
  194. | Program (_, ann)
  195. | Param (_, _, ann)
  196. | Dim (_, ann)
  197. | FunDec (_, _, _, ann)
  198. | FunDef (_, _, _, _, _, ann)
  199. | GlobalDec (_, _, ann)
  200. | GlobalDef (_, _, _, _, ann)
  201. | VarDec (_, _, _, ann)
  202. | Assign (_, _, _, ann)
  203. | VarLet (_, _, _, ann)
  204. | Return (_, ann)
  205. | If (_, _, ann)
  206. | IfElse (_, _, _, ann)
  207. | While (_, _, ann)
  208. | DoWhile (_, _, ann)
  209. | For (_, _, _, _, _, ann)
  210. | Allocate (_, _, ann)
  211. | Const (BoolVal _, ann)
  212. | Const (IntVal _, ann)
  213. | Const (FloatVal _, ann)
  214. | ArrayConst (_, ann)
  215. | Var (_, _, ann)
  216. | Monop (_, _, ann)
  217. | Binop (_, _, _, ann)
  218. | Cond (_, _, _, ann)
  219. | TypeCast (_, _, ann)
  220. | VarUse (_, _, ann)
  221. | FunUse (_, _, ann)
  222. | FunCall (_, _, ann) -> ann
  223. | ArrayInit (value, _)
  224. | ArrayScalar value
  225. | Expr value
  226. | Arg value -> annof value
  227. | _ -> raise InvalidNode
  228. let locof node =
  229. let rec trav = function
  230. | [] -> noloc
  231. | Loc loc :: _ -> loc
  232. | _ :: tl -> trav tl
  233. in trav (annof node)
  234. let depthof node =
  235. let rec trav = function
  236. | [] ->
  237. prerr_string "cannot get depth for: ";
  238. prt_node node;
  239. raise InvalidNode
  240. | Depth depth :: _ -> depth
  241. | _ :: tl -> trav tl
  242. in trav (annof node)
  243. let indexof node =
  244. let rec trav = function
  245. | [] ->
  246. prerr_string "cannot get index for: ";
  247. prt_node node;
  248. raise InvalidNode
  249. | Index index :: _ -> index
  250. | _ :: tl -> trav tl
  251. in trav (annof node)
  252. let typeof = function
  253. (* Some nodes have their type as property *)
  254. | VarDec (ctype, _, _, _)
  255. | Param (ctype, _, _)
  256. | FunDec (ctype, _, _, _)
  257. | FunDef (_, ctype, _, _, _, _)
  258. | GlobalDec (ctype, _, _)
  259. | GlobalDef (_, ctype, _, _, _)
  260. | TypeCast (ctype, _, _)
  261. -> ctype
  262. (* Other nodes must be annotated during typechecking *)
  263. | node ->
  264. let rec trav = function
  265. | [] ->
  266. prerr_string "cannot get type for: ";
  267. prt_node node;
  268. raise InvalidNode
  269. | Type t :: _ -> t
  270. | _ :: tl -> trav tl
  271. in trav (annof node)
  272. let labelof node =
  273. let rec trav = function
  274. | [] ->
  275. prerr_string "cannot get label for: ";
  276. prt_node node;
  277. raise InvalidNode
  278. | LabelName label :: _ -> label
  279. | _ :: tl -> trav tl
  280. in trav (annof node)
  281. let const_type = function
  282. | BoolVal _ -> Bool
  283. | IntVal _ -> Int
  284. | FloatVal _ -> Float
  285. let prerr_loc (fname, ystart, yend, xstart, xend) =
  286. let file = open_in fname in
  287. (* skip lines until the first matched line *)
  288. for i = 1 to ystart - 1 do let _ = input_line file in () done;
  289. (* for each line in `loc`, print the source line with an underline *)
  290. for l = ystart to yend do
  291. let line = input_line file in
  292. let linewidth = String.length line in
  293. let left = if l = ystart then xstart else 1 in
  294. let right = if l = yend then xend else linewidth in
  295. if linewidth > 0 then (
  296. prerr_endline line;
  297. for i = 1 to left - 1 do prerr_char ' ' done;
  298. for i = left to right do prerr_char '^' done;
  299. prerr_endline "";
  300. )
  301. done;
  302. ()
  303. let prerr_loc_msg loc msg =
  304. if args.verbose >= 1 then (
  305. let (fname, ystart, yend, xstart, xend) = loc in
  306. if loc != noloc then (
  307. let line_s = if yend != ystart
  308. then sprintf "lines %d-%d" ystart yend
  309. else sprintf "line %d" ystart
  310. in
  311. let char_s = if xend != xstart || yend != ystart
  312. then sprintf "characters %d-%d" xstart xend
  313. else sprintf "character %d" xstart
  314. in
  315. eprintf "File \"%s\", %s, %s:\n" fname line_s char_s;
  316. );
  317. eprintf "%s\n" msg;
  318. if args.verbose >= 2 && loc != noloc then prerr_loc loc
  319. );
  320. ()
  321. let block_body = function
  322. | Block nodes -> nodes
  323. | _ -> raise InvalidNode
  324. let rec list_size = function
  325. | [] -> 0
  326. | hd :: tl -> 1 + (list_size tl)
  327. let basetypeof node = match typeof node with
  328. | Array (ctype, _)
  329. | FlatArray ctype
  330. | ctype -> ctype
  331. let array_depth = function
  332. | Array (_, dims) -> list_size dims
  333. | _ -> raise InvalidNode
  334. let nameof = function
  335. | GlobalDec (_, name, _)
  336. | GlobalDef (_, _, name, _, _)
  337. | FunDec (_, name, _, _)
  338. | FunDef (_, _, name, _, _, _)
  339. | VarDec (_, name, _, _)
  340. | Param (_, name, _)
  341. | Dim (name, _) -> name
  342. | _ -> raise InvalidNode
  343. let optmap f = function
  344. | None -> None
  345. | Some lst -> Some (List.map f lst)
  346. let optmapl f = function
  347. | None -> []
  348. | Some lst -> List.map f lst
  349. let mapi f lst =
  350. let rec trav i = function
  351. | [] -> []
  352. | hd :: tl -> f i hd :: (trav (i + 1) tl)
  353. in trav 0 lst
  354. let is_immediate_const const =
  355. if args.optimize then List.mem const immediate_consts else false