* pretty-print now properly detects and prints cyclic and shared
data structures
This commit is contained in:
parent
d1db554eee
commit
123e2f9e10
BIN
src/ikarus.boot
BIN
src/ikarus.boot
Binary file not shown.
|
@ -198,6 +198,12 @@
|
||||||
[(vector? x) (boxify-vector x)]
|
[(vector? x) (boxify-vector x)]
|
||||||
[(list? x) (boxify-list x '())]
|
[(list? x) (boxify-list x '())]
|
||||||
[(pair? x) (boxify-pair x)]
|
[(pair? x) (boxify-pair x)]
|
||||||
|
[(setbox? x)
|
||||||
|
(let ([i (format "#~a=" (setbox-idx x))]
|
||||||
|
[b (boxify (setbox-data x))])
|
||||||
|
(make-cbox (+ (string-length i) (box-length b))
|
||||||
|
(list i b)))]
|
||||||
|
[(refbox? x) (format "#~a#" (refbox-idx x))]
|
||||||
[else (format "~s" x)]))
|
[else (format "~s" x)]))
|
||||||
(define string-esc-table
|
(define string-esc-table
|
||||||
'((7 . "a")
|
'((7 . "a")
|
||||||
|
@ -433,9 +439,141 @@
|
||||||
(f x p 0)
|
(f x p 0)
|
||||||
(newline p))
|
(newline p))
|
||||||
;;;
|
;;;
|
||||||
|
|
||||||
|
(define (hasher x h)
|
||||||
|
(define (vec-graph x i j)
|
||||||
|
(unless (fx= i j)
|
||||||
|
(graph (vector-ref x i))
|
||||||
|
(vec-graph x (fxadd1 i) j h)))
|
||||||
|
(define (vec-dynamic x i j)
|
||||||
|
(unless (fx= i j)
|
||||||
|
(dynamic (vector-ref x i))
|
||||||
|
(vec-dynamic x (fxadd1 i) j)))
|
||||||
|
(define rv #f)
|
||||||
|
(define (graph x)
|
||||||
|
(cond
|
||||||
|
[(pair? x)
|
||||||
|
(cond
|
||||||
|
[(get-hash-table h x #f) =>
|
||||||
|
(lambda (n)
|
||||||
|
(set! rv #t)
|
||||||
|
(put-hash-table! h x (fxadd1 n)))]
|
||||||
|
[else
|
||||||
|
(put-hash-table! h x 0)
|
||||||
|
(graph (car x))
|
||||||
|
(graph (cdr x))])]
|
||||||
|
[(vector? x)
|
||||||
|
(cond
|
||||||
|
[(get-hash-table h x #f) =>
|
||||||
|
(lambda (n)
|
||||||
|
(set! rv #t)
|
||||||
|
(put-hash-table! h x (fxadd1 n)))]
|
||||||
|
[else
|
||||||
|
(put-hash-table! h x 0)
|
||||||
|
(vec-graph x 0 (vector-length x))])]
|
||||||
|
[(gensym? x)
|
||||||
|
(cond
|
||||||
|
[(get-hash-table h x #f) =>
|
||||||
|
(lambda (n)
|
||||||
|
(set! rv #t)
|
||||||
|
(put-hash-table! h x (fxadd1 n)))])]))
|
||||||
|
(define (dynamic x)
|
||||||
|
(cond
|
||||||
|
[(pair? x)
|
||||||
|
(cond
|
||||||
|
[(get-hash-table h x #f) =>
|
||||||
|
(lambda (n)
|
||||||
|
(set! rv #t)
|
||||||
|
(put-hash-table! h x (fxadd1 n)))]
|
||||||
|
[else
|
||||||
|
(put-hash-table! h x 0)
|
||||||
|
(dynamic (car x))
|
||||||
|
(dynamic (cdr x))
|
||||||
|
(when (and (get-hash-table h x #f)
|
||||||
|
(fxzero? (get-hash-table h x #f)))
|
||||||
|
(put-hash-table! h x #f))])]
|
||||||
|
[(vector? x)
|
||||||
|
(cond
|
||||||
|
[(get-hash-table h x #f) =>
|
||||||
|
(lambda (n)
|
||||||
|
(set! rv #t)
|
||||||
|
(put-hash-table! h x (fxadd1 n)))]
|
||||||
|
[else
|
||||||
|
(put-hash-table! h x 0)
|
||||||
|
(vec-dynamic x 0 (vector-length x))
|
||||||
|
(when (and (get-hash-table h x #f)
|
||||||
|
(fxzero? (get-hash-table h x #f)))
|
||||||
|
(put-hash-table! h x #f))])]))
|
||||||
|
(if (print-graph)
|
||||||
|
(graph x)
|
||||||
|
(dynamic x))
|
||||||
|
rv)
|
||||||
|
|
||||||
|
(define-record setbox (idx data))
|
||||||
|
(define-record refbox (idx))
|
||||||
|
|
||||||
|
(define (rewrite-shared x h)
|
||||||
|
(define counter 0)
|
||||||
|
(let f ([x x])
|
||||||
|
(cond
|
||||||
|
[(pair? x)
|
||||||
|
(cond
|
||||||
|
[(get-hash-table h x #f) =>
|
||||||
|
(lambda (n)
|
||||||
|
(cond
|
||||||
|
[(setbox? n)
|
||||||
|
(make-refbox (setbox-idx n))]
|
||||||
|
[(and (fixnum? n) (fx> n 0))
|
||||||
|
(let ([box (make-setbox counter #f)])
|
||||||
|
(set! counter (add1 counter))
|
||||||
|
(put-hash-table! h x box)
|
||||||
|
(let* ([a (f (car x))]
|
||||||
|
[d (f (cdr x))])
|
||||||
|
(set-setbox-data! box (cons a d))
|
||||||
|
box))]
|
||||||
|
[else
|
||||||
|
(let* ([a (f (car x))]
|
||||||
|
[d (f (cdr x))])
|
||||||
|
(if (and (eq? a (car x))
|
||||||
|
(eq? d (cdr x)))
|
||||||
|
x
|
||||||
|
(cons a d)))]))]
|
||||||
|
[else
|
||||||
|
(let* ([a (f (car x))]
|
||||||
|
[d (f (cdr x))])
|
||||||
|
(if (and (eq? a (car x))
|
||||||
|
(eq? d (cdr x)))
|
||||||
|
x
|
||||||
|
(cons a d)))])]
|
||||||
|
[(vector? x)
|
||||||
|
(cond
|
||||||
|
[(get-hash-table h x #f) =>
|
||||||
|
(lambda (n)
|
||||||
|
(cond
|
||||||
|
[(setbox? n)
|
||||||
|
(make-refbox (setbox-idx n))]
|
||||||
|
[(and (fixnum? n) (fx> n 0))
|
||||||
|
(let ([box (make-setbox counter #f)])
|
||||||
|
(set! counter (add1 counter))
|
||||||
|
(put-hash-table! h x box)
|
||||||
|
(set-setbox-data! box
|
||||||
|
(list->vector
|
||||||
|
(map1ltr f (vector->list x))))
|
||||||
|
box)]
|
||||||
|
[else
|
||||||
|
(list->vector (map1ltr f (vector->list x)))]))]
|
||||||
|
[else
|
||||||
|
(list->vector (map1ltr f (vector->list x)))])]
|
||||||
|
[else x])))
|
||||||
|
|
||||||
|
(define (unshare x)
|
||||||
|
(let ([h (make-hash-table)])
|
||||||
|
(if (hasher x h)
|
||||||
|
(rewrite-shared x h)
|
||||||
|
x)))
|
||||||
|
;;;
|
||||||
(define (pretty x p)
|
(define (pretty x p)
|
||||||
(let ([x (boxify x)])
|
(output (boxify (unshare x)) p))
|
||||||
(output x p)))
|
|
||||||
;;;
|
;;;
|
||||||
(define *pretty-format* '*pretty-format*)
|
(define *pretty-format* '*pretty-format*)
|
||||||
(define (set-fmt! name fmt)
|
(define (set-fmt! name fmt)
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
|
|
||||||
(library (ikarus writer)
|
(library (ikarus writer)
|
||||||
(export write display format printf print-error error-handler
|
(export write display format printf print-error error-handler
|
||||||
error print-unicode)
|
error print-unicode print-graph)
|
||||||
(import
|
(import
|
||||||
(ikarus system $chars)
|
(ikarus system $chars)
|
||||||
(ikarus system $strings)
|
(ikarus system $strings)
|
||||||
|
@ -12,7 +12,7 @@
|
||||||
(ikarus system $bytevectors)
|
(ikarus system $bytevectors)
|
||||||
(only (ikarus unicode-data) unicode-printable-char?)
|
(only (ikarus unicode-data) unicode-printable-char?)
|
||||||
(except (ikarus) write display format printf print-error
|
(except (ikarus) write display format printf print-error
|
||||||
error-handler error print-unicode))
|
error-handler error print-unicode print-graph))
|
||||||
|
|
||||||
(define print-unicode
|
(define print-unicode
|
||||||
(make-parameter #t))
|
(make-parameter #t))
|
||||||
|
|
|
@ -588,6 +588,7 @@
|
||||||
[pretty-print i]
|
[pretty-print i]
|
||||||
[comment-handler i]
|
[comment-handler i]
|
||||||
[print-gensym i symbols]
|
[print-gensym i symbols]
|
||||||
|
[print-graph i]
|
||||||
[print-unicode i]
|
[print-unicode i]
|
||||||
[char-general-category i]
|
[char-general-category i]
|
||||||
[gensym-count i symbols]
|
[gensym-count i symbols]
|
||||||
|
|
Loading…
Reference in New Issue