@@ -280,15 +280,34 @@ ValueWithRealFlags<Real<W, P>> Real<W, P>::SQRT(Rounding rounding) const {
280
280
// SQRT(+Inf) == +Inf
281
281
result.value = Infinity (false );
282
282
} else {
283
- // Slow but reliable bit-at-a-time method. Start with a clear significand
284
- // and half the unbiased exponent, and then try to set significand bits
285
- // in descending order of magnitude without exceeding the exact result.
286
283
int expo{UnbiasedExponent ()};
287
- if (IsSubnormal ()) {
288
- expo -= GetFraction ().LEADZ ();
284
+ if (expo < -1 || expo > 1 ) {
285
+ // Reduce the range to [0.5 .. 4.0) by dividing by an integral power
286
+ // of four to avoid trouble with very large and very small values
287
+ // (esp. truncation of subnormals).
288
+ // SQRT(2**(2a) * x) = SQRT(2**(2a)) * SQRT(x) = 2**a * SQRT(x)
289
+ Real scaled;
290
+ int adjust{expo / 2 };
291
+ scaled.Normalize (false , expo - 2 * adjust + exponentBias, GetFraction ());
292
+ result = scaled.SQRT (rounding);
293
+ result.value .Normalize (false ,
294
+ result.value .UnbiasedExponent () + adjust + exponentBias,
295
+ result.value .GetFraction ());
296
+ return result;
289
297
}
298
+ // Compute the square root of the reduced value with the slow but
299
+ // reliable bit-at-a-time method. Start with a clear significand and
300
+ // half of the unbiased exponent, and then try to set significand bits
301
+ // in descending order of magnitude without exceeding the exact result.
290
302
expo = expo / 2 + exponentBias;
291
303
result.value .Normalize (false , expo, Fraction::MASKL (1 ));
304
+ Real initialSq{result.value .Multiply (result.value ).value };
305
+ if (Compare (initialSq) == Relation::Less) {
306
+ // Initial estimate is too large; this can happen for values just
307
+ // under 1.0.
308
+ --expo;
309
+ result.value .Normalize (false , expo, Fraction::MASKL (1 ));
310
+ }
292
311
for (int bit{significandBits - 1 }; bit >= 0 ; --bit) {
293
312
Word word{result.value .word_ };
294
313
result.value .word_ = word.IBSET (bit);
@@ -299,10 +318,10 @@ ValueWithRealFlags<Real<W, P>> Real<W, P>::SQRT(Rounding rounding) const {
299
318
result.value .word_ = word;
300
319
}
301
320
}
302
- // The computed square root, when squared, has a square that's not greater
303
- // than the original argument. Check this square against the square of the
304
- // next Real value, and return that one if its square is closer in magnitude
305
- // to the original argument.
321
+ // The computed square root has a square that's not greater than the
322
+ // original argument. Check this square against the square of the next
323
+ // larger Real and return that one if its square is closer in magnitude to
324
+ // the original argument.
306
325
Real resultSq{result.value .Multiply (result.value ).value };
307
326
Real diff{Subtract (resultSq).value .ABS ()};
308
327
if (diff.IsZero ()) {
0 commit comments