From 0dea4acb4216f9ee1182a6dc49483ec8d42babc5 Mon Sep 17 00:00:00 2001 From: William Harold Newman Date: Wed, 8 Aug 2001 17:21:59 +0000 Subject: [PATCH] 0.pre7.4: moved more contrib/*-extras.lisp stuff to main system.. ..TRUNCATE, FLOOR, and CEILING ..SORT ..VECTOR-PUSH-EXTEND tweaked TRUNCATE, FLOOR, and CEILING DEFTRANSFORMs so that they work when &OPTIONAL argument is missing --- contrib/code-extras.lisp | 135 ------------------------- contrib/compiler-extras.lisp | 33 ------ src/code/array.lisp | 36 ++++++- src/code/sort.lisp | 226 +++++++++++++++++++++--------------------- src/compiler/float-tran.lisp | 57 ++++++++--- version.lisp-expr | 2 +- 6 files changed, 194 insertions(+), 295 deletions(-) diff --git a/contrib/code-extras.lisp b/contrib/code-extras.lisp index aedfa21..fe1b715 100644 --- a/contrib/code-extras.lisp +++ b/contrib/code-extras.lisp @@ -4,141 +4,6 @@ (declaim (optimize (speed 3) (space 1))) -;;; Like CMU CL, we use HEAPSORT. However, instead of trying to -;;; generalize the CMU CL code to allow START and END values, this -;;; code has been written from scratch following Chapter 7 of -;;; _Introduction to Algorithms_ by Corman, Rivest, and Shamir. -(macrolet ((%index (x) `(truly-the index ,x)) - (%parent (i) `(ash ,i -1)) - (%left (i) `(%index (ash ,i 1))) - (%right (i) `(%index (1+ (ash ,i 1)))) - (%heapify (i) - `(do* ((i ,i) - (left (%left i) (%left i))) - ((> left current-heap-size)) - (declare (type index i left)) - (let* ((i-elt (%elt i)) - (i-key (funcall keyfun i-elt)) - (left-elt (%elt left)) - (left-key (funcall keyfun left-elt))) - (multiple-value-bind (large large-elt large-key) - (if (funcall predicate i-key left-key) - (values left left-elt left-key) - (values i i-elt i-key)) - (let ((right (%right i))) - (multiple-value-bind (largest largest-elt) - (if (> right current-heap-size) - (values large large-elt) - (let* ((right-elt (%elt right)) - (right-key (funcall keyfun right-elt))) - (if (funcall predicate large-key right-key) - (values right right-elt) - (values large large-elt)))) - (cond ((= largest i) - (return)) - (t - (setf (%elt i) largest-elt - (%elt largest) i-elt - i largest))))))))) - (%srt-vector (keyfun &optional (vtype 'vector)) - `(macrolet (;; In SBCL ca. 0.6.10, I had trouble getting - ;; type inference to propagate all the way - ;; through this tangled mess of inlining. The - ;; TRULY-THE here works around that. -- WHN - (%elt (i) - `(aref (truly-the ,',vtype vector) - (%index (+ (%index ,i) start-1))))) - (let ((start-1 (1- start)) ; Heaps prefer 1-based addressing. - (current-heap-size (- end start)) - (keyfun ,keyfun)) - (declare (type (integer -1 #.(1- most-positive-fixnum)) - start-1)) - (declare (type index current-heap-size)) - (declare (type function keyfun)) - (/noshow "doing SRT-VECTOR" keyfun) - (loop for i of-type index - from (ash current-heap-size -1) downto 1 do - (/noshow vector "about to %HEAPIFY" i) - (%heapify i)) - (loop - (/noshow current-heap-size vector) - (when (< current-heap-size 2) - (/noshow "returning") - (return)) - (/noshow "setting" current-heap-size "element to" (%elt 1)) - (rotatef (%elt 1) (%elt current-heap-size)) - (decf current-heap-size) - (%heapify 1)) - (/noshow "falling out of %SRT-VECTOR"))))) - - (declaim (inline srt-vector)) - (defun srt-vector (vector start end predicate key) - (declare (type vector vector)) - (declare (type index start end)) - (declare (type function predicate)) - (declare (type (or function null) key)) - (declare (optimize (speed 3) (safety 3) (debug 1) (space 1))) - (if (typep vector 'simple-vector) - ;; (VECTOR T) is worth optimizing for, and SIMPLE-VECTOR is - ;; what we get from (VECTOR T) inside WITH-ARRAY-DATA. - (if (null key) - ;; Special-casing the KEY=NIL case lets us avoid some - ;; function calls. - (%srt-vector #'identity simple-vector) - (%srt-vector key simple-vector)) - ;; It's hard to imagine many important applications for - ;; sorting vector types other than (VECTOR T), so we just lump - ;; them all together in one slow dynamically typed mess. - (locally - (declare (optimize (speed 2) (space 2) (inhibit-warnings 3))) - (error "stub: suppressed to hide notes") - #+nil (%srt-vector (or key #'identity)))))) - -(declaim (maybe-inline sort)) -(defun sort (sequence predicate &key key) - (let ((predicate-function (%coerce-callable-to-function predicate)) - (key-function (and key (%coerce-callable-to-function key)))) - (typecase sequence - (list (sort-list sequence predicate-function key-function)) - (vector - (with-array-data ((vector (the vector sequence)) - (start 0) - (end (length sequence))) - (srt-vector vector start end predicate-function key-function)) - (/noshow "back from SRT-VECTOR" sequence) - sequence) - (t - (error 'simple-type-error - :datum sequence - :expected-type 'sequence - :format-control "~S is not a sequence." - :format-arguments (list sequence)))))) - -(defun vector-push-extend (new-element - vector - &optional - (extension nil extension-p)) - (declare (type vector vector)) - (let ((old-fill-pointer (fill-pointer vector))) - (declare (type index old-fill-pointer)) - (when (= old-fill-pointer (%array-available-elements vector)) - (adjust-array vector (+ old-fill-pointer - (if extension-p - (the (integer 1 #.most-positive-fixnum) - extension) - (1+ old-fill-pointer))))) - (setf (%array-fill-pointer vector) - (1+ old-fill-pointer)) - ;; Wrapping the type test and the AREF in the same WITH-ARRAY-DATA - ;; saves some time. - (with-array-data ((v vector) (i old-fill-pointer) (end) - :force-inline t) - (declare (ignore end) (optimize (safety 0))) - (if (simple-vector-p v) ; if common special case - (setf (aref v i) new-element) - (setf (aref v i) new-element))) - old-fill-pointer)) - ;;; FIXME: should DEFUN REPLACE in terms of same expansion as ;;; DEFTRANSFORM #+nil diff --git a/contrib/compiler-extras.lisp b/contrib/compiler-extras.lisp index 15fa799..1d8ca7f 100644 --- a/contrib/compiler-extras.lisp +++ b/contrib/compiler-extras.lisp @@ -253,37 +253,4 @@ (check-inlineability-of-find-position-if sequence from-end) '(%find-position-vector-macro item sequence from-end start end key test)) - -;;;; optimizations for floating point FLOOR, CEILING, TRUNCATE, and -;;;; ROUND, lifted from CMU CL 18c -;;;; -;;;; (Without these optimizations, these functions cons!) - -;;; Convert (TRUNCATE x y) to the obvious implementation. We only want -;;; this when under certain conditions and let the generic TRUNCATE -;;; handle the rest. (Note: if Y = 1, the divide and multiply by Y -;;; should be removed by other DEFTRANSFORMs.) -(deftransform truncate ((x &optional y) - (float &optional (or float integer))) - '(let ((res (%unary-truncate (/ x y)))) - (values res (- x (* y res))))) - -(deftransform floor ((number &optional divisor) - (float &optional (or integer float))) - '(multiple-value-bind (tru rem) (truncate number divisor) - (if (and (not (zerop rem)) - (if (minusp divisor) - (plusp number) - (minusp number))) - (values (1- tru) (+ rem divisor)) - (values tru rem)))) -(deftransform ceiling ((number &optional divisor) - (float &optional (or integer float))) - '(multiple-value-bind (tru rem) (truncate number divisor) - (if (and (not (zerop rem)) - (if (minusp divisor) - (minusp number) - (plusp number))) - (values (1+ tru) (- rem divisor)) - (values tru rem)))) diff --git a/src/code/array.lisp b/src/code/array.lisp index 90891e1..e5647cd 100644 --- a/src/code/array.lisp +++ b/src/code/array.lisp @@ -620,6 +620,11 @@ :format-control "~S is not an array with a fill pointer." :format-arguments (list vector)))) +;;; FIXME: It'd probably make sense to use a MACROLET to share the +;;; guts of VECTOR-PUSH between VECTOR-PUSH-EXTEND. Such a macro +;;; should probably be based on the VECTOR-PUSH-EXTEND code (which is +;;; new ca. sbcl-0.7.0) rather than the VECTOR-PUSH code (which dates +;;; back to CMU CL). (defun vector-push (new-el array) #!+sb-doc "Attempt to set the element of ARRAY designated by its fill pointer @@ -639,10 +644,35 @@ (defun vector-push-extend (new-element vector &optional - (extension (1+ (length vector)))) + (extension nil extension-p)) #!+sb-doc - "This is like Vector-Push except that if the fill pointer gets too - large, the Vector is extended rather than Nil being returned." + "This is like VECTOR-PUSH except that if the fill pointer gets too + large, VECTOR is extended to allow the push to work." + (declare (type vector vector)) + (let ((old-fill-pointer (fill-pointer vector))) + (declare (type index old-fill-pointer)) + (when (= old-fill-pointer (%array-available-elements vector)) + (adjust-array vector (+ old-fill-pointer + (if extension-p + (the (integer 1 #.most-positive-fixnum) + extension) + (1+ old-fill-pointer))))) + (setf (%array-fill-pointer vector) + (1+ old-fill-pointer)) + ;; Wrapping the type test and the AREF in the same WITH-ARRAY-DATA + ;; saves some time. + (with-array-data ((v vector) (i old-fill-pointer) (end) + :force-inline t) + (declare (ignore end) (optimize (safety 0))) + (if (simple-vector-p v) ; if common special case + (setf (aref v i) new-element) + (setf (aref v i) new-element))) + old-fill-pointer)) + +(defun vector-push-extend (new-element + vector + &optional + (extension (1+ (length vector)))) (declare (vector vector) (fixnum extension)) (let ((fill-pointer (fill-pointer vector))) (declare (fixnum fill-pointer)) diff --git a/src/code/sort.lisp b/src/code/sort.lisp index 7a140fa..372f2c5 100644 --- a/src/code/sort.lisp +++ b/src/code/sort.lisp @@ -11,110 +11,115 @@ (in-package "SB!IMPL") +;;; Like CMU CL, we use HEAPSORT. However, other than that, this code +;;; isn't really related to the CMU CL code, since instead of trying +;;; to generalize the CMU CL code to allow START and END values, this +;;; code has been written from scratch following Chapter 7 of +;;; _Introduction to Algorithms_ by Corman, Rivest, and Shamir. +(macrolet ((%index (x) `(truly-the index ,x)) + (%parent (i) `(ash ,i -1)) + (%left (i) `(%index (ash ,i 1))) + (%right (i) `(%index (1+ (ash ,i 1)))) + (%heapify (i) + `(do* ((i ,i) + (left (%left i) (%left i))) + ((> left current-heap-size)) + (declare (type index i left)) + (let* ((i-elt (%elt i)) + (i-key (funcall keyfun i-elt)) + (left-elt (%elt left)) + (left-key (funcall keyfun left-elt))) + (multiple-value-bind (large large-elt large-key) + (if (funcall predicate i-key left-key) + (values left left-elt left-key) + (values i i-elt i-key)) + (let ((right (%right i))) + (multiple-value-bind (largest largest-elt) + (if (> right current-heap-size) + (values large large-elt) + (let* ((right-elt (%elt right)) + (right-key (funcall keyfun right-elt))) + (if (funcall predicate large-key right-key) + (values right right-elt) + (values large large-elt)))) + (cond ((= largest i) + (return)) + (t + (setf (%elt i) largest-elt + (%elt largest) i-elt + i largest))))))))) + (%sort-vector (keyfun &optional (vtype 'vector)) + `(macrolet (;; KLUDGE: In SBCL ca. 0.6.10, I had trouble getting + ;; type inference to propagate all the way + ;; through this tangled mess of inlining. The + ;; TRULY-THE here works around that. -- WHN + (%elt (i) + `(aref (truly-the ,',vtype vector) + (%index (+ (%index ,i) start-1))))) + (let ((start-1 (1- start)) ; Heaps prefer 1-based addressing. + (current-heap-size (- end start)) + (keyfun ,keyfun)) + (declare (type (integer -1 #.(1- most-positive-fixnum)) + start-1)) + (declare (type index current-heap-size)) + (declare (type function keyfun)) + (loop for i of-type index + from (ash current-heap-size -1) downto 1 do + (%heapify i)) + (loop + (when (< current-heap-size 2) + (return)) + (rotatef (%elt 1) (%elt current-heap-size)) + (decf current-heap-size) + (%heapify 1)))))) + + (declaim (inline sort-vector)) + (defun sort-vector (vector start end predicate key) + (declare (type vector vector)) + (declare (type index start end)) + (declare (type function predicate)) + (declare (type (or function null) key)) + (declare (optimize (speed 3) (safety 3) (debug 1) (space 1))) + (if (typep vector 'simple-vector) + ;; (VECTOR T) is worth optimizing for, and SIMPLE-VECTOR is + ;; what we get from (VECTOR T) inside WITH-ARRAY-DATA. + (if (null key) + ;; Special-casing the KEY=NIL case lets us avoid some + ;; function calls. + (%sort-vector #'identity simple-vector) + (%sort-vector key simple-vector)) + ;; It's hard to anticipate many speed-critical applications for + ;; sorting vector types other than (VECTOR T), so we just lump + ;; them all together in one slow dynamically typed mess. + (locally + (declare (optimize (speed 2) (space 2) (inhibit-warnings 3))) + (%sort-vector (or key #'identity)))))) + +;;; This is MAYBE-INLINE because it's not too hard to have an +;;; application where sorting is a major bottleneck, and inlining it +;;; allows the compiler to make enough optimizations that it might be +;;; worth the (large) cost in space. +(declaim (maybe-inline sort)) (defun sort (sequence predicate &key key) #!+sb-doc "Destructively sort SEQUENCE. PREDICATE should return non-NIL if ARG1 is to precede ARG2." - (typecase sequence - (simple-vector - (if (> (the fixnum (length (the simple-vector sequence))) 0) - (sort-simple-vector sequence predicate key) - sequence)) - (list - (sort-list sequence predicate key)) - (vector - (if (> (the fixnum (length sequence)) 0) - (sort-vector sequence predicate key) - sequence)) - (t - (error 'simple-type-error - :datum sequence - :expected-type 'sequence - :format-control "~S is not a SEQUENCE." - :format-arguments (list sequence))))) - -;;;; sorting vectors - -;;; Make sorting functions for SIMPLE-VECTOR and miscellaneous other VECTORs. -(macrolet (;; BUILD-HEAP rearranges seq elements into a heap to start heap - ;; sorting. - (build-heap (seq type len-1 pred key) - (let ((i (gensym))) - `(do ((,i (floor ,len-1 2) (1- ,i))) - ((minusp ,i) ,seq) - (declare (fixnum ,i)) - (heapify ,seq ,type ,i ,len-1 ,pred ,key)))) - ;; HEAPIFY, assuming both sons of root are heaps, - ;; percolates the root element through the sons to form a - ;; heap at root. Root and max are zero based coordinates, - ;; but the heap algorithm only works on arrays indexed from - ;; 1 through N (not 0 through N-1); This is because a root - ;; at I has sons at 2*I and 2*I+1 which does not work for a - ;; root at 0. Because of this, boundaries, roots, and - ;; termination are computed using 1..N indexes. - (heapify (seq vector-ref root max pred key) - (let ((heap-root (gensym)) - (heap-max (gensym)) - (root-ele (gensym)) - (root-key (gensym)) - (heap-max/2 (gensym)) - (heap-l-son (gensym)) - (one-son (gensym)) - (one-son-ele (gensym)) - (one-son-key (gensym)) - (r-son-ele (gensym)) - (r-son-key (gensym)) - (var-root (gensym))) - `(let* ((,var-root ,root) ; (necessary to not clobber calling - ; root var) - (,heap-root (1+ ,root)) - (,heap-max (1+ ,max)) - (,root-ele (,vector-ref ,seq ,root)) - (,root-key (apply-key ,key ,root-ele)) - (,heap-max/2 (ash ,heap-max -1))) ; (floor heap-max 2) - (declare (fixnum ,var-root ,heap-root ,heap-max ,heap-max/2)) - (loop - (if (> ,heap-root ,heap-max/2) (return)) - (let* ((,heap-l-son (ash ,heap-root 1)) ; (* 2 heap-root) - ;; l-son index in seq (0..N-1) is one less than heap - ;; computation. - (,one-son (1- ,heap-l-son)) - (,one-son-ele (,vector-ref ,seq ,one-son)) - (,one-son-key (apply-key ,key ,one-son-ele))) - (declare (fixnum ,heap-l-son ,one-son)) - (if (< ,heap-l-son ,heap-max) - ;; There is a right son. - (let* ((,r-son-ele (,vector-ref ,seq ,heap-l-son)) - (,r-son-key (apply-key ,key ,r-son-ele))) - ;; Choose the greater of the two sons. - (when (funcall ,pred ,one-son-key ,r-son-key) - (setf ,one-son ,heap-l-son) - (setf ,one-son-ele ,r-son-ele) - (setf ,one-son-key ,r-son-key)))) - ;; If greater son is less than root, then we've - ;; formed a heap again.. - (if (funcall ,pred ,one-son-key ,root-key) (return)) - ;; ..else put greater son at root and make - ;; greater son node be the root. - (setf (,vector-ref ,seq ,var-root) ,one-son-ele) - (setf ,heap-root (1+ ,one-son)) ; (one plus to be in heap coordinates) - (setf ,var-root ,one-son))) ; actual index into vector for root ele - ;; Now really put percolated value into heap at the - ;; appropriate root node. - (setf (,vector-ref ,seq ,var-root) ,root-ele)))) - (def-vector-sort-fun (fun-name vector-ref) - `(defun ,fun-name (seq pred key) - (let ((len-1 (1- (length (the vector seq))))) - (declare (fixnum len-1)) - (build-heap seq ,vector-ref len-1 pred key) - (do* ((i len-1 i-1) - (i-1 (1- i) (1- i-1))) - ((zerop i) seq) - (declare (fixnum i i-1)) - (rotatef (,vector-ref seq 0) (,vector-ref seq i)) - (heapify seq ,vector-ref 0 i-1 pred key)))))) - (def-vector-sort-fun sort-vector aref) - (def-vector-sort-fun sort-simple-vector svref)) + (let ((predicate-function (%coerce-callable-to-function predicate)) + (key-function (and key (%coerce-callable-to-function key)))) + (typecase sequence + (list (sort-list sequence predicate-function key-function)) + (vector + (with-array-data ((vector (the vector sequence)) + (start 0) + (end (length sequence))) + (sort-vector vector start end predicate-function key-function)) + sequence) + (t + (error 'simple-type-error + :datum sequence + :expected-type 'sequence + :format-control "~S is not a sequence." + :format-arguments (list sequence)))))) ;;;; stable sorting @@ -280,10 +285,10 @@ (incf ,i))) (incf ,target-i))))) -;;; VECTOR-MERGE-SORT is the same algorithm used to stable sort lists, but -;;; it uses a temporary vector. Direction determines whether we are merging -;;; into the temporary (T) or back into the given vector (NIL). - +;;; VECTOR-MERGE-SORT is the same algorithm used to stable sort lists, +;;; but it uses a temporary vector. DIRECTION determines whether we +;;; are merging into the temporary (T) or back into the given vector +;;; (NIL). (sb!xc:defmacro vector-merge-sort (vector pred key vector-ref) (let ((vector-len (gensym)) (n (gensym)) (direction (gensym)) (unsorted (gensym)) @@ -351,7 +356,7 @@ ) ; EVAL-when -;;; Temporary vector for stable sorting vectors. +;;; temporary vector for stable sorting vectors (defvar *merge-sort-temp-vector* (make-array 50)) @@ -369,10 +374,9 @@ (eval-when (:compile-toplevel :execute) ;;; MERGE-VECTORS returns a new vector which contains an interleaving -;;; of the elements of vector-1 and vector-2. Elements from vector-2 are -;;; chosen only if they are strictly less than elements of vector-1, -;;; (pred elt-2 elt-1), as specified in the manual. - +;;; of the elements of VECTOR-1 and VECTOR-2. Elements from VECTOR-2 +;;; are chosen only if they are strictly less than elements of +;;; VECTOR-1, (PRED ELT-2 ELT-1), as specified in the manual. (sb!xc:defmacro merge-vectors (vector-1 length-1 vector-2 length-2 result-vector pred key access) (let ((result-i (gensym)) @@ -411,8 +415,8 @@ (defun merge (result-type sequence1 sequence2 predicate &key key) #!+sb-doc - "The sequences SEQUENCE1 and SEQUENCE2 are destructively merged into - a sequence of type RESULT-TYPE using PREDICATE to order the elements." + "Merge the sequences SEQUENCE1 and SEQUENCE2 destructively into a + sequence of type RESULT-TYPE using PREDICATE to order the elements." (if (eq result-type 'list) (let ((result (merge-lists* (coerce sequence1 'list) (coerce sequence2 'list) diff --git a/src/compiler/float-tran.lisp b/src/compiler/float-tran.lisp index bc7a1cd..ba12b8e 100644 --- a/src/compiler/float-tran.lisp +++ b/src/compiler/float-tran.lisp @@ -30,18 +30,6 @@ (deftransform %double-float ((n) (double-float) * :when :both) 'n) -;;; not strictly float functions, but primarily useful on floats: -(macrolet ((frob (fun ufun) - `(progn - (defknown ,ufun (real) integer (movable foldable flushable)) - (deftransform ,fun ((x &optional by) - (* &optional - (constant-argument (member 1)))) - '(let ((res (,ufun x))) - (values res (- x res))))))) - (frob truncate %unary-truncate) - (frob round %unary-round)) - ;;; RANDOM (macrolet ((frob (fun type) `(deftransform random ((num &optional state) @@ -1260,3 +1248,48 @@ #'cis)) ) ; PROGN + +;;;; TRUNCATE, FLOOR, CEILING, and ROUND + +(macrolet ((define-frobs (fun ufun) + `(progn + (defknown ,ufun (real) integer (movable foldable flushable)) + (deftransform ,fun ((x &optional by) + (* &optional + (constant-argument (member 1)))) + '(let ((res (,ufun x))) + (values res (- x res))))))) + (define-frobs truncate %unary-truncate) + (define-frobs round %unary-round)) + +;;; Convert (TRUNCATE x y) to the obvious implementation. We only want +;;; this when under certain conditions and let the generic TRUNCATE +;;; handle the rest. (Note: if Y = 1, the divide and multiply by Y +;;; should be removed by other DEFTRANSFORMs.) +(deftransform truncate ((x &optional y) + (float &optional (or float integer))) + (let ((defaulted-y (if y 'y 1))) + `(let ((res (%unary-truncate (/ x ,defaulted-y)))) + (values res (- x (* ,defaulted-y res)))))) + +(deftransform floor ((number &optional divisor) + (float &optional (or integer float))) + (let ((defaulted-divisor (if divisor 'divisor 1))) + `(multiple-value-bind (tru rem) (truncate number ,defaulted-divisor) + (if (and (not (zerop rem)) + (if (minusp ,defaulted-divisor) + (plusp number) + (minusp number))) + (values (1- tru) (+ rem ,defaulted-divisor)) + (values tru rem))))) + +(deftransform ceiling ((number &optional divisor) + (float &optional (or integer float))) + (let ((defaulted-divisor (if divisor 'divisor 1))) + `(multiple-value-bind (tru rem) (truncate number ,defaulted-divisor) + (if (and (not (zerop rem)) + (if (minusp ,defaulted-divisor) + (minusp number) + (plusp number))) + (values (1+ tru) (- rem ,defaulted-divisor)) + (values tru rem))))) diff --git a/version.lisp-expr b/version.lisp-expr index ca1ccf5..dc1ba94 100644 --- a/version.lisp-expr +++ b/version.lisp-expr @@ -16,4 +16,4 @@ ;;; four numeric fields, is used for versions which aren't released ;;; but correspond only to CVS tags or snapshots. -"0.pre7.3" +"0.pre7.4" -- 1.7.10.4