From: Nikodemus Siivola Date: Fri, 8 Jun 2007 12:47:50 +0000 (+0000) Subject: 1.0.6.37: thread safety test for the CLOS cache X-Git-Url: http://repo.macrolet.net/gitweb/?a=commitdiff_plain;h=d8659f1e656234e8f0f47d5295b503dd6cff4aba;p=sbcl.git 1.0.6.37: thread safety test for the CLOS cache ...better late then never. --- diff --git a/tests/clos-cache.impure.lisp b/tests/clos-cache.impure.lisp new file mode 100644 index 0000000..e7a2971 --- /dev/null +++ b/tests/clos-cache.impure.lisp @@ -0,0 +1,89 @@ +;;;; testing clos cache + +;;;; This software is part of the SBCL system. See the README file for +;;;; more information. +;;;; +;;;; While most of SBCL is derived from the CMU CL system, the test +;;;; files (like this one) were written from scratch after the fork +;;;; from CMU CL. +;;;; +;;;; This software is in the public domain and is provided with +;;;; absolutely no warranty. See the COPYING and CREDITS files for +;;;; more information. + +(defpackage "CLOS-CACHE-TEST" + (:use "COMMON-LISP")) + +(in-package "CLOS-CACHE-TEST") + +;;;; Make a GF, populate it with a ton of methods, and then hammer +;;;; it with multiple threads. On 1.0.6 this would have failed with +;;;; "NIL is not an SB-KERNEL:LAYOUT" pretty quickly. + +(defgeneric cache-test (x y)) + +(defvar *cache-test-classes* nil) + +(macrolet ((def () + `(progn + (defmethod cache-test (x y) + (list t t)) + ,@(loop for i from 0 upto 128 + collect + (let ((c (intern (format nil "CACHE-TEST-CLASS-~S" i)))) + `(progn + (defclass ,c () ()) + (defmethod cache-test ((x ,c) (y ,c)) + (list x y)) + (defmethod cache-test ((x ,c) y) + (list x t)) + (defmethod cache-test (x (y ,c)) + (list t y)) + (push (find-class ',c) *cache-test-classes*))))))) + (def)) + +(defvar *run-cache-test* nil) + +(let* ((instances (map 'vector #'make-instance *cache-test-classes*)) + (limit (length instances))) + (defun test-cache () + (let* ((r (random limit)) + (instance (svref instances r))) + (if (logbitp 0 r) + (if (logbitp 1 r) + (assert (equal (cache-test r r) '(t t))) + (assert (equal (cache-test r instance) (list t instance)))) + (if (logbitp 1 r) + (assert (equal (cache-test instance r) (list instance t))) + (assert (equal (cache-test instance instance) (list instance instance)))))))) + +(let ((lock (sb-thread:make-mutex))) + (defun note (control &rest args) + (let ((string (apply #'format nil control args))) + (sb-thread:with-mutex (lock) + (write-line string))))) + +(defun test-loop () + (note "/~S waiting for permission to run" sb-thread:*current-thread*) + (loop until *run-cache-test*) + (note "/~S joining the tundering herd" sb-thread:*current-thread*) + (handler-case + (loop repeat 1024 do (test-cache)) + (error (e) + (note "~&Error in cache test in ~S:~%~A~%...aborting" + sb-thread:*current-thread* e) + (sb-ext:quit :unix-status 1))) + (note "/~S done" sb-thread:*current-thread*)) + +#+sb-thread +(let ((threads (loop repeat 32 + collect (sb-thread:make-thread 'test-loop)))) + (setf *run-cache-test* t) + (mapcar #'sb-thread:join-thread threads)) + +#-sb-thread +(loop repeat 4 + do (test-loop)) + +;;; Check that the test tests what it was supposed to test: the cache. +(assert (sb-pcl::cache-p (sb-pcl::gf-dfun-cache #'cache-test))) diff --git a/version.lisp-expr b/version.lisp-expr index fc46069..754ab61 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.6.36" +"1.0.6.37"