From 15ea3d80dacc4bb0919655e0f16e808ce83b5cf9 Mon Sep 17 00:00:00 2001 From: Scott McMurray Date: Fri, 3 Nov 2017 21:48:33 -0700 Subject: [PATCH 1/3] Fix #18604: next_power_of_two should panic on overflow --- src/libcore/num/mod.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/src/libcore/num/mod.rs b/src/libcore/num/mod.rs index 5799d37c19cc6..914b3762d808e 100644 --- a/src/libcore/num/mod.rs +++ b/src/libcore/num/mod.rs @@ -2222,6 +2222,7 @@ macro_rules! uint_impl { /// ``` #[stable(feature = "rust1", since = "1.0.0")] #[inline] + #[rustc_inherit_overflow_checks] pub fn next_power_of_two(self) -> Self { self.one_less_than_next_power_of_two() + 1 } From 1b19e643244ab71a0842ee502d7ca7d8aa705209 Mon Sep 17 00:00:00 2001 From: Scott McMurray Date: Fri, 3 Nov 2017 22:33:34 -0700 Subject: [PATCH 2/3] Add overflow tests for next_power_of_two --- .../next-power-of-two-overflow-debug.rs | 36 +++++++++++++++++++ .../next-power-of-two-overflow-ndebug.rs | 24 +++++++++++++ 2 files changed, 60 insertions(+) create mode 100644 src/test/run-pass/next-power-of-two-overflow-debug.rs create mode 100644 src/test/run-pass/next-power-of-two-overflow-ndebug.rs diff --git a/src/test/run-pass/next-power-of-two-overflow-debug.rs b/src/test/run-pass/next-power-of-two-overflow-debug.rs new file mode 100644 index 0000000000000..a3e7ffd4e49d7 --- /dev/null +++ b/src/test/run-pass/next-power-of-two-overflow-debug.rs @@ -0,0 +1,36 @@ +// Copyright 2017 The Rust Project Developers. See the COPYRIGHT +// file at the top-level directory of this distribution and at +// http://rust-lang.org/COPYRIGHT. +// +// Licensed under the Apache License, Version 2.0 or the MIT license +// , at your +// option. This file may not be copied, modified, or distributed +// except according to those terms. + +// compile-flags: -C debug_assertions=yes + +#![feature(i128_type)] + +use std::panic; + +fn main() { + macro_rules! overflow_test { + ($t:ident) => ( + let r = panic::catch_unwind(|| { + ($t::max_value()).next_power_of_two() + }); + assert!(r.is_err()); + + let r = panic::catch_unwind(|| { + (($t::max_value() >> 1) + 2).next_power_of_two() + }); + assert!(r.is_err()); + ) + } + overflow_test!(u8); + overflow_test!(u16); + overflow_test!(u32); + overflow_test!(u64); + overflow_test!(u128); +} diff --git a/src/test/run-pass/next-power-of-two-overflow-ndebug.rs b/src/test/run-pass/next-power-of-two-overflow-ndebug.rs new file mode 100644 index 0000000000000..f8bcb961c6833 --- /dev/null +++ b/src/test/run-pass/next-power-of-two-overflow-ndebug.rs @@ -0,0 +1,24 @@ +// Copyright 2017 The Rust Project Developers. See the COPYRIGHT +// file at the top-level directory of this distribution and at +// http://rust-lang.org/COPYRIGHT. +// +// Licensed under the Apache License, Version 2.0 or the MIT license +// , at your +// option. This file may not be copied, modified, or distributed +// except according to those terms. + +// compile-flags: -C debug_assertions=no + +#![feature(i128_type)] + +fn main() { + for i in 129..256 { + assert_eq!((i as u8).next_power_of_two(), 0); + } + + assert_eq!(((1u16 << 15) + 1).next_power_of_two(), 0); + assert_eq!(((1u32 << 31) + 1).next_power_of_two(), 0); + assert_eq!(((1u64 << 63) + 1).next_power_of_two(), 0); + assert_eq!(((1u128 << 127) + 1).next_power_of_two(), 0); +} From 0d745af29a7566538277bd7a3b16faf09df7fe63 Mon Sep 17 00:00:00 2001 From: Scott McMurray Date: Sat, 4 Nov 2017 17:10:51 -0700 Subject: [PATCH 3/3] Use Add::add for overflow checks instead of [rustc_inherit_overflow_checks] --- src/libcore/num/mod.rs | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/libcore/num/mod.rs b/src/libcore/num/mod.rs index 914b3762d808e..a50779bedfdc1 100644 --- a/src/libcore/num/mod.rs +++ b/src/libcore/num/mod.rs @@ -15,6 +15,7 @@ use convert::{Infallible, TryFrom}; use fmt; use intrinsics; +use ops; use str::FromStr; /// Provides intentionally-wrapped arithmetic on `T`. @@ -2222,9 +2223,9 @@ macro_rules! uint_impl { /// ``` #[stable(feature = "rust1", since = "1.0.0")] #[inline] - #[rustc_inherit_overflow_checks] pub fn next_power_of_two(self) -> Self { - self.one_less_than_next_power_of_two() + 1 + // Call the trait to get overflow checks + ops::Add::add(self.one_less_than_next_power_of_two(), 1) } /// Returns the smallest power of two greater than or equal to `n`. If