]> git.eshelyaron.com Git - emacs.git/commitdiff
Fix (round FLOAT BIGNUM) bug
authorPaul Eggert <eggert@cs.ucla.edu>
Tue, 4 Sep 2018 18:49:41 +0000 (11:49 -0700)
committerPaul Eggert <eggert@cs.ucla.edu>
Tue, 4 Sep 2018 18:50:54 +0000 (11:50 -0700)
* src/floatfns.c (rounding_driver): Fix bug when one
argument is a float and the other is a bignum.
* test/src/floatfns-tests.el (bignum-round): Test for the bug.

src/floatfns.c
test/src/floatfns-tests.el

index 2f33b8652b29d651444315b5bb0a96a4d0e64ec2..13ab7b0359f9e8e727a14755833821f02f286638 100644 (file)
@@ -355,6 +355,8 @@ rounding_driver (Lisp_Object arg, Lisp_Object divisor,
       CHECK_NUMBER (divisor);
       if (!FLOATP (arg) && !FLOATP (divisor))
        {
+         /* Divide as integers.  Converting to double might lose
+            info, even for fixnums; also see the FIXME below.  */
          if (EQ (divisor, make_fixnum (0)))
            xsignal0 (Qarith_error);
          int_divide (mpz[0],
@@ -363,10 +365,11 @@ rounding_driver (Lisp_Object arg, Lisp_Object divisor,
          return make_integer_mpz ();
        }
 
-      double f1 = FLOATP (arg) ? XFLOAT_DATA (arg) : XFIXNUM (arg);
-      double f2 = FLOATP (divisor) ? XFLOAT_DATA (divisor) : XFIXNUM (divisor);
+      double f1 = XFLOATINT (arg);
+      double f2 = XFLOATINT (divisor);
       if (! IEEE_FLOATING_POINT && f2 == 0)
        xsignal0 (Qarith_error);
+      /* FIXME: This division rounds, so the result is double-rounded.  */
       d = f1 / f2;
     }
 
index d41b08f7965c0e92b20934149ab41db14f41936e..9a382058b4349400cc67aa2ab450d6f0d1986819 100644 (file)
       (should (= n (floor n)))
       (should (= n (round n)))
       (should (= n (truncate n)))
+      (let ((-n (- n))
+           (f (float n))
+           (-f (- (float n))))
+       (should (= 1 (round n f) (round -n -f) (round f n) (round -f -n)))
+       (should (= -1 (round -n f) (round n -f) (round f -n) (round -f n))))
       (dolist (d ns)
         (let ((q (/ n d))
               (r (% n d))