Be even more thorough about releasing port locks after exceptions.
This commit is contained in:
parent
f5d853712b
commit
b4312d028c
|
@ -80,11 +80,7 @@
|
||||||
(define (one-arg-proc->handler proc signal-exception)
|
(define (one-arg-proc->handler proc signal-exception)
|
||||||
(lambda (opcode reason port)
|
(lambda (opcode reason port)
|
||||||
(if (= reason (enum exception buffer-full/empty))
|
(if (= reason (enum exception buffer-full/empty))
|
||||||
(begin
|
(protect-port-op port (lambda () (proc port)))
|
||||||
(obtain-port-lock port)
|
|
||||||
(let ((value (proc port)))
|
|
||||||
(release-port-lock port)
|
|
||||||
value))
|
|
||||||
(signal-exception opcode reason port))))
|
(signal-exception opcode reason port))))
|
||||||
|
|
||||||
; This could combined with on-arg-... if the port were the first argument.
|
; This could combined with on-arg-... if the port were the first argument.
|
||||||
|
@ -92,11 +88,7 @@
|
||||||
(define (two-arg-proc->handler proc signal-exception)
|
(define (two-arg-proc->handler proc signal-exception)
|
||||||
(lambda (opcode reason arg port)
|
(lambda (opcode reason arg port)
|
||||||
(if (= reason (enum exception buffer-full/empty))
|
(if (= reason (enum exception buffer-full/empty))
|
||||||
(begin
|
(protect-port-op port (lambda () (proc arg port)))
|
||||||
(obtain-port-lock port)
|
|
||||||
(let ((value (proc arg port)))
|
|
||||||
(release-port-lock port)
|
|
||||||
value))
|
|
||||||
(signal-exception opcode reason arg port))))
|
(signal-exception opcode reason arg port))))
|
||||||
|
|
||||||
; If a character is available, use it; if there is an EOF waiting, use that;
|
; If a character is available, use it; if there is an EOF waiting, use that;
|
||||||
|
@ -163,31 +155,42 @@
|
||||||
(define port-flushed? port-pending-eof?)
|
(define port-flushed? port-pending-eof?)
|
||||||
(define set-port-flushed?! set-port-pending-eof?!)
|
(define set-port-flushed?! set-port-pending-eof?!)
|
||||||
|
|
||||||
|
(define (protect-port-op port thunk)
|
||||||
|
(obtain-port-lock port)
|
||||||
|
(let ((result
|
||||||
|
(with-handler
|
||||||
|
(lambda (condition punt)
|
||||||
|
(release-port-lock port)
|
||||||
|
(punt))
|
||||||
|
thunk)))
|
||||||
|
(release-port-lock port)
|
||||||
|
result))
|
||||||
|
|
||||||
;----------------
|
;----------------
|
||||||
; Closing is done with the appropriate handler.
|
; Closing is done with the appropriate handler.
|
||||||
; R4RS says that CLOSE-... is idempotent.
|
; R4RS says that CLOSE-... is idempotent.
|
||||||
|
|
||||||
(define (close-input-port port)
|
(define (close-input-port port)
|
||||||
(if (input-port? port)
|
(if (input-port? port)
|
||||||
(begin
|
(protect-port-op
|
||||||
(obtain-port-lock port)
|
port
|
||||||
(if (open-input-port? port)
|
(lambda ()
|
||||||
(begin
|
(if (open-input-port? port)
|
||||||
(make-input-port-closed! port)
|
(begin
|
||||||
((port-handler-close (port-handler port)) (port-data port))))
|
(make-input-port-closed! port)
|
||||||
(release-port-lock port))
|
((port-handler-close (port-handler port)) (port-data port))))))
|
||||||
(call-error "invalid argument" close-input-port port)))
|
(call-error "invalid argument" close-input-port port)))
|
||||||
|
|
||||||
(define (close-output-port port)
|
(define (close-output-port port)
|
||||||
(if (output-port? port)
|
(if (output-port? port)
|
||||||
(begin
|
(protect-port-op
|
||||||
(obtain-port-lock port)
|
port
|
||||||
(if (open-output-port? port)
|
(lambda ()
|
||||||
(begin
|
(if (open-output-port? port)
|
||||||
(really-force-output port)
|
(begin
|
||||||
(make-output-port-closed! port)
|
(make-output-port-closed! port)
|
||||||
((port-handler-close (port-handler port)) (port-data port))))
|
(really-force-output port)
|
||||||
(release-port-lock port))
|
((port-handler-close (port-handler port)) (port-data port))))))
|
||||||
(call-error "invalid argument" close-output-port port)))
|
(call-error "invalid argument" close-output-port port)))
|
||||||
|
|
||||||
;----------------
|
;----------------
|
||||||
|
@ -203,22 +206,12 @@
|
||||||
((define-port-op (?id ?args ...) ?port ?predicate ?body)
|
((define-port-op (?id ?args ...) ?port ?predicate ?body)
|
||||||
(define (?id ?args ...)
|
(define (?id ?args ...)
|
||||||
(if ?predicate ; if args are okay
|
(if ?predicate ; if args are okay
|
||||||
(begin
|
(protect-port-op
|
||||||
(obtain-port-lock ?port) ; lock the port
|
?port
|
||||||
|
(lambda ()
|
||||||
(if (open-port? ?port) ; check that it's open
|
(if (open-port? ?port) ; check that it's open
|
||||||
(let ((value ; do the work
|
?body
|
||||||
(with-handler
|
(call-error "invalid argument" ?id ?args ...))))
|
||||||
(lambda (condition punt)
|
|
||||||
|
|
||||||
(release-port-lock ?port)
|
|
||||||
(punt))
|
|
||||||
(lambda ()
|
|
||||||
?body))))
|
|
||||||
(release-port-lock ?port) ; release the lock
|
|
||||||
value) ; return
|
|
||||||
(begin
|
|
||||||
(release-port-lock ?port)
|
|
||||||
(call-error "invalid argument" ?id ?args ...))))
|
|
||||||
(call-error "invalid argument" ?id ?args ...))))
|
(call-error "invalid argument" ?id ?args ...))))
|
||||||
|
|
||||||
; ?port defaults to the first argument
|
; ?port defaults to the first argument
|
||||||
|
@ -340,14 +333,19 @@
|
||||||
(call-error "invalid argument" output-port-ready? port))
|
(call-error "invalid argument" output-port-ready? port))
|
||||||
((not (maybe-obtain-port-lock port))
|
((not (maybe-obtain-port-lock port))
|
||||||
#f)
|
#f)
|
||||||
((not (open-port? port)) ; have to check again after the lock call
|
((not (open-port? port)) ; have to check again after the lock call
|
||||||
(release-port-lock port)
|
(release-port-lock port)
|
||||||
(call-error "invalid argument" output-port-ready? port))
|
(call-error "invalid argument" output-port-ready? port))
|
||||||
(else
|
(else
|
||||||
(let ((val ((port-handler-ready? (port-handler port))
|
(with-handler
|
||||||
port)))
|
(lambda (condition punt)
|
||||||
(release-port-lock port)
|
(release-port-lock port)
|
||||||
val))))
|
(punt))
|
||||||
|
(lambda ()
|
||||||
|
(let ((val ((port-handler-ready? (port-handler port))
|
||||||
|
port)))
|
||||||
|
(release-port-lock port)
|
||||||
|
val))))))
|
||||||
|
|
||||||
; Copy the bytes into the buffer if there is room, otherwise write out anything
|
; Copy the bytes into the buffer if there is room, otherwise write out anything
|
||||||
; in the buffer and then write BUFFER.
|
; in the buffer and then write BUFFER.
|
||||||
|
@ -390,11 +388,11 @@
|
||||||
|
|
||||||
(define (force-output-if-open port)
|
(define (force-output-if-open port)
|
||||||
(if (output-port? port)
|
(if (output-port? port)
|
||||||
(begin
|
(protect-port-op
|
||||||
(obtain-port-lock port)
|
port
|
||||||
(if (open-output-port? port)
|
(lambda ()
|
||||||
(really-force-output port))
|
(if (open-output-port? port)
|
||||||
(release-port-lock port))
|
(really-force-output port))))
|
||||||
(call-error "invalid argument" force-output-if-open port)))
|
(call-error "invalid argument" force-output-if-open port)))
|
||||||
|
|
||||||
;----------------
|
;----------------
|
||||||
|
@ -608,18 +606,34 @@
|
||||||
(if (and (report-errors-as-warnings
|
(if (and (report-errors-as-warnings
|
||||||
(lambda ()
|
(lambda ()
|
||||||
(if (maybe-obtain-port-lock port)
|
(if (maybe-obtain-port-lock port)
|
||||||
(begin
|
(with-handler
|
||||||
(really-force-output port)
|
(lambda (condition punt)
|
||||||
(release-port-lock port))))
|
(release-port-lock port)
|
||||||
|
(punt))
|
||||||
|
(lambda ()
|
||||||
|
(really-force-output port)
|
||||||
|
(release-port-lock port)))))
|
||||||
"error when flushing buffer; closing port"
|
"error when flushing buffer; closing port"
|
||||||
port)
|
port)
|
||||||
(open-output-port? port))
|
(open-output-port? port))
|
||||||
(report-errors-as-warnings
|
(report-errors-as-warnings
|
||||||
(lambda ()
|
(lambda ()
|
||||||
(make-output-port-closed! port)
|
(make-output-port-closed! port)
|
||||||
((port-handler-close (port-handler port)) (port-data port)))
|
((port-handler-close (port-handler port)) (port-data port)))
|
||||||
"error when closing port"
|
"error when closing port"
|
||||||
port))))
|
port))))
|
||||||
|
|
||||||
|
;----------------
|
||||||
|
; The following is used to make the REPL's input, output, and error ports
|
||||||
|
; available after a keyboard interrupt. If PORT is a locked channel port
|
||||||
|
; we save the its state and then reinitialize it. The OS is told to
|
||||||
|
; abort any pending operation on the port's channel. Finally, the owning
|
||||||
|
; thread's continuation is munged to restore the port when the thread
|
||||||
|
; resumes. Any buffered input is thrown away at that point (it could
|
||||||
|
; be saved away with the channel).
|
||||||
|
;
|
||||||
|
; If the port is locked by us or one of our ancestors there is no point in
|
||||||
|
; trying to grab it.
|
||||||
|
|
||||||
(define (steal-port! port)
|
(define (steal-port! port)
|
||||||
(begin
|
(begin
|
||||||
|
@ -652,17 +666,25 @@
|
||||||
(interrupt-thread owner
|
(interrupt-thread owner
|
||||||
(lambda results
|
(lambda results
|
||||||
(obtain-port-lock port)
|
(obtain-port-lock port)
|
||||||
(cond ((output-port? port)
|
(let ((cleanup
|
||||||
(really-force-output port))
|
(lambda ()
|
||||||
((< (port-index port)
|
(set-port-buffer! port buffer)
|
||||||
(port-limit port))
|
(set-port-index! port index)
|
||||||
(warn "dropping input from port" port)))
|
(set-port-limit! port limit)
|
||||||
(set-port-buffer! port buffer)
|
(set-port-pending-eof?! port eof?)
|
||||||
(set-port-index! port index)
|
(set-port-lock! port lock))))
|
||||||
(set-port-limit! port limit)
|
(with-handler
|
||||||
(set-port-pending-eof?! port eof?)
|
(lambda (condition punt)
|
||||||
(set-port-lock! port lock)
|
(cleanup)
|
||||||
(or status (apply values results))))
|
(punt))
|
||||||
|
(lambda ()
|
||||||
|
(cond ((output-port? port)
|
||||||
|
(really-force-output port))
|
||||||
|
((< (port-index port)
|
||||||
|
(port-limit port))
|
||||||
|
(warn "dropping input from port" port)))
|
||||||
|
(cleanup)
|
||||||
|
(or status (apply values results)))))))
|
||||||
; if we took OWNER off a channel-wait queue we need to make it ready to run
|
; if we took OWNER off a channel-wait queue we need to make it ready to run
|
||||||
(if status (make-ready owner))))
|
(if status (make-ready owner))))
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue