diff --git a/src/libcore/nonzero.rs b/src/libcore/nonzero.rs index 1b5fa4e0e950b..9e9c371b573da 100644 --- a/src/libcore/nonzero.rs +++ b/src/libcore/nonzero.rs @@ -14,6 +14,9 @@ use marker::Sized; use ops::{CoerceUnsized, Deref}; +use mem; +use option::Option; +use ptr; /// Unsafe trait to indicate what types are usable with the NonZero struct pub unsafe trait Zeroable {} @@ -44,6 +47,20 @@ impl NonZero { pub unsafe fn new(inner: T) -> NonZero { NonZero(inner) } + /// Tries to create an instance of NonZero with the provided value. + /// If the value is actually "non-zero", the function will succeed, + /// returning `Some(NonZero(value))`. Otherwise, `None` will be + /// returned. + #[inline(always)] + pub fn new_option(inner: T) -> Option> { + unsafe { + // Work around the fact that `mem::transmute` will + // complain about generic types: + let result = ptr::read(&inner as *const T as *const Option>); + mem::forget(inner); + result + } + } } impl Deref for NonZero { diff --git a/src/libcoretest/nonzero.rs b/src/libcoretest/nonzero.rs index 7a367ddeec8d4..0be2045b630dd 100644 --- a/src/libcoretest/nonzero.rs +++ b/src/libcoretest/nonzero.rs @@ -41,6 +41,31 @@ fn test_match_on_nonzero_option() { } } +#[test] +fn test_nonzero_new_option() { + let a = NonZero::new_option(42); + match a { + Some(val) => assert_eq!(*val, 42), + None => panic!("unexpected None while matching on NonZero::new_option(42)") + } + + match NonZero::new_option(43) { + Some(val) => assert_eq!(*val, 43), + None => panic!("unexpected None while matching on NonZero::new_option(43)") + } + + let b = NonZero::new_option(0); + match b { + Some(_) => panic!("unexpected Some(_) while matching on NonZero::new_option(0)"), + None => () + } + + match NonZero::new_option(0) { + Some(_) => panic!("unexpected Some(_) while matching on NonZero::new_option(0)"), + None => () + } +} + #[test] fn test_match_option_empty_vec() { let a: Option> = Some(vec![]);