dump.scm 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. ;;; Continuation-passing style (CPS) intermediate language (IL)
  2. ;; Copyright (C) 2013, 2014, 2015, 2017, 2018, 2019, 2020, 2021 Free Software Foundation, Inc.
  3. ;;;; This library is free software; you can redistribute it and/or
  4. ;;;; modify it under the terms of the GNU Lesser General Public
  5. ;;;; License as published by the Free Software Foundation; either
  6. ;;;; version 3 of the License, or (at your option) any later version.
  7. ;;;;
  8. ;;;; This library is distributed in the hope that it will be useful,
  9. ;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. ;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  11. ;;;; Lesser General Public License for more details.
  12. ;;;;
  13. ;;;; You should have received a copy of the GNU Lesser General Public
  14. ;;;; License along with this library; if not, write to the Free Software
  15. ;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  16. ;;; Commentary:
  17. ;;;
  18. ;;; Helper facilities for working with CPS.
  19. ;;;
  20. ;;; Code:
  21. (define-module (language cps dump)
  22. #:use-module (ice-9 format)
  23. #:use-module (ice-9 match)
  24. #:use-module (srfi srfi-1)
  25. #:use-module (language cps)
  26. #:use-module (language cps intset)
  27. #:use-module (language cps intmap)
  28. #:use-module (language cps graphs)
  29. #:use-module (language cps utils)
  30. #:export (dump))
  31. ;; ideas: unused vars print as _
  32. ;; print all labels
  33. ;; call bb headers with values
  34. ;; annotate blocks with available bindings? live bindings?
  35. ;; how to print calls...
  36. ;; dot graph
  37. (define (cont-successors cont)
  38. (match cont
  39. (($ $kargs _ _ term)
  40. (match term
  41. (($ $continue k) (list k))
  42. (($ $branch kf kt) (list kf kt))
  43. (($ $switch kf kt*) (cons kf kt*))
  44. (($ $prompt k kh) (list k kh))
  45. (($ $throw) '())))
  46. (($ $kclause _ kbody kalternate)
  47. (if kalternate
  48. (list kbody kalternate)
  49. (list kbody)))
  50. (($ $kfun src meta self ktail kentry)
  51. (list ktail kentry))
  52. (($ $kreceive arity kargs) (list kargs))
  53. (($ $ktail) '())))
  54. (define (compute-block-entries cps kfun body all-labels?)
  55. (if all-labels?
  56. body
  57. (let ((preds (compute-predecessors cps kfun #:labels body)))
  58. ;; Conts whose predecessor count is not 1 start blocks.
  59. (define (add-entry label blocks)
  60. (match (intmap-ref preds label)
  61. ((_) blocks)
  62. (_ (intset-add! blocks label))))
  63. ;; Continuations of branches start blocks.
  64. (define (add-exits label blocks)
  65. (fold1 (lambda (succ blocks)
  66. (intset-add! blocks succ))
  67. (match (cont-successors (intmap-ref cps label))
  68. ((_) '())
  69. (succs succs))
  70. blocks))
  71. (persistent-intset
  72. (intset-fold
  73. (lambda (label blocks)
  74. (add-exits label (add-entry label blocks)))
  75. body
  76. empty-intset)))))
  77. (define (collect-blocks cps entries)
  78. (define (collect-block entry)
  79. (let ((cont (intmap-ref cps entry)))
  80. (acons entry cont
  81. (match (cont-successors (intmap-ref cps entry))
  82. ((succ)
  83. (if (intset-ref entries succ)
  84. '()
  85. (collect-block succ)))
  86. (_ '())))))
  87. (persistent-intmap
  88. (intset-fold
  89. (lambda (start blocks)
  90. (intmap-add! blocks start (collect-block start)))
  91. entries
  92. empty-intmap)))
  93. (define (compute-block-succs blocks)
  94. (intmap-map (lambda (entry conts)
  95. (match conts
  96. (((_ . _) ... (exit . cont))
  97. (fold1 (lambda (succ succs)
  98. (intset-add succs succ))
  99. (cont-successors cont)
  100. empty-intset))))
  101. blocks))
  102. (define (dump-block cps port labelled-conts)
  103. (define (format-label label) (format #f "L~a" label))
  104. (define (format-name name) (if name (symbol->string name) "_"))
  105. (define (format-var var) (format #f "v~a" var))
  106. (define (format-loc src)
  107. (and src
  108. (format #f "~a:~a:~a"
  109. (or (assq-ref src 'filename) "<unknown>")
  110. (1+ (assq-ref src 'line))
  111. (assq-ref src 'column))))
  112. (define (arg-list strs) (string-join strs ", "))
  113. (define (false-if-empty str) (if (string-null? str) #f str))
  114. (define (format-arity arity)
  115. (match arity
  116. (($ $arity req opt rest kw aok?)
  117. (arg-list
  118. `(,@(map format-name req)
  119. ,@(map (lambda (name)
  120. (format #f "[~a]" (format-name name)))
  121. opt)
  122. ,@(map (match-lambda
  123. ((kw name var)
  124. (format #f "~a" kw)))
  125. kw)
  126. ,@(if aok? '("[#:allow-other-keys]") '())
  127. ,@(if rest
  128. (list (string-append (format-name rest) "..."))
  129. '()))))))
  130. (define (format-primcall op param args)
  131. (format #f "~a~@[[~s]~](~a)" op param (arg-list (map format-var args))))
  132. (define (format-exp exp)
  133. (match exp
  134. (($ $const val)
  135. (format #f "const ~s" val))
  136. (($ $prim name)
  137. (format #f "prim ~s" name))
  138. (($ $fun body)
  139. (format #f "fun ~a" (format-label body)))
  140. (($ $rec names syms funs)
  141. (format #f "rec(~a)" (arg-list (map format-exp funs))))
  142. (($ $const-fun label)
  143. (format #f "const-fun ~a" (format-label label)))
  144. (($ $code label)
  145. (format #f "code ~a" (format-label label)))
  146. (($ $call proc args)
  147. (format #f "call ~a(~a)"
  148. (format-var proc) (arg-list (map format-var args))))
  149. (($ $callk k proc args)
  150. (format #f "callk ~a(~a)" (format-label k)
  151. (arg-list
  152. (cons (if proc (format-var proc) "_")
  153. (map format-var args)))))
  154. (($ $primcall name param args)
  155. (format-primcall name param args))
  156. (($ $values args)
  157. (arg-list (map format-var args)))))
  158. (define (dump-annotation ann src)
  159. (when (or ann src)
  160. (format port "~45t ; ~@[~a ~]" ann)
  161. (when src
  162. (let* ((src (format-loc src))
  163. (col (- 80 4 (string-length src))))
  164. (format port "~vt at ~a" col src))))
  165. (newline port))
  166. (define (dump-definition src names vars fmt . args)
  167. (define (take formatter val)
  168. (cond
  169. ((not val) #f)
  170. ((string? val) (false-if-empty val))
  171. ((null? val) #f)
  172. (else (arg-list (map formatter val)))))
  173. (let ((names (take format-name names))
  174. (vars (take format-var vars)))
  175. (format port " ~@[~a := ~]~?" vars fmt args)
  176. (dump-annotation names src)))
  177. (define (dump-statement src ann fmt . args)
  178. (format port " ~?" fmt args)
  179. (dump-annotation (and ann (false-if-empty ann)) src))
  180. (define (dump-block-header label cont)
  181. (match cont
  182. (($ $kargs names vars)
  183. (format port "~a(~a):"
  184. (format-label label)
  185. (arg-list (map format-var vars)))
  186. (dump-annotation (false-if-empty (arg-list (map format-name names)))
  187. #f))
  188. (($ $ktail)
  189. (values))
  190. (($ $kfun src meta self ktail kentry)
  191. (let ((name (assq-ref meta 'name)))
  192. (format port "~a:" (format-label label))
  193. (dump-annotation name src)))
  194. ((or ($ $kreceive) ($ $kclause))
  195. (format port "~a:\n" (format-label label)))))
  196. (define (dump-block-body label cont)
  197. (match cont
  198. (($ $kargs _ _ ($ $continue k src exp))
  199. (match (intmap-ref cps k)
  200. (($ $kargs names vars)
  201. (dump-definition src names vars "~a" (format-exp exp)))
  202. (_
  203. (dump-definition src #f #f "~a" (format-exp exp)))))
  204. (($ $kreceive arity kargs)
  205. (match (intmap-ref cps kargs)
  206. (($ $kargs names vars)
  207. (dump-definition #f names vars
  208. "receive(~a)" (format-arity arity)))))
  209. (($ $ktail)
  210. (values))
  211. (($ $kclause arity kbody #f)
  212. (match (intmap-ref cps kbody)
  213. (($ $kargs names vars)
  214. (dump-definition #f names vars
  215. "receive(~a)" (format-arity arity)))))))
  216. (define (dump-block-exit label cont)
  217. (match cont
  218. (($ $kargs _ _ term)
  219. (match term
  220. (($ $continue k src exp)
  221. (match (intmap-ref cps k)
  222. (($ $ktail)
  223. (match exp
  224. (($ $values vals)
  225. (dump-statement src #f
  226. "return ~a" (arg-list (map format-var vals))))
  227. (_
  228. (dump-statement src #f
  229. "tail ~a" (format-exp exp)))))
  230. (_
  231. (dump-statement src #f
  232. "~a(~a)" (format-label k) (format-exp exp)))))
  233. (($ $branch kf kt src op param args)
  234. (dump-statement src #f
  235. "~a ? ~a() : ~a()"
  236. (format-primcall op param args)
  237. (format-label kt)
  238. (format-label kf)))
  239. (($ $switch kf kt* src arg)
  240. (dump-statement src #f
  241. "[~a]~a() or ~a()"
  242. (arg-list (map format-label kt*))
  243. (format-var arg)
  244. (format-label kf)))
  245. (($ $prompt k kh src escape? tag)
  246. (dump-statement src #f
  247. "~a(prompt(kh:~a,~a tag:~a)"
  248. (format-label k)
  249. (format-label kh)
  250. (if escape? ", escape-only" "")
  251. (format-var tag)))
  252. (($ $throw src op param args)
  253. (dump-statement src #f
  254. "throw ~a" (format-primcall op param args)))))
  255. (($ $kreceive arity kargs)
  256. (dump-statement #f #f
  257. "~a(receive(~a))"
  258. (format-label kargs)
  259. (format-arity arity)))
  260. (($ $kfun src meta self ktail kentry)
  261. (for-each (match-lambda
  262. ((k . v)
  263. (unless (eq? k 'name)
  264. (format port " meta: ~a: ~s\n" k v))))
  265. meta)
  266. ;; (format port " tail: ~a:\n" (format-label ktail))
  267. (when self
  268. (format port " ~a := self\n" (format-var self)))
  269. (format port " ~a(...)\n" (format-label kentry)))
  270. (($ $kclause arity kbody kalt)
  271. (dump-statement #f #f
  272. "~a(receive(~a))~@[or ~a()~]\n"
  273. (format-label kbody)
  274. (format-arity arity)
  275. (and=> kalt format-label)))
  276. (($ $ktail)
  277. (values))))
  278. (match labelled-conts
  279. (((label . cont) . _)
  280. (dump-block-header label cont)))
  281. (let lp ((labelled-conts labelled-conts))
  282. (match labelled-conts
  283. (((label . cont))
  284. (dump-block-exit label cont))
  285. (((label . cont) . labelled-conts)
  286. (dump-block-body label cont)
  287. (lp labelled-conts)))))
  288. (define (dump-function cps port kfun body all-labels?)
  289. (define entries (compute-block-entries cps kfun body all-labels?))
  290. (define blocks (collect-blocks cps entries))
  291. (define block-succs (compute-block-succs blocks))
  292. (define block-order (compute-reverse-post-order block-succs kfun))
  293. (for-each (lambda (entry)
  294. (dump-block cps port (intmap-ref blocks entry)))
  295. block-order)
  296. (values))
  297. (define* (dump cps #:key
  298. (port (current-output-port))
  299. (entry (intmap-next cps))
  300. (all-labels? #f))
  301. (let ((functions (compute-reachable-functions cps entry)))
  302. (intmap-fold (lambda (kfun body)
  303. (unless (eqv? kfun entry) (newline port))
  304. (dump-function cps port kfun body all-labels?))
  305. functions)))