From: William Harold Newman Date: Thu, 22 Mar 2001 18:04:23 +0000 (+0000) Subject: 0.6.11.25: X-Git-Url: http://repo.macrolet.net/gitweb/?a=commitdiff_plain;h=ce62508ec1a0f39008c18a2a5a06461eabe662c0;p=sbcl.git 0.6.11.25: fixed bug 45a: restored COMPLEX-ACOS and friends from sbcl-2.4.8's #-OLD-SPECFUN code, so that e.g. (ACOS (COMPLEX 1 1)) works now --- diff --git a/BUGS b/BUGS index 6857912..5400789 100644 --- a/BUGS +++ b/BUGS @@ -356,9 +356,7 @@ returning an array as first value always. 45: a slew of floating-point-related errors reported by Peter Van Eynde on July 25, 2000: - a: (SQRT -9.0) fails, because SB-KERNEL::COMPLEX-SQRT is undefined. - Similarly, COMPLEX-ASIN, COMPLEX-ACOS, COMPLEX-ACOSH, and others - aren't found. + a: (fixed in sbcl-0.6.11.25) b: SBCL's value for LEAST-POSITIVE-SHORT-FLOAT is bogus, and should probably be 1.4012985e-45. In SBCL, (/ LEAST-POSITIVE-SHORT-FLOAT 2) returns a number smaller @@ -372,10 +370,7 @@ returning an array as first value always. (EXPT 10.0d0 1000) PVE's regression tests want them to raise errors. SBCL generates the infinities instead, which may or may not be - conforming behavior, but then blow it by being unable to - output the infinities, since support for infinities is generally - broken, and in particular SB-IMPL::OUTPUT-FLOAT-INFINITY is - undefined. + conforming behavior. d: (in section12.erg) various forms a la (FLOAT 1 DOUBLE-FLOAT-EPSILON) don't give the right behavior. @@ -831,6 +826,25 @@ Error in function C::GET-LAMBDA-TO-COMPILE: bootstrap on a system which uses a different value of CHAR-CODE-LIMIT than SBCL does. +91: + (subtypep '(or (integer -1 1) + unsigned-byte) + '(or (rational -1 7) + unsigned-byte + (integer -1 1))) => NIL,T + An analogous problem with SINGLE-FLOAT and REAL types was fixed in + sbcl-0.6.11.22, but some peculiarites of the RATIO type makes it + awkward to generalize the fix to INTEGER and RATIONAL. It's not + clear what's the best fix. (See the "bug in type handling" discussion + on cmucl-imp ca. 2001-03-22 and ca. 2001-02-12.) + +92: + (< SB-EXT:SINGLE-FLOAT-POSITIVE-INFINITY 100) signals an error: + error in function SB-KERNEL:INTEGER-DECODE-SINGLE-FLOAT: + can't decode NaN or infinity: #.EXT:SINGLE-FLOAT-POSITIVE-INFINITY + This is a bug in the original CMU CL code. I reported it to cmucl-imp + 2001-03-22 in hopes that they'll fix it for us. + KNOWN BUGS RELATED TO THE IR1 INTERPRETER diff --git a/NEWS b/NEWS index f0ed713..2a79085 100644 --- a/NEWS +++ b/NEWS @@ -685,16 +685,19 @@ changes in sbcl-0.6.12 relative to sbcl-0.6.11: * many patches ported from CMU CL by Martin Atzmueller, with half a dozen bug fixes in pretty-printing and the debugger, and half a dozen others elsewhere -* improved support for type intersection and union, fixing bug 12 - (e.g., now (SUBTYPEP 'KEYWORD 'SYMBOL)=>T,T) and some other - smaller bugs as well +* fixed bug 13: Floating point infinities are now supported again. +* fixed bug 45a: Various internal functions required to support + complex special functions have been merged from CMU CL sources. + (When I was first setting up SBCL, I misunderstood a compile-time + conditional #-OLD-SPECFUN, and so accidentally deleted them.) ?? The :PROPAGATE-FLOAT-TYPE and :PROPAGATE-FUN-TYPE features are now supported, and enabled by default. Thus, the compiler can handle many floating point and complex operations much less inefficiently. (Thus e.g. you can implement a complex FFT without consing!) -?? unscrewed floating point infinities (bug 13) in order to support - :PROPAGATE-FLOAT-TYPE and :PROPAGATE-FUN-TYPE features +* improved support for type intersection and union, fixing bug 12 + (e.g., now (SUBTYPEP 'KEYWORD 'SYMBOL)=>T,T) and some other + more obscure bugs as well * various fixes to make the cross-compiler more portable to ANSI-conforming-but-different cross-compilation hosts (notably Lispworks for Windows, following bug reports from Arthur Lemmens) diff --git a/src/code/irrat.lisp b/src/code/irrat.lisp index 761940d..555c952 100644 --- a/src/code/irrat.lisp +++ b/src/code/irrat.lisp @@ -619,7 +619,6 @@ ;;; HP-UX does not supply a C version of log1p, so ;;; use the definition. - #!+hpux #!-sb-fluid (declaim (inline %log1p)) #!+hpux @@ -627,4 +626,438 @@ (declare (double-float number) (optimize (speed 3) (safety 0))) (the double-float (log (the (double-float 0d0) (+ number 1d0))))) + +;;;; OLD-SPECFUN stuff +;;;; +;;;; (This was conditional on #-OLD-SPECFUN in the CMU CL sources, +;;;; but OLD-SPECFUN was mentioned nowhere else, so it seems to be +;;;; the standard special function system.) +;;;; +;;;; This is a set of routines that implement many elementary +;;;; transcendental functions as specified by ANSI Common Lisp. The +;;;; implementation is based on Kahan's paper. +;;;; +;;;; I believe I have accurately implemented the routines and are +;;;; correct, but you may want to check for your self. +;;;; +;;;; These functions are written for CMU Lisp and take advantage of +;;;; some of the features available there. It may be possible, +;;;; however, to port this to other Lisps. +;;;; +;;;; Some functions are significantly more accurate than the original +;;;; definitions in CMU Lisp. In fact, some functions in CMU Lisp +;;;; give the wrong answer like (acos #c(-2.0 0.0)), where the true +;;;; answer is pi + i*log(2-sqrt(3)). +;;;; +;;;; All of the implemented functions will take any number for an +;;;; input, but the result will always be a either a complex +;;;; single-float or a complex double-float. +;;;; +;;;; general functions: +;;;; complex-sqrt +;;;; complex-log +;;;; complex-atanh +;;;; complex-tanh +;;;; complex-acos +;;;; complex-acosh +;;;; complex-asin +;;;; complex-asinh +;;;; complex-atan +;;;; complex-tan +;;;; +;;;; utility functions: +;;;; scalb logb +;;;; +;;;; internal functions: +;;;; square coerce-to-complex-type cssqs complex-log-scaled +;;;; +;;;; references: +;;;; Kahan, W. "Branch Cuts for Complex Elementary Functions, or Much +;;;; Ado About Nothing's Sign Bit" in Iserles and Powell (eds.) "The +;;;; State of the Art in Numerical Analysis", pp. 165-211, Clarendon +;;;; Press, 1987 +;;;; +;;;; The original CMU CL code requested: +;;;; Please send any bug reports, comments, or improvements to +;;;; Raymond Toy at toy@rtp.ericsson.se. + +;;; FIXME: In SBCL, the floating point infinity constants like +;;; SB!EXT:DOUBLE-FLOAT-POSITIVE-INFINITY aren't available as +;;; constants at cross-compile time, because the cross-compilation +;;; host might not have support for floating point infinities. Thus, +;;; they're effectively implemented as special variable references, +;;; and the code below which uses them might be unnecessarily +;;; inefficient. Perhaps some sort of MAKE-LOAD-TIME-VALUE hackery +;;; should be used instead? + +(declaim (inline square)) +(declaim (ftype (function (double-float) (double-float 0d0)) square)) +(defun square (x) + (declare (double-float x) + (values (double-float 0d0))) + (* x x)) + +;;; original CMU CL comment, apparently re. SCALB and LOGB and +;;; perhaps CSSQS: +;;; If you have these functions in libm, perhaps they should be used +;;; instead of these Lisp versions. These versions are probably good +;;; enough, especially since they are portable. + +;;; Compute 2^N * X without computing 2^N first. (Use properties of +;;; the underlying floating-point format.) +(declaim (inline scalb)) +(defun scalb (x n) + (declare (type double-float x) + (type double-float-exponent n)) + (scale-float x n)) + +;;; Compute an integer N such that 1 <= |2^N * x| < 2. +;;; For the special cases, the following values are used: +;;; x logb +;;; NaN NaN +;;; +/- infinity +infinity +;;; 0 -infinity +(defun logb (x) + (declare (type double-float x)) + (cond ((float-nan-p x) + x) + ((float-infinity-p x) + sb!ext:double-float-positive-infinity) + ((zerop x) + ;; The answer is negative infinity, but we are supposed to + ;; signal divide-by-zero. + ;; (error 'division-by-zero :operation 'logb :operands (list x)) + (/ -1.0d0 x) + ) + (t + (multiple-value-bind (signif expon sign) + (decode-float x) + (declare (ignore signif sign)) + ;; DECODE-FLOAT is almost right, except that the exponent + ;; is off by one. + (1- expon))))) + +;;; This function is used to create a complex number of the +;;; appropriate type: +;;; Create complex number with real part X and imaginary part Y +;;; such that has the same type as Z. If Z has type (complex +;;; rational), the X and Y are coerced to single-float. +#!+long-float (eval-when (:compile-toplevel :load-toplevel :execute) + (error "needs work for long float support")) +(declaim (inline coerce-to-complex-type)) +(defun coerce-to-complex-type (x y z) + (declare (double-float x y) + (number z)) + (if (subtypep (type-of (realpart z)) 'double-float) + (complex x y) + ;; Convert anything that's not a DOUBLE-FLOAT to a SINGLE-FLOAT. + (complex (float x 1.0) + (float y 1.0)))) + +;;; Compute |(x+i*y)/2^k|^2 scaled to avoid over/underflow. The +;;; result is r + i*k, where k is an integer. +#!+long-float (eval-when (:compile-toplevel :load-toplevel :execute) + (error "needs work for long float support")) +(defun cssqs (z) + ;; Save all FP flags + (let ((x (float (realpart z) 1d0)) + (y (float (imagpart z) 1d0)) + (k 0) + (rho 0d0)) + (declare (double-float x y) + (type (double-float 0d0) rho) + (fixnum k)) + ;; Would this be better handled using an exception handler to + ;; catch the overflow or underflow signal? For now, we turn all + ;; traps off and look at the accrued exceptions to see if any + ;; signal would have been raised. + (with-float-traps-masked (:underflow :overflow) + (setf rho (+ (square x) (square y))) + (cond ((and (or (float-nan-p rho) + (float-infinity-p rho)) + (or (float-infinity-p (abs x)) + (float-infinity-p (abs y)))) + (setf rho sb!ext:double-float-positive-infinity)) + ((let ((threshold #.(/ least-positive-double-float + double-float-epsilon)) + (traps (ldb sb!vm::float-sticky-bits + (sb!vm:floating-point-modes)))) + ;; overflow raised or (underflow raised and rho < lambda/eps) + (or (not (zerop (logand sb!vm:float-overflow-trap-bit traps))) + (and (not (zerop (logand sb!vm:float-underflow-trap-bit + traps))) + (< rho threshold)))) + (setf k (logb (max (abs x) (abs y)))) + (setf rho (+ (square (scalb x (- k))) + (square (scalb y (- k)))))))) + (values rho k))) + +;;; principal square root of Z +;;; +;;; Z may be any NUMBER, but the result is always a COMPLEX. +(defun complex-sqrt (z) + (declare (number z)) + (multiple-value-bind (rho k) + (cssqs z) + (declare (type (double-float 0d0) rho) + (fixnum k)) + (let ((x (float (realpart z) 1.0d0)) + (y (float (imagpart z) 1.0d0)) + (eta 0d0) + (nu 0d0)) + (declare (double-float x y eta nu)) + + (if (not (float-nan-p x)) + (setf rho (+ (scalb (abs x) (- k)) (sqrt rho)))) + + (cond ((oddp k) + (setf k (ash k -1))) + (t + (setf k (1- (ash k -1))) + (setf rho (+ rho rho)))) + + (setf rho (scalb (sqrt rho) k)) + + (setf eta rho) + (setf nu y) + + (when (/= rho 0d0) + (when (not (float-infinity-p (abs nu))) + (setf nu (/ (/ nu rho) 2d0))) + (when (< x 0d0) + (setf eta (abs nu)) + (setf nu (float-sign y rho)))) + (coerce-to-complex-type eta nu z)))) + +;;; Compute log(2^j*z). +;;; +;;; This is for use with J /= 0 only when |z| is huge. +(defun complex-log-scaled (z j) + (declare (number z) + (fixnum j)) + ;; The constants t0, t1, t2 should be evaluated to machine + ;; precision. In addition, Kahan says the accuracy of log1p + ;; influences the choices of these constants but doesn't say how to + ;; choose them. We'll just assume his choices matches our + ;; implementation of log1p. + (let ((t0 #.(/ 1 (sqrt 2.0d0))) + (t1 1.2d0) + (t2 3d0) + (ln2 #.(log 2d0)) + (x (float (realpart z) 1.0d0)) + (y (float (imagpart z) 1.0d0))) + (multiple-value-bind (rho k) + (cssqs z) + (declare (type (double-float 0d0) rho) + (fixnum k)) + (let ((beta (max (abs x) (abs y))) + (theta (min (abs x) (abs y)))) + (declare (type (double-float 0d0) beta theta)) + (if (and (zerop k) + (< t0 beta) + (or (<= beta t1) + (< rho t2))) + (setf rho (/ (%log1p (+ (* (- beta 1.0d0) + (+ beta 1.0d0)) + (* theta theta))) + 2d0)) + (setf rho (+ (/ (log rho) 2d0) + (* (+ k j) ln2)))) + (setf theta (atan y x)) + (coerce-to-complex-type rho theta z))))) + +;;; log of Z = log |Z| + i * arg Z +;;; +;;; Z may be any number, but the result is always a complex. +(defun complex-log (z) + (declare (number z)) + (complex-log-scaled z 0)) + +;;; KLUDGE: Let us note the following "strange" behavior. atanh 1.0d0 +;;; is +infinity, but the following code returns approx 176 + i*pi/4. +;;; The reason for the imaginary part is caused by the fact that arg +;;; i*y is never 0 since we have positive and negative zeroes. -- rtoy +;;; Compute atanh z = (log(1+z) - log(1-z))/2. +(defun complex-atanh (z) + (declare (number z)) + (let* (;; constants + (theta #.(/ (sqrt most-positive-double-float) 4.0d0)) + (rho #.(/ 4.0d0 (sqrt most-positive-double-float))) + (half-pi #.(/ pi 2.0d0)) + (rp (float (realpart z) 1.0d0)) + (beta (float-sign rp 1.0d0)) + (x (* beta rp)) + (y (* beta (- (float (imagpart z) 1.0d0)))) + (eta 0.0d0) + (nu 0.0d0)) + (declare (double-float theta rho half-pi rp beta y eta nu) + (type (double-float 0d0) x)) + (cond ((or (> x theta) + (> (abs y) theta)) + ;; to avoid overflow... + (setf eta (float-sign y half-pi)) + ;; nu is real part of 1/(x + iy). This is x/(x^2+y^2), + ;; which can cause overflow. Arrange this computation so + ;; that it won't overflow. + (setf nu (let* ((x-bigger (> x (abs y))) + (r (if x-bigger (/ y x) (/ x y))) + (d (+ 1.0d0 (* r r)))) + (declare (double-float r d)) + (if x-bigger + (/ (/ x) d) + (/ (/ r y) d))))) + ((= x 1.0d0) + ;; Should this be changed so that if y is zero, eta is set + ;; to +infinity instead of approx 176? In any case + ;; tanh(176) is 1.0d0 within working precision. + (let ((t1 (+ 4d0 (square y))) + (t2 (+ (abs y) rho))) + (declare (type (double-float 0d0) t1 t2)) + #+nil + (setf eta (log (/ (sqrt (sqrt t1))) + (sqrt t2))) + (setf eta (* 0.5d0 (log (the (double-float 0.0d0) + (/ (sqrt t1) t2))))) + (setf nu (* 0.5d0 + (float-sign y + (+ half-pi (atan (* 0.5d0 t2)))))))) + (t + (let ((t1 (+ (abs y) rho))) + (declare (double-float t1)) + ;; normal case using log1p(x) = log(1 + x) + (setf eta (* 0.25d0 + (%log1p (/ (* 4.0d0 x) + (+ (square (- 1.0d0 x)) + (square t1)))))) + (setf nu (* 0.5d0 + (atan (* 2.0d0 y) + (- (* (- 1.0d0 x) + (+ 1.0d0 x)) + (square t1)))))))) + (coerce-to-complex-type (* beta eta) + (- (* beta nu)) + z))) + +;;; Compute tanh z = sinh z / cosh z. +(defun complex-tanh (z) + (declare (number z)) + (let ((x (float (realpart z) 1.0d0)) + (y (float (imagpart z) 1.0d0))) + (declare (double-float x y)) + (cond ((> (abs x) + #-(or linux hpux) #.(/ (asinh most-positive-double-float) 4d0) + ;; This is more accurate under linux. + #+(or linux hpux) #.(/ (+ (log 2.0d0) + (log most-positive-double-float)) + 4d0)) + (complex (float-sign x) + (float-sign y 0.0d0))) + (t + (let* ((tv (%tan y)) + (beta (+ 1.0d0 (* tv tv))) + (s (sinh x)) + (rho (sqrt (+ 1.0d0 (* s s))))) + (declare (double-float tv s) + (type (double-float 0.0d0) beta rho)) + (if (float-infinity-p (abs tv)) + (coerce-to-complex-type (/ rho s) + (/ tv) + z) + (let ((den (+ 1.0d0 (* beta s s)))) + (coerce-to-complex-type (/ (* beta rho s) + den) + (/ tv den) + z)))))))) + +;;; Compute acos z = pi/2 - asin z. +;;; +;;; Z may be any NUMBER, but the result is always a COMPLEX. +(defun complex-acos (z) + ;; Kahan says we should only compute the parts needed. Thus, the + ;; REALPART's below should only compute the real part, not the whole + ;; complex expression. Doing this can be important because we may get + ;; spurious signals that occur in the part that we are not using. + ;; + ;; However, we take a pragmatic approach and just use the whole + ;; expression. + ;; + ;; NOTE: The formula given by Kahan is somewhat ambiguous in whether + ;; it's the conjugate of the square root or the square root of the + ;; conjugate. This needs to be checked. + ;; + ;; I checked. It doesn't matter because (conjugate (sqrt z)) is the + ;; same as (sqrt (conjugate z)) for all z. This follows because + ;; + ;; (conjugate (sqrt z)) = exp(0.5*log |z|)*exp(-0.5*j*arg z). + ;; + ;; (sqrt (conjugate z)) = exp(0.5*log|z|)*exp(0.5*j*arg conj z) + ;; + ;; and these two expressions are equal if and only if arg conj z = + ;; -arg z, which is clearly true for all z. + (declare (number z)) + (let ((sqrt-1+z (complex-sqrt (+ 1 z))) + (sqrt-1-z (complex-sqrt (- 1 z)))) + (with-float-traps-masked (:divide-by-zero) + (complex (* 2 (atan (/ (realpart sqrt-1-z) + (realpart sqrt-1+z)))) + (asinh (imagpart (* (conjugate sqrt-1+z) + sqrt-1-z))))))) + +;;; Compute acosh z = 2 * log(sqrt((z+1)/2) + sqrt((z-1)/2)) +;;; +;;; Z may be any NUMBER, but the result is always a COMPLEX. +(defun complex-acosh (z) + (declare (number z)) + (let ((sqrt-z-1 (complex-sqrt (- z 1))) + (sqrt-z+1 (complex-sqrt (+ z 1)))) + (with-float-traps-masked (:divide-by-zero) + (complex (asinh (realpart (* (conjugate sqrt-z-1) + sqrt-z+1))) + (* 2 (atan (/ (imagpart sqrt-z-1) + (realpart sqrt-z+1)))))))) + +;;; Compute asin z = asinh(i*z)/i. +;;; +;;; Z may be any NUMBER, but the result is always a COMPLEX. +(defun complex-asin (z) + (declare (number z)) + (let ((sqrt-1-z (complex-sqrt (- 1 z))) + (sqrt-1+z (complex-sqrt (+ 1 z)))) + (with-float-traps-masked (:divide-by-zero) + (complex (atan (/ (realpart z) + (realpart (* sqrt-1-z sqrt-1+z)))) + (asinh (imagpart (* (conjugate sqrt-1-z) + sqrt-1+z))))))) + +;;; Compute asinh z = log(z + sqrt(1 + z*z)). +;;; +;;; Z may be any number, but the result is always a complex. +(defun complex-asinh (z) + (declare (number z)) + ;; asinh z = -i * asin (i*z) + (let* ((iz (complex (- (imagpart z)) (realpart z))) + (result (complex-asin iz))) + (complex (imagpart result) + (- (realpart result))))) + +;;; Compute atan z = atanh (i*z) / i. +;;; +;;; Z may be any number, but the result is always a complex. +(defun complex-atan (z) + (declare (number z)) + ;; atan z = -i * atanh (i*z) + (let* ((iz (complex (- (imagpart z)) (realpart z))) + (result (complex-atanh iz))) + (complex (imagpart result) + (- (realpart result))))) +;;; Compute tan z = -i * tanh(i * z) +;;; +;;; Z may be any number, but the result is always a complex. +(defun complex-tan (z) + (declare (number z)) + ;; tan z = -i * tanh(i*z) + (let* ((iz (complex (- (imagpart z)) (realpart z))) + (result (complex-tanh iz))) + (complex (imagpart result) + (- (realpart result))))) diff --git a/src/code/target-numbers.lisp b/src/code/target-numbers.lisp index 8cc52ab..4b02217 100644 --- a/src/code/target-numbers.lisp +++ b/src/code/target-numbers.lisp @@ -792,40 +792,39 @@ (,op (rational x) y))) (((foreach bignum fixnum ratio) float) (,op x (rational y))))) +) ; EVAL-WHEN -(sb!xc:defmacro two-arg- (name op ratio-arg1 ratio-arg2 &rest cases) - `(defun ,name (x y) - (number-dispatch ((x real) (y real)) - (basic-compare ,op) - - (((foreach fixnum bignum) ratio) - (,op x (,ratio-arg2 (numerator y) (denominator y)))) - ((ratio integer) - (,op (,ratio-arg1 (numerator x) (denominator x)) y)) - ((ratio ratio) - (,op (* (numerator (truly-the ratio x)) - (denominator (truly-the ratio y))) - (* (numerator (truly-the ratio y)) - (denominator (truly-the ratio x))))) - ,@cases))) - -); Eval-When (Compile Eval) - -(two-arg- two-arg-< < floor ceiling - ((fixnum bignum) - (bignum-plus-p y)) - ((bignum fixnum) - (not (bignum-plus-p x))) - ((bignum bignum) - (minusp (bignum-compare x y)))) - -(two-arg- two-arg-> > ceiling floor - ((fixnum bignum) - (not (bignum-plus-p y))) - ((bignum fixnum) - (bignum-plus-p x)) - ((bignum bignum) - (plusp (bignum-compare x y)))) +(macrolet ((def-two-arg- (name op ratio-arg1 ratio-arg2 &rest cases) + `(defun ,name (x y) + (number-dispatch ((x real) (y real)) + (basic-compare ,op) + (((foreach fixnum bignum) ratio) + (,op x (,ratio-arg2 (numerator y) + (denominator y)))) + ((ratio integer) + (,op (,ratio-arg1 (numerator x) + (denominator x)) + y)) + ((ratio ratio) + (,op (* (numerator (truly-the ratio x)) + (denominator (truly-the ratio y))) + (* (numerator (truly-the ratio y)) + (denominator (truly-the ratio x))))) + ,@cases)))) + (def-two-arg- two-arg-< < floor ceiling + ((fixnum bignum) + (bignum-plus-p y)) + ((bignum fixnum) + (not (bignum-plus-p x))) + ((bignum bignum) + (minusp (bignum-compare x y)))) + (def-two-arg- two-arg-> > ceiling floor + ((fixnum bignum) + (not (bignum-plus-p y))) + ((bignum fixnum) + (bignum-plus-p x)) + ((bignum bignum) + (plusp (bignum-compare x y))))) (defun two-arg-= (x y) (number-dispatch ((x number) (y number)) diff --git a/tests/float.pure.lisp b/tests/float.pure.lisp index 9b87d63..e5d62ac 100644 --- a/tests/float.pure.lisp +++ b/tests/float.pure.lisp @@ -24,5 +24,7 @@ (assert (= (/ -ifni 0.1) -ifni)) (assert (= (/ -ifni 100/3) -ifni)) (assert (< -ifni +ifni)) - (assert (not (< +ifni 100))) + ;; FIXME: Reenable this when bug 92 is fixed. + ;; (assert (not (< +ifni 100))) + (assert (not (< +ifni 100.0))) (assert (not (< +ifni -ifni)))) diff --git a/tests/type.impure.lisp b/tests/type.impure.lisp index 4533165..31b42e9 100644 --- a/tests/type.impure.lisp +++ b/tests/type.impure.lisp @@ -13,7 +13,13 @@ integer fixnum (integer 0 10) single-float (single-float -1.0 1.0) (single-float 0.1) (real 4 8) (real -1 7) (real 2 11) - (member #\a #\b #\c) (member 1 #\a) (member 3.0 3.3)))) + (member #\a #\b #\c) (member 1 #\a) (member 3.0 3.3) + ;; FIXME: When bug 91 is fixed, add these to the list: + ;; (INTEGER -1 1) + ;; UNSIGNED-BYTE + ;; (RATIONAL -1 7) (RATIONAL -2 4) + ;; RATIO + ))) (dolist (i types) (format t "type I=~S~%" i) (dolist (j types) diff --git a/version.lisp-expr b/version.lisp-expr index bfc0f9f..0cafb49 100644 --- a/version.lisp-expr +++ b/version.lisp-expr @@ -15,4 +15,4 @@ ;;; versions, and a string like "0.6.5.12" is used for versions which ;;; aren't released but correspond only to CVS tags or snapshots. -"0.6.11.24" +"0.6.11.25"