scm-scheme.scm 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408
  1. ; Part of Scheme 48 1.9. See file COPYING for notices and license.
  2. ; Authors: Richard Kelsey
  3. ; Primitives that directly correspond to primops.
  4. ;
  5. ; (define-primitive (id (arg-pred arg-type) ...) result-type . maybe-primop-id)
  6. ;
  7. ; Primitives that are n-ary or have other weirdness.
  8. ;
  9. ; (define-complex-primitive (id . argument-predicates)
  10. ; eval-fn inference-rule source . maybe-expander)
  11. ;
  12. ; Primitives that have only source but not a primop.
  13. ;
  14. ; (define-semi-primitive (id . argument-predicates)
  15. ; eval-fn inference-rule source maybe-expander)
  16. ;
  17. ; Primitives available only at load time.
  18. ;
  19. ; (define-load-time-primitive (id . argument-predicates) eval-fn)
  20. ; (really-define-primitive (id . argument-predicates)
  21. ; eval-fn inference-rule source expander expands-in-place?)
  22. (define-syntax really-define-primitive
  23. (lambda (exp r c)
  24. (let* ((spec (cadr exp))
  25. (id (car spec))
  26. (arg-predicates (cdr spec))
  27. (eval (caddr exp))
  28. (rest (cdddr exp))
  29. (inference-rule (car rest))
  30. (source (cadr rest))
  31. (expander (caddr rest))
  32. (expands-in-place? (cadddr rest)))
  33. `(let ((,(r 'predicates) ,(let recur ((preds arg-predicates))
  34. (cond ((pair? preds)
  35. `(cons ,(car preds)
  36. ,(recur (cdr preds))))
  37. ((null? preds)
  38. '(quote ()))
  39. (else
  40. preds)))))
  41. (define-prescheme! ',id
  42. #f ; location
  43. (make-primitive ',id
  44. ,(r 'predicates)
  45. ,eval
  46. ',source
  47. ,expander
  48. ,expands-in-place?
  49. ,inference-rule))))))
  50. (define-syntax define-complex-primitive
  51. (lambda (exp r c)
  52. `(really-define-primitive ,@(cdr exp) #t)))
  53. (define-syntax define-primitive
  54. (lambda (exp r c)
  55. (let* ((id (cadr exp))
  56. (args (caddr exp))
  57. (result (cadddr exp))
  58. (primop (if (null? (cddddr exp)) (cadr exp) (car (cddddr exp))))
  59. (names (map (lambda (a b) b)
  60. args
  61. '(x1 x2 x3 x4 x5 x6 x7 x8 x9))))
  62. `(define-complex-primitive (,id . ,(map car args)) ,id
  63. (lambda (args node depth return?)
  64. (if (not (= (length args)
  65. ,(length args)))
  66. (user-error "wrong number of arguments in ~S" (schemify node)))
  67. ,@(do ((i 0 (+ i 1))
  68. (args args (cdr args))
  69. (res '() (cons `(check-arg-type args ,i ,(cadar args) depth node)
  70. res)))
  71. ((null? args)
  72. (reverse res)))
  73. ,result)
  74. (lambda ,names (,id . ,names))
  75. (lambda (args type)
  76. (make-primop-call-node (get-prescheme-primop ',primop) args type))))))
  77. (define-syntax define-semi-primitive
  78. (lambda (exp r c)
  79. `(really-define-primitive ,@(cdr exp) #f #f)))
  80. (define-syntax define-load-time-primitive
  81. (lambda (exp r c)
  82. `(define-semi-primitive ,(cadr exp)
  83. ,(caddr exp)
  84. (make-load-time-only-rule ',(caadr exp))
  85. #f)))
  86. (define (make-load-time-only-rule id)
  87. (lambda (args node depth return?)
  88. (user-error "~S is only available at load time ~S" id (schemify node))))
  89. ;----------------------------------------------------------------
  90. ; Boolean stuff
  91. (define-semi-primitive (not #f) not
  92. (lambda (args node depth return?)
  93. (check-arg-type args 0 type/boolean depth node)
  94. type/boolean)
  95. (lambda (x) (if x #f #t)))
  96. (define-load-time-primitive (boolean? #f) boolean?)
  97. (define-complex-primitive (eq? #f #f) eq?
  98. (lambda (args node depth return?)
  99. (unify! (infer-type (car args) depth)
  100. (infer-type (cadr args) depth)
  101. node)
  102. type/boolean)
  103. (lambda (x y) (eq? x y))
  104. (lambda (args type)
  105. (make-primop-call-node (get-prescheme-primop 'eq?) args type)))
  106. (define-load-time-primitive (eqv? #f) eqv?)
  107. (define-load-time-primitive (equal? #f) equal?)
  108. ;----------------------------------------------------------------
  109. ; Characters
  110. (define (ascii-value? n)
  111. (and (integer? n)
  112. (>= n 0)
  113. (< n ascii-limit)))
  114. (define-primitive ascii->char ((ascii-value? type/integer)) type/char)
  115. (define-primitive char->ascii ((char? type/char)) type/integer)
  116. (define (char-comparison-rule args node depth return?)
  117. (check-arg-type args 0 type/char depth node)
  118. (check-arg-type args 1 type/char depth node)
  119. type/boolean)
  120. (define-syntax define-char-comparison
  121. (lambda (exp r c)
  122. (let ((id (cadr exp))
  123. (op (caddr exp)))
  124. `(define-complex-primitive (,id char? char?) ,id
  125. char-comparison-rule
  126. (lambda (x y) (,op x y))
  127. (lambda (args type)
  128. (make-primop-call-node (get-prescheme-primop ',op) args type))))))
  129. (define-char-comparison char=? =)
  130. (define-char-comparison char<? <)
  131. (define-char-comparison char>? >)
  132. (define-char-comparison char<=? <=)
  133. (define-char-comparison char>=? >=)
  134. ; Plus lots more...
  135. ;----------------------------------------------------------------
  136. ; Data manipulation
  137. (define (any? x) #t)
  138. (define (positive-integer? x)
  139. (and (integer? x)
  140. (<= 0 x)))
  141. (define (unsigned-byte? x)
  142. (and (positive-integer? x)
  143. (<= x 256)))
  144. (define-complex-primitive (make-vector positive-integer? any?) make-vector
  145. (lambda (args node depth return?)
  146. (let ((uvar (make-uvar 'v depth)))
  147. (make-nonpolymorphic! uvar)
  148. (check-arg-type args 0 type/integer depth node)
  149. (check-arg-type args 1 uvar depth node)
  150. (make-pointer-type uvar)))
  151. (lambda (size init)
  152. (make-vector size init))
  153. (lambda (args type)
  154. (make-primop-call-node (get-prescheme-primop 'make-vector) args type)))
  155. (define-load-time-primitive (vector-length vector?) vector-length)
  156. (define-complex-primitive (vector-ref vector? positive-integer?) vector-ref
  157. (lambda (args node depth return?)
  158. (let ((elt-type (make-uvar 'v depth)))
  159. (check-arg-type args 0 (make-pointer-type elt-type) depth node)
  160. (check-arg-type args 1 type/integer depth node)
  161. elt-type))
  162. (lambda (vector index)
  163. (vector-ref vector index))
  164. (lambda (args type)
  165. (make-primop-call-node (get-prescheme-primop 'vector-ref) args type)))
  166. (define-complex-primitive (vector-set! vector? positive-integer? any?)
  167. vector-set!
  168. (lambda (args node depth return?)
  169. (let ((elt-type (make-uvar 'v depth)))
  170. (check-arg-type args 0 (make-pointer-type elt-type) depth node)
  171. (check-arg-type args 1 type/integer depth node)
  172. (check-arg-type args 2 elt-type depth node)
  173. type/unit))
  174. (lambda (vector index value)
  175. (vector-set! vector index value))
  176. (lambda (args type)
  177. (make-primop-call-node (get-prescheme-primop 'vector-set!) args type)))
  178. (define-primitive make-string ((integer? type/integer)) type/string)
  179. (define-primitive string-length ((string? type/string)) type/integer)
  180. (define-primitive string-ref
  181. ((string? type/string) (integer? type/integer))
  182. type/char)
  183. (define-primitive string-set!
  184. ((string? type/string) (integer? type/integer) (char? type/char))
  185. type/unit)
  186. (define-complex-primitive (deallocate any?) (lambda (x) (values))
  187. (lambda (args node depth return?)
  188. (let ((type (make-pointer-type (make-uvar 'p depth))))
  189. (check-arg-type args 0 type depth node)
  190. type/unit))
  191. (lambda (thing)
  192. (deallocate thing))
  193. (lambda (args type)
  194. (make-primop-call-node (get-prescheme-primop 'deallocate) args type)))
  195. (define-complex-primitive (null-pointer? any?) (lambda (x) #f)
  196. (lambda (args node depth return?)
  197. (let ((type (make-pointer-type (make-uvar 'p depth))))
  198. (check-arg-type args 0 type depth node)
  199. type/boolean))
  200. (lambda (thing)
  201. (null-pointer? thing))
  202. (lambda (args type)
  203. (make-primop-call-node (get-prescheme-primop 'null-pointer?) args type)))
  204. (define-complex-primitive (null-pointer) (lambda () #f)
  205. (lambda (args node depth return?)
  206. (make-pointer-type (make-uvar 'null depth)))
  207. (lambda (type)
  208. (null-pointer type))
  209. (lambda (args type)
  210. (make-primop-call-node (get-prescheme-primop 'null-pointer)
  211. (list (make-literal-node type))
  212. type)))
  213. ;----------------------------------------------------------------
  214. ; I/O
  215. (define-primitive current-input-port () type/input-port stdin)
  216. (define-primitive current-output-port () type/output-port stdout)
  217. (define-primitive current-error-port () type/output-port stderr)
  218. (define type/status type/integer)
  219. (let ((return (make-tuple-type (list type/input-port type/status))))
  220. (define-primitive open-input-file ((string? type/string)) return))
  221. (let ((return (make-tuple-type (list type/output-port type/status))))
  222. (define-primitive open-output-file ((string? type/string)) return))
  223. (define-primitive close-input-port ((input-port? type/input-port)) type/status)
  224. (define-primitive close-output-port ((output-port? type/output-port)) type/status)
  225. (define char-return-type
  226. (make-tuple-type (list type/char type/boolean type/status)))
  227. (define-primitive read-char ((input-port? type/input-port)) char-return-type)
  228. (define-primitive peek-char ((input-port? type/input-port)) char-return-type)
  229. (define integer-return-type
  230. (make-tuple-type (list type/integer type/boolean type/status)))
  231. (define-primitive read-integer ((input-port? type/input-port)) integer-return-type)
  232. (define-primitive write-char
  233. ((char? type/char) (output-port? type/output-port))
  234. type/status)
  235. (define-primitive write-string
  236. ((string? type/string) (output-port? type/output-port))
  237. type/status)
  238. (define-primitive write-integer
  239. ((integer? type/integer) (output-port? type/output-port))
  240. type/status)
  241. (define-complex-primitive (newline output-port?) newline
  242. (lambda (args node depth return?)
  243. (check-arg-type args 0 type/output-port depth node)
  244. type/status)
  245. (lambda (out)
  246. (write-char #\newline out))
  247. (lambda (args type)
  248. (make-primop-call-node (get-prescheme-primop 'write-char)
  249. (cons (make-literal-node #\newline) args)
  250. type)))
  251. (define-primitive force-output ((output-port? type/output-port)) type/status)
  252. (define-primitive error-string
  253. ((positive-integer? type/status))
  254. type/string)
  255. ;----------------------------------------------------------------
  256. (define-complex-primitive (values . any?) values
  257. (lambda (args node depth return?)
  258. (make-tuple-type (infer-types args depth)))
  259. #f
  260. (lambda (args type)
  261. (let ((node (make-node values-operator (cons 'values args))))
  262. (node-set! node 'type type)
  263. node)))
  264. (define values-operator (get-operator 'values))
  265. ; CALL-WITH-VALUES that uses closures instead of procedures.
  266. (define (ps-call-with-values producer consumer)
  267. (call-with-values
  268. (lambda ()
  269. (apply-closure producer '()))
  270. (lambda args
  271. (apply-closure consumer args))))
  272. (define-complex-primitive (call-with-values closure? closure?)
  273. ps-call-with-values
  274. (lambda (args node depth return?)
  275. (if (not (lambda-node? (cadr args)))
  276. (user-error
  277. "second argument to CALL-WITH-VALUES must be a lambda node~% ~S"
  278. (schemify node)))
  279. (let* ((consumer-type (infer-type (cadr args) depth))
  280. (arg-types (arrow-type-args consumer-type))
  281. (result-type (arrow-type-result consumer-type)))
  282. (unify! (infer-type (car args) depth)
  283. (make-arrow-type '() (make-tuple-type arg-types))
  284. node)
  285. (if (not return?) ; so we cause a check for illegal tuples
  286. (unify! result-type (make-uvar 'temp depth) node))
  287. result-type))
  288. #f
  289. (lambda (args type)
  290. (let* ((tuple-type (arrow-type-result
  291. (maybe-follow-uvar (node-ref (car args) 'type))))
  292. (node (make-node call-with-values-operator
  293. (list 'call-with-values
  294. (make-call-node (car args) '() tuple-type)
  295. (cadr args)))))
  296. (node-set! node 'type type)
  297. node)))
  298. (define lambda-node? (node-predicate 'lambda))
  299. (define call-with-values-operator (get-operator 'call-with-values))
  300. (define-primitive unspecific () type/unit)
  301. (define-complex-primitive (error string? . integer?) error
  302. (lambda (args node depth return?)
  303. (check-arg-type args 0 type/string depth node)
  304. (do ((args (cdr args) (cdr args)))
  305. ((null? args))
  306. (check-arg-type args 0 type/integer depth node))
  307. type/null)
  308. (lambda (error string)
  309. (error string))
  310. (lambda (args type)
  311. (make-primop-call-node (get-prescheme-primop 'error) args type)))
  312. ; For enumerated types that are shared with C
  313. (define-load-time-primitive (make-external-constant symbol? symbol? string?)
  314. make-external-constant)
  315. ;----------------------------------------------------------------
  316. ; Utilities for making nodes
  317. (define call-operator (get-operator 'call))
  318. (define literal-operator (get-operator 'literal))
  319. (define name-operator (get-operator 'name))
  320. (define primitive-operator (get-operator 'primitive))
  321. (define (make-call-node proc args type)
  322. (let ((node (make-node call-operator (cons proc args))))
  323. (node-set! node 'type type)
  324. node))
  325. (define (make-literal-node value)
  326. (make-node literal-operator value))
  327. (define (make-primop-call-node primop args type)
  328. (make-call-node (make-literal-node primop) args type))
  329. (define (make-reference-node id binding)
  330. (let ((node (make-node name-operator id)))
  331. (node-set! node 'binding binding)
  332. node))
  333. (define (var->name-node var)
  334. (make-reference-node ((structure-ref variable variable-name) var)
  335. (make-binding #f var #f)))