#lang scheme
(require
(for-syntax scheme))
(provide current-heap)
(define current-heap (make-parameter false))
(define (format-cell cell)
(let* ([str (format "~s" cell)]
[len (string-length str)])
(if (<= len 10)
(string-append str (build-string (- 10 len) (λ (_) #\space)))
(substring str 0 10))))
(provide heap-as-string)
(define (heap-as-string)
(let ([step 0])
(apply string-append
(for/list ([elt (in-vector (current-heap))])
(cond
[(= step 0)
(begin
(set! step (add1 step))
(format-cell elt))]
[(= step 9)
(begin
(set! step 0)
(string-append (format-cell elt) "\n"))]
[else
(begin
(set! step (add1 step))
(string-append " " (format-cell elt)))])))))
(provide/contract [heap-value? (any/c . -> . boolean?)])
(define (heap-value? v)
(or (number? v) (symbol? v) (boolean? v) (empty? v) (procedure? v)))
(provide location?)
(define (location? v)
(if (vector? (current-heap))
(and (exact-nonnegative-integer? v) (< v (vector-length (current-heap))))
(error "Heap is unintialized")))
(provide/contract (init-heap! (exact-nonnegative-integer? . -> . void?)))
(define (init-heap! size)
(current-heap (build-vector size (λ (ix) false))))
(provide/contract (heap-set! (location? heap-value? . -> . void?)))
(define (heap-set! location value)
(vector-set! (current-heap) location value)
(when gui
(send gui update-view #:location location)))
(provide/contract (heap-ref (location? . -> . heap-value?)))
(define (heap-ref location)
(vector-ref (current-heap) location))
(provide/contract (heap-size (-> (or/c false/c exact-nonnegative-integer?))))
(define (heap-size)
(and (vector? (current-heap)) (vector-length (current-heap))))
(provide gc-roots-key)
(define gc-roots-key (gensym 'gc-roots-key))
(provide root? root-name make-root)
(define-struct root (name get set!)
#:property prop:custom-write (λ (v port write?)
(display (format "#<root:~a>" (root-name v)) port)))
(provide make-env-root)
(define-syntax (make-env-root stx)
(syntax-case stx ()
[(_ id) (identifier? #'id)
#`(make-root 'id (λ () id) (λ (loc) (set! id loc)))]))
(provide/contract (stack-roots (-> (listof root?))))
(define (stack-roots)
(filter is-mutable-root?
(apply append (continuation-mark-set->list (current-continuation-marks) gc-roots-key))))
(define (is-mutable-root? root)
(location? ((root-get root))))
(provide/contract (make-stack-root (symbol? location? . -> . root?)))
(define (make-stack-root id location)
(make-root id (λ () location) (λ (new-location) (set! location new-location))))
(provide/contract (read-root (root? . -> . location?)))
(define (read-root root)
((root-get root)))
(provide/contract (set-root! (root? location? . -> . any)))
(define (set-root! root loc)
((root-set! root) loc))
(provide/contract (get-global-roots (-> (listof root?))))
(define (get-global-roots)
(filter is-mutable-root? global-roots))
(define global-roots empty)
(provide/contract (add-global-root! (root? . -> . void?)))
(define (add-global-root! root)
(set! global-roots (cons root global-roots)))
(provide get-root-set)
(define-syntax (get-root-set stx)
(syntax-case stx ()
[(_ root-id ...)
(andmap identifier? (syntax->list #'(root-id ...)))
#`(begin
(append
(list (make-root 'root-id (λ () root-id)
(λ (loc)
(set! root-id loc)))
...)
(get-global-roots)
(stack-roots)))]
[(_ e ...)
(let ([err (ormap (λ (x) (and (not (identifier? x)) x)) (syntax->list #'(e ...)))])
(raise-syntax-error false
"expected an identifier to treat as a root"
stx
err))]
[_ (raise-syntax-error false
"missing open parenthesis"
stx)]))
(define closure-envs (make-weak-hash))
(provide/contract (add-closure-env! (procedure? (listof root?) . -> . any)))
(define (add-closure-env! proc roots)
(hash-set! closure-envs proc roots))
(provide/contract (get-closure-env (procedure? . -> . (or/c false/c (listof root?)))))
(define (get-closure-env proc)
(hash-ref closure-envs proc false))
(provide/contract (procedure-roots (procedure? . -> . (listof root?))))
(define (procedure-roots proc)
(filter is-mutable-root? (hash-ref closure-envs proc empty)))
(provide set-ui!)
(define (set-ui! ui%)
(set! gui (new ui% [heap-vec (current-heap)])))
(define gui false)