From 7128169273aeacedc82848820d5f52112dcf253f Mon Sep 17 00:00:00 2001 From: Nikodemus Siivola Date: Fri, 29 Feb 2008 11:04:43 +0000 Subject: [PATCH] 1.0.15.6: split main part of SUB-SERVE-EVENT into SUB-SUB-SERVE-EVENT * Easier to understand, fixes periodic polling. Patch by Espen S Johnsen. * NEWS entry for 1.0.15.5. as well. --- NEWS | 3 + src/code/serve-event.lisp | 150 ++++++++++++++++++++++----------------------- version.lisp-expr | 2 +- 3 files changed, 77 insertions(+), 78 deletions(-) diff --git a/NEWS b/NEWS index f588a4b..81fda56 100644 --- a/NEWS +++ b/NEWS @@ -2,6 +2,9 @@ changes in sbcl-1.0.16 relative to 1.0.15: * minor incompatible change: change PROBE-FILE back to returning NIL whenever we can't get a truename, as was the case before 1.0.14. + * bug fix: periodic polling was broken. (thanks to Espen S Johnsen) + * bug fix: copying output from RUN-PROGRAM to a stream signalled + bogus errors if select() was interrupted. * enhancement: add support for fcntl's struct flock to SB-POSIX. changes in sbcl-1.0.15 relative to sbcl-1.0.14: diff --git a/src/code/serve-event.lisp b/src/code/serve-event.lisp index aa35856..b1b8aaf 100644 --- a/src/code/serve-event.lisp +++ b/src/code/serve-event.lisp @@ -215,85 +215,81 @@ threads, unless locally bound. EXPERIMENTAL.") at maximum, before calling the *PERIODIC-POLLING-FUNCTION* \(if any.) Shared between all threads, unless locally bound. EXPERIMENTAL.") -;;; Takes timeout broken into seconds and microseconds. +;;; Takes timeout broken into seconds and microseconds, NIL timeout means +;;; to wait as long as needed. (defun sub-serve-event (to-sec to-usec deadlinep) - ;; Figure out our peridic polling needs. MORE-SEC/USEC is the amount - ;; of actual waiting left after we poll (assuming we are polling.) - (multiple-value-bind (poll more-sec more-usec) - (when *periodic-polling-function* - (multiple-value-bind (p-sec p-usec) - (decode-internal-time - (seconds-to-internal-time *periodic-polling-period*)) - (when (or (not to-sec) (> to-sec p-sec) - (and (= to-sec p-sec) (> to-usec p-usec))) - (multiple-value-prog1 - (values *periodic-polling-function* - (when to-sec (- to-sec p-sec)) - (when to-sec (- to-usec p-usec))) - (setf to-sec p-sec - to-usec p-sec))))) + (or + (if *periodic-polling-function* + (multiple-value-bind (p-sec p-usec) + (decode-internal-time + (seconds-to-internal-time *periodic-polling-period*)) + (if to-sec + (loop repeat (/ (+ to-sec (/ to-usec 1e6)) + *periodic-polling-period*) + thereis (sub-sub-serve-event p-sec p-usec) + do (funcall *periodic-polling-function*)) + (loop thereis (sub-sub-serve-event p-sec p-usec) + do (funcall *periodic-polling-function*)))) + (sub-sub-serve-event to-sec to-usec)) + (when deadlinep + (signal-deadline)))) - ;; Next, wait for something to happen. - (sb!alien:with-alien ((read-fds (sb!alien:struct sb!unix:fd-set)) - (write-fds (sb!alien:struct sb!unix:fd-set))) - (sb!unix:fd-zero read-fds) - (sb!unix:fd-zero write-fds) - (let ((count 0)) - (declare (type index count)) +;;; Handles the work of the above, except for periodic polling. Returns +;;; true if something of interest happened. +(defun sub-sub-serve-event (to-sec to-usec) + (sb!alien:with-alien ((read-fds (sb!alien:struct sb!unix:fd-set)) + (write-fds (sb!alien:struct sb!unix:fd-set))) + (sb!unix:fd-zero read-fds) + (sb!unix:fd-zero write-fds) + (let ((count 0)) + (declare (type index count)) - ;; Initialize the fd-sets for UNIX-SELECT and return the active - ;; descriptor count. - (map-descriptor-handlers - (lambda (handler) - ;; FIXME: If HANDLER-ACTIVE ever is reinstanted, it needs - ;; to be checked here in addition to HANDLER-BOGUS - (unless (handler-bogus handler) - (let ((fd (handler-descriptor handler))) - (ecase (handler-direction handler) - (:input (sb!unix:fd-set fd read-fds)) - (:output (sb!unix:fd-set fd write-fds))) - (when (> fd count) - (setf count fd)))))) - (incf count) + ;; Initialize the fd-sets for UNIX-SELECT and return the active + ;; descriptor count. + (map-descriptor-handlers + (lambda (handler) + ;; FIXME: If HANDLER-ACTIVE ever is reinstanted, it needs + ;; to be checked here in addition to HANDLER-BOGUS + (unless (handler-bogus handler) + (let ((fd (handler-descriptor handler))) + (ecase (handler-direction handler) + (:input (sb!unix:fd-set fd read-fds)) + (:output (sb!unix:fd-set fd write-fds))) + (when (> fd count) + (setf count fd)))))) + (incf count) - ;; Next, wait for something to happen. - (multiple-value-bind (value err) - (sb!unix:unix-fast-select count - (sb!alien:addr read-fds) - (sb!alien:addr write-fds) - nil to-sec to-usec) - #!+win32 - (declare (ignore err)) - ;; Now see what it was (if anything) - (cond ((not value) - ;; Interrupted or one of the file descriptors is bad. - ;; FIXME: Check for other errnos. Why do we return true - ;; when interrupted? - #!-win32 - (if (eql err sb!unix:eintr) - t - (handler-descriptors-error)) - #!+win32 - (handler-descriptors-error)) - ((plusp value) - ;; Got something. Call file descriptor handlers - ;; according to the readable and writable masks - ;; returned by select. - (dolist (handler - (select-descriptor-handlers - (lambda (handler) - (let ((fd (handler-descriptor handler))) - (ecase (handler-direction handler) - (:input (sb!unix:fd-isset fd read-fds)) - (:output (sb!unix:fd-isset fd write-fds))))))) - (funcall (handler-function handler) - (handler-descriptor handler))) - t) - ((zerop value) - ;; Timeout. - (cond (poll - (funcall poll) - (sub-serve-event more-sec more-usec deadlinep)) - (deadlinep - (signal-deadline)))))))))) + ;; Next, wait for something to happen. + (multiple-value-bind (value err) + (sb!unix:unix-fast-select count + (sb!alien:addr read-fds) + (sb!alien:addr write-fds) + nil to-sec to-usec) + #!+win32 + (declare (ignore err)) + ;; Now see what it was (if anything) + (cond ((not value) + ;; Interrupted or one of the file descriptors is bad. + ;; FIXME: Check for other errnos. Why do we return true + ;; when interrupted? + #!-win32 + (if (eql err sb!unix:eintr) + t + (handler-descriptors-error)) + #!+win32 + (handler-descriptors-error)) + ((plusp value) + ;; Got something. Call file descriptor handlers + ;; according to the readable and writable masks + ;; returned by select. + (dolist (handler + (select-descriptor-handlers + (lambda (handler) + (let ((fd (handler-descriptor handler))) + (ecase (handler-direction handler) + (:input (sb!unix:fd-isset fd read-fds)) + (:output (sb!unix:fd-isset fd write-fds))))))) + (funcall (handler-function handler) + (handler-descriptor handler))) + t)))))) diff --git a/version.lisp-expr b/version.lisp-expr index b9a44bb..a1dc484 100644 --- a/version.lisp-expr +++ b/version.lisp-expr @@ -17,4 +17,4 @@ ;;; checkins which aren't released. (And occasionally for internal ;;; versions, especially for internal versions off the main CVS ;;; branch, it gets hairier, e.g. "0.pre7.14.flaky4.13".) -"1.0.15.5" +"1.0.15.6" -- 1.7.10.4