+ Lock for the process table

+ Use wait-lock to glue waiting and the process object together
+ Delete the pid/weak-pointer pair in the process table if the process object
  gets finalized
This commit is contained in:
mainzelm 2002-02-13 14:56:11 +00:00
parent 47d41db14e
commit 54efae2318
1 changed files with 60 additions and 37 deletions

View File

@ -42,11 +42,16 @@
(make-reinitializer (lambda () (make-reinitializer (lambda ()
(set! process-table (make-integer-table)))) (set! process-table (make-integer-table))))
(define process-table-lock (make-lock))
(define (process-table-ref n) (define (process-table-ref n)
(weak-table-ref process-table n)) (with-lock process-table-lock
(lambda ()
(weak-table-ref process-table n))))
(define (process-table-set! n val) (define (process-table-set! n val)
(weak-table-set! process-table n val)) (with-lock process-table-lock
(lambda ()
(weak-table-set! process-table n val))))
(define (maybe-pid->proc pid) (define (maybe-pid->proc pid)
(process-table-ref pid)) (process-table-ref pid))
@ -150,8 +155,8 @@
(let lp ((event (most-recent-sigevent))) (let lp ((event (most-recent-sigevent)))
(let ((next-event (next-sigevent event interrupt/post-gc))) (let ((next-event (next-sigevent event interrupt/post-gc)))
(*post/gc-handler*) (*post/gc-handler*)
(lp next-event)))) (lp next-event))))
'*post/gc-handler*-thread)) '*post/gc-handler*-thread))
(define set-post/gc-handler! start-set-post/gc-handler!) (define set-post/gc-handler! start-set-post/gc-handler!)
@ -195,10 +200,13 @@
;;; reap this special pid ;;; reap this special pid
;;; return status or #f ;;; return status or #f
(define (reap-pid pid) (define (reap-pid pid)
(let ((status (atomic-wait pid wait/poll))) (with-lock
(if status wait-lock
(waited-by-reap pid status)) (lambda ()
status)) (let ((status (atomic-wait pid wait/poll)))
(if status
(waited-by-reap pid status))
status))))
;;; Handler for SIGCHLD according policy ;;; Handler for SIGCHLD according policy
(define (late-sigchld-handler) #f) (define (late-sigchld-handler) #f)
@ -212,6 +220,7 @@
;;; Finalizer for procobjs ;;; Finalizer for procobjs
;;; ;;;
(define (procobj-finalizer procobj) (define (procobj-finalizer procobj)
(process-table-set! (proc:pid procobj) #f)
(if (not (proc:finished? procobj)) (if (not (proc:finished? procobj))
(need-reaping-add! (proc:pid procobj)))) (need-reaping-add! (proc:pid procobj))))
@ -222,14 +231,18 @@
(define (reap-zombies) (define (reap-zombies)
(let lp () (let lp ()
(obtain-lock wait-lock)
(receive (pid status) (receive (pid status)
(%wait-any (bitwise-ior wait/poll wait/stopped-children)) (%wait-any (bitwise-ior wait/poll wait/stopped-children))
(if pid (if pid
(begin (waited-by-reap pid status) (begin (waited-by-reap pid status)
(release-lock wait-lock)
; (format (current-error-port) ; (format (current-error-port)
; "Reaping ~d[~d]~%" pid status) ; "Reaping ~d[~d]~%" pid status)
(lp)) (lp))
status)))) (begin
(release-lock wait-lock)
status)))))
@ -260,7 +273,7 @@
;;; With this lock, we ensure that only one thread may call ;;; With this lock, we ensure that only one thread may call
;;; really-wait for a given pid ;;; really-wait for a given pid and manipulates the associated process object
(define wait-lock (make-lock)) (define wait-lock (make-lock))
@ -272,26 +285,29 @@
status))) status)))
;; save the event before we check for finished ;; save the event before we check for finished
(let ((pre-event (most-recent-sigevent))) (let ((pre-event (most-recent-sigevent)))
(cond ((atomic-wait proc (bitwise-ior flags wait/poll)) => win) (with-lock
wait-lock
(lambda ()
(cond ((atomic-wait proc (bitwise-ior flags wait/poll)) => win)
((zero? (bitwise-and flags wait/poll)) ((zero? (bitwise-and flags wait/poll))
;; we have to block and hence use the event system ;; we have to block and hence use the event system
(let lp ((pre-event pre-event)) (let lp ((pre-event pre-event))
(cond ((atomic-wait proc (bitwise-ior flags wait/poll)) (cond ((atomic-wait proc (bitwise-ior flags wait/poll))
=> win) => win)
(else (else
(lp (next-sigevent pre-event interrupt/chld)))))) (release-lock wait-lock)
(else #f))))) (let ((next-event (next-sigevent pre-event interrupt/chld)))
(obtain-lock wait-lock)
(lp next-event))))))
(else #f)))))))
;;; -> process-object proc status/#f ;;; -> process-object proc status/#f
(define (atomic-wait proc flags) (define (atomic-wait proc flags)
(with-lock (cond ((proc:finished? proc)
wait-lock (placeholder-value (proc:status proc)))
(lambda () (else (really-wait (proc:pid proc) (bitwise-ior flags wait/poll)))))
(cond ((proc:finished? proc)
(placeholder-value (proc:status proc)))
(else (really-wait (proc:pid proc) (bitwise-ior flags wait/poll)))))))
;;; This one is used, to wait on a positive pid ;;; This one is used, to wait on a positive pid
;;; We NEVER do a blocking wait syscall ;;; We NEVER do a blocking wait syscall
@ -315,7 +331,8 @@
(cond ((maybe-pid->proc pid) => (cond ((maybe-pid->proc pid) =>
(lambda (proc) (lambda (proc)
(obituary proc status) (obituary proc status)
(push-reaped-proc proc))))) (push-reaped-proc proc)
))))
;;; All you have to do, if a wait on proc was successful ;;; All you have to do, if a wait on proc was successful
@ -373,13 +390,16 @@
(define (really-wait-any flags) (define (really-wait-any flags)
(if (zero? (bitwise-and flags wait/poll)) (if (zero? (bitwise-and flags wait/poll))
(error "real-wait-any without wait/poll" flags)) (error "real-wait-any without wait/poll" flags))
(receive (pid status) (with-lock
(%wait-any flags) wait-lock
(if pid (lambda ()
(let ((proc (new-child-proc pid))) (receive (pid status)
(waited-by-wait proc status) (%wait-any flags)
(values proc status)) (if pid
(values #f #f)))) (let ((proc (new-child-proc pid)))
(waited-by-wait proc status)
(values proc status))
(values #f #f))))))
;;; (wait-process-group [proc-group flags]) => [proc status] ;;; (wait-process-group [proc-group flags]) => [proc status]
@ -424,7 +444,10 @@
;;; (%wait-any flags) (%wait-pid pid flags) (%wait-process-group pgrp flags) ;;; (%wait-any flags) (%wait-pid pid flags) (%wait-process-group pgrp flags)
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; Direct interfaces to waitpid(2) call. ;;; Direct interfaces to waitpid(2) call. As opposed to %wait-pid this
;;; waits on any child (using -1) and gets along if no child is alive
;;; at all (i.e. catches errno/child).
;;; [#f #f] means no processes ready on a non-blocking wait. ;;; [#f #f] means no processes ready on a non-blocking wait.
;;; [#f #t] means no waitable process on wait-any. ;;; [#f #t] means no waitable process on wait-any.