diff options
| author | Peter Reid <peter.d.reid@gmail.com> | 2015-09-04 22:10:39 -0400 |
|---|---|---|
| committer | Peter Reid <peter.d.reid@gmail.com> | 2015-09-04 22:15:57 -0400 |
| commit | 1927a869d4917cb7bf47cc069dde574e01c26de3 (patch) | |
| tree | c81a8229f8f57e052f343b0d05c9b7a6e8ae0f7b /src | |
| parent | 1110f1e873e9c082333b8239422e482b66d6645f (diff) | |
| download | rust-1927a869d4917cb7bf47cc069dde574e01c26de3.tar.gz rust-1927a869d4917cb7bf47cc069dde574e01c26de3.zip | |
Fix for issue #28012: pow overflow inconsistency
Overflows in integer pow() computations would be missed if they preceded a 0 bit of the exponent being processed. This made calls such as 2i32.pow(1024) not trigger an overflow.
Diffstat (limited to 'src')
| -rw-r--r-- | src/libcore/num/mod.rs | 26 |
1 files changed, 11 insertions, 15 deletions
diff --git a/src/libcore/num/mod.rs b/src/libcore/num/mod.rs index 05c7e8b8de4..127f8d3b5a2 100644 --- a/src/libcore/num/mod.rs +++ b/src/libcore/num/mod.rs @@ -537,25 +537,21 @@ macro_rules! int_impl { let mut base = self; let mut acc = Self::one(); - let mut prev_base = self; - let mut base_oflo = false; - while exp > 0 { + while exp > 1 { if (exp & 1) == 1 { - if base_oflo { - // ensure overflow occurs in the same manner it - // would have otherwise (i.e. signal any exception - // it would have otherwise). - acc = acc * (prev_base * prev_base); - } else { - acc = acc * base; - } + acc = acc * base; } - prev_base = base; - let (new_base, new_base_oflo) = base.overflowing_mul(base); - base = new_base; - base_oflo = new_base_oflo; exp /= 2; + base = base * base; + } + + // Deal with the final bit of the exponent separately, since + // squaring the base afterwards is not necessary and may cause a + // needless overflow. + if exp == 1 { + acc = acc * base; } + acc } |
