#lang racket/base (require (for-syntax racket/base racket/syntax) racket/private/dict) ;; No-contract version. (define-struct id-table (hash phase)) ;; where hash maps symbol => (listof (cons identifier value)) ;; phase is a phase-level (integer or #f) (define (make-id-table/constructor who init-dict phase make identifier->symbol identifier=?) (let ([t (make (make-hasheq) phase)]) (for ([(k v) (in-dict init-dict)]) (unless (identifier? k) (raise-type-error who "dictionary with identifier keys" init-dict)) (id-table-set! who t k v identifier->symbol identifier=?)) t)) (define (make-immutable-id-table/constructor who init-dict phase make identifier->symbol identifier=?) (for/fold ([t (make '#hasheq() phase)]) ([(k v) (in-dict init-dict)]) (unless (identifier? k) (raise-type-error who "dictionary with identifier keys" init-dict)) (id-table-set/constructor who t k v make identifier->symbol identifier=?))) (define (id-table-ref who d id default identifier->symbol identifier=?) (let ([phase (id-table-phase d)]) (let ([i (for/first ([i (in-list (hash-ref (id-table-hash d) (identifier->symbol id phase) null))] #:when (identifier=? (car i) id phase)) i)]) (if i (cdr i) (cond [(eq? default not-given) (error who "no mapping for ~e" id)] [(procedure? default) (default)] [else default]))))) (define (id-table-set! who d id v identifier->symbol identifier=?) (let* ([phase (id-table-phase d)] [sym (identifier->symbol id phase)] [l (hash-ref (id-table-hash d) sym null)] [new-l (alist-set identifier=? phase l id v)]) (hash-set! (id-table-hash d) sym new-l))) (define (id-table-remove! who d id identifier->symbol identifier=?) (let* ([phase (id-table-phase d)] [sym (identifier->symbol id phase)] [l (hash-ref (id-table-hash d) sym null)] [newl (alist-remove identifier=? phase l id)]) (if (pair? newl) (hash-set! (id-table-hash d) sym newl) (hash-remove! (id-table-hash d) sym)))) (define (id-table-set/constructor who d id v constructor identifier->symbol identifier=?) (let* ([phase (id-table-phase d)] [sym (identifier->symbol id phase)] [l (hash-ref (id-table-hash d) sym null)] [new-l (alist-set identifier=? phase l id v)]) (constructor (hash-set (id-table-hash d) sym new-l) phase))) (define (id-table-remove/constructor who d id constructor identifier->symbol identifier=?) (let* ([phase (id-table-phase d)] [sym (identifier->symbol id phase)] [l (hash-ref (id-table-hash d) sym null)] [newl (alist-remove identifier=? phase l id)]) (constructor (if (pair? newl) (hash-set (id-table-hash d) sym newl) (hash-remove (id-table-hash d) sym)) phase))) (define (id-table-count d) (apply + (hash-map (id-table-hash d) (lambda (k v) (length v))))) (define (id-table-for-each d p) (define (pp i) (p (car i) (cdr i))) (hash-for-each (id-table-hash d) (lambda (k v) (for-each pp v)))) (define (id-table-map d f) (define (fp i) (f (car i) (cdr i))) (apply append (hash-map (id-table-hash d) (lambda (k v) (map fp v))))) (define-struct id-table-iter (d a br b)) ;; where d is an id-table ;; a is hash iter ;; br is non-empty-alist "root" ;; b is non-empty-alist current position, b = (list-tail br N) for some N #| For mutable table, the alist iter can be invalidated because we replace entire alist on update. So, we store alist root and if root has been replaced, we try to re-find the previous position in the new alist (see rebase-iter). Not an issue for immutable tables. Notes (FIXME?): - we don't (can't?) check that hash iter is okay (would require hash revision?) so internal errors raised by hash-iterate-{next,value} can slip out |# (define (id-table-iterate-first d) (let* ([h (id-table-hash d)] [a (hash-iterate-first h)]) (and a (let ([b (hash-iterate-value h a)]) (make-id-table-iter d a b b))))) (define (id-table-iterate-next who d pos) (let-values ([(h a br b) (rebase-iter who d pos)]) (let ([b2 (cdr b)]) (if (pair? b2) (make-id-table-iter d a br b2) (let ([a2 (hash-iterate-next h a)]) (and a2 (let ([b2 (hash-iterate-value h a2)]) (make-id-table-iter d a2 b2 b2)))))))) (define (id-table-iterate-key who d pos) (let-values ([(h a br b) (rebase-iter who d pos)]) (caar b))) (define (id-table-iterate-value who d pos) (let-values ([(h a br b) (rebase-iter who d pos)]) (cdar b))) (define (rebase-iter who d pos) (unless (eq? d (id-table-iter-d pos)) (error who "invalid iteration position for identifier table")) (let* ([h (id-table-hash d)] [a (id-table-iter-a pos)] [br (id-table-iter-br pos)] [b (id-table-iter-b pos)] [v (hash-iterate-value h a)]) ;; FIXME: may expose internal error (if (eq? br v) (values h a br b) ;; hash entry has changed from br to v, so find (caar b) in v (let ([id (caar b)]) (let loop ([v* v]) (cond [(null? v*) (error who "invalid iteration position for identifier table")] [(eq? (caar v*) id) ;; relies on id staying same; see alist-set (values h a v v*)] [else (loop (cdr v*))])))))) ;; ======== (define (alist-set identifier=? phase l0 id v) ;; To minimize allocation ;; - add new pairs to front ;; - avoid allocation on idempotent sets (let* ([not-found? #f] [new-l (let loop ([l l0]) (cond [(null? l) (begin (set! not-found? #t) null)] [(identifier=? (caar l) id phase) (if (eq? v (cdar l)) ;; idempotent; just leave it alone l ;; Note: use (caar l) instead of id so that rebase-iter ;; above can find entry by stxobj identity. (cons (cons (caar l) v) (cdr l)))] [else (let ([rest* (loop (cdr l))]) (if (eq? (cdr l) rest*) l (cons (car l) rest*)))]))]) (if not-found? (cons (cons id v) l0) new-l))) (define (alist-remove identifier=? phase l0 id) ;; To minimize allocation ;; - avoid allocation on idempotent removes (let loop ([l l0]) (cond [(null? l) null] [(identifier=? (caar l) id phase) (cdr l)] [else (let ([rest* (loop (cdr l))]) (if (eq? (cdr l) rest*) l (cons (car l) rest*)))]))) (define not-given (gensym 'not-given)) ;; ======== (define-syntax (make-code stx) (syntax-case stx () [(_ idtbl identifier->symbol identifier=?) (with-syntax ([mutable-idtbl (format-id #'idtbl "mutable-~a" (syntax-e #'idtbl))] [immutable-idtbl (format-id #'idtbl "immutable-~a" (syntax-e #'idtbl))] [make-idtbl (format-id #'idtbl "make-~a" (syntax-e #'idtbl))] [make-mutable-idtbl (format-id #'idtbl "make-mutable-~a" (syntax-e #'idtbl))] [make-immutable-idtbl (format-id #'idtbl "make-immutable-~a" (syntax-e #'idtbl))] [mutable-idtbl? (format-id #'idtbl "mutable-~a?" (syntax-e #'idtbl))] [immutable-idtbl? (format-id #'idtbl "immutable-~a?" (syntax-e #'idtbl))]) (define (s x) (format-id #'idtbl "~a~a" (syntax-e #'idtbl) x)) (with-syntax ([idtbl? (s '?)] [idtbl-ref (s '-ref)] [idtbl-set! (s '-set!)] [idtbl-set (s '-set)] [idtbl-remove! (s '-remove!)] [idtbl-remove (s '-remove)] [idtbl-set/constructor (s '-set/constructor)] [idtbl-remove/constructor (s '-remove/constructor)] [idtbl-count (s '-count)] [idtbl-iterate-first (s '-iterate-first)] [idtbl-iterate-next (s '-iterate-next)] [idtbl-iterate-key (s '-iterate-key)] [idtbl-iterate-value (s '-iterate-value)] [idtbl-map (s '-map)] [idtbl-for-each (s '-for-each)] [idtbl-mutable-methods (s '-mutable-methods)] [idtbl-immutable-methods (s '-immutable-methods)]) #'(begin ;; Struct defs at end, so that dict methods can refer to earlier procs (define (make-idtbl [init-dict null] #:phase [phase (syntax-local-phase-level)]) (make-id-table/constructor 'make-idtbl init-dict phase mutable-idtbl identifier->symbol identifier=?)) (define (make-immutable-idtbl [init-dict null] #:phase [phase (syntax-local-phase-level)]) (make-immutable-id-table/constructor 'make-immutable-idtbl init-dict phase immutable-idtbl identifier->symbol identifier=?)) (define (idtbl-ref d id [default not-given]) (id-table-ref 'idtbl-ref d id default identifier->symbol identifier=?)) (define (idtbl-set! d id v) (id-table-set! 'idtbl-set! d id v identifier->symbol identifier=?)) (define (idtbl-set/constructor d id v constructor) (id-table-set/constructor 'idtbl-set d id v constructor identifier->symbol identifier=?)) (define (idtbl-set d id v) (idtbl-set/constructor d id v immutable-idtbl)) (define (idtbl-remove! d id) (id-table-remove! 'idtbl-remove! d id identifier->symbol identifier=?)) (define (idtbl-remove/constructor d id constructor) (id-table-remove/constructor 'idtbl-remove d id constructor identifier->symbol identifier=?)) (define (idtbl-remove d id) (idtbl-remove/constructor d id immutable-idtbl)) (define (idtbl-count d) (id-table-count d)) (define (idtbl-for-each d p) (id-table-for-each d p)) (define (idtbl-map d f) (id-table-map d f)) (define (idtbl-iterate-first d) (id-table-iterate-first d)) (define (idtbl-iterate-next d pos) (id-table-iterate-next 'idtbl-iterate-next d pos)) (define (idtbl-iterate-key d pos) (id-table-iterate-key 'idtbl-iterate-key d pos)) (define (idtbl-iterate-value d pos) (id-table-iterate-value 'idtbl-iterate-value d pos)) (define idtbl-mutable-methods (vector-immutable idtbl-ref idtbl-set! #f idtbl-remove! #f id-table-count idtbl-iterate-first idtbl-iterate-next idtbl-iterate-key idtbl-iterate-value)) (define idtbl-immutable-methods (vector-immutable idtbl-ref #f idtbl-set #f idtbl-remove id-table-count idtbl-iterate-first idtbl-iterate-next idtbl-iterate-key idtbl-iterate-value)) (struct idtbl id-table ()) (struct mutable-idtbl idtbl () #:property prop:dict idtbl-mutable-methods) (struct immutable-idtbl idtbl () #:property prop:dict idtbl-immutable-methods) (provide make-idtbl make-immutable-idtbl idtbl? mutable-idtbl? immutable-idtbl? idtbl-ref idtbl-set! idtbl-set idtbl-remove! idtbl-remove idtbl-count idtbl-iterate-first idtbl-iterate-next idtbl-iterate-key idtbl-iterate-value idtbl-map idtbl-for-each ;; just for use/extension by syntax/id-table idtbl-set/constructor idtbl-remove/constructor idtbl-mutable-methods mutable-idtbl immutable-idtbl))))])) (define (bound-identifier->symbol id phase) (syntax-e id)) (make-code bound-id-table bound-identifier->symbol bound-identifier=?) (define (free-identifier->symbol id phase) (let ([binding (identifier-binding id phase)]) (if (pair? binding) (cadr binding) (syntax-e id)))) (make-code free-id-table free-identifier->symbol free-identifier=?) (provide id-table-iter?)