picrin/etc/R7RS/src/hashtable0.sch

115 lines
3.9 KiB
Scheme

;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
; Copyright 2007 William D Clinger.
;
; Permission to copy this software, in whole or in part, to use this
; software for any lawful purpose, and to redistribute this software
; is granted subject to the restriction that all copies made of this
; software must include this copyright notice in full.
;
; I also request that you send me a copy of any improvements that you
; make to this software so that they may be incorporated within it to
; the benefit of the Scheme community.
;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
; Hashtable benchmark.
;
; Tests only eq? and eqv? hashtables.
;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
(import (rnrs base)
(rnrs control)
(rnrs io simple)
(rnrs hashtables))
; Crude test rig, just for benchmarking.
(define failures '())
(define (report-failure! n)
(set! failures (cons n failures))
(display "******** TEST FAILED ******** ")
(write n)
(newline))
; The parameter n2 is the number of items to be added to the table
; during the stress phase.
(define (hashtable-eq-tests n2 . rest)
(call-with-current-continuation
(lambda (exit)
(let ((maker (if (null? rest) make-eq-hashtable (car rest)))
(test (lambda (n passed?)
(if (not passed?)
(report-failure! n)))))
(let ((t (maker))
(not-found (list 'not-found))
(x1 (string #\a #\b #\c))
(sym1 'sym1)
(vec1 (vector 'vec1))
(pair1 (list -1))
(n1 1000) ; population added in first phase
;(n2 10000) ; population added in second phase
(n3 1000)) ; population added in third phase
(define (hashtable-get t key)
(hashtable-ref t key #f))
(test 1 (eq? not-found (hashtable-ref t x1 not-found)))
(hashtable-set! t x1 'a)
(test 2 (eq? 'a (hashtable-get t x1)))
(hashtable-set! t sym1 'b)
(test 3 (eq? 'a (hashtable-get t x1)))
(test 4 (eq? 'b (hashtable-get t sym1)))
(hashtable-set! t vec1 'c)
(test 5 (eq? 'a (hashtable-get t x1)))
(test 6 (eq? 'b (hashtable-get t sym1)))
(test 7 (eq? 'c (hashtable-get t vec1)))
(hashtable-set! t n2 'd)
(test 8 (eq? 'a (hashtable-get t x1)))
(test 9 (eq? 'b (hashtable-get t sym1)))
(test 10 (eq? 'c (hashtable-get t vec1)))
(test 11 (eq? 'd (hashtable-get t n2)))
(hashtable-set! t pair1 'e)
(do ((i 0 (+ i 1)))
((= i n1))
(hashtable-set! t (list i) i))
(test 12 (eq? 'e (hashtable-get t pair1)))
(do ((i 0 (+ i 1)))
((= i n2))
(if (and #f (zero? (mod i 1000))) (display "."))
(hashtable-set! t (list i) i))
(test 13 (eq? 'e (hashtable-get t pair1)))
(do ((i 0 (+ i 1)))
((= i n3))
(test 14 (eq? 'e (hashtable-get t pair1)))
(hashtable-set! t (list i) i))
(test 15 (eq? 'a (hashtable-get t x1)))
(test 16 (eq? 'b (hashtable-get t sym1)))
(test 17 (eq? 'c (hashtable-get t vec1)))
(test 18 (eq? 'd (hashtable-get t n2)))
(test 19 (eq? 'e (hashtable-get t pair1)))
(hashtable-size t))))))
(define (main)
(let* ((count (read))
(input1 (read))
(input2 (read))
(output (read))
(s2 (number->string count))
(s1 (number->string input1))
(name "hashtable0"))
(run-r6rs-benchmark
(string-append name ":" s1 ":" s2)
count
(lambda ()
(hashtable-eq-tests (hide count input1) make-eq-hashtable)
(hashtable-eq-tests (hide count input2) make-eqv-hashtable))
(lambda (result) (and (null? failures) (equal? result output))))))