reify-primitives.scm 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607
  1. ;;; Continuation-passing style (CPS) intermediate language (IL)
  2. ;; Copyright (C) 2013-2021,2023 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. ;;; A pass to reify lone $prim's that were never folded into a
  19. ;;; $primcall, and $primcall's to primitives that don't have a
  20. ;;; corresponding VM op.
  21. ;;;
  22. ;;; Code:
  23. (define-module (language cps guile-vm reify-primitives)
  24. #:use-module (ice-9 match)
  25. #:use-module ((language tree-il primitives)
  26. #:select ((primitive-module . tree-il:primitive-module)))
  27. #:use-module (language cps)
  28. #:use-module (language cps utils)
  29. #:use-module (language cps with-cps)
  30. #:use-module (language cps intmap)
  31. #:use-module (language bytecode)
  32. #:use-module (system base target)
  33. #:use-module (system base types internal)
  34. #:export (reify-primitives))
  35. (define (primitive-module name)
  36. (tree-il:primitive-module name))
  37. (define (primitive-ref cps name k src)
  38. (with-cps cps
  39. (letv box)
  40. (letk kbox ($kargs ('box) (box)
  41. ($continue k src
  42. ($primcall 'scm-ref/immediate '(box . 1) (box)))))
  43. ($ ((hashq-ref *ephemeral-reifiers* 'cached-module-box)
  44. kbox src (list (primitive-module name) name #f #t) '()))))
  45. (define (builtin-ref cps idx k src)
  46. (with-cps cps
  47. (build-term
  48. ($continue k src ($primcall 'builtin-ref idx ())))))
  49. (define (reify-clause cps)
  50. (with-cps cps
  51. (let$ body
  52. (with-cps-constants ((wna 'wrong-number-of-args)
  53. (args '(#f "Wrong number of arguments" () #f)))
  54. (build-term ($throw #f 'throw #f (wna args)))))
  55. (letk kbody ($kargs () () ,body))
  56. (letk kclause ($kclause ('() '() #f '() #f) kbody #f))
  57. kclause))
  58. (define (wrap-unary cps k src wrap unwrap op param a)
  59. (with-cps cps
  60. (letv a* res*)
  61. (letk kres ($kargs ('res*) (res*)
  62. ($continue k src
  63. ($primcall 'u64->s64 #f (res*)))))
  64. (letk ka ($kargs ('a*) (a*)
  65. ($continue kres src
  66. ($primcall op param (a*)))))
  67. (build-term
  68. ($continue ka src
  69. ($primcall 's64->u64 #f (a))))))
  70. (define (wrap-binary cps k src wrap unwrap op param a b)
  71. (with-cps cps
  72. (letv a* b* res*)
  73. (letk kres ($kargs ('res*) (res*)
  74. ($continue k src
  75. ($primcall 'u64->s64 #f (res*)))))
  76. (letk kb ($kargs ('b*) (b*)
  77. ($continue kres src
  78. ($primcall op param (a* b*)))))
  79. (letk ka ($kargs ('a*) (a*)
  80. ($continue kb src
  81. ($primcall 's64->u64 #f (b)))))
  82. (build-term
  83. ($continue ka src
  84. ($primcall 's64->u64 #f (a))))))
  85. (define (wrap-binary/exp cps k src wrap unwrap op param a b-exp)
  86. (with-cps cps
  87. (letv a* b* res*)
  88. (letk kres ($kargs ('res*) (res*)
  89. ($continue k src
  90. ($primcall 'u64->s64 #f (res*)))))
  91. (letk kb ($kargs ('b*) (b*)
  92. ($continue kres src
  93. ($primcall op param (a* b*)))))
  94. (letk ka ($kargs ('a*) (a*)
  95. ($continue kb src ,b-exp)))
  96. (build-term
  97. ($continue ka src
  98. ($primcall 's64->u64 #f (a))))))
  99. ;; Primitives that we need to remove.
  100. (define *ephemeral-reifiers* (make-hash-table))
  101. (define-syntax-rule (define-ephemeral (name cps k src param arg ...)
  102. . body)
  103. (hashq-set! *ephemeral-reifiers* 'name
  104. (lambda (cps k src param args)
  105. (match args ((arg ...) (let () . body))))))
  106. (define-ephemeral (fadd/immediate cps k src param a)
  107. (with-cps cps
  108. (letv b)
  109. (letk kb ($kargs ('b) (b)
  110. ($continue k src
  111. ($primcall 'fadd #f (a b)))))
  112. (build-term
  113. ($continue kb src
  114. ($primcall 'load-f64 param ())))))
  115. (define-syntax-rule (define-binary-signed-ephemeral name uname)
  116. (define-ephemeral (name cps k src param a b)
  117. (wrap-binary cps k src 's64->u64 'u64->s64 'uname #f a b)))
  118. (define-binary-signed-ephemeral sadd uadd)
  119. (define-binary-signed-ephemeral ssub usub)
  120. (define-binary-signed-ephemeral smul umul)
  121. (define-syntax-rule (define-binary-signed-ephemeral/imm name/imm
  122. uname/imm uname)
  123. (define-ephemeral (name/imm cps k src param a)
  124. (if (and (exact-integer? param) (<= 0 param 255))
  125. (wrap-unary cps k src 's64->u64 'u64->s64 'uname/imm param a)
  126. (wrap-binary/exp cps k src 's64->u64 'u64->s64 'uname #f a
  127. (let ((param (logand param (1- (ash 1 64)))))
  128. (build-exp ($primcall 'load-u64 param ())))))))
  129. (define-binary-signed-ephemeral/imm sadd/immediate uadd/immediate uadd)
  130. (define-binary-signed-ephemeral/imm ssub/immediate usub/immediate usub)
  131. (define-binary-signed-ephemeral/imm smul/immediate umul/immediate umul)
  132. (define-ephemeral (slsh cps k src param a b)
  133. (wrap-binary/exp cps k src 's64->u64 'u64->s64 'ulsh #f a
  134. (build-exp ($values (b)))))
  135. (define-ephemeral (slsh/immediate cps k src param a)
  136. (wrap-unary cps k src 's64->u64 'u64->s64 'ulsh/immediate param a))
  137. (define (reify-lookup cps src mod-var name assert-bound? have-var)
  138. (with-cps cps
  139. (letv name-var var)
  140. (let$ body (have-var var))
  141. (letk kres ($kargs ('var) (var) ,body))
  142. (letk klookup ($kargs ('name) (name-var)
  143. ($continue kres src
  144. ($primcall (if assert-bound? 'lookup-bound 'lookup) #f
  145. (mod-var name-var)))))
  146. (build-term ($continue klookup src ($const name)))))
  147. (define (reify-resolve-module cps k src module public?)
  148. (with-cps cps
  149. (letv mod-name)
  150. (letk kresolve
  151. ($kargs ('mod-name) (mod-name)
  152. ($continue k src
  153. ($primcall 'resolve-module public? (mod-name)))))
  154. (build-term
  155. ($continue kresolve src ($const module)))))
  156. (define-ephemeral (cached-module-box cps k src param)
  157. (match param
  158. ((module name public? #t)
  159. (let ((cache-key param))
  160. (with-cps cps
  161. (letv cached var)
  162. (letk k* ($kargs () () ($continue k src ($values (var)))))
  163. (letk kcache ($kargs ('var) (var)
  164. ($continue k* src
  165. ($primcall 'cache-set! cache-key (var)))))
  166. (letk kinit ($kargs () ()
  167. ($continue kcache src
  168. ($primcall (if public?
  169. 'lookup-bound-public
  170. 'lookup-bound-private)
  171. (list module name) ()))))
  172. (letk kok ($kargs () ()
  173. ($continue k src ($values (cached)))))
  174. (letk ktest
  175. ($kargs ('cached) (cached)
  176. ($branch kinit kok src 'heap-object? #f (cached))))
  177. (build-term
  178. ($continue ktest src
  179. ($primcall 'cache-ref cache-key ()))))))
  180. ((module name public? bound?)
  181. (let ((cache-key param))
  182. (with-cps cps
  183. (letv mod cached)
  184. (let$ lookup
  185. (reify-lookup
  186. src mod name bound?
  187. (lambda (cps var)
  188. (with-cps cps
  189. (letk k* ($kargs () () ($continue k src ($values (var)))))
  190. (build-term
  191. ($continue k* src
  192. ($primcall 'cache-set! cache-key (var))))))))
  193. (letk kmod ($kargs ('mod) (mod) ,lookup))
  194. (let$ module (reify-resolve-module kmod src module public?))
  195. (letk kinit ($kargs () () ,module))
  196. (letk kok ($kargs () () ($continue k src ($values (cached)))))
  197. (letk ktest
  198. ($kargs ('cached) (cached)
  199. ($branch kinit kok src 'heap-object? #f (cached))))
  200. (build-term
  201. ($continue ktest src
  202. ($primcall 'cache-ref cache-key ()))))))))
  203. (define-ephemeral (cache-current-module! cps k src param mod)
  204. (match param
  205. ((scope)
  206. (with-cps cps
  207. (build-term
  208. ($continue k src
  209. ($primcall 'cache-set! scope (mod))))))))
  210. (define-ephemeral (cached-toplevel-box cps k src param)
  211. (match param
  212. ((scope name bound?)
  213. (let ((cache-key (cons scope name)))
  214. (with-cps cps
  215. (letv mod cached)
  216. (let$ lookup
  217. (reify-lookup
  218. src mod name bound?
  219. (lambda (cps var)
  220. (with-cps cps
  221. (letk k* ($kargs () () ($continue k src ($values (var)))))
  222. (build-term
  223. ($continue k* src
  224. ($primcall 'cache-set! cache-key (var))))))))
  225. (letk kmod ($kargs ('mod) (mod) ,lookup))
  226. (letk kinit ($kargs () ()
  227. ($continue kmod src ($primcall 'cache-ref scope ()))))
  228. (letk kok ($kargs () () ($continue k src ($values (cached)))))
  229. (letk ktest
  230. ($kargs ('cached) (cached)
  231. ($branch kinit kok src 'heap-object? #f (cached))))
  232. (build-term
  233. ($continue ktest src
  234. ($primcall 'cache-ref cache-key ()))))))))
  235. (define-ephemeral (mul/immediate cps k src param a)
  236. (with-cps cps
  237. (letv imm)
  238. (letk kop ($kargs ('imm) (imm)
  239. ($continue k src ($primcall 'mul #f (a imm)))))
  240. (build-term
  241. ($continue kop src ($const param)))))
  242. (define-ephemeral (logand/immediate cps k src param a)
  243. (with-cps cps
  244. (letv imm)
  245. (letk kop ($kargs ('imm) (imm)
  246. ($continue k src ($primcall 'logand #f (a imm)))))
  247. (build-term
  248. ($continue kop src ($const param)))))
  249. ;; FIXME: Instead of having to check this, instead every primcall that's
  250. ;; not ephemeral should be handled by compile-bytecode.
  251. (define (compute-known-primitives)
  252. (define *macro-instructions*
  253. '(add
  254. add/immediate
  255. sub
  256. sub/immediate
  257. mul
  258. div
  259. quo
  260. rem
  261. mod
  262. inexact
  263. sqrt
  264. abs
  265. floor
  266. ceiling
  267. sin
  268. cos
  269. tan
  270. asin
  271. acos
  272. atan
  273. atan2
  274. fsqrt
  275. fabs
  276. ffloor
  277. fceiling
  278. fsin
  279. fcos
  280. ftan
  281. fasin
  282. facos
  283. fatan
  284. fatan2
  285. logand
  286. logior
  287. logxor
  288. logsub
  289. string-set!
  290. string->number
  291. string->symbol
  292. symbol->keyword
  293. symbol->string
  294. string-utf8-length string->utf8 utf8->string
  295. class-of
  296. scm->f64
  297. s64->u64 s64->scm scm->s64
  298. u64->s64 u64->scm scm->u64 scm->u64/truncate
  299. wind unwind
  300. push-fluid pop-fluid fluid-ref fluid-set!
  301. push-dynamic-state pop-dynamic-state
  302. lsh rsh lsh/immediate rsh/immediate
  303. cache-ref cache-set!
  304. current-module resolve-module
  305. module-variable define!
  306. lookup lookup-bound lookup-bound-public lookup-bound-private))
  307. (let ((table (make-hash-table)))
  308. (for-each
  309. (match-lambda ((inst . _) (hashq-set! table inst #t)))
  310. (instruction-list))
  311. (for-each
  312. (lambda (prim) (hashq-set! table prim #t))
  313. *macro-instructions*)
  314. table))
  315. (define *known-primitives* (delay (compute-known-primitives)))
  316. (define (known-primitive? name)
  317. "Is @var{name} a primitive that can be lowered to bytecode?"
  318. (hashq-ref (force *known-primitives*) name))
  319. (define (reify-primitives cps)
  320. (define (visit-cont label cont cps)
  321. (define (resolve-prim cps name k src)
  322. (cond
  323. ((builtin-name->index name)
  324. => (lambda (idx) (builtin-ref cps idx k src)))
  325. (else
  326. (primitive-ref cps name k src))))
  327. (match cont
  328. (($ $kfun src meta self tail #f)
  329. (with-cps cps
  330. (let$ clause (reify-clause))
  331. (setk label ($kfun src meta self tail clause))))
  332. (($ $kargs names vars ($ $throw src op param args))
  333. (match op
  334. ('raise-type-error
  335. (match (cons param args)
  336. ((#(proc-name pos what) val)
  337. (define msg
  338. (format #f
  339. "Wrong type argument in position ~a (expecting ~a): ~~S"
  340. pos what))
  341. (with-cps cps
  342. (setk label
  343. ($kargs names vars
  344. ($throw src 'throw/value+data
  345. (vector 'wrong-type-arg proc-name msg)
  346. (val))))))))
  347. ('raise-range-error
  348. (match (cons param args)
  349. ((#(proc-name pos) val)
  350. (define msg
  351. (format #f "Argument ~a out of range: ~~S" pos))
  352. (with-cps cps
  353. (setk label
  354. ($kargs names vars
  355. ($throw src 'throw/value+data
  356. (vector 'out-of-range proc-name msg)
  357. (val))))))))
  358. ('raise-arity-error
  359. (match (cons param args)
  360. ((#(proc-name) val)
  361. (define msg "Wrong number of arguments to ~A")
  362. (with-cps cps
  363. (setk label
  364. ($kargs names vars
  365. ($throw src 'throw/value
  366. (vector 'wrong-number-of-args proc-name msg)
  367. (val))))))))
  368. ('raise-exception
  369. (match (cons param args)
  370. ((#f exn)
  371. (with-cps cps
  372. (letv ignored prim)
  373. (letk kdie ($kargs (#f) (ignored)
  374. ($throw src 'unreachable #f ())))
  375. (letk kret ($kreceive '() 'rest kdie))
  376. (letk kcall ($kargs ('raise-exception) (prim)
  377. ($continue kret src ($call prim (exn)))))
  378. (let$ body (resolve-prim 'raise-exception kcall src))
  379. (setk label ($kargs names vars ,body))))))
  380. ((or 'unreachable 'throw 'throw/value 'throw/value+data) cps)))
  381. (($ $kargs names vars ($ $continue k src ($ $prim name)))
  382. (with-cps cps
  383. (let$ body (resolve-prim name k src))
  384. (setk label ($kargs names vars ,body))))
  385. (($ $kargs names vars
  386. ($ $continue k src ($ $primcall 'call-thunk/no-inline #f (proc))))
  387. (with-cps cps
  388. (setk label ($kargs names vars ($continue k src ($call proc ()))))))
  389. (($ $kargs names vars
  390. ($ $continue k src ($ $primcall 'u64->scm/unlikely #f (u64))))
  391. (with-cps cps
  392. (setk label ($kargs names vars
  393. ($continue k src ($primcall 'u64->scm #f (u64)))))))
  394. (($ $kargs names vars
  395. ($ $continue k src ($ $primcall 's64->scm/unlikely #f (s64))))
  396. (with-cps cps
  397. (setk label ($kargs names vars
  398. ($continue k src ($primcall 's64->scm #f (s64)))))))
  399. (($ $kargs names vars
  400. ($ $continue k src ($ $primcall 'tag-fixnum/unlikely #f (s64))))
  401. (with-cps cps
  402. (setk label ($kargs names vars
  403. ($continue k src ($primcall 'tag-fixnum #f (s64)))))))
  404. (($ $kargs names vars
  405. ($ $continue k src ($ $primcall 'load-const/unlikely val ())))
  406. (with-cps cps
  407. (setk label ($kargs names vars ($continue k src ($const val))))))
  408. (($ $kargs names vars
  409. ($ $continue k src
  410. ($ $primcall (or 'assume-u64 'assume-s64) (lo . hi) (val))))
  411. (with-cps cps
  412. (setk label ($kargs names vars
  413. ($continue k src ($values (val)))))))
  414. (($ $kargs names vars ($ $continue k src ($ $primcall name param args)))
  415. (cond
  416. ((hashq-ref *ephemeral-reifiers* name)
  417. => (lambda (reify)
  418. (with-cps cps
  419. (let$ body (reify k src param args))
  420. (setk label ($kargs names vars ,body)))))
  421. ((known-primitive? name)
  422. ;; Assume arities are correct.
  423. (let ()
  424. (define (u6? val) (and (exact-integer? val) (<= 0 val 63)))
  425. (define (u8? val) (and (exact-integer? val) (<= 0 val 255)))
  426. (define-syntax-rule (reify-constants
  427. wrap
  428. ((op (pred? c) in ...) (op* out ...))
  429. ...
  430. (_ default))
  431. (match name
  432. ('op
  433. (if (pred? param)
  434. cps
  435. (match args
  436. ((in ...)
  437. (with-cps cps
  438. (letv c)
  439. (letk kconst ($kargs ('c) (c)
  440. ($continue k src
  441. ($primcall 'op* #f (out ...)))))
  442. (setk label
  443. ($kargs names vars
  444. ($continue kconst src wrap))))))))
  445. ...
  446. (_ default)))
  447. (define-syntax-rule (reify-scm-constants clause ...)
  448. (reify-constants ($const param) clause ...))
  449. (define-syntax-rule (reify-u64-constants clause ...)
  450. (reify-constants ($primcall 'load-u64 param ()) clause ...))
  451. (reify-scm-constants
  452. ((add/immediate (u8? y) x) (add x y))
  453. ((sub/immediate (u8? y) x) (sub x y))
  454. (_
  455. (reify-u64-constants
  456. ((uadd/immediate (u8? y) x) (uadd x y))
  457. ((usub/immediate (u8? y) x) (usub x y))
  458. ((umul/immediate (u8? y) x) (umul x y))
  459. ((rsh/immediate (u6? y) x) (rsh x y))
  460. ((lsh/immediate (u6? y) x) (lsh x y))
  461. ;; These should all be u6's by construction.
  462. ;; ((ursh/immediate (u6? y) x) (ursh x y))
  463. ;; ((srsh/immediate (u6? y) x) (srsh x y))
  464. ;; ((ulsh/immediate (u6? y) x) (ulsh x y))
  465. ((ulogand/immediate (u8? y) x) (ulogand x y))
  466. (_
  467. (match (cons name args)
  468. (((or 'allocate-words/immediate
  469. 'allocate-pointerless-words/immediate))
  470. (define op
  471. (match name
  472. ('allocate-words/immediate
  473. 'allocate-words)
  474. ('allocate-pointerless-words/immediate
  475. 'allocate-pointerless-words)))
  476. (match param
  477. ((ann . n)
  478. (if (u8? n)
  479. cps
  480. (with-cps cps
  481. (letv n*)
  482. (letk kop ($kargs ('n) (n*)
  483. ($continue k src
  484. ($primcall op ann (n*)))))
  485. (setk label ($kargs names vars
  486. ($continue kop src
  487. ($primcall 'load-u64 n ())))))))))
  488. ;; Assume (tail-)pointer-ref/immediate is within u8 range.
  489. (((or 'word-ref/immediate 'scm-ref/immediate) obj)
  490. (match param
  491. ((ann . idx)
  492. (if (u8? idx)
  493. cps
  494. (let ((op (match name
  495. ('word-ref/immediate 'word-ref)
  496. ('scm-ref/immediate 'scm-ref))))
  497. (with-cps cps
  498. (letv idx*)
  499. (letk kop ($kargs ('idx) (idx*)
  500. ($continue k src
  501. ($primcall op ann (obj idx*)))))
  502. (setk label ($kargs names vars
  503. ($continue kop src
  504. ($primcall 'load-u64 idx ()))))))))))
  505. (((or 'word-set!/immediate 'scm-set!/immediate) obj val)
  506. (match param
  507. ((ann . idx)
  508. (if (u8? idx)
  509. cps
  510. (let ((op (match name
  511. ('word-set!/immediate 'word-set!)
  512. ('scm-set!/immediate 'scm-set!))))
  513. (with-cps cps
  514. (letv idx*)
  515. (letk kop ($kargs ('idx) (idx*)
  516. ($continue k src
  517. ($primcall op ann (obj idx* val)))))
  518. (setk label ($kargs names vars
  519. ($continue kop src
  520. ($primcall 'load-u64 idx ()))))))))))
  521. (_ cps))))))))
  522. (param (error "unexpected param to reified primcall" name))
  523. (else
  524. (with-cps cps
  525. (letv proc)
  526. (letk krecv ($kreceive '(res) #f k))
  527. (letk kproc ($kargs ('proc) (proc)
  528. ($continue krecv src ($call proc args))))
  529. (let$ body (resolve-prim name kproc src))
  530. (setk label ($kargs names vars ,body))))))
  531. (($ $kargs names vars ($ $branch kf kt src name param args))
  532. (let ()
  533. (define (u11? val) (<= 0 val #x7ff))
  534. (define (u12? val) (<= 0 val #xfff))
  535. (define (s12? val) (<= (- #x800) val #x7ff))
  536. (define (imm16? val)
  537. (and=> (scm->immediate-bits val)
  538. (lambda (bits)
  539. (truncate-bits bits 16 #t))))
  540. (define (load-u64 k param)
  541. (build-term ($continue k src ($primcall 'load-u64 param ()))))
  542. (define (load-s64 k param)
  543. (build-term ($continue k src ($primcall 'load-s64 param ()))))
  544. (define (load-const k param)
  545. (build-term ($continue k src ($const param))))
  546. (define-syntax-rule (reify-constants ((op (pred? c) in ...)
  547. wrap (op* out ...))
  548. ...
  549. (_ default))
  550. (match name
  551. ('op
  552. (if (pred? param)
  553. cps
  554. (match args
  555. ((in ...)
  556. (with-cps cps
  557. (letv c)
  558. (letk kconst
  559. ($kargs ('c) (c)
  560. ($branch kf kt src 'op* #f (out ...))))
  561. (setk label
  562. ($kargs names vars ,(wrap kconst param))))))))
  563. ...
  564. (_ default)))
  565. (reify-constants
  566. ((u64-imm-= (u11? b) a) load-u64 (u64-= a b))
  567. ((u64-imm-< (u12? b) a) load-u64 (u64-< a b))
  568. ((imm-u64-< (u12? a) b) load-u64 (u64-< a b))
  569. ((s64-imm-= (s12? b) a) load-s64 (s64-= a b))
  570. ((s64-imm-< (s12? b) a) load-s64 (s64-< a b))
  571. ((imm-s64-< (s12? a) b) load-s64 (s64-< a b))
  572. ((eq-constant? (imm16? b) a) load-const (eq? a b))
  573. (_ cps))))
  574. (_ cps)))
  575. (with-fresh-name-state cps
  576. (persistent-intmap (intmap-fold visit-cont cps cps))))