Use a weak integer-table to store the fdports dropping the limit on

the number of ports (fixes #433867).
This commit is contained in:
mainzelm 2001-06-21 08:30:51 +00:00
parent 4659f69310
commit 921ecb9320
1 changed files with 27 additions and 25 deletions

View File

@ -6,36 +6,40 @@
channel channel
revealed) revealed)
(define max-fdport 255)
; This stuff is _weak_. ; This stuff is _weak_.
; Vector of weak pointers mapping fd -> fdport. ; Vector of weak pointers mapping fd -> fdport.
(define (weak-vector-set! vector number set-me) (define (weak-table-set! table number set-me)
(vector-set! vector number (make-weak-pointer set-me))) (table-set! table number (make-weak-pointer set-me)))
(define (weak-vector-ref vector number) (define (weak-table-ref table number)
(let ((ref (vector-ref vector number))) (let ((ref (table-ref table number)))
(if (weak-pointer? ref) (weak-pointer-ref ref) ref))) (if (weak-pointer? ref) (weak-pointer-ref ref) ref)))
(define (strengthen-weak-vector-ref vector number) (define (weak-table-walk proc table)
(vector-set! vector number (weak-vector-ref vector number))) (table-walk
(lambda (number value)
(if (weak-pointer? value) (weak-pointer-ref value) value))
table))
(define (weaken-weak-vector-ref vector number) (define (strengthen-weak-table-ref table number)
(weak-vector-set! vector number (weak-vector-ref vector number))) (table-set! table number (weak-table-ref vector number)))
(define fdports (make-vector max-fdport #f)) (define (weaken-weak-table-ref table number)
(weak-table-set! table number (weak-table-ref vector number)))
(define fdports (make-integer-table))
(define (install-fdport fdport) (define (install-fdport fdport)
(let* ((fdport* (fdport-data fdport)) (let* ((fdport* (fdport-data fdport))
(ch (fdport-data:channel fdport*)) (ch (fdport-data:channel fdport*))
(ch-number (channel-os-index ch))) (ch-number (channel-os-index ch)))
(if (not (= (fdport-data:revealed fdport*) 0)) (if (not (= (fdport-data:revealed fdport*) 0))
(vector-set! fdports ch-number fdport) (table-set! fdports ch-number fdport)
(weak-vector-set! fdports ch-number fdport)))) (weak-table-set! fdports ch-number fdport))))
(define (maybe-fdes->port fdes) (define (maybe-fdes->port fdes)
(weak-vector-ref fdports fdes)) (weak-table-ref fdports fdes))
;Hmm... these shouldn't be necessary. But still. ;Hmm... these shouldn't be necessary. But still.
;Fake defrec routines for backwards compatibility. ;Fake defrec routines for backwards compatibility.
@ -133,7 +137,7 @@
(define null-func (lambda args #t)) (define null-func (lambda args #t))
(define (close-fdport* fdport*) (define (close-fdport* fdport*)
(vector-set! fdports (channel-os-index (fdport-data:channel fdport*)) #f) (table-set! fdports (channel-os-index (fdport-data:channel fdport*)) #f)
(close-channel (fdport-data:channel fdport*))) (close-channel (fdport-data:channel fdport*)))
;The handlers drop straight through to the convenient channel routines. ;The handlers drop straight through to the convenient channel routines.
@ -376,7 +380,7 @@
(set-fdport-data:revealed data newcount) (set-fdport-data:revealed data newcount)
(if (and (zero? count) (> newcount 0)) ; We just became revealed, (if (and (zero? count) (> newcount 0)) ; We just became revealed,
(begin (begin
(strengthen-weak-vector-ref fdports (fdport-data:fd data)) (strengthen-weak-table-ref fdports (fdport-data:fd data))
(%set-cloexec (fdport-data:fd data) #f)))); so don't close on exec(). (%set-cloexec (fdport-data:fd data) #f)))); so don't close on exec().
(release-port-lock port)) (release-port-lock port))
@ -392,7 +396,7 @@
(set-fdport-data:revealed data new-rev) (set-fdport-data:revealed data new-rev)
(if (zero? new-rev) ; We just became unrevealed, so (if (zero? new-rev) ; We just became unrevealed, so
(begin ; the fd can be closed on exec. (begin ; the fd can be closed on exec.
(weaken-weak-vector-ref fdports (fdport-data:fd data)) (weaken-weak-table-ref fdports (fdport-data:fd data))
(%set-cloexec (fdport-data:fd data) #t)))))) (%set-cloexec (fdport-data:fd data) #t))))))
(release-port-lock port)) (release-port-lock port))
@ -520,12 +524,12 @@
(old-fd (channel-os-index ch)) (old-fd (channel-os-index ch))
(old-vector-ref (vector-ref fdports old-fd))) (old-vector-ref (vector-ref fdports old-fd)))
(set-fdport-data:revealed fdport* new-revealed) (set-fdport-data:revealed fdport* new-revealed)
(vector-set! fdports old-fd #f) (table-set! fdports old-fd #f)
(close-channel ch) (close-channel ch)
(set-fdport-data:channel (set-fdport-data:channel
fdport* fdport*
(make-fd-channel port fd)) (make-fd-channel port fd))
(vector-set! fdports fd old-vector-ref) (table-set! fdports fd old-vector-ref)
(%set-cloexec fd (not new-revealed))) (%set-cloexec fd (not new-revealed)))
(release-port-lock port) (release-port-lock port)
#f) ; JMG: It used to return #f on succes in 0.5.1, so we do the same #f) ; JMG: It used to return #f on succes in 0.5.1, so we do the same
@ -542,12 +546,10 @@
(force-output fdport)) (force-output fdport))
(define (flush-all-ports) (define (flush-all-ports)
(let loop ((i 0)) (weak-table-walk
(if (< i max-fdport) (lambda (i fdport)
(begin (if (and fdport (output-port? fdport)) (flush-fdport fdport)))
(let ((fdport (weak-vector-ref fdports i))) fdports))
(if (and fdport (output-port? fdport) ) (flush-fdport fdport)))
(loop (+ i 1))))))
;;; Extend R4RS i/o ops to handle file descriptors. ;;; Extend R4RS i/o ops to handle file descriptors.
;;; ----------------------------------------------- ;;; -----------------------------------------------