From 6ab32453060c4dd2b399164e3b0703525dec7ff0 Mon Sep 17 00:00:00 2001 From: Juho Snellman Date: Mon, 18 Jun 2007 16:15:54 +0000 Subject: [PATCH] 1.0.6.50: better arglists for generic functions * Parameter names in the lambda list of the generic function are given priority to those of the gf's methods, since the names given in the generic function are likely to be more general. * Fix merging of &key parameters of all the methods. * Patch by Tobias Rittweiler. --- NEWS | 4 + contrib/sb-introspect/test-driver.lisp | 49 ++++++++++++- contrib/sb-introspect/test.lisp | 9 ++- src/pcl/generic-functions.lisp | 4 +- src/pcl/methods.lisp | 125 +++++++++++++++----------------- version.lisp-expr | 2 +- 6 files changed, 119 insertions(+), 74 deletions(-) diff --git a/NEWS b/NEWS index fa541ee..1b686e1 100644 --- a/NEWS +++ b/NEWS @@ -17,6 +17,8 @@ changes in sbcl-1.0.7 relative to sbcl-1.0.6: "a constant string". * enhancement: SB-POSIX now supports lockf(). (Thanks to Zach Beane.) * enhancement: SB-POSIX now supports getcwd(). (Thanks to Tassilo Horn.) + * enhancement: SB-INTROSPECT:FUNCTION-ARGLIST shows nicer argument lists + for generic functions. (Thanks to Tobias C. Rittweiler) * optimization: bignum printing speed has been improved by 20-40% (depending on the bignum size.) * bug fix: WITH-MUTEX and WITH-RECURSIVE-LOCK are now interrupt safe @@ -29,6 +31,8 @@ changes in sbcl-1.0.7 relative to sbcl-1.0.6: fixed.) * bug fix: ADD/REMOVE-METHOD is now thread and interrupt safe. * bug fix: interning EQL-specializers is now thread and interrupt safe. + * bug fix: asdf systems with dependencies to the SB-POSIX or + SB-BSD-SOCKETS contribs can be loaded with :FORCE T. changes in sbcl-1.0.6 relative to sbcl-1.0.5: * new contrib: sb-cover, an experimental code coverage tool, is included diff --git a/contrib/sb-introspect/test-driver.lisp b/contrib/sb-introspect/test-driver.lisp index ee24543..15e7ccc 100644 --- a/contrib/sb-introspect/test-driver.lisp +++ b/contrib/sb-introspect/test-driver.lisp @@ -75,9 +75,54 @@ (assert (matchp-name :function 'cl-user::one 2)) (sb-profile:unprofile cl-user::one) -;;; Test the xref facility + +;;;; Check correctness of FUNCTION-ARGLIST. + +(assert (equal (function-arglist 'cl-user::one) + '(cl-user::a cl-user::b cl-user::c))) +(assert (equal (function-arglist 'the) + '(type sb-c::value))) + +;;; Check wrt. interplay of generic functions and their methods. + +(defgeneric xuuq (gf.a gf.b &rest gf.rest &key gf.k-X)) +(defmethod xuuq ((m1.a number) m1.b &rest m1.rest &key gf.k-X m1.k-Y m1.k-Z) + (declare (ignore m1.a m1.b m1.rest gf.k-X m1.k-Y m1.k-Z)) + 'm1) +(defmethod xuuq ((m2.a string) m2.b &rest m2.rest &key gf.k-X m1.k-Y m2.k-Q) + (declare (ignore m2.a m2.b m2.rest gf.k-X m1.k-Y m2.k-Q)) + 'm2) + +;; XUUQ's lambda list should look similiar to +;; +;; (GF.A GF.B &REST GF.REST &KEY GF.K-X M1.K-Z M1.K-Y M2.K-Q) +;; +(multiple-value-bind (required optional restp rest keyp keys allowp + auxp aux morep more-context more-count) + (sb-int:parse-lambda-list (function-arglist #'xuuq)) + (assert (equal required '(gf.a gf.b))) + (assert (null optional)) + (assert (and restp (eql rest 'gf.rest))) + (assert (and keyp + (member 'gf.k-X keys) + (member 'm1.k-Y keys) + (member 'm1.k-Z keys) + (member 'm2.k-Q keys))) + (assert (not allowp)) + (assert (and (not auxp) (null aux))) + (assert (and (not morep) (null more-context) (not more-count)))) + +;;; Check what happens when there's no explicit DEFGENERIC. + +(defmethod kroolz (r1 r2 &optional opt &aux aux) + (declare (ignore r1 r2 opt aux)) + 'kroolz) +(assert (equal (function-arglist #'kroolz) '(r1 r2 &optional opt))) + + +;;;; Test the xref facility (load (merge-pathnames "xref-test.lisp" *load-pathname*)) -;;; Unix success convention for exit codes +;;;; Unix success convention for exit codes (sb-ext:quit :unix-status 0) diff --git a/contrib/sb-introspect/test.lisp b/contrib/sb-introspect/test.lisp index 1b66b2b..064e262 100644 --- a/contrib/sb-introspect/test.lisp +++ b/contrib/sb-introspect/test.lisp @@ -1,4 +1,10 @@ -;; Do not alter this file unless you edit test-driver.lisp to match + +;;; +;;; The order of the forms must not change, as the order is checked in +;;; `test-driver.lisp'. Thus do not alter this file unless you edit +;;; test-driver.lisp to match. +;;; + (declaim (optimize (debug 3))) (in-package :cl-user) @@ -64,4 +70,3 @@ (define-setf-expander s (a b) (format t "~a ~a~%" a b)) - diff --git a/src/pcl/generic-functions.lisp b/src/pcl/generic-functions.lisp index 547557a..5712219 100644 --- a/src/pcl/generic-functions.lisp +++ b/src/pcl/generic-functions.lisp @@ -257,7 +257,7 @@ (defgeneric finalize-inheritance (class)) -(defgeneric function-keywords (method)) +(defgeneric function-keyword-parameters (method)) (defgeneric generic-function-argument-precedence-order (gf)) @@ -275,8 +275,6 @@ (defgeneric method-lambda-list (m)) -(defgeneric method-pretty-arglist (method)) - (defgeneric method-qualifiers (m)) (defgeneric method-specializers (m)) diff --git a/src/pcl/methods.lisp b/src/pcl/methods.lisp index c6fee7d..4a45c20 100644 --- a/src/pcl/methods.lisp +++ b/src/pcl/methods.lisp @@ -1550,13 +1550,14 @@ (reinitialize-instance generic-function :name new-value) new-value) -(defmethod function-keywords ((method standard-method)) - (multiple-value-bind (nreq nopt keysp restp allow-other-keys-p keywords) +(defmethod function-keyword-parameters ((method standard-method)) + (multiple-value-bind (nreq nopt keysp restp allow-other-keys-p + keywords keyword-parameters) (analyze-lambda-list (if (consp method) (early-method-lambda-list method) (method-lambda-list method))) - (declare (ignore nreq nopt keysp restp)) - (values keywords allow-other-keys-p))) + (declare (ignore nreq nopt keysp restp keywords)) + (values keyword-parameters allow-other-keys-p))) (defun method-ll->generic-function-ll (ll) (multiple-value-bind @@ -1568,67 +1569,59 @@ (eq s '&allow-other-keys))) ll))) -;;; This is based on the rules of method lambda list congruency defined in -;;; the spec. The lambda list it constructs is the pretty union of the -;;; lambda lists of all the methods. It doesn't take method applicability -;;; into account at all yet. +;;; This is based on the rules of method lambda list congruency +;;; defined in the spec. The lambda list it constructs is the pretty +;;; union of the lambda lists of the generic function and of all its +;;; methods. It doesn't take method applicability into account at all +;;; yet. + +;;; (Notice that we ignore &AUX variables as they're not part of the +;;; "public interface" of a function.) + (defmethod generic-function-pretty-arglist ((generic-function standard-generic-function)) - (let ((methods (generic-function-methods generic-function))) - (if methods - (let ((arglist ())) - ;; arglist is constructed from the GF's methods - maybe with - ;; keys and rest stuff added - (multiple-value-bind (required optional rest key allow-other-keys) - (method-pretty-arglist (car methods)) - (dolist (m (cdr methods)) - (multiple-value-bind (method-key-keywords - method-allow-other-keys - method-key) - (function-keywords m) - ;; we've modified function-keywords to return what we want as - ;; the third value, no other change here. - (declare (ignore method-key-keywords)) - (setq key (union key method-key)) - (setq allow-other-keys (or allow-other-keys - method-allow-other-keys)))) - (when allow-other-keys - (setq arglist '(&allow-other-keys))) - (when key - (setq arglist (nconc (list '&key) key arglist))) - (when rest - (setq arglist (nconc (list '&rest rest) arglist))) - (when optional - (setq arglist (nconc (list '&optional) optional arglist))) - (nconc required arglist))) - ;; otherwise we take the lambda-list from the GF directly, with no - ;; other 'keys' added ... - (let ((lambda-list (generic-function-lambda-list generic-function))) - lambda-list)))) - -(defmethod method-pretty-arglist ((method standard-method)) - (let ((required ()) - (optional ()) - (rest nil) - (key ()) - (allow-other-keys nil) - (state 'required) - (arglist (method-lambda-list method))) - (dolist (arg arglist) - (cond ((eq arg '&optional) (setq state 'optional)) - ((eq arg '&rest) (setq state 'rest)) - ((eq arg '&key) (setq state 'key)) - ((eq arg '&allow-other-keys) (setq allow-other-keys t)) - ((memq arg lambda-list-keywords)) - (t - (ecase state - (required (push arg required)) - (optional (push arg optional)) - (key (push arg key)) - (rest (setq rest arg)))))) - (values (nreverse required) - (nreverse optional) - rest - (nreverse key) - allow-other-keys))) - + (let ((gf-lambda-list (generic-function-lambda-list generic-function)) + (methods (generic-function-methods generic-function))) + (if (null methods) + gf-lambda-list + (multiple-value-bind (gf.required gf.optional gf.rest gf.keys gf.allowp) + (%split-arglist gf-lambda-list) + ;; Possibly extend the keyword parameters of the gf by + ;; additional key parameters of its methods: + (let ((methods.keys nil) (methods.allowp nil)) + (dolist (m methods) + (multiple-value-bind (m.keyparams m.allow-other-keys) + (function-keyword-parameters m) + (setq methods.keys (union methods.keys m.keyparams :key #'maybe-car)) + (setq methods.allowp (or methods.allowp m.allow-other-keys)))) + (let ((arglist '())) + (when (or gf.allowp methods.allowp) + (push '&allow-other-keys arglist)) + (when (or gf.keys methods.keys) + ;; We make sure that the keys of the gf appear before + ;; those of its methods, since they're probably more + ;; generally appliable. + (setq arglist (nconc (list '&key) gf.keys + (nset-difference methods.keys gf.keys) + arglist))) + (when gf.rest + (setq arglist (nconc (list '&rest gf.rest) arglist))) + (when gf.optional + (setq arglist (nconc (list '&optional) gf.optional arglist))) + (nconc gf.required arglist))))))) + +(defun maybe-car (thing) + (if (listp thing) + (car thing) + thing)) + + +(defun %split-arglist (lambda-list) + ;; This function serves to shrink the number of returned values of + ;; PARSE-LAMBDA-LIST to something handier. + (multiple-value-bind (required optional restp rest keyp keys allowp + auxp aux morep more-context more-count) + (parse-lambda-list lambda-list) + (declare (ignore restp keyp auxp aux morep)) + (declare (ignore more-context more-count)) + (values required optional rest keys allowp))) \ No newline at end of file diff --git a/version.lisp-expr b/version.lisp-expr index 92cd704..a8763bf 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.49" +"1.0.6.50" -- 1.7.10.4