From 284c8f6833589a6bddf22a5af30d3ac4eafcd2cc Mon Sep 17 00:00:00 2001 From: Nikodemus Siivola Date: Mon, 13 Aug 2007 13:55:15 +0000 Subject: [PATCH] 1.0.8.25: refactor CAN-OPTIMIZE-ACCESS usage and interface * Make the constant slot-name and the possible new-value-form additional return values. * Use CONSTANT-FORM-VALUE instead of EVAL to get the constant slot-name. Also use the environment correctly to CONSTANTP and CONSTANT-FORM-VALUE. * Call CAN-OPTIMIZE-ACCESS in the various instance-access optimizers and not around calls to them, so that they can directly use the slot-name and new-value-form results from CAN-OPTIMIZE-ACCESS instead of having to recompute them. --- src/pcl/boot.lisp | 15 ++--- src/pcl/vector.lisp | 172 ++++++++++++++++++++++----------------------------- version.lisp-expr | 2 +- 3 files changed, 80 insertions(+), 109 deletions(-) diff --git a/src/pcl/boot.lisp b/src/pcl/boot.lisp index e8a0d5c..b056940 100644 --- a/src/pcl/boot.lisp +++ b/src/pcl/boot.lisp @@ -1506,15 +1506,12 @@ bootstrapping. (t nil)))) ((and (memq (car form) '(slot-value set-slot-value slot-boundp)) - (constantp (caddr form))) - (let ((parameter (can-optimize-access form - required-parameters - env))) - (let ((fun (ecase (car form) - (slot-value #'optimize-slot-value) - (set-slot-value #'optimize-set-slot-value) - (slot-boundp #'optimize-slot-boundp)))) - (funcall fun slots parameter form)))) + (constantp (caddr form) env)) + (let ((fun (ecase (car form) + (slot-value #'optimize-slot-value) + (set-slot-value #'optimize-set-slot-value) + (slot-boundp #'optimize-slot-boundp)))) + (funcall fun form slots required-parameters env))) (t form)))) (let ((walked-lambda (walk-form method-lambda env #'walk-function))) diff --git a/src/pcl/vector.lisp b/src/pcl/vector.lisp index 9e4dcd3..4b7fcfe 100644 --- a/src/pcl/vector.lisp +++ b/src/pcl/vector.lisp @@ -304,31 +304,34 @@ (incf param)))))) (defun can-optimize-access (form required-parameters env) - (let ((type (ecase (car form) - (slot-value 'reader) - (set-slot-value 'writer) - (slot-boundp 'boundp))) - (var (extract-the (cadr form))) - (slot-name (eval (caddr form)))) ; known to be constant - (when (symbolp var) - (let* ((rebound? (caddr (var-declaration '%variable-rebinding var env))) - (parameter-or-nil (car (memq (or rebound? var) - required-parameters)))) - (when parameter-or-nil - (let* ((class-name (caddr (var-declaration '%class - parameter-or-nil - env))) - (class (find-class class-name nil))) - (when (or (not (eq *boot-state* 'complete)) - (and class (not (class-finalized-p class)))) - (setq class nil)) - (when (and class-name (not (eq class-name t))) - (when (or (null type) - (not (and class - (memq *the-class-structure-object* - (class-precedence-list class)))) - (optimize-slot-value-by-class-p class slot-name type)) - (cons parameter-or-nil (or class class-name)))))))))) + (destructuring-bind (op var-form slot-name-form &optional new-value) form + (let ((type (ecase op + (slot-value 'reader) + (set-slot-value 'writer) + (slot-boundp 'boundp))) + (var (extract-the var-form)) + (slot-name (constant-form-value slot-name-form env))) + (when (symbolp var) + (let* ((rebound? (caddr (var-declaration '%variable-rebinding var env))) + (parameter-or-nil (car (memq (or rebound? var) + required-parameters)))) + (when parameter-or-nil + (let* ((class-name (caddr (var-declaration '%class + parameter-or-nil + env))) + (class (find-class class-name nil))) + (when (or (not (eq *boot-state* 'complete)) + (and class (not (class-finalized-p class)))) + (setq class nil)) + (when (and class-name (not (eq class-name t))) + (when (or (null type) + (not (and class + (memq *the-class-structure-object* + (class-precedence-list class)))) + (optimize-slot-value-by-class-p class slot-name type)) + (values (cons parameter-or-nil (or class class-name)) + slot-name + new-value)))))))))) ;;; Check whether the binding of the named variable is modified in the ;;; method body. @@ -336,32 +339,31 @@ (let ((modified-variables (macroexpand '%parameter-binding-modified env))) (memq parameter-name modified-variables))) -(defun optimize-slot-value (slots sparameter form) - (if sparameter - (let ((optimized-form - (destructuring-bind (ignore1 ignore2 slot-name-form) form - (declare (ignore ignore1 ignore2)) - (let ((slot-name (eval slot-name-form))) - (optimize-instance-access slots :read sparameter - slot-name nil))))) - ;; We don't return the optimized form directly, since there's - ;; still a chance that we'll find out later on that the - ;; optimization should not have been done, for example due to - ;; the walker encountering a SETQ on SPARAMETER later on in - ;; the body [ see for example clos.impure.lisp test with :name - ;; ((:setq :method-parameter) slot-value)) ]. Instead we defer - ;; the decision until the compiler macroexpands - ;; OPTIMIZED-SLOT-VALUE. - ;; - ;; Note that we must still call OPTIMIZE-INSTANCE-ACCESS at - ;; this point (instead of when expanding - ;; OPTIMIZED-SLOT-VALUE), since it mutates the structure of - ;; SLOTS. If that mutation isn't done during the walking, - ;; MAKE-METHOD-LAMBDA-INTERNAL won't wrap a correct PV-BINDING - ;; form around the body, and compilation will fail. -- JES, - ;; 2006-09-18 - `(optimized-slot-value ,form ,(car sparameter) ,optimized-form)) - `(accessor-slot-value ,@(cdr form)))) +(defun optimize-slot-value (form slots required-parameters env) + (multiple-value-bind (sparameter slot-name) + (can-optimize-access form required-parameters env) + (if sparameter + (let ((optimized-form + (optimize-instance-access slots :read sparameter + slot-name nil))) + ;; We don't return the optimized form directly, since there's + ;; still a chance that we'll find out later on that the + ;; optimization should not have been done, for example due to + ;; the walker encountering a SETQ on SPARAMETER later on in + ;; the body [ see for example clos.impure.lisp test with :name + ;; ((:setq :method-parameter) slot-value)) ]. Instead we defer + ;; the decision until the compiler macroexpands + ;; OPTIMIZED-SLOT-VALUE. + ;; + ;; Note that we must still call OPTIMIZE-INSTANCE-ACCESS at + ;; this point (instead of when expanding + ;; OPTIMIZED-SLOT-VALUE), since it mutates the structure of + ;; SLOTS. If that mutation isn't done during the walking, + ;; MAKE-METHOD-LAMBDA-INTERNAL won't wrap a correct PV-BINDING + ;; form around the body, and compilation will fail. -- JES, + ;; 2006-09-18 + `(optimized-slot-value ,form ,(car sparameter) ,optimized-form)) + `(accessor-slot-value ,@(cdr form))))) (defmacro optimized-slot-value (form parameter-name optimized-form &environment env) @@ -371,20 +373,16 @@ `(accessor-slot-value ,@(cdr form)) optimized-form)) -(defun optimize-set-slot-value (slots sparameter form) - (if sparameter - (let ((optimized-form - (destructuring-bind (ignore1 ignore2 slot-name-form new-value) form - (declare (ignore ignore1 ignore2)) - (let ((slot-name (eval slot-name-form))) - (optimize-instance-access slots - :write - sparameter - slot-name - new-value))))) - ;; See OPTIMIZE-SLOT-VALUE - `(optimized-set-slot-value ,form ,(car sparameter) ,optimized-form)) - `(accessor-set-slot-value ,@(cdr form)))) +(defun optimize-set-slot-value (form slots required-parameters env) + (multiple-value-bind (sparameter slot-name new-value) + (can-optimize-access form required-parameters env) + (if sparameter + (let ((optimized-form + (optimize-instance-access slots :write sparameter + slot-name new-value))) + ;; See OPTIMIZE-SLOT-VALUE + `(optimized-set-slot-value ,form ,(car sparameter) ,optimized-form)) + `(accessor-set-slot-value ,@(cdr form))))) (defmacro optimized-set-slot-value (form parameter-name optimized-form &environment env) @@ -399,40 +397,16 @@ (t optimized-form))) -(defun optimize-slot-boundp (slots sparameter form) - (if sparameter - (let ((optimized-form - (destructuring-bind - ;; FIXME: In CMU CL ca. 19991205, this binding list - ;; had a fourth element in it, NEW-VALUE. It's hard - ;; to see how that could possibly be right, since - ;; SLOT-BOUNDP has no NEW-VALUE. Since it was - ;; causing a failure in building PCL for SBCL, so I - ;; changed it to match the definition of - ;; SLOT-BOUNDP (and also to match the list used in - ;; the similar OPTIMIZE-SLOT-VALUE, - ;; above). However, I'm weirded out by this, since - ;; this is old code which has worked for ages to - ;; build PCL for CMU CL, so it's hard to see why it - ;; should need a patch like this in order to build - ;; PCL for SBCL. I'd like to return to this and - ;; find a test case which exercises this function - ;; both in CMU CL, to see whether it's really a - ;; previously-unexercised bug or whether I've - ;; misunderstood something (and, presumably, - ;; patched it wrong). - (slot-boundp-symbol instance slot-name-form) - form - (declare (ignore slot-boundp-symbol instance)) - (let ((slot-name (eval slot-name-form))) - (optimize-instance-access slots - :boundp - sparameter - slot-name - nil))))) - ;; See OPTIMIZE-SLOT-VALUE - `(optimized-slot-boundp ,form ,(car sparameter) ,optimized-form)) - `(accessor-slot-boundp ,@(cdr form)))) +(defun optimize-slot-boundp (form slots required-parameters env) + (multiple-value-bind (sparameter slot-name) + (can-optimize-access form required-parameters env) + (if sparameter + (let ((optimized-form + (optimize-instance-access slots :boundp sparameter + slot-name nil))) + ;; See OPTIMIZE-SLOT-VALUE + `(optimized-slot-boundp ,form ,(car sparameter) ,optimized-form)) + `(accessor-slot-boundp ,@(cdr form))))) (defmacro optimized-slot-boundp (form parameter-name optimized-form &environment env) diff --git a/version.lisp-expr b/version.lisp-expr index b03f964..de59784 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.8.24" +"1.0.8.25" -- 1.7.10.4