source: clfs-embedded/patches/mpfr-2.3.2-branch_update-1.patch@ c1f3dad

Last change on this file since c1f3dad was 06beb7a, checked in by Jim Gifford <clfs@…>, 16 years ago

Added: Patches

  • Property mode set to 100644
File size: 16.4 KB
RevLine 
[06beb7a]1Submitted By: Jim Gifford (jim at cross-lfs dot org)
2Date: 2009-01-05
3Initial Package Version: 2.3.2
4Origin: MPFR Website
5Upstream Status: Fixed
6Description: See http://www.mpfr.org Website Under Bugs
7
8diff -Naur mpfr-2.3.2.orig/strtofr.c mpfr-2.3.2/strtofr.c
9--- mpfr-2.3.2.orig/strtofr.c 2008-09-12 06:47:25.000000000 -0700
10+++ mpfr-2.3.2/strtofr.c 2009-01-05 03:26:12.000000000 -0800
11@@ -31,14 +31,24 @@
12 #define MPFR_MAX_BASE 62
13
14 struct parsed_string {
15- int negative;
16- int base;
17- unsigned char *mantissa, *mant;
18- size_t prec, alloc;
19- mp_exp_t exp_base, exp_bin;
20+ int negative; /* non-zero iff the number is negative */
21+ int base; /* base of the string */
22+ unsigned char *mantissa; /* raw significand (without any point) */
23+ unsigned char *mant; /* stripped significand (without starting and
24+ ending zeroes) */
25+ size_t prec; /* length of mant (zero for +/-0) */
26+ size_t alloc; /* allocation size of mantissa */
27+ mp_exp_t exp_base; /* number of digits before the point */
28+ mp_exp_t exp_bin; /* exponent in case base=2 or 16, and the pxxx
29+ format is used (i.e., exponent is given in
30+ base 10) */
31 };
32
33-/* This table has been generated by the following program */
34+/* This table has been generated by the following program.
35+ For 2 <= b <= MPFR_MAX_BASE,
36+ RedInvLog2Table[b-2][0] / RedInvLog2Table[b-2][1]
37+ is an upper approximation of log(2)/log(b).
38+*/
39 static const unsigned long RedInvLog2Table[MPFR_MAX_BASE-1][2] = {
40 {1UL, 1UL},
41 {53UL, 84UL},
42@@ -441,7 +451,7 @@
43 MPFR_ZIV_DECL (loop);
44 MPFR_TMP_DECL (marker);
45
46- /* determine the minimal precision for the computation */
47+ /* initialize the working precision */
48 prec = MPFR_PREC (x) + MPFR_INT_CEIL_LOG2 (MPFR_PREC (x));
49
50 /* compute y as long as rounding is not possible */
51@@ -449,60 +459,88 @@
52 MPFR_ZIV_INIT (loop, prec);
53 for (;;)
54 {
55- /* initialize y to the value of 0.mant_s[0]...mant_s[pr-1] */
56+ /* Set y to the value of the ~prec most significant bits of pstr->mant
57+ (as long as we guarantee correct rounding, we don't need to get
58+ exactly prec bits). */
59 ysize = (prec - 1) / BITS_PER_MP_LIMB + 1;
60+ /* prec bits corresponds to ysize limbs */
61 ysize_bits = ysize * BITS_PER_MP_LIMB;
62+ /* and to ysize_bits >= prec > MPFR_PREC (x) bits */
63 y = (mp_limb_t*) MPFR_TMP_ALLOC ((2 * ysize + 1) * sizeof (mp_limb_t));
64- y += ysize;
65+ y += ysize; /* y has (ysize+1) allocated limbs */
66
67- /* required precision for str to have ~ysize limbs
68- We must have (2^(BITS_PER_MP_LIMB))^ysize ~= base^pstr_size
69- aka pstr_size = ceil (ysize*BITS_PER_MP_LIMB/log2(base))
70- ysize ~ prec/BITS_PER_MP_LIMB and prec < Umax/2 =>
71+ /* pstr_size is the number of characters we read in pstr->mant
72+ to have at least ysize full limbs.
73+ We must have base^(pstr_size-1) >= (2^(BITS_PER_MP_LIMB))^ysize
74+ (in the worst case, the first digit is one and all others are zero).
75+ i.e., pstr_size >= 1 + ysize*BITS_PER_MP_LIMB/log2(base)
76+ Since ysize ~ prec/BITS_PER_MP_LIMB and prec < Umax/2 =>
77 ysize*BITS_PER_MP_LIMB can not overflow.
78- Compute pstr_size = ysize_bits * Num / Den
79- Where Num/Den ~ 1/log2(base)
80+ We compute pstr_size = 1 + ceil(ysize_bits * Num / Den)
81+ where Num/Den >= 1/log2(base)
82 It is not exactly ceil(1/log2(base)) but could be one more (base 2)
83- Quite ugly since it tries to avoid overflow */
84- pstr_size = ( ysize_bits / RedInvLog2Table[pstr->base-2][1]
85- * RedInvLog2Table[pstr->base-2][0] )
86- + (( ysize_bits % RedInvLog2Table[pstr->base-2][1])
87- * RedInvLog2Table[pstr->base-2][0]
88- / RedInvLog2Table[pstr->base-2][1] )
89- + 1;
90+ Quite ugly since it tries to avoid overflow:
91+ let Num = RedInvLog2Table[pstr->base-2][0]
92+ and Den = RedInvLog2Table[pstr->base-2][1],
93+ and ysize_bits = a*Den+b,
94+ then ysize_bits * Num/Den = a*Num + (b * Num)/Den,
95+ thus ceil(ysize_bits * Num/Den) = a*Num + floor(b * Num + Den - 1)/Den
96+ */
97+ {
98+ unsigned long Num = RedInvLog2Table[pstr->base-2][0];
99+ unsigned long Den = RedInvLog2Table[pstr->base-2][1];
100+ pstr_size = ((ysize_bits / Den) * Num)
101+ + (((ysize_bits % Den) * Num + Den - 1) / Den)
102+ + 1;
103+ }
104+
105+ /* since pstr_size corresponds to at least ysize_bits full bits,
106+ and ysize_bits > prec, the weight of the neglected part of
107+ pstr->mant (if any) is < ulp(y) < ulp(x) */
108
109+ /* if the number of wanted characters is more than what we have in
110+ pstr->mant, round it down */
111 if (pstr_size >= pstr->prec)
112 pstr_size = pstr->prec;
113- MPFR_ASSERTD ((mp_exp_t) pstr_size == (mp_exp_t) pstr_size);
114+ MPFR_ASSERTD (pstr_size == (mp_exp_t) pstr_size);
115
116- /* convert str into binary */
117+ /* convert str into binary: note that pstr->mant is big endian,
118+ thus no offset is needed */
119 real_ysize = mpn_set_str (y, pstr->mant, pstr_size, pstr->base);
120- MPFR_ASSERTD ( real_ysize <= ysize+1);
121+ MPFR_ASSERTD (real_ysize <= ysize+1);
122
123 /* normalize y: warning we can get even get ysize+1 limbs! */
124- MPFR_ASSERTD (y[real_ysize - 1] != 0);
125+ MPFR_ASSERTD (y[real_ysize - 1] != 0); /* mpn_set_str guarantees this */
126 count_leading_zeros (count, y[real_ysize - 1]);
127- exact = (real_ysize <= ysize);
128- if (exact != 0) /* shift y to the left in that case y should be exact */
129+ /* exact means that the number of limbs of the output of mpn_set_str
130+ is less or equal to ysize */
131+ exact = real_ysize <= ysize;
132+ if (exact) /* shift y to the left in that case y should be exact */
133 {
134+ /* we have enough limbs to store {y, real_ysize} */
135 /* shift {y, num_limb} for count bits to the left */
136 if (count != 0)
137- mpn_lshift (y, y, real_ysize, count);
138- /* shift {y, num_limb} for (ysize-num_limb) limbs to the left */
139+ mpn_lshift (y + ysize - real_ysize, y, real_ysize, count);
140 if (real_ysize != ysize)
141 {
142- MPN_COPY_DECR (y + ysize - real_ysize, y, real_ysize);
143+ if (count == 0)
144+ MPN_COPY_DECR (y + ysize - real_ysize, y, real_ysize);
145 MPN_ZERO (y, ysize - real_ysize);
146 }
147 /* for each bit shift decrease exponent of y */
148 /* (This should not overflow) */
149 exp = - ((ysize - real_ysize) * BITS_PER_MP_LIMB + count);
150 }
151- else /* shift y for the right */
152+ else /* shift y to the right, by doing this we might lose some
153+ bits from the result of mpn_set_str (in addition to the
154+ characters neglected from pstr->mant) */
155 {
156 /* shift {y, num_limb} for (BITS_PER_MP_LIMB - count) bits
157- to the right */
158- mpn_rshift (y, y, real_ysize, BITS_PER_MP_LIMB - count);
159+ to the right. FIXME: can we prove that count cannot be zero here,
160+ since mpn_rshift does not accept a shift of BITS_PER_MP_LIMB? */
161+ MPFR_ASSERTD (count != 0);
162+ exact = mpn_rshift (y, y, real_ysize, BITS_PER_MP_LIMB - count) ==
163+ MPFR_LIMB_ZERO;
164 /* for each bit shift increase exponent of y */
165 exp = BITS_PER_MP_LIMB - count;
166 }
167@@ -542,7 +580,7 @@
168 result = y;
169 err = 0;
170 }
171- /* case pstr->exp_base > pstr_size */
172+ /* case non-power-of-two-base, and pstr->exp_base > pstr_size */
173 else if (pstr->exp_base > (mp_exp_t) pstr_size)
174 {
175 mp_limb_t *z;
176@@ -559,6 +597,13 @@
177 goto overflow;
178 exact = exact && (err == -1);
179
180+ /* If exact is non zero, then z equals exactly the value of the
181+ pstr_size most significant digits from pstr->mant, i.e., the
182+ only difference can come from the neglected pstr->prec-pstr_size
183+ least significant digits of pstr->mant.
184+ If exact is zero, then z is rounded towards zero with respect
185+ to that value. */
186+
187 /* multiply(y = 0.mant_s[0]...mant_s[pr-1])_base by base^(exp_s-g) */
188 mpn_mul_n (result, y, z, ysize);
189
190@@ -588,6 +633,8 @@
191 exp --;
192 }
193
194+ /* if the low ysize limbs of {result, 2*ysize} are all zero,
195+ then the result is still "exact" (if it was before) */
196 exact = exact && (mpn_scan1 (result, 0)
197 >= (unsigned long) ysize_bits);
198 result += ysize;
199@@ -598,7 +645,7 @@
200 mp_limb_t *z;
201 mp_exp_t exp_z;
202
203- result = (mp_limb_t*) MPFR_TMP_ALLOC ( (3*ysize+1) * BYTES_PER_MP_LIMB);
204+ result = (mp_limb_t*) MPFR_TMP_ALLOC ((3*ysize+1) * BYTES_PER_MP_LIMB);
205
206 /* set y to y * K^ysize */
207 y = y - ysize; /* we have allocated ysize limbs at y - ysize */
208@@ -622,7 +669,7 @@
209 /* compute y / z */
210 /* result will be put into result + n, and remainder into result */
211 mpn_tdiv_qr (result + ysize, result, (mp_size_t) 0, y,
212- 2*ysize, z, ysize);
213+ 2 * ysize, z, ysize);
214
215 /* exp -= exp_z + ysize_bits with overflow checking
216 and check that we can add/substract 2 to exp without overflow */
217@@ -635,6 +682,8 @@
218 MPFR_EXP_MIN+2, MPFR_EXP_MAX-2,
219 goto overflow, goto underflow);
220 err += 2;
221+ /* if the remainder of the division is zero, then the result is
222+ still "exact" if it was before */
223 exact = exact && (mpn_popcount (result, ysize) == 0);
224
225 /* normalize result */
226@@ -648,7 +697,7 @@
227 }
228 result += ysize;
229 }
230- /* case exp_base = pstr_size */
231+ /* case exp_base = pstr_size: no multiplication or division needed */
232 else
233 {
234 /* base^(exp_s-pr) = 1 nothing to compute */
235@@ -656,6 +705,17 @@
236 err = 0;
237 }
238
239+ /* If result is exact, we still have to consider the neglected part
240+ of the input string. For a directed rounding, in that case we could
241+ still correctly round, since the neglected part is less than
242+ one ulp, but that would make the code more complex, and give a
243+ speedup for rare cases only. */
244+ exact = exact && (pstr_size == pstr->prec);
245+
246+ /* at this point, result is an approximation rounded towards zero
247+ of the pstr_size most significant digits of pstr->mant, with
248+ equality in case exact is non-zero. */
249+
250 /* test if rounding is possible, and if so exit the loop */
251 if (exact || mpfr_can_round_raw (result, ysize,
252 (pstr->negative) ? -1 : 1,
253@@ -679,6 +739,13 @@
254 exp ++;
255 }
256
257+ if (res == 0) /* fix ternary value */
258+ {
259+ exact = exact && (pstr_size == pstr->prec);
260+ if (!exact)
261+ res = (pstr->negative) ? 1 : -1;
262+ }
263+
264 /* Set sign of x before exp since check_range needs a valid sign */
265 (pstr->negative) ? MPFR_SET_NEG (x) : MPFR_SET_POS (x);
266
267diff -Naur mpfr-2.3.2.orig/tests/tset_str.c mpfr-2.3.2/tests/tset_str.c
268--- mpfr-2.3.2.orig/tests/tset_str.c 2008-09-12 06:47:25.000000000 -0700
269+++ mpfr-2.3.2/tests/tset_str.c 2009-01-05 03:25:12.000000000 -0800
270@@ -81,6 +81,24 @@
271 mpfr_clear (a);
272 }
273
274+/* Bug found by Christoph Lauter. */
275+static void
276+bug20081028 (void)
277+{
278+ mpfr_t x;
279+ const char *s = "0.10000000000000000000000000000001E1";
280+
281+ mpfr_init2 (x, 32);
282+ mpfr_set_str (x, "1.00000000000000000006", 10, GMP_RNDU);
283+ if (! mpfr_greater_p (x, __gmpfr_one))
284+ {
285+ printf ("Error in bug20081028:\nExpected %s\nGot ", s);
286+ mpfr_dump (x);
287+ exit (1);
288+ }
289+ mpfr_clear (x);
290+}
291+
292 int
293 main (int argc, char *argv[])
294 {
295@@ -845,6 +863,7 @@
296 mpfr_clear (y);
297
298 check_underflow ();
299+ bug20081028 ();
300
301 tests_end_mpfr ();
302 return 0;
303diff -Naur mpfr-2.3.2.orig/tests/tstrtofr.c mpfr-2.3.2/tests/tstrtofr.c
304--- mpfr-2.3.2.orig/tests/tstrtofr.c 2008-09-12 06:47:25.000000000 -0700
305+++ mpfr-2.3.2/tests/tstrtofr.c 2009-01-05 03:26:12.000000000 -0800
306@@ -27,28 +27,7 @@
307
308 #include "mpfr-test.h"
309
310-static void check_reftable (void);
311-static void check_special (void);
312-static void check_retval (void);
313-static void check_overflow (void);
314-static void check_parse (void);
315-
316-int
317-main (int argc, char *argv[])
318-{
319- tests_start_mpfr ();
320-
321- check_special();
322- check_reftable ();
323- check_parse ();
324- check_overflow ();
325- check_retval ();
326-
327- tests_end_mpfr ();
328- return 0;
329-}
330-
331-void
332+static void
333 check_special (void)
334 {
335 mpfr_t x, y;
336@@ -551,8 +530,7 @@
337 "1.001000010110011011000101100000101111101001100011101101001111110111000010010110010001100e-16920"}
338 };
339
340-
341-void
342+static void
343 check_reftable ()
344 {
345 int i, base;
346@@ -597,7 +575,7 @@
347 mpfr_clear (x);
348 }
349
350-void
351+static void
352 check_parse (void)
353 {
354 mpfr_t x;
355@@ -951,3 +929,104 @@
356
357 mpfr_clear (x);
358 }
359+
360+/* Bug found by Christoph Lauter (in mpfr_set_str). */
361+static struct bug20081025_test {
362+ mpfr_rnd_t rnd;
363+ int inexact;
364+ const char *str;
365+ const char *binstr;
366+} Bug20081028Table[] = {
367+ {GMP_RNDN, -1, "1.00000000000000000006", "1"},
368+ {GMP_RNDZ, -1, "1.00000000000000000006", "1"},
369+ {GMP_RNDU, +1, "1.00000000000000000006",
370+ "10000000000000000000000000000001e-31"},
371+ {GMP_RNDD, -1, "1.00000000000000000006", "1"},
372+
373+
374+ {GMP_RNDN, +1, "-1.00000000000000000006", "-1"},
375+ {GMP_RNDZ, +1, "-1.00000000000000000006", "-1"},
376+ {GMP_RNDU, +1, "-1.00000000000000000006", "-1"},
377+ {GMP_RNDD, -1, "-1.00000000000000000006",
378+ "-10000000000000000000000000000001e-31"},
379+
380+ {GMP_RNDN, +1, "0.999999999999999999999999999999999999999999999", "1"},
381+ {GMP_RNDZ, -1, "0.999999999999999999999999999999999999999999999",
382+ "11111111111111111111111111111111e-32"},
383+ {GMP_RNDU, +1, "0.999999999999999999999999999999999999999999999", "1"},
384+ {GMP_RNDD, -1, "0.999999999999999999999999999999999999999999999",
385+ "11111111111111111111111111111111e-32"},
386+
387+ {GMP_RNDN, -1, "-0.999999999999999999999999999999999999999999999", "-1"},
388+ {GMP_RNDZ, +1, "-0.999999999999999999999999999999999999999999999",
389+ "-11111111111111111111111111111111e-32"},
390+ {GMP_RNDU, +1, "-0.999999999999999999999999999999999999999999999",
391+ "-11111111111111111111111111111111e-32"},
392+ {GMP_RNDD, -1, "-0.999999999999999999999999999999999999999999999", "-1"}
393+};
394+
395+static void
396+bug20081028 (void)
397+{
398+ int i;
399+ int inexact, res;
400+ mpfr_rnd_t rnd;
401+ mpfr_t x, y;
402+ char *s;
403+
404+ mpfr_init2 (x, 32);
405+ mpfr_init2 (y, 32);
406+ for (i = 0 ; i < numberof (Bug20081028Table) ; i++)
407+ {
408+ rnd = Bug20081028Table[i].rnd;
409+ inexact = Bug20081028Table[i].inexact;
410+ mpfr_set_str_binary (x, Bug20081028Table[i].binstr);
411+ res = mpfr_strtofr (y, Bug20081028Table[i].str, &s, 10, rnd);
412+ if (s == NULL || *s != 0)
413+ {
414+ printf ("Error in Bug20081028: strtofr didn't parse entire input\n"
415+ "for (i=%d) Str=\"%s\"", i, Bug20081028Table[i].str);
416+ exit (1);
417+ }
418+ if (! SAME_SIGN (res, inexact))
419+ {
420+ printf ("Error in Bug20081028: expected %s ternary value, "
421+ "got %d\nfor (i=%d) Rnd=%s Str=\"%s\"\n Set binary gives: ",
422+ inexact > 0 ? "positive" : "negative",
423+ res, i, mpfr_print_rnd_mode(rnd), Bug20081028Table[i].str);
424+ mpfr_dump (x);
425+ printf (" strtofr gives: ");
426+ mpfr_dump (y);
427+ exit (1);
428+ }
429+ if (mpfr_cmp (x, y))
430+ {
431+ printf ("Error in Bug20081028: Results differ between strtofr and "
432+ "set_binary\nfor (i=%d) Rnd=%s Str=\"%s\"\n"
433+ " Set binary gives: ",
434+ i, mpfr_print_rnd_mode(rnd), Bug20081028Table[i].str);
435+ mpfr_dump (x);
436+ printf (" strtofr gives: ");
437+ mpfr_dump (y);
438+ exit (1);
439+ }
440+ }
441+ mpfr_clear (y);
442+ mpfr_clear (x);
443+}
444+
445+int
446+main (int argc, char *argv[])
447+{
448+ tests_start_mpfr ();
449+
450+ check_special();
451+ check_reftable ();
452+ check_parse ();
453+ check_overflow ();
454+ check_retval ();
455+ bug20081028 ();
456+
457+ tests_end_mpfr ();
458+ return 0;
459+}
Note: See TracBrowser for help on using the repository browser.