fpu/softfloat: Canonicalize NaN fraction

Shift the NaN fraction to a canonical position, much like we
do for the fraction of normal numbers. This will facilitate
manipulation of NaNs within the shared code paths.

Backports commit 94933df0e5c34d1a50fc950553f9c9649cae5320 from qemu
This commit is contained in:
Richard Henderson 2018-05-19 22:17:38 -04:00 committed by Lioncash
parent 3873bfe133
commit f89c1fc034
No known key found for this signature in database
GPG key ID: 4E3C3CC1031BA9C7

View file

@ -330,10 +330,11 @@ static FloatParts canonicalize(FloatParts part, const FloatFmt *parm,
if (part.frac == 0) { if (part.frac == 0) {
part.cls = float_class_inf; part.cls = float_class_inf;
} else { } else {
part.frac <<= parm->frac_shift;
#ifdef NO_SIGNALING_NANS #ifdef NO_SIGNALING_NANS
part.cls = float_class_qnan; part.cls = float_class_qnan;
#else #else
int64_t msb = part.frac << (parm->frac_shift + 2); int64_t msb = part.frac << 2;
if ((msb < 0) == status->snan_bit_is_one) { if ((msb < 0) == status->snan_bit_is_one) {
part.cls = float_class_snan; part.cls = float_class_snan;
} else { } else {
@ -480,6 +481,7 @@ static FloatParts round_canonical(FloatParts p, float_status *s,
case float_class_qnan: case float_class_qnan:
case float_class_snan: case float_class_snan:
exp = exp_max; exp = exp_max;
frac >>= parm->frac_shift;
break; break;
default: default:
@ -503,6 +505,7 @@ static float16 float16_round_pack_canonical(FloatParts p, float_status *s)
case float_class_dnan: case float_class_dnan:
return float16_default_nan(s); return float16_default_nan(s);
case float_class_msnan: case float_class_msnan:
p.frac >>= float16_params.frac_shift;
return float16_maybe_silence_nan(float16_pack_raw(p), s); return float16_maybe_silence_nan(float16_pack_raw(p), s);
default: default:
p = round_canonical(p, s, &float16_params); p = round_canonical(p, s, &float16_params);
@ -521,6 +524,7 @@ static float32 float32_round_pack_canonical(FloatParts p, float_status *s)
case float_class_dnan: case float_class_dnan:
return float32_default_nan(s); return float32_default_nan(s);
case float_class_msnan: case float_class_msnan:
p.frac >>= float32_params.frac_shift;
return float32_maybe_silence_nan(float32_pack_raw(p), s); return float32_maybe_silence_nan(float32_pack_raw(p), s);
default: default:
p = round_canonical(p, s, &float32_params); p = round_canonical(p, s, &float32_params);
@ -539,6 +543,7 @@ static float64 float64_round_pack_canonical(FloatParts p, float_status *s)
case float_class_dnan: case float_class_dnan:
return float64_default_nan(s); return float64_default_nan(s);
case float_class_msnan: case float_class_msnan:
p.frac >>= float64_params.frac_shift;
return float64_maybe_silence_nan(float64_pack_raw(p), s); return float64_maybe_silence_nan(float64_pack_raw(p), s);
default: default:
p = round_canonical(p, s, &float64_params); p = round_canonical(p, s, &float64_params);