Raymond Toy pushed to branch master at cmucl / cmucl
Commits:
1 changed file:
Changes:
tests/fdlibm.lisp
--- a/tests/fdlibm.lisp
+++ b/tests/fdlibm.lisp
@@ -403,3 +403,115 @@
(assert-error 'floating-point-overflow (exp 1000d0))
;; exp(-1000) underflow
(assert-eql 0d0 (exp -1000d0)))
+
+(define-test log-basic-tests
+ (:tag :fdlibm)
+ (assert-eql 0d0 (log 1d0))
+ (assert-eql 1d0 (log (exp 1d0)))
+ (assert-eql -1d0 (log (exp -1d0)))
+ (assert-eql 0.5d0 (log (sqrt (exp 1d0))))
+ (assert-eql -0.5d0 (log (sqrt (exp -1d0))))
+ ;; Test a denormal arg
+ (assert-eql -709.08956571282410d0 (log (scale-float 1d0 -1023)))
+ ;; Largest double value
+ (assert-eql 709.7827128933840d0 (log most-positive-double-float))
+ ;; Tests case 0 < f < 2^-20, k = 0
+ ;; log(1+2^-21)
+ (assert-eql 4.7683704451632344d-7 (log (+ 1 (scale-float 1d0 -21))))
+ ;; Tests case 0 < f < 2^-20, k = 1
+ ;; log(2 + 2^-20)
+ (assert-eql 0.6931476573969898d0 (log (+ 2(scale-float 1d0 -20))))
+ (assert-eql 1.3862943611198906d0 (log 4d0))
+ ;; Tests main path, i > 0, k = 0
+ (assert-eql 0.3220828910287846d0
+ (log (kernel:make-double-float (+ #x3ff00000 #x6147a) 0)))
+ ;; Tests main path, i > 0, k = 1
+ (assert-eql 0.35065625373947773d0
+ (log (kernel:make-double-float (+ #x3ff00000 #x6b851) 0)))
+ ;; Tests main path, i > 0, k = -1
+ (assert-eql -0.3710642895311607d0
+ (log (kernel:make-double-float (+ #x3fe00000 #x6147a) 0)))
+ ;; Tests main path, i < 0, k = 0
+ (assert-eql 0.3220821999597803d0
+ (log (kernel:make-double-float (+ #x3ff00000 #x61479) 0)))
+ ;; Tests main path, i < 0, k = 1
+ (assert-eql 1.0152293805197257d0
+ (log (kernel:make-double-float (+ #x40000000 #x61479) 0)))
+ ;; Tests main path, i < 0, k = -1
+ (assert-eql -0.37106498060016496d0
+ (log (kernel:make-double-float (+ #x3fe00000 #x61479) 0))))
+
+(define-test log-consistency
+ (:tag :fdlibm)
+ ;; |log(x) + log(1/x)| < 1.77635684e-15, x = 1.2^k, 0 <= k < 2000
+ ;; The threshold is experimentally determined
+ (let ((x 1d0)
+ (max-value -1d0))
+ (declare (double-float max-value)
+ (type (double-float 1d0) x))
+ (dotimes (k 2000)
+ (let ((y (abs (+ (log x) (log (/ x))))))
+ (setf max-value (max max-value y))
+ (setf x (* x 1.4d0))))
+ (assert-true (< max-value 1.77635684d-15)))
+ ;; |exp(log(x)) - x|/x < 5.6766649d-14, x = 1.4^k, 0 <= k < 2000
+ (let ((x 1d0)
+ (max-error 0d0))
+ (declare (double-float max-error)
+ (type (double-float 1d0) x))
+ (dotimes (k 2000)
+ (let ((y (abs (/ (- (exp (log x)) x) x))))
+ (setf max-error (max max-error y))
+ (setf x (* x 1.4d0))))
+ (assert-true (< max-error 5.6766649d-14)))
+ ;; |exp(log(x)) - x|/x < 5.68410245d-14, x = 1.4^(-k), 0 <= k < 2000
+ (let ((x 1d0)
+ (max-error 0d0))
+ (declare (double-float max-error)
+ (type (double-float (0d0)) x))
+ (dotimes (k 2000)
+ (let ((y (abs (/ (- (exp (log x)) x) x))))
+ (setf max-error (max max-error y))
+ (setf x (/ x 1.4d0))))
+ (assert-true (< max-error 5.68410245d-14))))
+
+(define-test sinh-basic-tests
+ (:tag :fdlibm)
+ (assert-eql +0d0 (sinh 0d0))
+ (assert-eql -0d0 (sinh -0d0))
+ ;; sinh(x) = x, |x| < 2^-28
+ (let ((x (scale-float 1d0 -29)))
+ (assert-eql x (sinh x))
+ (assert-eql (- x) (sinh (- x))))
+ ;; case |x| < 1
+ (assert-eql 0.5210953054937474d0 (sinh 0.5d0))
+ (assert-eql -0.5210953054937474d0 (sinh -0.5d0))
+ ;; sinh(10*log(2)) = 1048575/2048, case |x| < 22
+ (let ((x (* 10 (log 2d0)))
+ (y (float 1048575/2048 1d0)))
+ (assert-eql y (sinh x))
+ (assert-eql (- y) (sinh (- x))))
+ ;; sinh(10), case |x| < 22
+ (let ((y 11013.232874703393d0))
+ (assert-eql y (sinh 10d0))
+ (assert-eql (- y) (sinh -10d0)))
+ ;; sinh(32*log(2)), case |x| in [22, log(maxdouble)]
+ (let ((x (* 32 (log 2d0)))
+ (y 2.1474836479999983d9))
+ (assert-eql y (sinh x))
+ (assert-eql (- y) (sinh (- x))))
+ ;; sinh(100), case |x| in [22, log(maxdouble)]
+ (let ((y 1.3440585709080678d43))
+ (assert-eql y (sinh 100d0))
+ (assert-eql (- y) (sinh -100d0)))
+ ;; sinh(710....), no overflow, case |x| in [log(maxdouble), overflowthreshold]
+ (let ((x 710.4758600739439d0)
+ (y 1.7976931348621744d308))
+ (assert-eql y (sinh x))
+ (assert-eql (- y) (sinh (- x))))
+ ;; sinh(710.475860073944), overflow, case |x| > ovfthreshold]
+ (let ((x 710.475860073944d0))
+ (assert-error 'floating-point-overflow (sinh x))
+ (assert-error 'floating-point-overflow (sinh (- x))))
+ (assert-error 'floating-point-overflow (sinh 1000d0))
+ (assert-error 'floating-point-overflow (sinh -1000d0)))