softfloat-specialize.h 30.6 KB
Newer Older
1 2 3 4 5
/*
 * QEMU float support
 *
 * Derived from SoftFloat.
 */
B
bellard 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37

/*============================================================================

This C source fragment is part of the SoftFloat IEC/IEEE Floating-point
Arithmetic Package, Release 2b.

Written by John R. Hauser.  This work was made possible in part by the
International Computer Science Institute, located at Suite 600, 1947 Center
Street, Berkeley, California 94704.  Funding was partially provided by the
National Science Foundation under grant MIP-9311980.  The original version
of this code was written as part of a project to build a fixed-point vector
processor in collaboration with the University of California at Berkeley,
overseen by Profs. Nelson Morgan and John Wawrzynek.  More information
is available through the Web page `http://www.cs.berkeley.edu/~jhauser/
arithmetic/SoftFloat.html'.

THIS SOFTWARE IS DISTRIBUTED AS IS, FOR FREE.  Although reasonable effort has
been made to avoid it, THIS SOFTWARE MAY CONTAIN FAULTS THAT WILL AT TIMES
RESULT IN INCORRECT BEHAVIOR.  USE OF THIS SOFTWARE IS RESTRICTED TO PERSONS
AND ORGANIZATIONS WHO CAN AND WILL TAKE FULL RESPONSIBILITY FOR ALL LOSSES,
COSTS, OR OTHER PROBLEMS THEY INCUR DUE TO THE SOFTWARE, AND WHO FURTHERMORE
EFFECTIVELY INDEMNIFY JOHN HAUSER AND THE INTERNATIONAL COMPUTER SCIENCE
INSTITUTE (possibly via similar legal warning) AGAINST ALL LOSSES, COSTS, OR
OTHER PROBLEMS INCURRED BY THEIR CUSTOMERS AND CLIENTS DUE TO THE SOFTWARE.

Derivative works are acceptable, even for commercial purposes, so long as
(1) the source code for the derivative work includes prominent notice that
the work is derivative, and (2) the source code includes prominent notice with
these four paragraphs for those parts of this code that are retained.

=============================================================================*/

38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
#if defined(TARGET_MIPS) || defined(TARGET_SH4) || defined(TARGET_UNICORE32)
#define SNAN_BIT_IS_ONE		1
#else
#define SNAN_BIT_IS_ONE		0
#endif

/*----------------------------------------------------------------------------
| The pattern for a default generated half-precision NaN.
*----------------------------------------------------------------------------*/
#if defined(TARGET_ARM)
const float16 float16_default_nan = const_float16(0x7E00);
#elif SNAN_BIT_IS_ONE
const float16 float16_default_nan = const_float16(0x7DFF);
#else
const float16 float16_default_nan = const_float16(0xFE00);
#endif

/*----------------------------------------------------------------------------
| The pattern for a default generated single-precision NaN.
*----------------------------------------------------------------------------*/
#if defined(TARGET_SPARC)
const float32 float32_default_nan = const_float32(0x7FFFFFFF);
#elif defined(TARGET_PPC) || defined(TARGET_ARM) || defined(TARGET_ALPHA)
const float32 float32_default_nan = const_float32(0x7FC00000);
#elif SNAN_BIT_IS_ONE
const float32 float32_default_nan = const_float32(0x7FBFFFFF);
#else
const float32 float32_default_nan = const_float32(0xFFC00000);
#endif

/*----------------------------------------------------------------------------
| The pattern for a default generated double-precision NaN.
*----------------------------------------------------------------------------*/
#if defined(TARGET_SPARC)
const float64 float64_default_nan = const_float64(LIT64( 0x7FFFFFFFFFFFFFFF ));
#elif defined(TARGET_PPC) || defined(TARGET_ARM) || defined(TARGET_ALPHA)
const float64 float64_default_nan = const_float64(LIT64( 0x7FF8000000000000 ));
#elif SNAN_BIT_IS_ONE
const float64 float64_default_nan = const_float64(LIT64( 0x7FF7FFFFFFFFFFFF ));
#else
const float64 float64_default_nan = const_float64(LIT64( 0xFFF8000000000000 ));
#endif

/*----------------------------------------------------------------------------
| The pattern for a default generated extended double-precision NaN.
*----------------------------------------------------------------------------*/
#if SNAN_BIT_IS_ONE
#define floatx80_default_nan_high 0x7FFF
#define floatx80_default_nan_low  LIT64( 0xBFFFFFFFFFFFFFFF )
#else
#define floatx80_default_nan_high 0xFFFF
#define floatx80_default_nan_low  LIT64( 0xC000000000000000 )
#endif

const floatx80 floatx80_default_nan = make_floatx80(floatx80_default_nan_high,
                                                    floatx80_default_nan_low);

/*----------------------------------------------------------------------------
| The pattern for a default generated quadruple-precision NaN.  The `high' and
| `low' values hold the most- and least-significant bits, respectively.
*----------------------------------------------------------------------------*/
#if SNAN_BIT_IS_ONE
#define float128_default_nan_high LIT64( 0x7FFF7FFFFFFFFFFF )
#define float128_default_nan_low  LIT64( 0xFFFFFFFFFFFFFFFF )
#else
#define float128_default_nan_high LIT64( 0xFFFF800000000000 )
#define float128_default_nan_low  LIT64( 0x0000000000000000 )
#endif

const float128 float128_default_nan = make_float128(float128_default_nan_high,
                                                    float128_default_nan_low);

B
bellard 已提交
110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
/*----------------------------------------------------------------------------
| Raises the exceptions specified by `flags'.  Floating-point traps can be
| defined here if desired.  It is currently not possible for such a trap
| to substitute a result value.  If traps are not implemented, this routine
| should be simply `float_exception_flags |= flags;'.
*----------------------------------------------------------------------------*/

void float_raise( int8 flags STATUS_PARAM )
{
    STATUS(float_exception_flags) |= flags;
}

/*----------------------------------------------------------------------------
| Internal canonical NaN format.
*----------------------------------------------------------------------------*/
typedef struct {
    flag sign;
127
    uint64_t high, low;
B
bellard 已提交
128 129
} commonNaNT;

130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167
/*----------------------------------------------------------------------------
| Returns 1 if the half-precision floating-point value `a' is a quiet
| NaN; otherwise returns 0.
*----------------------------------------------------------------------------*/

int float16_is_quiet_nan(float16 a_)
{
    uint16_t a = float16_val(a_);
#if SNAN_BIT_IS_ONE
    return (((a >> 9) & 0x3F) == 0x3E) && (a & 0x1FF);
#else
    return ((a & ~0x8000) >= 0x7c80);
#endif
}

/*----------------------------------------------------------------------------
| Returns 1 if the half-precision floating-point value `a' is a signaling
| NaN; otherwise returns 0.
*----------------------------------------------------------------------------*/

int float16_is_signaling_nan(float16 a_)
{
    uint16_t a = float16_val(a_);
#if SNAN_BIT_IS_ONE
    return ((a & ~0x8000) >= 0x7c80);
#else
    return (((a >> 9) & 0x3F) == 0x3E) && (a & 0x1FF);
#endif
}

/*----------------------------------------------------------------------------
| Returns a quiet NaN if the half-precision floating point value `a' is a
| signaling NaN; otherwise returns `a'.
*----------------------------------------------------------------------------*/
float16 float16_maybe_silence_nan(float16 a_)
{
    if (float16_is_signaling_nan(a_)) {
#if SNAN_BIT_IS_ONE
168
#  if defined(TARGET_MIPS) || defined(TARGET_SH4) || defined(TARGET_UNICORE32)
169 170 171 172 173 174 175 176 177 178 179 180 181
        return float16_default_nan;
#  else
#    error Rules for silencing a signaling NaN are target-specific
#  endif
#else
        uint16_t a = float16_val(a_);
        a |= (1 << 9);
        return make_float16(a);
#endif
    }
    return a_;
}

182 183 184 185 186 187 188 189 190 191 192 193 194
/*----------------------------------------------------------------------------
| Returns the result of converting the half-precision floating-point NaN
| `a' to the canonical NaN format.  If `a' is a signaling NaN, the invalid
| exception is raised.
*----------------------------------------------------------------------------*/

static commonNaNT float16ToCommonNaN( float16 a STATUS_PARAM )
{
    commonNaNT z;

    if ( float16_is_signaling_nan( a ) ) float_raise( float_flag_invalid STATUS_VAR );
    z.sign = float16_val(a) >> 15;
    z.low = 0;
195
    z.high = ((uint64_t) float16_val(a))<<54;
196 197 198
    return z;
}

199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219
/*----------------------------------------------------------------------------
| Returns the result of converting the canonical NaN `a' to the half-
| precision floating-point format.
*----------------------------------------------------------------------------*/

static float16 commonNaNToFloat16(commonNaNT a STATUS_PARAM)
{
    uint16_t mantissa = a.high>>54;

    if (STATUS(default_nan_mode)) {
        return float16_default_nan;
    }

    if (mantissa) {
        return make_float16(((((uint16_t) a.sign) << 15)
                             | (0x1F << 10) | mantissa));
    } else {
        return float16_default_nan;
    }
}

B
bellard 已提交
220
/*----------------------------------------------------------------------------
T
ths 已提交
221 222
| Returns 1 if the single-precision floating-point value `a' is a quiet
| NaN; otherwise returns 0.
B
bellard 已提交
223 224
*----------------------------------------------------------------------------*/

225
int float32_is_quiet_nan( float32 a_ )
B
bellard 已提交
226
{
P
pbrook 已提交
227
    uint32_t a = float32_val(a_);
T
ths 已提交
228
#if SNAN_BIT_IS_ONE
T
ths 已提交
229 230
    return ( ( ( a>>22 ) & 0x1FF ) == 0x1FE ) && ( a & 0x003FFFFF );
#else
231
    return ( 0xFF800000 <= (uint32_t) ( a<<1 ) );
T
ths 已提交
232
#endif
B
bellard 已提交
233 234 235 236 237 238 239
}

/*----------------------------------------------------------------------------
| Returns 1 if the single-precision floating-point value `a' is a signaling
| NaN; otherwise returns 0.
*----------------------------------------------------------------------------*/

P
pbrook 已提交
240
int float32_is_signaling_nan( float32 a_ )
B
bellard 已提交
241
{
P
pbrook 已提交
242
    uint32_t a = float32_val(a_);
T
ths 已提交
243
#if SNAN_BIT_IS_ONE
244
    return ( 0xFF800000 <= (uint32_t) ( a<<1 ) );
T
ths 已提交
245
#else
B
bellard 已提交
246
    return ( ( ( a>>22 ) & 0x1FF ) == 0x1FE ) && ( a & 0x003FFFFF );
T
ths 已提交
247
#endif
B
bellard 已提交
248 249
}

250 251 252 253 254 255 256 257 258
/*----------------------------------------------------------------------------
| Returns a quiet NaN if the single-precision floating point value `a' is a
| signaling NaN; otherwise returns `a'.
*----------------------------------------------------------------------------*/

float32 float32_maybe_silence_nan( float32 a_ )
{
    if (float32_is_signaling_nan(a_)) {
#if SNAN_BIT_IS_ONE
259
#  if defined(TARGET_MIPS) || defined(TARGET_SH4) || defined(TARGET_UNICORE32)
260 261 262 263
        return float32_default_nan;
#  else
#    error Rules for silencing a signaling NaN are target-specific
#  endif
264
#else
265
        uint32_t a = float32_val(a_);
266 267
        a |= (1 << 22);
        return make_float32(a);
268
#endif
269 270 271 272
    }
    return a_;
}

B
bellard 已提交
273 274 275 276 277 278 279 280 281 282 283
/*----------------------------------------------------------------------------
| Returns the result of converting the single-precision floating-point NaN
| `a' to the canonical NaN format.  If `a' is a signaling NaN, the invalid
| exception is raised.
*----------------------------------------------------------------------------*/

static commonNaNT float32ToCommonNaN( float32 a STATUS_PARAM )
{
    commonNaNT z;

    if ( float32_is_signaling_nan( a ) ) float_raise( float_flag_invalid STATUS_VAR );
P
pbrook 已提交
284
    z.sign = float32_val(a)>>31;
B
bellard 已提交
285
    z.low = 0;
286
    z.high = ( (uint64_t) float32_val(a) )<<41;
B
bellard 已提交
287 288 289 290 291 292 293 294
    return z;
}

/*----------------------------------------------------------------------------
| Returns the result of converting the canonical NaN `a' to the single-
| precision floating-point format.
*----------------------------------------------------------------------------*/

295
static float32 commonNaNToFloat32( commonNaNT a STATUS_PARAM)
B
bellard 已提交
296
{
297
    uint32_t mantissa = a.high>>41;
298 299 300 301 302

    if ( STATUS(default_nan_mode) ) {
        return float32_default_nan;
    }

303 304
    if ( mantissa )
        return make_float32(
305
            ( ( (uint32_t) a.sign )<<31 ) | 0x7F800000 | ( a.high>>41 ) );
306 307
    else
        return float32_default_nan;
B
bellard 已提交
308 309
}

310 311 312 313 314 315 316
/*----------------------------------------------------------------------------
| Select which NaN to propagate for a two-input operation.
| IEEE754 doesn't specify all the details of this, so the
| algorithm is target-specific.
| The routine is passed various bits of information about the
| two NaNs and should return 0 to select NaN a and 1 for NaN b.
| Note that signalling NaNs are always squashed to quiet NaNs
317 318
| by the caller, by calling floatXX_maybe_silence_nan() before
| returning them.
319 320 321 322 323 324 325 326
|
| aIsLargerSignificand is only valid if both a and b are NaNs
| of some kind, and is true if a has the larger significand,
| or if both a and b have the same significand but a is
| positive but b is negative. It is only needed for the x87
| tie-break rule.
*----------------------------------------------------------------------------*/

327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347
#if defined(TARGET_ARM)
static int pickNaN(flag aIsQNaN, flag aIsSNaN, flag bIsQNaN, flag bIsSNaN,
                    flag aIsLargerSignificand)
{
    /* ARM mandated NaN propagation rules: take the first of:
     *  1. A if it is signaling
     *  2. B if it is signaling
     *  3. A (quiet)
     *  4. B (quiet)
     * A signaling NaN is always quietened before returning it.
     */
    if (aIsSNaN) {
        return 0;
    } else if (bIsSNaN) {
        return 1;
    } else if (aIsQNaN) {
        return 0;
    } else {
        return 1;
    }
}
348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374
#elif defined(TARGET_MIPS)
static int pickNaN(flag aIsQNaN, flag aIsSNaN, flag bIsQNaN, flag bIsSNaN,
                    flag aIsLargerSignificand)
{
    /* According to MIPS specifications, if one of the two operands is
     * a sNaN, a new qNaN has to be generated. This is done in
     * floatXX_maybe_silence_nan(). For qNaN inputs the specifications
     * says: "When possible, this QNaN result is one of the operand QNaN
     * values." In practice it seems that most implementations choose
     * the first operand if both operands are qNaN. In short this gives
     * the following rules:
     *  1. A if it is signaling
     *  2. B if it is signaling
     *  3. A (quiet)
     *  4. B (quiet)
     * A signaling NaN is always silenced before returning it.
     */
    if (aIsSNaN) {
        return 0;
    } else if (bIsSNaN) {
        return 1;
    } else if (aIsQNaN) {
        return 0;
    } else {
        return 1;
    }
}
375 376 377 378 379 380 381 382 383 384 385 386 387 388 389
#elif defined(TARGET_PPC)
static int pickNaN(flag aIsQNaN, flag aIsSNaN, flag bIsQNaN, flag bIsSNaN,
                   flag aIsLargerSignificand)
{
    /* PowerPC propagation rules:
     *  1. A if it sNaN or qNaN
     *  2. B if it sNaN or qNaN
     * A signaling NaN is always silenced before returning it.
     */
    if (aIsSNaN || aIsQNaN) {
        return 0;
    } else {
        return 1;
    }
}
390
#else
391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419
static int pickNaN(flag aIsQNaN, flag aIsSNaN, flag bIsQNaN, flag bIsSNaN,
                    flag aIsLargerSignificand)
{
    /* This implements x87 NaN propagation rules:
     * SNaN + QNaN => return the QNaN
     * two SNaNs => return the one with the larger significand, silenced
     * two QNaNs => return the one with the larger significand
     * SNaN and a non-NaN => return the SNaN, silenced
     * QNaN and a non-NaN => return the QNaN
     *
     * If we get down to comparing significands and they are the same,
     * return the NaN with the positive sign bit (if any).
     */
    if (aIsSNaN) {
        if (bIsSNaN) {
            return aIsLargerSignificand ? 0 : 1;
        }
        return bIsQNaN ? 1 : 0;
    }
    else if (aIsQNaN) {
        if (bIsSNaN || !bIsQNaN)
            return 0;
        else {
            return aIsLargerSignificand ? 0 : 1;
        }
    } else {
        return 1;
    }
}
420
#endif
421

B
bellard 已提交
422 423 424 425 426 427 428 429
/*----------------------------------------------------------------------------
| Takes two single-precision floating-point values `a' and `b', one of which
| is a NaN, and returns the appropriate NaN result.  If either `a' or `b' is a
| signaling NaN, the invalid exception is raised.
*----------------------------------------------------------------------------*/

static float32 propagateFloat32NaN( float32 a, float32 b STATUS_PARAM)
{
430 431
    flag aIsQuietNaN, aIsSignalingNaN, bIsQuietNaN, bIsSignalingNaN;
    flag aIsLargerSignificand;
432
    uint32_t av, bv;
B
bellard 已提交
433

434
    aIsQuietNaN = float32_is_quiet_nan( a );
B
bellard 已提交
435
    aIsSignalingNaN = float32_is_signaling_nan( a );
436
    bIsQuietNaN = float32_is_quiet_nan( b );
B
bellard 已提交
437
    bIsSignalingNaN = float32_is_signaling_nan( b );
P
pbrook 已提交
438 439
    av = float32_val(a);
    bv = float32_val(b);
440

B
bellard 已提交
441
    if ( aIsSignalingNaN | bIsSignalingNaN ) float_raise( float_flag_invalid STATUS_VAR);
442

A
Aurelien Jarno 已提交
443 444 445
    if ( STATUS(default_nan_mode) )
        return float32_default_nan;

446
    if ((uint32_t)(av<<1) < (uint32_t)(bv<<1)) {
447
        aIsLargerSignificand = 0;
448
    } else if ((uint32_t)(bv<<1) < (uint32_t)(av<<1)) {
449 450 451
        aIsLargerSignificand = 1;
    } else {
        aIsLargerSignificand = (av < bv) ? 1 : 0;
B
bellard 已提交
452
    }
453

454
    if (pickNaN(aIsQuietNaN, aIsSignalingNaN, bIsQuietNaN, bIsSignalingNaN,
455
                aIsLargerSignificand)) {
456
        return float32_maybe_silence_nan(b);
457
    } else {
458
        return float32_maybe_silence_nan(a);
B
bellard 已提交
459 460 461 462
    }
}

/*----------------------------------------------------------------------------
T
ths 已提交
463 464
| Returns 1 if the double-precision floating-point value `a' is a quiet
| NaN; otherwise returns 0.
B
bellard 已提交
465 466
*----------------------------------------------------------------------------*/

467
int float64_is_quiet_nan( float64 a_ )
B
bellard 已提交
468
{
469
    uint64_t a = float64_val(a_);
T
ths 已提交
470
#if SNAN_BIT_IS_ONE
T
ths 已提交
471 472 473 474
    return
           ( ( ( a>>51 ) & 0xFFF ) == 0xFFE )
        && ( a & LIT64( 0x0007FFFFFFFFFFFF ) );
#else
475
    return ( LIT64( 0xFFF0000000000000 ) <= (uint64_t) ( a<<1 ) );
T
ths 已提交
476
#endif
B
bellard 已提交
477 478 479 480 481 482 483
}

/*----------------------------------------------------------------------------
| Returns 1 if the double-precision floating-point value `a' is a signaling
| NaN; otherwise returns 0.
*----------------------------------------------------------------------------*/

P
pbrook 已提交
484
int float64_is_signaling_nan( float64 a_ )
B
bellard 已提交
485
{
486
    uint64_t a = float64_val(a_);
T
ths 已提交
487
#if SNAN_BIT_IS_ONE
488
    return ( LIT64( 0xFFF0000000000000 ) <= (uint64_t) ( a<<1 ) );
T
ths 已提交
489
#else
B
bellard 已提交
490 491 492
    return
           ( ( ( a>>51 ) & 0xFFF ) == 0xFFE )
        && ( a & LIT64( 0x0007FFFFFFFFFFFF ) );
T
ths 已提交
493
#endif
B
bellard 已提交
494 495
}

496 497 498 499 500 501 502 503 504
/*----------------------------------------------------------------------------
| Returns a quiet NaN if the double-precision floating point value `a' is a
| signaling NaN; otherwise returns `a'.
*----------------------------------------------------------------------------*/

float64 float64_maybe_silence_nan( float64 a_ )
{
    if (float64_is_signaling_nan(a_)) {
#if SNAN_BIT_IS_ONE
505
#  if defined(TARGET_MIPS) || defined(TARGET_SH4) || defined(TARGET_UNICORE32)
506 507 508 509
        return float64_default_nan;
#  else
#    error Rules for silencing a signaling NaN are target-specific
#  endif
510
#else
511
        uint64_t a = float64_val(a_);
512 513
        a |= LIT64( 0x0008000000000000 );
        return make_float64(a);
514
#endif
515 516 517 518
    }
    return a_;
}

B
bellard 已提交
519 520 521 522 523 524 525 526 527 528 529
/*----------------------------------------------------------------------------
| Returns the result of converting the double-precision floating-point NaN
| `a' to the canonical NaN format.  If `a' is a signaling NaN, the invalid
| exception is raised.
*----------------------------------------------------------------------------*/

static commonNaNT float64ToCommonNaN( float64 a STATUS_PARAM)
{
    commonNaNT z;

    if ( float64_is_signaling_nan( a ) ) float_raise( float_flag_invalid STATUS_VAR);
P
pbrook 已提交
530
    z.sign = float64_val(a)>>63;
B
bellard 已提交
531
    z.low = 0;
P
pbrook 已提交
532
    z.high = float64_val(a)<<12;
B
bellard 已提交
533 534 535 536 537 538 539 540
    return z;
}

/*----------------------------------------------------------------------------
| Returns the result of converting the canonical NaN `a' to the double-
| precision floating-point format.
*----------------------------------------------------------------------------*/

541
static float64 commonNaNToFloat64( commonNaNT a STATUS_PARAM)
B
bellard 已提交
542
{
543
    uint64_t mantissa = a.high>>12;
544

545 546 547 548
    if ( STATUS(default_nan_mode) ) {
        return float64_default_nan;
    }

549 550
    if ( mantissa )
        return make_float64(
551
              ( ( (uint64_t) a.sign )<<63 )
552 553 554 555
            | LIT64( 0x7FF0000000000000 )
            | ( a.high>>12 ));
    else
        return float64_default_nan;
B
bellard 已提交
556 557 558 559 560 561 562 563 564 565
}

/*----------------------------------------------------------------------------
| Takes two double-precision floating-point values `a' and `b', one of which
| is a NaN, and returns the appropriate NaN result.  If either `a' or `b' is a
| signaling NaN, the invalid exception is raised.
*----------------------------------------------------------------------------*/

static float64 propagateFloat64NaN( float64 a, float64 b STATUS_PARAM)
{
566 567
    flag aIsQuietNaN, aIsSignalingNaN, bIsQuietNaN, bIsSignalingNaN;
    flag aIsLargerSignificand;
568
    uint64_t av, bv;
B
bellard 已提交
569

570
    aIsQuietNaN = float64_is_quiet_nan( a );
B
bellard 已提交
571
    aIsSignalingNaN = float64_is_signaling_nan( a );
572
    bIsQuietNaN = float64_is_quiet_nan( b );
B
bellard 已提交
573
    bIsSignalingNaN = float64_is_signaling_nan( b );
P
pbrook 已提交
574 575
    av = float64_val(a);
    bv = float64_val(b);
576

B
bellard 已提交
577
    if ( aIsSignalingNaN | bIsSignalingNaN ) float_raise( float_flag_invalid STATUS_VAR);
578

A
Aurelien Jarno 已提交
579 580 581
    if ( STATUS(default_nan_mode) )
        return float64_default_nan;

582
    if ((uint64_t)(av<<1) < (uint64_t)(bv<<1)) {
583
        aIsLargerSignificand = 0;
584
    } else if ((uint64_t)(bv<<1) < (uint64_t)(av<<1)) {
585 586 587
        aIsLargerSignificand = 1;
    } else {
        aIsLargerSignificand = (av < bv) ? 1 : 0;
B
bellard 已提交
588
    }
589

590
    if (pickNaN(aIsQuietNaN, aIsSignalingNaN, bIsQuietNaN, bIsSignalingNaN,
591
                aIsLargerSignificand)) {
592
        return float64_maybe_silence_nan(b);
593
    } else {
594
        return float64_maybe_silence_nan(a);
B
bellard 已提交
595 596 597 598 599
    }
}

/*----------------------------------------------------------------------------
| Returns 1 if the extended double-precision floating-point value `a' is a
600 601
| quiet NaN; otherwise returns 0. This slightly differs from the same
| function for other types as floatx80 has an explicit bit.
B
bellard 已提交
602 603
*----------------------------------------------------------------------------*/

604
int floatx80_is_quiet_nan( floatx80 a )
B
bellard 已提交
605
{
T
ths 已提交
606
#if SNAN_BIT_IS_ONE
607
    uint64_t aLow;
B
bellard 已提交
608

T
ths 已提交
609 610 611
    aLow = a.low & ~ LIT64( 0x4000000000000000 );
    return
           ( ( a.high & 0x7FFF ) == 0x7FFF )
612
        && (uint64_t) ( aLow<<1 )
T
ths 已提交
613 614
        && ( a.low == aLow );
#else
615
    return ( ( a.high & 0x7FFF ) == 0x7FFF )
616
        && (LIT64( 0x8000000000000000 ) <= ((uint64_t) ( a.low<<1 )));
T
ths 已提交
617
#endif
B
bellard 已提交
618 619 620 621
}

/*----------------------------------------------------------------------------
| Returns 1 if the extended double-precision floating-point value `a' is a
622 623
| signaling NaN; otherwise returns 0. This slightly differs from the same
| function for other types as floatx80 has an explicit bit.
B
bellard 已提交
624 625
*----------------------------------------------------------------------------*/

626
int floatx80_is_signaling_nan( floatx80 a )
B
bellard 已提交
627
{
T
ths 已提交
628
#if SNAN_BIT_IS_ONE
629
    return ( ( a.high & 0x7FFF ) == 0x7FFF )
630
        && (LIT64( 0x8000000000000000 ) <= ((uint64_t) ( a.low<<1 )));
T
ths 已提交
631
#else
632
    uint64_t aLow;
B
bellard 已提交
633 634 635 636

    aLow = a.low & ~ LIT64( 0x4000000000000000 );
    return
           ( ( a.high & 0x7FFF ) == 0x7FFF )
637
        && (uint64_t) ( aLow<<1 )
B
bellard 已提交
638
        && ( a.low == aLow );
T
ths 已提交
639
#endif
B
bellard 已提交
640 641
}

642 643 644 645 646 647 648 649 650
/*----------------------------------------------------------------------------
| Returns a quiet NaN if the extended double-precision floating point value
| `a' is a signaling NaN; otherwise returns `a'.
*----------------------------------------------------------------------------*/

floatx80 floatx80_maybe_silence_nan( floatx80 a )
{
    if (floatx80_is_signaling_nan(a)) {
#if SNAN_BIT_IS_ONE
651
#  if defined(TARGET_MIPS) || defined(TARGET_SH4) || defined(TARGET_UNICORE32)
652 653 654 655 656 657 658 659 660 661 662 663 664
        a.low = floatx80_default_nan_low;
        a.high = floatx80_default_nan_high;
#  else
#    error Rules for silencing a signaling NaN are target-specific
#  endif
#else
        a.low |= LIT64( 0xC000000000000000 );
        return a;
#endif
    }
    return a;
}

B
bellard 已提交
665 666 667 668 669 670 671 672 673 674 675
/*----------------------------------------------------------------------------
| Returns the result of converting the extended double-precision floating-
| point NaN `a' to the canonical NaN format.  If `a' is a signaling NaN, the
| invalid exception is raised.
*----------------------------------------------------------------------------*/

static commonNaNT floatx80ToCommonNaN( floatx80 a STATUS_PARAM)
{
    commonNaNT z;

    if ( floatx80_is_signaling_nan( a ) ) float_raise( float_flag_invalid STATUS_VAR);
676 677 678 679 680 681 682 683 684
    if ( a.low >> 63 ) {
        z.sign = a.high >> 15;
        z.low = 0;
        z.high = a.low << 1;
    } else {
        z.sign = floatx80_default_nan_high >> 15;
        z.low = 0;
        z.high = floatx80_default_nan_low << 1;
    }
B
bellard 已提交
685 686 687 688 689 690 691 692
    return z;
}

/*----------------------------------------------------------------------------
| Returns the result of converting the canonical NaN `a' to the extended
| double-precision floating-point format.
*----------------------------------------------------------------------------*/

693
static floatx80 commonNaNToFloatx80( commonNaNT a STATUS_PARAM)
B
bellard 已提交
694 695 696
{
    floatx80 z;

697 698 699 700 701 702
    if ( STATUS(default_nan_mode) ) {
        z.low = floatx80_default_nan_low;
        z.high = floatx80_default_nan_high;
        return z;
    }

703 704 705 706
    if (a.high >> 1) {
        z.low = LIT64( 0x8000000000000000 ) | a.high >> 1;
        z.high = ( ( (uint16_t) a.sign )<<15 ) | 0x7FFF;
    } else {
707
        z.low = floatx80_default_nan_low;
708 709 710
        z.high = floatx80_default_nan_high;
    }

B
bellard 已提交
711 712 713 714 715 716 717 718 719 720 721
    return z;
}

/*----------------------------------------------------------------------------
| Takes two extended double-precision floating-point values `a' and `b', one
| of which is a NaN, and returns the appropriate NaN result.  If either `a' or
| `b' is a signaling NaN, the invalid exception is raised.
*----------------------------------------------------------------------------*/

static floatx80 propagateFloatx80NaN( floatx80 a, floatx80 b STATUS_PARAM)
{
722 723
    flag aIsQuietNaN, aIsSignalingNaN, bIsQuietNaN, bIsSignalingNaN;
    flag aIsLargerSignificand;
B
bellard 已提交
724

725
    aIsQuietNaN = floatx80_is_quiet_nan( a );
B
bellard 已提交
726
    aIsSignalingNaN = floatx80_is_signaling_nan( a );
727
    bIsQuietNaN = floatx80_is_quiet_nan( b );
B
bellard 已提交
728
    bIsSignalingNaN = floatx80_is_signaling_nan( b );
729

B
bellard 已提交
730
    if ( aIsSignalingNaN | bIsSignalingNaN ) float_raise( float_flag_invalid STATUS_VAR);
731

A
Aurelien Jarno 已提交
732 733 734 735 736 737
    if ( STATUS(default_nan_mode) ) {
        a.low = floatx80_default_nan_low;
        a.high = floatx80_default_nan_high;
        return a;
    }

738 739 740 741 742 743
    if (a.low < b.low) {
        aIsLargerSignificand = 0;
    } else if (b.low < a.low) {
        aIsLargerSignificand = 1;
    } else {
        aIsLargerSignificand = (a.high < b.high) ? 1 : 0;
B
bellard 已提交
744
    }
745

746
    if (pickNaN(aIsQuietNaN, aIsSignalingNaN, bIsQuietNaN, bIsSignalingNaN,
747
                aIsLargerSignificand)) {
748
        return floatx80_maybe_silence_nan(b);
749
    } else {
750
        return floatx80_maybe_silence_nan(a);
B
bellard 已提交
751 752 753 754
    }
}

/*----------------------------------------------------------------------------
T
ths 已提交
755 756
| Returns 1 if the quadruple-precision floating-point value `a' is a quiet
| NaN; otherwise returns 0.
B
bellard 已提交
757 758
*----------------------------------------------------------------------------*/

759
int float128_is_quiet_nan( float128 a )
B
bellard 已提交
760
{
T
ths 已提交
761 762 763 764 765
#if SNAN_BIT_IS_ONE
    return
           ( ( ( a.high>>47 ) & 0xFFFF ) == 0xFFFE )
        && ( a.low || ( a.high & LIT64( 0x00007FFFFFFFFFFF ) ) );
#else
B
bellard 已提交
766
    return
767
           ( LIT64( 0xFFFE000000000000 ) <= (uint64_t) ( a.high<<1 ) )
B
bellard 已提交
768
        && ( a.low || ( a.high & LIT64( 0x0000FFFFFFFFFFFF ) ) );
T
ths 已提交
769
#endif
B
bellard 已提交
770 771 772 773 774 775 776
}

/*----------------------------------------------------------------------------
| Returns 1 if the quadruple-precision floating-point value `a' is a
| signaling NaN; otherwise returns 0.
*----------------------------------------------------------------------------*/

777
int float128_is_signaling_nan( float128 a )
B
bellard 已提交
778
{
T
ths 已提交
779 780
#if SNAN_BIT_IS_ONE
    return
781
           ( LIT64( 0xFFFE000000000000 ) <= (uint64_t) ( a.high<<1 ) )
T
ths 已提交
782 783
        && ( a.low || ( a.high & LIT64( 0x0000FFFFFFFFFFFF ) ) );
#else
B
bellard 已提交
784 785 786
    return
           ( ( ( a.high>>47 ) & 0xFFFF ) == 0xFFFE )
        && ( a.low || ( a.high & LIT64( 0x00007FFFFFFFFFFF ) ) );
T
ths 已提交
787
#endif
B
bellard 已提交
788 789
}

790 791 792 793 794 795 796 797 798
/*----------------------------------------------------------------------------
| Returns a quiet NaN if the quadruple-precision floating point value `a' is
| a signaling NaN; otherwise returns `a'.
*----------------------------------------------------------------------------*/

float128 float128_maybe_silence_nan( float128 a )
{
    if (float128_is_signaling_nan(a)) {
#if SNAN_BIT_IS_ONE
799
#  if defined(TARGET_MIPS) || defined(TARGET_SH4) || defined(TARGET_UNICORE32)
800 801 802 803 804 805 806 807 808 809 810 811 812
        a.low = float128_default_nan_low;
        a.high = float128_default_nan_high;
#  else
#    error Rules for silencing a signaling NaN are target-specific
#  endif
#else
        a.high |= LIT64( 0x0000800000000000 );
        return a;
#endif
    }
    return a;
}

B
bellard 已提交
813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833
/*----------------------------------------------------------------------------
| Returns the result of converting the quadruple-precision floating-point NaN
| `a' to the canonical NaN format.  If `a' is a signaling NaN, the invalid
| exception is raised.
*----------------------------------------------------------------------------*/

static commonNaNT float128ToCommonNaN( float128 a STATUS_PARAM)
{
    commonNaNT z;

    if ( float128_is_signaling_nan( a ) ) float_raise( float_flag_invalid STATUS_VAR);
    z.sign = a.high>>63;
    shortShift128Left( a.high, a.low, 16, &z.high, &z.low );
    return z;
}

/*----------------------------------------------------------------------------
| Returns the result of converting the canonical NaN `a' to the quadruple-
| precision floating-point format.
*----------------------------------------------------------------------------*/

834
static float128 commonNaNToFloat128( commonNaNT a STATUS_PARAM)
B
bellard 已提交
835 836 837
{
    float128 z;

838 839 840 841 842 843
    if ( STATUS(default_nan_mode) ) {
        z.low = float128_default_nan_low;
        z.high = float128_default_nan_high;
        return z;
    }

B
bellard 已提交
844
    shift128Right( a.high, a.low, 16, &z.high, &z.low );
845
    z.high |= ( ( (uint64_t) a.sign )<<63 ) | LIT64( 0x7FFF000000000000 );
B
bellard 已提交
846 847 848 849 850 851 852 853 854 855 856
    return z;
}

/*----------------------------------------------------------------------------
| Takes two quadruple-precision floating-point values `a' and `b', one of
| which is a NaN, and returns the appropriate NaN result.  If either `a' or
| `b' is a signaling NaN, the invalid exception is raised.
*----------------------------------------------------------------------------*/

static float128 propagateFloat128NaN( float128 a, float128 b STATUS_PARAM)
{
857 858
    flag aIsQuietNaN, aIsSignalingNaN, bIsQuietNaN, bIsSignalingNaN;
    flag aIsLargerSignificand;
B
bellard 已提交
859

860
    aIsQuietNaN = float128_is_quiet_nan( a );
B
bellard 已提交
861
    aIsSignalingNaN = float128_is_signaling_nan( a );
862
    bIsQuietNaN = float128_is_quiet_nan( b );
B
bellard 已提交
863
    bIsSignalingNaN = float128_is_signaling_nan( b );
864

B
bellard 已提交
865
    if ( aIsSignalingNaN | bIsSignalingNaN ) float_raise( float_flag_invalid STATUS_VAR);
866

A
Aurelien Jarno 已提交
867 868 869 870 871 872
    if ( STATUS(default_nan_mode) ) {
        a.low = float128_default_nan_low;
        a.high = float128_default_nan_high;
        return a;
    }

873 874 875 876 877 878
    if (lt128(a.high<<1, a.low, b.high<<1, b.low)) {
        aIsLargerSignificand = 0;
    } else if (lt128(b.high<<1, b.low, a.high<<1, a.low)) {
        aIsLargerSignificand = 1;
    } else {
        aIsLargerSignificand = (a.high < b.high) ? 1 : 0;
B
bellard 已提交
879
    }
880

881
    if (pickNaN(aIsQuietNaN, aIsSignalingNaN, bIsQuietNaN, bIsSignalingNaN,
882
                aIsLargerSignificand)) {
883
        return float128_maybe_silence_nan(b);
884
    } else {
885
        return float128_maybe_silence_nan(a);
B
bellard 已提交
886 887 888
    }
}