From ba649fc0ec1d25dae4bf97d22611d78d42a7d187 Mon Sep 17 00:00:00 2001 From: Nikodemus Siivola Date: Thu, 30 Jul 2009 09:51:57 +0000 Subject: [PATCH] 1.0.30.17: generalize the previous COERCE optimization a bit * As noted by Christophe Rhodes, this is simple enough to apply to non-simple one-dimensional recognizable subtypes of ARRAY. ...and (COERCE X 'STRING) is so tempting to write that it is worth optimizing too. Need to take some care with things like (COERCE X '(BIT-VECTOR (NOT SIMPLE-BIT-VECTOR))) though. Add compiler notes as well. --- NEWS | 6 ++-- src/compiler/typetran.lisp | 71 +++++++++++++++++++++++++++++--------------- version.lisp-expr | 2 +- 3 files changed, 51 insertions(+), 28 deletions(-) diff --git a/NEWS b/NEWS index d65d68b..9ee33af 100644 --- a/NEWS +++ b/NEWS @@ -6,9 +6,9 @@ changes relative to sbcl-1.0.30: * new feature: experimental :EMIT-CFASL parameter to COMPILE-FILE can be used to output toplevel compile-time effects into a separate .CFASL file. - * optimization: COERCE to SIMPLE-STRING and recognizable one-dimenstional - subtypes of SIMPLE-ARRAY is upto 70% faster when the coercion is actually - needed. + * optimization: COERCE to STRING, SIMPLE-STRING and recognizable + one-dimenstional subtypes of ARRAY is upto 70% faster when the coercion is + actually needed. * optimization: division of floating point numbers by constants uses multiplication by reciprocal when an exact reciprocal exists. * optimization: multiplication of single- and double-floats floats by diff --git a/src/compiler/typetran.lisp b/src/compiler/typetran.lisp index b9106f4..eb23711 100644 --- a/src/compiler/typetran.lisp +++ b/src/compiler/typetran.lisp @@ -602,7 +602,8 @@ (deftransform coerce ((x type) (* *) * :node node) (unless (constant-lvar-p type) (give-up-ir1-transform)) - (let ((tspec (ir1-transform-specifier-type (lvar-value type)))) + (let* ((tval (lvar-value type)) + (tspec (ir1-transform-specifier-type tval))) (if (csubtypep (lvar-type x) tspec) 'x ;; Note: The THE here makes sure that specifiers like @@ -614,29 +615,51 @@ ;; FIXME: #!+long-float (t ,(error "LONG-FLOAT case needed")) ((csubtypep tspec (specifier-type 'float)) '(%single-float x)) - ;; Special case this one: SIMPLE-STRING is a union-type. - ((type= tspec (specifier-type 'simple-string)) - `(if (typep x 'simple-string) + ;; Special case STRING and SIMPLE-STRING as they are union types + ;; in SBCL. + ((member tval '(string simple-string)) + `(if (typep x ',tval) x (replace (make-array (length x) :element-type 'character) x))) - ;; Handle specialized element types. - ((csubtypep tspec (specifier-type '(simple-array * (*)))) - (dolist (etype sb!kernel::*specialized-array-element-types* - (give-up-ir1-transform)) - (when etype - (let ((spec `(simple-array ,etype (*)))) - (when (and (csubtypep tspec (specifier-type spec)) - ;; Can we avoid checking for dimension issues like (COERCE FOO - ;; '(SIMPLE-VECTOR 5)) returning a vector of length 6? - (or (policy node (< safety 3)) ; no need in unsafe code - (and (array-type-p tspec) ; no need when no dimensions - (equal (array-type-dimensions tspec) '(*))))) - (return - `(if (typep x ',spec) - x - (replace (make-array (length x) :element-type ',etype) x)))) - (give-up-ir1-transform))))) + ;; Handle specialized element types for 1D arrays. + ((csubtypep tspec (specifier-type '(array * (*)))) + ;; Can we avoid checking for dimension issues like (COERCE FOO + ;; '(SIMPLE-VECTOR 5)) returning a vector of length 6? + (if (or (policy node (< safety 3)) ; no need in unsafe code + (and (array-type-p tspec) ; no need when no dimensions + (equal (array-type-dimensions tspec) '(*)))) + ;; We can! + (let ((array-type + (if (csubtypep tspec (specifier-type 'simple-array)) + 'simple-array + 'array))) + (dolist (etype + #+sb-xc-host '(t bit character) + #-sb-xc-host sb!kernel::*specialized-array-element-types* + (give-up-ir1-transform)) + (when etype + (let ((spec `(,array-type ,etype (*)))) + (when (csubtypep tspec (specifier-type spec)) + ;; Is the result required to be non-simple? + (let ((result-simple + (or (eq 'simple-array array-type) + (neq *empty-type* + (type-intersection + tspec (specifier-type 'simple-array)))))) + (return + `(if (typep x ',spec) + x + (replace + (make-array (length x) :element-type ',etype + ,@(unless result-simple + (list :fill-pointer t + :adjustable t))) + x))))))))) + ;; No, duh. Dimension checking required. + (give-up-ir1-transform + "~@<~S specifies dimensions other than (*) in safe code.~:@>" + tval))) (t - (give-up-ir1-transform))))))) - - + (give-up-ir1-transform + "~@" + tval))))))) diff --git a/version.lisp-expr b/version.lisp-expr index ad8a46e..12bd78d 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.30.16" +"1.0.30.17" -- 1.7.10.4