X-Git-Url: http://repo.macrolet.net/gitweb/?a=blobdiff_plain;f=src%2Fcode%2Fthread.lisp;h=378fb4b159ce8909996660047bd21c660901f7c6;hb=48ec282d877900caf5ea4ab42e9d87e566ce6b43;hp=1e622ce9e51f88a82419bc074a2b238b8444d137;hpb=ec066d84dd46611428943d152749b3891a3f4b7c;p=sbcl.git diff --git a/src/code/thread.lisp b/src/code/thread.lisp index 1e622ce..378fb4b 100644 --- a/src/code/thread.lisp +++ b/src/code/thread.lisp @@ -11,15 +11,36 @@ (in-package "SB!THREAD") -(sb!xc:defmacro with-mutex ((mutex &key value (wait-p t)) &body body) +(def!struct mutex + #!+sb-doc + "Mutex type." + (name nil :type (or null simple-string)) + (value nil)) + +(def!struct spinlock + #!+sb-doc + "Spinlock type." + (name nil :type (or null simple-string)) + (value 0)) + +(sb!xc:defmacro with-mutex ((mutex &key (value '*current-thread*) (wait-p t)) + &body body) + #!+sb-doc + "Acquire MUTEX for the dynamic scope of BODY, setting it to +NEW-VALUE or some suitable default value if NIL. If WAIT-P is non-NIL +and the mutex is in use, sleep until it is available" #!-sb-thread (declare (ignore mutex value wait-p)) #!+sb-thread - (with-unique-names (got) - `(let ((,got (get-mutex ,mutex ,value ,wait-p))) - (when ,got - (unwind-protect - (locally ,@body) - (release-mutex ,mutex))))) + (with-unique-names (got mutex1) + `(let ((,mutex1 ,mutex) + ,got) + (unwind-protect + ;; FIXME: async unwind in SETQ form + (when (setq ,got (get-mutex ,mutex1 ,value ,wait-p)) + (locally + ,@body)) + (when ,got + (release-mutex ,mutex1))))) ;; KLUDGE: this separate expansion for (NOT SB-THREAD) is not ;; strictly necessary; GET-MUTEX and RELEASE-MUTEX are implemented. ;; However, there would be a (possibly slight) performance hit in @@ -28,29 +49,24 @@ `(locally ,@body)) (sb!xc:defmacro with-recursive-lock ((mutex) &body body) - #!-sb-thread (declare (ignore mutex)) + #!+sb-doc + "Acquires MUTEX for the dynamic scope of BODY. Within that scope +further recursive lock attempts for the same mutex succeed. It is +allowed to mix WITH-MUTEX and WITH-RECURSIVE-LOCK for the same mutex +provided the default value is used for the mutex." + #!-sb-thread + (declare (ignore mutex)) #!+sb-thread - (with-unique-names (cfp) - `(let ((,cfp (sb!kernel:current-fp))) - (unless (and (mutex-value ,mutex) - (sb!vm:control-stack-pointer-valid-p - (sb!sys:int-sap - (sb!kernel:get-lisp-obj-address (mutex-value ,mutex))))) - ;; this punning with MAKE-LISP-OBJ depends for its safety on - ;; the frame pointer being a lispobj-aligned integer. While - ;; it is, then MAKE-LISP-OBJ will always return a FIXNUM, so - ;; we're safe to do that. Should this ever change, this - ;; MAKE-LISP-OBJ could return something that looks like a - ;; pointer, but pointing into neverneverland, which will - ;; confuse GC completely. -- CSR, 2003-06-03 - (get-mutex ,mutex (sb!kernel:make-lisp-obj (sb!sys:sap-int ,cfp)))) - (unwind-protect - (locally ,@body) - (when (sb!sys:sap= (sb!sys:int-sap - (sb!kernel:get-lisp-obj-address - (mutex-value ,mutex))) - ,cfp) - (release-mutex ,mutex))))) + (with-unique-names (mutex1 inner-lock-p) + `(let* ((,mutex1 ,mutex) + (,inner-lock-p (eq (mutex-value ,mutex1) *current-thread*))) + (unwind-protect + (progn + (unless ,inner-lock-p + (get-mutex ,mutex1)) + (locally + ,@body)) + (unless ,inner-lock-p + (release-mutex ,mutex1))))) #!-sb-thread `(locally ,@body)) -