This patch packages the Steven G. Kargl's patch when MPFR 3.1.0 or newer 
is available and fixes the original code for older versions using the 
FX Coudert's machinery (see PR for details).

Tested on powerpc-apple-darwin9 and x86_64-apple-darwin10 without 
regression.

OK for the trunk? If yes, could someone commit it for me?

Since this is a wrong-code issue, I think the light version of the patch 
should be applied to 4.7 and 4.8. If this is accepted I'll do the testing 
and the packaging.

Dominique


2013-06-23  Steven G. Kargl  <s...@troutmask.apl.washington.edu>
            Francois-Xavier Coudert  <fxcoud...@gcc.gnu.org>
            Dominique d'Humieres  <domi...@lps.ens.fr>

        PR fortran/52413
        * simplify.c: Fix the sign of negative values in gfc_simplify_fraction.

2013-06-23  Francois-Xavier Coudert  <fxcoud...@gcc.gnu.org>
            Dominique d'Humieres  <domi...@lps.ens.fr>

        PR fortran/52413
        * gfortran.dg/fraction.f90: New.


--- ../_clean/gcc/fortran/simplify.c    2013-06-08 21:50:33.000000000 +0200
+++ gcc/fortran/simplify.c      2013-06-23 17:19:55.000000000 +0200
@@ -2342,16 +2342,26 @@ gfc_expr *
 gfc_simplify_fraction (gfc_expr *x)
 {
   gfc_expr *result;
+
+#if MPFR_VERSION < MPFR_VERSION_NUM(3,1,0)
   mpfr_t absv, exp, pow2;
+#else
+  mpfr_exp_t e;
+#endif
 
   if (x->expr_type != EXPR_CONSTANT)
     return NULL;
 
   result = gfc_get_constant_expr (BT_REAL, x->ts.kind, &x->where);
 
+#if MPFR_VERSION < MPFR_VERSION_NUM(3,1,0)
+
+  /* MPFR versions before 3.1.0 do not include mpfr_frexp.  
+     TODO: remove the kludge when MPFR 3.1.0 or newer will be required */
+
   if (mpfr_sgn (x->value.real) == 0)
     {
-      mpfr_set_ui (result->value.real, 0, GFC_RND_MODE);
+      mpfr_set (result->value.real, x->value.real, GFC_RND_MODE);
       return result;
     }
 
@@ -2368,10 +2378,16 @@ gfc_simplify_fraction (gfc_expr *x)
 
   mpfr_ui_pow (pow2, 2, exp, GFC_RND_MODE);
 
-  mpfr_div (result->value.real, absv, pow2, GFC_RND_MODE);
+  mpfr_div (result->value.real, x->value.real, pow2, GFC_RND_MODE);
 
   mpfr_clears (exp, absv, pow2, NULL);
 
+#else
+
+  mpfr_frexp (&e, result->value.real, x->value.real, GFC_RND_MODE);
+
+#endif
+
   return range_check (result, "FRACTION");
 }
 
--- ../_clean/gcc/testsuite/gfortran.dg/fraction.f90    1970-01-01 
01:00:00.000000000 +0100
+++ gcc/testsuite/gfortran.dg/fraction.f90      2013-06-23 22:06:00.000000000 
+0200
@@ -0,0 +1,15 @@
+! { dg-do run }
+!
+! Test for pr52413
+!
+
+program test_frac
+
+  real :: y
+  y=fraction (-2.0) 
+  if (fraction (-2.0) /= -0.5) call abort ()
+  if (fraction (-0.0) /= 0.0) call abort ()
+  if (sign(1.0, fraction(-0.0)) /= -1.0) call abort ()
+  if (fraction (-2.0_8) /=  -0.5) call abort ()
+
+end program test_frac

Reply via email to