1
2
3
4
5
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
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
use crate::{
num::BigFloatNumber,
RoundingMode,
defs::{Error, WORD_SIGNIFICANT_BIT, WORD_BIT_SIZE},
common::consts::ONE, Sign,
};
use crate::ops::consts::Consts;
impl BigFloatNumber {
pub fn exp(&self, rm: RoundingMode, cc: &mut Consts) -> Result<Self, Error> {
if self.is_zero() {
return Self::from_word(1, self.get_mantissa_max_bit_len());
}
let int = self.get_int_as_usize()?;
let e_int = if int > 0 {
let e_const = cc.e(self.get_mantissa_max_bit_len() + 2 + 2*core::mem::size_of::<usize>(), RoundingMode::None)?;
e_const.powi(int, RoundingMode::None)
} else {
ONE.clone()
}?;
let mut fract = self.fract()?;
fract.set_precision(fract.get_mantissa_max_bit_len() + 4, RoundingMode::None)?;
fract.set_sign(Sign::Pos);
let e_fract = fract.expf(RoundingMode::None)?;
let mut ret = e_int.mul(&e_fract, RoundingMode::None)?;
if self.is_negative() {
ret = ret.reciprocal(RoundingMode::None)?;
};
ret.set_precision(self.get_mantissa_max_bit_len(), rm)?;
Ok(ret)
}
pub fn powi(&self, mut i: usize, rm: RoundingMode) -> Result<Self, Error> {
if self.is_zero() || i == 1 {
return self.clone();
}
if i == 0 {
return Self::from_word(1, self.get_mantissa_max_bit_len());
}
let mut bit_pos = WORD_BIT_SIZE;
while bit_pos > 0 {
bit_pos -= 1;
i <<= 1;
if i & WORD_SIGNIFICANT_BIT as usize != 0 {
bit_pos -= 1;
i <<= 1;
break;
}
}
let mut ret = self.clone()?;
while bit_pos > 0 {
bit_pos -= 1;
ret = ret.mul(&ret, rm)?;
if i & WORD_SIGNIFICANT_BIT as usize != 0 {
ret = ret.mul(self, rm)?;
}
i <<= 1;
}
Ok(ret)
}
fn expf(self, rm: RoundingMode) -> Result<Self, Error> {
let sh = self.sinh_series(rm)?; let sq = sh.mul(&sh, rm)?;
let sq2 = sq.add(&ONE, rm)?;
let sq3 = sq2.sqrt(rm)?;
sq3.add(&sh, rm)
}
}