1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 22 /* 23 * Copyright 2011 Nexenta Systems, Inc. All rights reserved. 24 */ 25 /* 26 * Copyright 2006 Sun Microsystems, Inc. All rights reserved. 27 * Use is subject to license terms. 28 */ 29 30 #include <sys/isa_defs.h> 31 #include "libm_synonyms.h" 32 #include "libm_inlines.h" 33 34 #ifdef _LITTLE_ENDIAN 35 #define HI(x) *(1+(int*)x) 36 #define LO(x) *(unsigned*)x 37 #else 38 #define HI(x) *(int*)x 39 #define LO(x) *(1+(unsigned*)x) 40 #endif 41 42 #ifdef __RESTRICT 43 #define restrict _Restrict 44 #else 45 #define restrict 46 #endif 47 48 /* double hypot(double x, double y) 49 * 50 * Method : 51 * 1. Special cases: 52 * x or y is +Inf or -Inf => +Inf 53 * x or y is NaN => QNaN 54 * 2. Computes hypot(x,y): 55 * hypot(x,y) = m * sqrt(xnm * xnm + ynm * ynm) 56 * Where: 57 * m = max(|x|,|y|) 58 * xnm = x * (1/m) 59 * ynm = y * (1/m) 60 * 61 * Compute xnm * xnm + ynm * ynm by simulating 62 * muti-precision arithmetic. 63 * 64 * Accuracy: 65 * Maximum error observed: less than 0.872 ulp after 16.777.216.000 66 * results. 67 */ 68 69 #define sqrt __sqrt 70 71 extern double sqrt( double ); 72 extern double fabs( double ); 73 74 static const unsigned long long LCONST[] = { 75 0x41b0000000000000ULL, /* D2ON28 = 2 ** 28 */ 76 0x0010000000000000ULL, /* D2ONM1022 = 2 ** -1022 */ 77 0x7fd0000000000000ULL /* D2ONP1022 = 2 ** 1022 */ 78 }; 79 80 static void 81 __vhypot_n( int n, double * restrict px, int stridex, double * restrict py, 82 int stridey, double * restrict pz, int stridez ); 83 84 #pragma no_inline(__vhypot_n) 85 86 #define RETURN(ret) \ 87 { \ 88 *pz = (ret); \ 89 py += stridey; \ 90 pz += stridez; \ 91 if ( n_n == 0 ) \ 92 { \ 93 hx0 = HI(px); \ 94 hy0 = HI(py); \ 95 spx = px; spy = py; spz = pz; \ 96 continue; \ 97 } \ 98 n--; \ 99 break; \ 100 } 101 102 void 103 __vhypot( int n, double * restrict px, int stridex, double * restrict py, 104 int stridey, double * restrict pz, int stridez ) 105 { 106 int hx0, hx1, hy0, j0, diff; 107 double x_hi, x_lo, y_hi, y_lo; 108 double scl = 0; 109 double x, y, res; 110 double *spx, *spy, *spz; 111 int n_n; 112 double D2ON28 = ((double*)LCONST)[0]; /* 2 ** 28 */ 113 double D2ONM1022 = ((double*)LCONST)[1]; /* 2 **-1022 */ 114 double D2ONP1022 = ((double*)LCONST)[2]; /* 2 ** 1022 */ 115 116 while ( n > 1 ) 117 { 118 n_n = 0; 119 spx = px; 120 spy = py; 121 spz = pz; 122 hx0 = HI(px); 123 hy0 = HI(py); 124 for ( ; n > 1 ; n-- ) 125 { 126 px += stridex; 127 hx0 &= 0x7fffffff; 128 hy0 &= 0x7fffffff; 129 130 if ( hx0 >= 0x7fe00000 ) /* |X| >= 2**1023 or Inf or NaN */ 131 { 132 diff = hy0 - hx0; 133 j0 = diff >> 31; 134 j0 = hy0 - (diff & j0); 135 j0 &= 0x7ff00000; 136 x = *(px - stridex); 137 y = *py; 138 x = fabs(x); 139 y = fabs(y); 140 if ( j0 >= 0x7ff00000 ) /* |X| or |Y| = Inf or NaN */ 141 { 142 int lx = LO((px - stridex)); 143 int ly = LO(py); 144 if ( hx0 == 0x7ff00000 && lx == 0 ) res = x == y ? y : x; 145 else if ( hy0 == 0x7ff00000 && ly == 0 ) res = x == y ? x : y; 146 else res = x + y; 147 RETURN ( res ) 148 } 149 else 150 { 151 j0 = diff >> 31; 152 if ( ((diff ^ j0) - j0) < 0x03600000 ) /* max(|X|,|Y|)/min(|X|,|Y|) < 2**54 */ 153 { 154 x *= D2ONM1022; 155 y *= D2ONM1022; 156 157 x_hi = ( x + D2ON28 ) - D2ON28; 158 x_lo = x - x_hi; 159 y_hi = ( y + D2ON28 ) - D2ON28; 160 y_lo = y - y_hi; 161 res = (x_hi * x_hi + y_hi * y_hi); 162 res += ((x + x_hi) * x_lo + (y + y_hi) * y_lo); 163 164 res = sqrt ( res ); 165 166 res = D2ONP1022 * res; 167 RETURN ( res ) 168 } 169 else RETURN ( x + y ) 170 } 171 } 172 if ( hy0 >= 0x7fe00000 ) /* |Y| >= 2**1023 or Inf or NaN */ 173 { 174 diff = hy0 - hx0; 175 j0 = diff >> 31; 176 j0 = hy0 - (diff & j0); 177 j0 &= 0x7ff00000; 178 x = *(px - stridex); 179 y = *py; 180 x = fabs(x); 181 y = fabs(y); 182 if ( j0 >= 0x7ff00000 ) /* |X| or |Y| = Inf or NaN */ 183 { 184 int lx = LO((px - stridex)); 185 int ly = LO(py); 186 if ( hx0 == 0x7ff00000 && lx == 0 ) res = x == y ? y : x; 187 else if ( hy0 == 0x7ff00000 && ly == 0 ) res = x == y ? x : y; 188 else res = x + y; 189 RETURN ( res ) 190 } 191 else 192 { 193 j0 = diff >> 31; 194 if ( ((diff ^ j0) - j0) < 0x03600000 ) /* max(|X|,|Y|)/min(|X|,|Y|) < 2**54 */ 195 { 196 x *= D2ONM1022; 197 y *= D2ONM1022; 198 199 x_hi = ( x + D2ON28 ) - D2ON28; 200 x_lo = x - x_hi; 201 y_hi = ( y + D2ON28 ) - D2ON28; 202 y_lo = y - y_hi; 203 res = (x_hi * x_hi + y_hi * y_hi); 204 res += ((x + x_hi) * x_lo + (y + y_hi) * y_lo); 205 206 res = sqrt ( res ); 207 208 res = D2ONP1022 * res; 209 RETURN ( res ) 210 } 211 else RETURN ( x + y ) 212 } 213 } 214 215 hx1 = HI(px); 216 217 if ( hx0 < 0x00100000 && hy0 < 0x00100000 ) /* X and Y are subnormal */ 218 { 219 x = *(px - stridex); 220 y = *py; 221 222 x *= D2ONP1022; 223 y *= D2ONP1022; 224 225 x_hi = ( x + D2ON28 ) - D2ON28; 226 x_lo = x - x_hi; 227 y_hi = ( y + D2ON28 ) - D2ON28; 228 y_lo = y - y_hi; 229 res = (x_hi * x_hi + y_hi * y_hi); 230 res += ((x + x_hi) * x_lo + (y + y_hi) * y_lo); 231 232 res = sqrt(res); 233 234 res = D2ONM1022 * res; 235 RETURN ( res ) 236 } 237 238 hx0 = hx1; 239 py += stridey; 240 pz += stridez; 241 n_n++; 242 hy0 = HI(py); 243 } 244 if ( n_n > 0 ) 245 __vhypot_n ( n_n, spx, stridex, spy, stridey, spz, stridez ); 246 } 247 248 if ( n > 0 ) 249 { 250 x = *px; 251 y = *py; 252 hx0 = HI(px); 253 hy0 = HI(py); 254 255 hx0 &= 0x7fffffff; 256 hy0 &= 0x7fffffff; 257 258 diff = hy0 - hx0; 259 j0 = diff >> 31; 260 j0 = hy0 - (diff & j0); 261 j0 &= 0x7ff00000; 262 263 if ( j0 >= 0x7fe00000 ) /* max(|X|,|Y|) >= 2**1023 or X or Y = Inf or NaN */ 264 { 265 x = fabs(x); 266 y = fabs(y); 267 if ( j0 >= 0x7ff00000 ) /* |X| or |Y| = Inf or NaN */ 268 { 269 int lx = LO(px); 270 int ly = LO(py); 271 if ( hx0 == 0x7ff00000 && lx == 0 ) res = x == y ? y : x; 272 else if ( hy0 == 0x7ff00000 && ly == 0 ) res = x == y ? x : y; 273 else res = x + y; 274 *pz = res; 275 return; 276 } 277 else 278 { 279 j0 = diff >> 31; 280 if ( ((diff ^ j0) - j0) < 0x03600000 ) /* max(|X|,|Y|)/min(|X|,|Y|) < 2**54 */ 281 { 282 x *= D2ONM1022; 283 y *= D2ONM1022; 284 285 x_hi = ( x + D2ON28 ) - D2ON28; 286 x_lo = x - x_hi; 287 y_hi = ( y + D2ON28 ) - D2ON28; 288 y_lo = y - y_hi; 289 res = (x_hi * x_hi + y_hi * y_hi); 290 res += ((x + x_hi) * x_lo + (y + y_hi) * y_lo); 291 292 res = sqrt ( res ); 293 294 res = D2ONP1022 * res; 295 *pz = res; 296 return; 297 } 298 else 299 { 300 *pz = x + y; 301 return; 302 } 303 } 304 } 305 306 if ( j0 < 0x00100000 ) /* X and Y are subnormal */ 307 { 308 x *= D2ONP1022; 309 y *= D2ONP1022; 310 311 x_hi = ( x + D2ON28 ) - D2ON28; 312 x_lo = x - x_hi; 313 y_hi = ( y + D2ON28 ) - D2ON28; 314 y_lo = y - y_hi; 315 res = (x_hi * x_hi + y_hi * y_hi); 316 res += ((x + x_hi) * x_lo + (y + y_hi) * y_lo); 317 318 res = sqrt(res); 319 320 res = D2ONM1022 * res; 321 *pz = res; 322 return; 323 } 324 325 HI(&scl) = (0x7fe00000 - j0); 326 327 x *= scl; 328 y *= scl; 329 330 x_hi = ( x + D2ON28 ) - D2ON28; 331 y_hi = ( y + D2ON28 ) - D2ON28; 332 x_lo = x - x_hi; 333 y_lo = y - y_hi; 334 335 res = (x_hi * x_hi + y_hi * y_hi); 336 res += ((x + x_hi) * x_lo + (y + y_hi) * y_lo); 337 338 res = sqrt(res); 339 340 HI(&scl) = j0; 341 342 res = scl * res; 343 *pz = res; 344 } 345 } 346 347 static void 348 __vhypot_n( int n, double * restrict px, int stridex, double * restrict py, 349 int stridey, double * restrict pz, int stridez ) 350 { 351 int hx0, hy0, j0, diff0; 352 double x_hi0, x_lo0, y_hi0, y_lo0, scl0 = 0; 353 double x0, y0, res0; 354 double D2ON28 = ((double*)LCONST)[0]; /* 2 ** 28 */ 355 356 for( ; n > 0 ; n-- ) 357 { 358 x0 = *px; 359 y0 = *py; 360 hx0 = HI(px); 361 hy0 = HI(py); 362 363 hx0 &= 0x7fffffff; 364 hy0 &= 0x7fffffff; 365 366 diff0 = hy0 - hx0; 367 j0 = diff0 >> 31; 368 j0 = hy0 - (diff0 & j0); 369 j0 &= 0x7ff00000; 370 371 px += stridex; 372 py += stridey; 373 374 HI(&scl0) = ( 0x7fe00000 - j0 ); 375 376 x0 *= scl0; 377 y0 *= scl0; 378 379 x_hi0 = ( x0 + D2ON28 ) - D2ON28; 380 y_hi0 = ( y0 + D2ON28 ) - D2ON28; 381 x_lo0 = x0 - x_hi0; 382 y_lo0 = y0 - y_hi0; 383 384 res0 = (x_hi0 * x_hi0 + y_hi0 * y_hi0); 385 res0 += ((x0 + x_hi0) * x_lo0 + (y0 + y_hi0) * y_lo0); 386 387 res0 = sqrt(res0); 388 389 HI(&scl0) = j0; 390 391 res0 = scl0 * res0; 392 *pz = res0; 393 394 pz += stridez; 395 } 396 } 397