bigdecimal/arithmetic/
sqrt.rs1use crate::*;
4
5
6pub(crate) fn impl_sqrt(n: &BigUint, scale: i64, ctx: &Context) -> BigDecimal {
7 let num_digits = count_decimal_digits_uint(n);
9 let scale_diff = BigInt::from(num_digits) - scale;
10
11 let prec = ctx.precision().get();
15 let extra_rounding_digit_count = 5;
16 let wanted_digits = 2 * (prec + extra_rounding_digit_count);
17 let exponent = wanted_digits.saturating_sub(num_digits) + u64::from(scale_diff.is_odd());
18 let sqrt_digits = (n * ten_to_the_uint(exponent)).sqrt();
19
20 let result_scale_digits = 2 * (2 * prec - scale_diff) - 1;
22 let result_scale_decimal: BigDecimal = BigDecimal::new(result_scale_digits, 0) / 4.0;
23 let mut result_scale = result_scale_decimal.with_scale_round(0, RoundingMode::HalfEven).int_val;
24
25 result_scale += count_decimal_digits_uint(&sqrt_digits).saturating_sub(prec);
27 let unrounded_result = BigDecimal::new(sqrt_digits.into(), result_scale.to_i64().unwrap());
28 unrounded_result.with_precision_round(ctx.precision(), ctx.rounding_mode())
29}
30
31#[cfg(test)]
32mod test {
33 use super::*;
34
35 include!("sqrt.tests.rs");
36}