diff options
| author | bors <bors@rust-lang.org> | 2016-09-19 19:03:52 -0700 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2016-09-19 19:03:52 -0700 |
| commit | cbd84aeb73ba1db2ce533fc764f28d748b9a0bb5 (patch) | |
| tree | 0b6a38e12ee08929baec700f3f4d4ed2039efb48 /src/libcore | |
| parent | e0547019eb1e1f78f14f46d014f15ee42589008f (diff) | |
| parent | b8c4e9c235db5f67129abdae8f336e06bf2bc8ba (diff) | |
| download | rust-cbd84aeb73ba1db2ce533fc764f28d748b9a0bb5.tar.gz rust-cbd84aeb73ba1db2ce533fc764f28d748b9a0bb5.zip | |
Auto merge of #34942 - porglezomp:master, r=sfackler
Fix overflow checking in unsigned pow() The pow() method for unsigned integers produced 0 instead of trapping overflow for certain inputs. Calls such as 2u32.pow(1024) produced 0 when they should trap an overflow. This also adds tests for the correctly handling overflow in unsigned pow(). This was previously fixed for signed integers in #28248, but it seems unsigned integers got missed that time. For issue number #34913
Diffstat (limited to 'src/libcore')
| -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 29ee29eb3eb..5c2cc671969 100644 --- a/src/libcore/num/mod.rs +++ b/src/libcore/num/mod.rs @@ -2211,25 +2211,21 @@ macro_rules! uint_impl { let mut base = self; let mut acc = 1; - 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 } |
