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    macro_rules! impl_case {
36        ($name:ident; $input:literal => $expected:literal) => {
37            #[test]
38            fn $name() {
39                let n: BigDecimal = $input.parse().unwrap();
40                let value = n.sqrt().unwrap();
41
42                let expected = $expected.parse().unwrap();
43                assert_eq!(value, expected);
44                }
46        };
47        ($name:ident; prec=$prec:literal; round=$round:ident; $input:literal => $expected:literal) => {
48            #[test]
49            fn $name() {
50                let ctx = Context::default()
51                                .with_prec($prec).unwrap()
52                                .with_rounding_mode(RoundingMode::$round);
53                let n: BigDecimal = $input.parse().unwrap();
54                let value = n.sqrt_with_context(&ctx).unwrap();
55
56                let expected = $expected.parse().unwrap();
57                assert_eq!(value, expected);
58                }
60        };
61    }
62
63    impl_case!(case_0d000; "0.000" => "0");
64    impl_case!(case_1en232; "1e-232" => "1e-116");
65    impl_case!(case_1d00; "1.00" => "1.00");
66    impl_case!(case_1d001; "1.001" => "1.000499875062460964823258287700109753027590031219780479551442971840836093890879944856933288426795152");
67    impl_case!(case_100d0; "100" => "10.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000");
68    impl_case!(case_49; "49" => "7.0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000");
69    impl_case!(case_d25; ".25" => ".5");
70    impl_case!(case_0d0152399025; "0.0152399025" => ".12345");
71    impl_case!(case_0d00400; "0.00400" => "0.06324555320336758663997787088865437067439110278650433653715009705585188877278476442688496216758600590");
72    impl_case!(case_0d1; "0.1" => "0.3162277660168379331998893544432718533719555139325216826857504852792594438639238221344248108379300295");
73    impl_case!(case_152399025; "152399025" => "12345");
74    impl_case!(case_2; "2" => "1.414213562373095048801688724209698078569671875376948073176679737990732478462107038850387534327641573");
75    impl_case!(case_125348; "125348" => "354.0451948551201563108487193176101314241016013304294520812832530590100407318465590778759640828114535");
76    impl_case!(case_121d000242000121; "121.000242000121000000" => "11.000011000");
77    impl_case!(case_0d01234567901234567901234567901234567901234567901234567901234567901234567901234567901234567901234567901; "0.01234567901234567901234567901234567901234567901234567901234567901234567901234567901234567901234567901" => "0.1111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111");
78    impl_case!(case_2e70; "2e70" => "141421356237309504880168872420969807.8569671875376948073176679737990732478462107038850387534327641573");
79    impl_case!(case_8d9793115997963468544185161590576171875en11; "8.9793115997963468544185161590576171875e-11" => "0.000009475922962855041517561783740144225422359796851494316346796373337470068631250135521161989831460407155");
80    impl_case!(case_18446744073709551616d1099511; "18446744073709551616.1099511" => "4294967296.000000000012799992691725492477397918722952224079252026972356303360555051219312462698703293");
81
82    impl_case!(case_3d1415926; "3.141592653589793115997963468544185161590576171875" => "1.772453850905515992751519103139248439290428205003682302442979619028063165921408635567477284443197875");
83    impl_case!(case_0d71777001; "0.7177700109762963922745342343167413624881759290454997218753321040760896053150388903350654937434826216803814031987652326749140535150336357405672040727695124057298138872112244784753994931999476811850580200000000000000000000000000000000" => "0.8472130847527653667042980517799020703921106560594525833177762276594388966885185567535692987624493813");
84    impl_case!(case_0d110889ddd444; "0.1108890000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000444" => "0.3330000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000667");
85
86    impl_case!(case_3e170; "3e170" => "17320508075688772935274463415058723669428052538103806280558069794519330169088000370811.46186757248576");
87    impl_case!(case_9e199; "9e199" => "9486832980505137995996680633298155601158665417975650480572514558377783315917714664032744325137900886");
88    impl_case!(case_7e200; "7e200" => "2645751311064590590501615753639260425710259183082450180368334459201068823230283627760392886474543611e1");
89    impl_case!(case_777e204; "777e204" => "2.787471972953270789531596912111625325974789615194854615319795902911796043681078997362635440358922503E+103");
90    impl_case!(case_777e600; "7e600" => "2.645751311064590590501615753639260425710259183082450180368334459201068823230283627760392886474543611E+300");
91    impl_case!(case_2e900; "2e900" => "1.414213562373095048801688724209698078569671875376948073176679737990732478462107038850387534327641573E+450");
92    impl_case!(case_7e999; "7e999" => "8.366600265340755479781720257851874893928153692986721998111915430804187725943170098308147119649515362E+499");
93    impl_case!(case_74908163946345982392040522594123773796e999; "74908163946345982392040522594123773796e999" => "2.736935584670307552030924971360722787091742391079630976117950955395149091570790266754718322365663909E+518");
94    impl_case!(case_20e1024; "20e1024" => "4.472135954999579392818347337462552470881236719223051448541794490821041851275609798828828816757564550E512");
95    impl_case!(case_3en1025; "3e-1025" => "5.477225575051661134569697828008021339527446949979832542268944497324932771227227338008584361638706258E-513");
96
97    impl_case!(case_3242053850483855en13_prec11_round_down; prec=11; round=Down; "324.2053850483855" => "18.005704236");
98    impl_case!(case_3242053850483855en13_prec11_round_up; prec=11; round=Up; "324.2053850483855" => "18.005704237");
99    impl_case!(case_3242053850483855en13_prec31_round_up; prec=31; round=Up; "324.2053850483855" => "18.00570423639090823994825477228");
100
101    impl_case!(case_5d085019992340351en10_prec25_round_down; prec=25; round=Down; "5.085019992340351e-10" => "0.00002254998889653906459324292");
102
103    impl_case!(case_3025d13579652399025_prec3_round_up; prec=3; round=Up; "3025.13579652399025" => "55.1");
104
105    impl_case!(case_3025d13579652399025_prec9_round_down; prec=9; round=Down; "3025.13579652399025" => "55.0012345");
106    impl_case!(case_3025d13579652399025_prec9_round_up; prec=9; round=Up; "3025.13579652399025" => "55.0012345");
107
108    impl_case!(case_3025d13579652399025_prec8_round_halfdown; prec=8; round=HalfDown; "3025.13579652399025" => "55.001234");
109    impl_case!(case_3025d13579652399025_prec8_round_halfeven; prec=8; round=HalfEven; "3025.13579652399025" => "55.001234");
110    impl_case!(case_3025d13579652399025_prec8_round_halfup; prec=8; round=HalfUp; "3025.13579652399025" => "55.001235");
111
112    #[test]
113    fn test_sqrt_rounding() {
114        let vals = vec![
115            ("1.21", "2", "1", "1", "1", "1", "1", "2"),
117            ("2.25", "2", "1", "1", "1", "2", "2", "2"),
119            ("6.25", "3", "2", "2", "2", "2", "3", "3"),
121            ("8.41", "3", "2", "2", "3", "3", "3", "3"),
123        ];
124        for &(val, ceiling, down, floor, half_down, half_even, half_up, up) in vals.iter() {
125            let val = BigDecimal::from_str(val).unwrap();
126            let ceiling = BigDecimal::from_str(ceiling).unwrap();
127            let down = BigDecimal::from_str(down).unwrap();
128            let floor = BigDecimal::from_str(floor).unwrap();
129            let half_down = BigDecimal::from_str(half_down).unwrap();
130            let half_even = BigDecimal::from_str(half_even).unwrap();
131            let half_up = BigDecimal::from_str(half_up).unwrap();
132            let up = BigDecimal::from_str(up).unwrap();
133            let ctx = Context::default().with_prec(1).unwrap();
134            assert_eq!(val.sqrt_with_context(&ctx.with_rounding_mode(RoundingMode::Ceiling)).unwrap(), ceiling);
135            assert_eq!(val.sqrt_with_context(&ctx.with_rounding_mode(RoundingMode::Down)).unwrap(), down);
136            assert_eq!(val.sqrt_with_context(&ctx.with_rounding_mode(RoundingMode::Floor)).unwrap(), floor);
137            assert_eq!(val.sqrt_with_context(&ctx.with_rounding_mode(RoundingMode::HalfDown)).unwrap(), half_down);
138            assert_eq!(val.sqrt_with_context(&ctx.with_rounding_mode(RoundingMode::HalfEven)).unwrap(), half_even);
139            assert_eq!(val.sqrt_with_context(&ctx.with_rounding_mode(RoundingMode::HalfUp)).unwrap(), half_up);
140            assert_eq!(val.sqrt_with_context(&ctx.with_rounding_mode(RoundingMode::Up)).unwrap(), up);
141        }
142    }
143
144    #[cfg(property_tests)]
145    mod prop {
146        use super::*;
147        use proptest::*;
148        use num_traits::FromPrimitive;
149
150        proptest! {
151            #[test]
152            fn sqrt_of_square_is_self(f: f64, prec in 15..50u64) {
153                prop_assume!(f.is_normal());
155
156                let n = BigDecimal::from_f64(f.abs()).unwrap().with_prec(prec);
157                let n_squared = n.square();
158                let x = n_squared.sqrt().unwrap();
159                prop_assert_eq!(x, n);
160            }
161        }
162    }
163}