bug-gnu-emacs
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

bug#5114: 23.1.50; (string-to-number (number-to-string most-positive-fix


From: Helmut Eller
Subject: bug#5114: 23.1.50; (string-to-number (number-to-string most-positive-fixnum))
Date: Sat, 05 Dec 2009 13:36:41 +0100
User-agent: Gnus/5.13 (Gnus v5.13) Emacs/23.1.50 (gnu/linux)

* Stefan Monnier [2009-12-03 21:52+0100] writes:

>> (setq x (string-to-number (number-to-string most-positive-fixnum)))
>> (= most-positive-fixnum x) => nil
>    
>> x is 2305843009213693440 but it should be most-positive-fixnum
>> which is 2305843009213693951.
>
>> The test
>> (= most-positive-fixnum 
>>    (string-to-number (number-to-string most-positive-fixnum)))
>> seems to work as expected on 32-bit machines but not so on 64 bit.
>
> Indeed, it passes through a floating point conversion, so there's only
> abour 52 bit of precesion.

Here is a patch for string-to-number to use the full fixnum range:

--- data.c.~1.308.~     2009-12-05 08:07:48.000000000 +0100
+++ data.c      2009-12-05 13:27:33.000000000 +0100
@@ -2393,23 +2393,26 @@
     p++;
 
   if (isfloat_string (p, 1) && b == 10)
-    val = make_float (sign * atof (p));
-  else
-    {
-      double v = 0;
-
-      while (1)
-       {
-         int digit = digit_to_number (*p++, b);
-         if (digit < 0)
-           break;
-         v = v * b + digit;
-       }
-
-      val = make_fixnum_or_float (sign * v);
-    }
-
-  return val;
+    return make_float (sign * atof (p));
+  else {
+    unsigned long u = 0;
+    while (1)
+      {
+       int digit = digit_to_number (*p++, b);
+       if (digit < 0)
+         return make_number (sign * u);
+       else if (u <= (MOST_POSITIVE_FIXNUM - digit) / b)
+         u = u * b + digit;
+       else
+         {
+           /* overflow to flonums */
+           double f = ((double)u) * b + digit;
+           while (digit = digit_to_number (*p++, b), digit >= 0)
+             f = f * b + digit;
+           return make_float (sign * f);
+         }
+      }
+  }
 }
 
 
Helmut

reply via email to

[Prev in Thread] Current Thread [Next in Thread]