olga / rpms / glibc

Forked from rpms/glibc 5 years ago
Clone

Blame SOURCES/glibc-rh1162895-3.patch

00db10
commit b0abbc21034f0e5edc49023d8fda0616173faf17
00db10
Author: Alan Modra <amodra@gmail.com>
00db10
Date:   Wed Apr 2 13:46:19 2014 +1030
00db10
00db10
    Correct IBM long double nextafterl.
00db10
    
00db10
    Fix for values near a power of two, and some tidies.
00db10
    
00db10
    	[BZ #16739]
00db10
    	* sysdeps/ieee754/ldbl-128ibm/s_nextafterl.c (__nextafterl): Correct
00db10
    	output when value is near a power of two.  Use int64_t for lx and
00db10
    	remove casts.  Use decimal rather than hex exponent constants.
00db10
    	Don't use long double multiplication when double will suffice.
00db10
    	* math/libm-test.inc (nextafter_test_data): Add tests.
00db10
    	* NEWS: Add 16739 and 16786 to bug list.
00db10
00db10
diff --git glibc-2.17-c758a686/math/libm-test.inc glibc-2.17-c758a686/math/libm-test.inc
00db10
index 19194f6..967b679 100644
00db10
--- glibc-2.17-c758a686/math/libm-test.inc
00db10
+++ glibc-2.17-c758a686/math/libm-test.inc
00db10
@@ -8199,6 +8208,14 @@ pow_test (void)
00db10
 #if defined TEST_DOUBLE || defined TEST_LDOUBLE
00db10
   TEST_ff_f (pow, -7.49321e+133, -9.80818e+16, 0, UNDERFLOW_EXCEPTION);
00db10
 #endif
00db10
+#if defined TEST_LDOUBLE && LDBL_MANT_DIG == 106
00db10
+    TEST_ff_f (nextafter, 1.0L, -10.0L, 1.0L-0x1p-106L, NO_EXCEPTION),
00db10
+    TEST_ff_f (nextafter, 1.0L, 10.0L, 1.0L+0x1p-105L, NO_EXCEPTION),
00db10
+    TEST_ff_f (nextafter, 1.0L-0x1p-106L, 10.0L, 1.0L, NO_EXCEPTION),
00db10
+    TEST_ff_f (nextafter, -1.0L, -10.0L, -1.0L-0x1p-105L, NO_EXCEPTION),
00db10
+    TEST_ff_f (nextafter, -1.0L, 10.0L, -1.0L+0x1p-106L, NO_EXCEPTION),
00db10
+    TEST_ff_f (nextafter, -1.0L+0x1p-106L, -10.0L, -1.0L, NO_EXCEPTION),
00db10
+#endif
00db10
 
00db10
   TEST_ff_f (pow, -1.0, -0xffffff, -1.0);
00db10
   TEST_ff_f (pow, -1.0, -0x1fffffe, 1.0);
00db10
diff --git glibc-2.17-c758a686/sysdeps/ieee754/ldbl-128ibm/s_nextafterl.c glibc-2.17-c758a686/sysdeps/ieee754/ldbl-128ibm/s_nextafterl.c
00db10
index 30b1540..bf57cb8 100644
00db10
--- glibc-2.17-c758a686/sysdeps/ieee754/ldbl-128ibm/s_nextafterl.c
00db10
+++ glibc-2.17-c758a686/sysdeps/ieee754/ldbl-128ibm/s_nextafterl.c
00db10
@@ -30,8 +30,7 @@ static char rcsid[] = "$NetBSD: $";
00db10
 
00db10
 long double __nextafterl(long double x, long double y)
00db10
 {
00db10
-	int64_t hx,hy,ihx,ihy;
00db10
-	uint64_t lx;
00db10
+	int64_t hx, hy, ihx, ihy, lx;
00db10
 	double xhi, xlo, yhi;
00db10
 
00db10
 	ldbl_unpack (x, &xhi, &xlo;;
00db10
@@ -79,19 +78,28 @@ long double __nextafterl(long double x, long double y)
00db10
 	      u = math_opt_barrier (x);
00db10
 	      x -= __LDBL_DENORM_MIN__;
00db10
 	      if (ihx < 0x0360000000000000LL
00db10
-		  || (hx > 0 && (int64_t) lx <= 0)
00db10
-		  || (hx < 0 && (int64_t) lx > 1)) {
00db10
+		  || (hx > 0 && lx <= 0)
00db10
+		  || (hx < 0 && lx > 1)) {
00db10
 		u = u * u;
00db10
 		math_force_eval (u);		/* raise underflow flag */
00db10
 	      }
00db10
 	      return x;
00db10
 	    }
00db10
-	    if (ihx < 0x06a0000000000000LL) { /* ulp will denormal */
00db10
-	      INSERT_WORDS64 (yhi, hx & (0x7ffLL<<52));
00db10
-	      u = yhi;
00db10
-	      u *= 0x1.0000000000000p-105L;
00db10
+	    /* If the high double is an exact power of two and the low
00db10
+	       double is the opposite sign, then 1ulp is one less than
00db10
+	       what we might determine from the high double.  Similarly
00db10
+	       if X is an exact power of two, and positive, because
00db10
+	       making it a little smaller will result in the exponent
00db10
+	       decreasing by one and normalisation of the mantissa.   */
00db10
+	    if ((hx & 0x000fffffffffffffLL) == 0
00db10
+		&& ((lx != 0 && (hx ^ lx) < 0)
00db10
+		    || (lx == 0 && hx >= 0)))
00db10
+	      ihx -= 1LL << 52;
00db10
+	    if (ihx < (106LL << 52)) { /* ulp will denormal */
00db10
+	      INSERT_WORDS64 (yhi, ihx & (0x7ffLL<<52));
00db10
+	      u = yhi * 0x1p-105;
00db10
 	    } else {
00db10
-	      INSERT_WORDS64 (yhi, (hx & (0x7ffLL<<52))-(0x069LL<<52));
00db10
+	      INSERT_WORDS64 (yhi, (ihx & (0x7ffLL<<52))-(105LL<<52));
00db10
 	      u = yhi;
00db10
 	    }
00db10
 	    return x - u;
00db10
@@ -109,8 +117,8 @@ long double __nextafterl(long double x, long double y)
00db10
 	      u = math_opt_barrier (x);
00db10
 	      x += __LDBL_DENORM_MIN__;
00db10
 	      if (ihx < 0x0360000000000000LL
00db10
-		  || (hx > 0 && (int64_t) lx < 0 && lx != 0x8000000000000001LL)
00db10
-		  || (hx < 0 && (int64_t) lx >= 0)) {
00db10
+		  || (hx > 0 && lx < 0 && lx != 0x8000000000000001LL)
00db10
+		  || (hx < 0 && lx >= 0)) {
00db10
 		u = u * u;
00db10
 		math_force_eval (u);		/* raise underflow flag */
00db10
 	      }
00db10
@@ -118,12 +126,21 @@ long double __nextafterl(long double x, long double y)
00db10
 		x = -0.0L;
00db10
 	      return x;
00db10
 	    }
00db10
-	    if (ihx < 0x06a0000000000000LL) { /* ulp will denormal */
00db10
-	      INSERT_WORDS64 (yhi, hx & (0x7ffLL<<52));
00db10
-	      u = yhi;
00db10
-	      u *= 0x1.0000000000000p-105L;
00db10
+	    /* If the high double is an exact power of two and the low
00db10
+	       double is the opposite sign, then 1ulp is one less than
00db10
+	       what we might determine from the high double.  Similarly
00db10
+	       if X is an exact power of two, and negative, because
00db10
+	       making it a little larger will result in the exponent
00db10
+	       decreasing by one and normalisation of the mantissa.   */
00db10
+	    if ((hx & 0x000fffffffffffffLL) == 0
00db10
+		&& ((lx != 0 && (hx ^ lx) < 0)
00db10
+		    || (lx == 0 && hx < 0)))
00db10
+	      ihx -= 1LL << 52;
00db10
+	    if (ihx < (106LL << 52)) { /* ulp will denormal */
00db10
+	      INSERT_WORDS64 (yhi, ihx & (0x7ffLL<<52));
00db10
+	      u = yhi * 0x1p-105;
00db10
 	    } else {
00db10
-	      INSERT_WORDS64 (yhi, (hx & (0x7ffLL<<52))-(0x069LL<<52));
00db10
+	      INSERT_WORDS64 (yhi, (ihx & (0x7ffLL<<52))-(105LL<<52));
00db10
 	      u = yhi;
00db10
 	    }
00db10
 	    return x + u;