cast.rs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593
  1. use core::mem::size_of;
  2. use core::num::Wrapping;
  3. use identities::Zero;
  4. use bounds::Bounded;
  5. use float::FloatCore;
  6. /// A generic trait for converting a value to a number.
  7. pub trait ToPrimitive {
  8. /// Converts the value of `self` to an `isize`.
  9. #[inline]
  10. fn to_isize(&self) -> Option<isize> {
  11. self.to_i64().and_then(|x| x.to_isize())
  12. }
  13. /// Converts the value of `self` to an `i8`.
  14. #[inline]
  15. fn to_i8(&self) -> Option<i8> {
  16. self.to_i64().and_then(|x| x.to_i8())
  17. }
  18. /// Converts the value of `self` to an `i16`.
  19. #[inline]
  20. fn to_i16(&self) -> Option<i16> {
  21. self.to_i64().and_then(|x| x.to_i16())
  22. }
  23. /// Converts the value of `self` to an `i32`.
  24. #[inline]
  25. fn to_i32(&self) -> Option<i32> {
  26. self.to_i64().and_then(|x| x.to_i32())
  27. }
  28. /// Converts the value of `self` to an `i64`.
  29. fn to_i64(&self) -> Option<i64>;
  30. /// Converts the value of `self` to a `usize`.
  31. #[inline]
  32. fn to_usize(&self) -> Option<usize> {
  33. self.to_u64().and_then(|x| x.to_usize())
  34. }
  35. /// Converts the value of `self` to an `u8`.
  36. #[inline]
  37. fn to_u8(&self) -> Option<u8> {
  38. self.to_u64().and_then(|x| x.to_u8())
  39. }
  40. /// Converts the value of `self` to an `u16`.
  41. #[inline]
  42. fn to_u16(&self) -> Option<u16> {
  43. self.to_u64().and_then(|x| x.to_u16())
  44. }
  45. /// Converts the value of `self` to an `u32`.
  46. #[inline]
  47. fn to_u32(&self) -> Option<u32> {
  48. self.to_u64().and_then(|x| x.to_u32())
  49. }
  50. /// Converts the value of `self` to an `u64`.
  51. #[inline]
  52. fn to_u64(&self) -> Option<u64>;
  53. /// Converts the value of `self` to an `f32`.
  54. #[inline]
  55. fn to_f32(&self) -> Option<f32> {
  56. self.to_f64().and_then(|x| x.to_f32())
  57. }
  58. /// Converts the value of `self` to an `f64`.
  59. #[inline]
  60. fn to_f64(&self) -> Option<f64> {
  61. self.to_i64().and_then(|x| x.to_f64())
  62. }
  63. }
  64. macro_rules! impl_to_primitive_int_to_int {
  65. ($SrcT:ty, $DstT:ty, $slf:expr) => (
  66. {
  67. if size_of::<$SrcT>() <= size_of::<$DstT>() {
  68. Some($slf as $DstT)
  69. } else {
  70. let n = $slf as i64;
  71. let min_value: $DstT = Bounded::min_value();
  72. let max_value: $DstT = Bounded::max_value();
  73. if min_value as i64 <= n && n <= max_value as i64 {
  74. Some($slf as $DstT)
  75. } else {
  76. None
  77. }
  78. }
  79. }
  80. )
  81. }
  82. macro_rules! impl_to_primitive_int_to_uint {
  83. ($SrcT:ty, $DstT:ty, $slf:expr) => (
  84. {
  85. let zero: $SrcT = Zero::zero();
  86. let max_value: $DstT = Bounded::max_value();
  87. if zero <= $slf && $slf as u64 <= max_value as u64 {
  88. Some($slf as $DstT)
  89. } else {
  90. None
  91. }
  92. }
  93. )
  94. }
  95. macro_rules! impl_to_primitive_int {
  96. ($T:ty) => (
  97. impl ToPrimitive for $T {
  98. #[inline]
  99. fn to_isize(&self) -> Option<isize> { impl_to_primitive_int_to_int!($T, isize, *self) }
  100. #[inline]
  101. fn to_i8(&self) -> Option<i8> { impl_to_primitive_int_to_int!($T, i8, *self) }
  102. #[inline]
  103. fn to_i16(&self) -> Option<i16> { impl_to_primitive_int_to_int!($T, i16, *self) }
  104. #[inline]
  105. fn to_i32(&self) -> Option<i32> { impl_to_primitive_int_to_int!($T, i32, *self) }
  106. #[inline]
  107. fn to_i64(&self) -> Option<i64> { impl_to_primitive_int_to_int!($T, i64, *self) }
  108. #[inline]
  109. fn to_usize(&self) -> Option<usize> { impl_to_primitive_int_to_uint!($T, usize, *self) }
  110. #[inline]
  111. fn to_u8(&self) -> Option<u8> { impl_to_primitive_int_to_uint!($T, u8, *self) }
  112. #[inline]
  113. fn to_u16(&self) -> Option<u16> { impl_to_primitive_int_to_uint!($T, u16, *self) }
  114. #[inline]
  115. fn to_u32(&self) -> Option<u32> { impl_to_primitive_int_to_uint!($T, u32, *self) }
  116. #[inline]
  117. fn to_u64(&self) -> Option<u64> { impl_to_primitive_int_to_uint!($T, u64, *self) }
  118. #[inline]
  119. fn to_f32(&self) -> Option<f32> { Some(*self as f32) }
  120. #[inline]
  121. fn to_f64(&self) -> Option<f64> { Some(*self as f64) }
  122. }
  123. )
  124. }
  125. impl_to_primitive_int!(isize);
  126. impl_to_primitive_int!(i8);
  127. impl_to_primitive_int!(i16);
  128. impl_to_primitive_int!(i32);
  129. impl_to_primitive_int!(i64);
  130. macro_rules! impl_to_primitive_uint_to_int {
  131. ($DstT:ty, $slf:expr) => (
  132. {
  133. let max_value: $DstT = Bounded::max_value();
  134. if $slf as u64 <= max_value as u64 {
  135. Some($slf as $DstT)
  136. } else {
  137. None
  138. }
  139. }
  140. )
  141. }
  142. macro_rules! impl_to_primitive_uint_to_uint {
  143. ($SrcT:ty, $DstT:ty, $slf:expr) => (
  144. {
  145. if size_of::<$SrcT>() <= size_of::<$DstT>() {
  146. Some($slf as $DstT)
  147. } else {
  148. let zero: $SrcT = Zero::zero();
  149. let max_value: $DstT = Bounded::max_value();
  150. if zero <= $slf && $slf as u64 <= max_value as u64 {
  151. Some($slf as $DstT)
  152. } else {
  153. None
  154. }
  155. }
  156. }
  157. )
  158. }
  159. macro_rules! impl_to_primitive_uint {
  160. ($T:ty) => (
  161. impl ToPrimitive for $T {
  162. #[inline]
  163. fn to_isize(&self) -> Option<isize> { impl_to_primitive_uint_to_int!(isize, *self) }
  164. #[inline]
  165. fn to_i8(&self) -> Option<i8> { impl_to_primitive_uint_to_int!(i8, *self) }
  166. #[inline]
  167. fn to_i16(&self) -> Option<i16> { impl_to_primitive_uint_to_int!(i16, *self) }
  168. #[inline]
  169. fn to_i32(&self) -> Option<i32> { impl_to_primitive_uint_to_int!(i32, *self) }
  170. #[inline]
  171. fn to_i64(&self) -> Option<i64> { impl_to_primitive_uint_to_int!(i64, *self) }
  172. #[inline]
  173. fn to_usize(&self) -> Option<usize> {
  174. impl_to_primitive_uint_to_uint!($T, usize, *self)
  175. }
  176. #[inline]
  177. fn to_u8(&self) -> Option<u8> { impl_to_primitive_uint_to_uint!($T, u8, *self) }
  178. #[inline]
  179. fn to_u16(&self) -> Option<u16> { impl_to_primitive_uint_to_uint!($T, u16, *self) }
  180. #[inline]
  181. fn to_u32(&self) -> Option<u32> { impl_to_primitive_uint_to_uint!($T, u32, *self) }
  182. #[inline]
  183. fn to_u64(&self) -> Option<u64> { impl_to_primitive_uint_to_uint!($T, u64, *self) }
  184. #[inline]
  185. fn to_f32(&self) -> Option<f32> { Some(*self as f32) }
  186. #[inline]
  187. fn to_f64(&self) -> Option<f64> { Some(*self as f64) }
  188. }
  189. )
  190. }
  191. impl_to_primitive_uint!(usize);
  192. impl_to_primitive_uint!(u8);
  193. impl_to_primitive_uint!(u16);
  194. impl_to_primitive_uint!(u32);
  195. impl_to_primitive_uint!(u64);
  196. macro_rules! impl_to_primitive_float_to_float {
  197. ($SrcT:ident, $DstT:ident, $slf:expr) => (
  198. if size_of::<$SrcT>() <= size_of::<$DstT>() {
  199. Some($slf as $DstT)
  200. } else {
  201. // Make sure the value is in range for the cast.
  202. // NaN and +-inf are cast as they are.
  203. let n = $slf as f64;
  204. let max_value: $DstT = ::core::$DstT::MAX;
  205. if !FloatCore::is_finite(n) || (-max_value as f64 <= n && n <= max_value as f64)
  206. {
  207. Some($slf as $DstT)
  208. } else {
  209. None
  210. }
  211. }
  212. )
  213. }
  214. macro_rules! impl_to_primitive_float {
  215. ($T:ident) => (
  216. impl ToPrimitive for $T {
  217. #[inline]
  218. fn to_isize(&self) -> Option<isize> { Some(*self as isize) }
  219. #[inline]
  220. fn to_i8(&self) -> Option<i8> { Some(*self as i8) }
  221. #[inline]
  222. fn to_i16(&self) -> Option<i16> { Some(*self as i16) }
  223. #[inline]
  224. fn to_i32(&self) -> Option<i32> { Some(*self as i32) }
  225. #[inline]
  226. fn to_i64(&self) -> Option<i64> { Some(*self as i64) }
  227. #[inline]
  228. fn to_usize(&self) -> Option<usize> { Some(*self as usize) }
  229. #[inline]
  230. fn to_u8(&self) -> Option<u8> { Some(*self as u8) }
  231. #[inline]
  232. fn to_u16(&self) -> Option<u16> { Some(*self as u16) }
  233. #[inline]
  234. fn to_u32(&self) -> Option<u32> { Some(*self as u32) }
  235. #[inline]
  236. fn to_u64(&self) -> Option<u64> { Some(*self as u64) }
  237. #[inline]
  238. fn to_f32(&self) -> Option<f32> { impl_to_primitive_float_to_float!($T, f32, *self) }
  239. #[inline]
  240. fn to_f64(&self) -> Option<f64> { impl_to_primitive_float_to_float!($T, f64, *self) }
  241. }
  242. )
  243. }
  244. impl_to_primitive_float!(f32);
  245. impl_to_primitive_float!(f64);
  246. /// A generic trait for converting a number to a value.
  247. pub trait FromPrimitive: Sized {
  248. /// Convert an `isize` to return an optional value of this type. If the
  249. /// value cannot be represented by this value, the `None` is returned.
  250. #[inline]
  251. fn from_isize(n: isize) -> Option<Self> {
  252. FromPrimitive::from_i64(n as i64)
  253. }
  254. /// Convert an `i8` to return an optional value of this type. If the
  255. /// type cannot be represented by this value, the `None` is returned.
  256. #[inline]
  257. fn from_i8(n: i8) -> Option<Self> {
  258. FromPrimitive::from_i64(n as i64)
  259. }
  260. /// Convert an `i16` to return an optional value of this type. If the
  261. /// type cannot be represented by this value, the `None` is returned.
  262. #[inline]
  263. fn from_i16(n: i16) -> Option<Self> {
  264. FromPrimitive::from_i64(n as i64)
  265. }
  266. /// Convert an `i32` to return an optional value of this type. If the
  267. /// type cannot be represented by this value, the `None` is returned.
  268. #[inline]
  269. fn from_i32(n: i32) -> Option<Self> {
  270. FromPrimitive::from_i64(n as i64)
  271. }
  272. /// Convert an `i64` to return an optional value of this type. If the
  273. /// type cannot be represented by this value, the `None` is returned.
  274. fn from_i64(n: i64) -> Option<Self>;
  275. /// Convert a `usize` to return an optional value of this type. If the
  276. /// type cannot be represented by this value, the `None` is returned.
  277. #[inline]
  278. fn from_usize(n: usize) -> Option<Self> {
  279. FromPrimitive::from_u64(n as u64)
  280. }
  281. /// Convert an `u8` to return an optional value of this type. If the
  282. /// type cannot be represented by this value, the `None` is returned.
  283. #[inline]
  284. fn from_u8(n: u8) -> Option<Self> {
  285. FromPrimitive::from_u64(n as u64)
  286. }
  287. /// Convert an `u16` to return an optional value of this type. If the
  288. /// type cannot be represented by this value, the `None` is returned.
  289. #[inline]
  290. fn from_u16(n: u16) -> Option<Self> {
  291. FromPrimitive::from_u64(n as u64)
  292. }
  293. /// Convert an `u32` to return an optional value of this type. If the
  294. /// type cannot be represented by this value, the `None` is returned.
  295. #[inline]
  296. fn from_u32(n: u32) -> Option<Self> {
  297. FromPrimitive::from_u64(n as u64)
  298. }
  299. /// Convert an `u64` to return an optional value of this type. If the
  300. /// type cannot be represented by this value, the `None` is returned.
  301. fn from_u64(n: u64) -> Option<Self>;
  302. /// Convert a `f32` to return an optional value of this type. If the
  303. /// type cannot be represented by this value, the `None` is returned.
  304. #[inline]
  305. fn from_f32(n: f32) -> Option<Self> {
  306. FromPrimitive::from_f64(n as f64)
  307. }
  308. /// Convert a `f64` to return an optional value of this type. If the
  309. /// type cannot be represented by this value, the `None` is returned.
  310. #[inline]
  311. fn from_f64(n: f64) -> Option<Self> {
  312. FromPrimitive::from_i64(n as i64)
  313. }
  314. }
  315. macro_rules! impl_from_primitive {
  316. ($T:ty, $to_ty:ident) => (
  317. #[allow(deprecated)]
  318. impl FromPrimitive for $T {
  319. #[inline] fn from_i8(n: i8) -> Option<$T> { n.$to_ty() }
  320. #[inline] fn from_i16(n: i16) -> Option<$T> { n.$to_ty() }
  321. #[inline] fn from_i32(n: i32) -> Option<$T> { n.$to_ty() }
  322. #[inline] fn from_i64(n: i64) -> Option<$T> { n.$to_ty() }
  323. #[inline] fn from_u8(n: u8) -> Option<$T> { n.$to_ty() }
  324. #[inline] fn from_u16(n: u16) -> Option<$T> { n.$to_ty() }
  325. #[inline] fn from_u32(n: u32) -> Option<$T> { n.$to_ty() }
  326. #[inline] fn from_u64(n: u64) -> Option<$T> { n.$to_ty() }
  327. #[inline] fn from_f32(n: f32) -> Option<$T> { n.$to_ty() }
  328. #[inline] fn from_f64(n: f64) -> Option<$T> { n.$to_ty() }
  329. }
  330. )
  331. }
  332. impl_from_primitive!(isize, to_isize);
  333. impl_from_primitive!(i8, to_i8);
  334. impl_from_primitive!(i16, to_i16);
  335. impl_from_primitive!(i32, to_i32);
  336. impl_from_primitive!(i64, to_i64);
  337. impl_from_primitive!(usize, to_usize);
  338. impl_from_primitive!(u8, to_u8);
  339. impl_from_primitive!(u16, to_u16);
  340. impl_from_primitive!(u32, to_u32);
  341. impl_from_primitive!(u64, to_u64);
  342. impl_from_primitive!(f32, to_f32);
  343. impl_from_primitive!(f64, to_f64);
  344. impl<T: ToPrimitive> ToPrimitive for Wrapping<T> {
  345. fn to_i64(&self) -> Option<i64> { self.0.to_i64() }
  346. fn to_u64(&self) -> Option<u64> { self.0.to_u64() }
  347. }
  348. impl<T: FromPrimitive> FromPrimitive for Wrapping<T> {
  349. fn from_u64(n: u64) -> Option<Self> { T::from_u64(n).map(Wrapping) }
  350. fn from_i64(n: i64) -> Option<Self> { T::from_i64(n).map(Wrapping) }
  351. }
  352. /// Cast from one machine scalar to another.
  353. ///
  354. /// # Examples
  355. ///
  356. /// ```
  357. /// # use num_traits as num;
  358. /// let twenty: f32 = num::cast(0x14).unwrap();
  359. /// assert_eq!(twenty, 20f32);
  360. /// ```
  361. ///
  362. #[inline]
  363. pub fn cast<T: NumCast, U: NumCast>(n: T) -> Option<U> {
  364. NumCast::from(n)
  365. }
  366. /// An interface for casting between machine scalars.
  367. pub trait NumCast: Sized + ToPrimitive {
  368. /// Creates a number from another value that can be converted into
  369. /// a primitive via the `ToPrimitive` trait.
  370. fn from<T: ToPrimitive>(n: T) -> Option<Self>;
  371. }
  372. macro_rules! impl_num_cast {
  373. ($T:ty, $conv:ident) => (
  374. impl NumCast for $T {
  375. #[inline]
  376. #[allow(deprecated)]
  377. fn from<N: ToPrimitive>(n: N) -> Option<$T> {
  378. // `$conv` could be generated using `concat_idents!`, but that
  379. // macro seems to be broken at the moment
  380. n.$conv()
  381. }
  382. }
  383. )
  384. }
  385. impl_num_cast!(u8, to_u8);
  386. impl_num_cast!(u16, to_u16);
  387. impl_num_cast!(u32, to_u32);
  388. impl_num_cast!(u64, to_u64);
  389. impl_num_cast!(usize, to_usize);
  390. impl_num_cast!(i8, to_i8);
  391. impl_num_cast!(i16, to_i16);
  392. impl_num_cast!(i32, to_i32);
  393. impl_num_cast!(i64, to_i64);
  394. impl_num_cast!(isize, to_isize);
  395. impl_num_cast!(f32, to_f32);
  396. impl_num_cast!(f64, to_f64);
  397. impl<T: NumCast> NumCast for Wrapping<T> {
  398. fn from<U: ToPrimitive>(n: U) -> Option<Self> {
  399. T::from(n).map(Wrapping)
  400. }
  401. }
  402. /// A generic interface for casting between machine scalars with the
  403. /// `as` operator, which admits narrowing and precision loss.
  404. /// Implementers of this trait AsPrimitive should behave like a primitive
  405. /// numeric type (e.g. a newtype around another primitive), and the
  406. /// intended conversion must never fail.
  407. ///
  408. /// # Examples
  409. ///
  410. /// ```
  411. /// # use num_traits::AsPrimitive;
  412. /// let three: i32 = (3.14159265f32).as_();
  413. /// assert_eq!(three, 3);
  414. /// ```
  415. ///
  416. /// # Safety
  417. ///
  418. /// Currently, some uses of the `as` operator are not entirely safe.
  419. /// In particular, it is undefined behavior if:
  420. ///
  421. /// - A truncated floating point value cannot fit in the target integer
  422. /// type ([#10184](https://github.com/rust-lang/rust/issues/10184));
  423. ///
  424. /// ```ignore
  425. /// # use num_traits::AsPrimitive;
  426. /// let x: u8 = (1.04E+17).as_(); // UB
  427. /// ```
  428. ///
  429. /// - Or a floating point value does not fit in another floating
  430. /// point type ([#15536](https://github.com/rust-lang/rust/issues/15536)).
  431. ///
  432. /// ```ignore
  433. /// # use num_traits::AsPrimitive;
  434. /// let x: f32 = (1e300f64).as_(); // UB
  435. /// ```
  436. ///
  437. pub trait AsPrimitive<T>: 'static + Copy
  438. where
  439. T: 'static + Copy
  440. {
  441. /// Convert a value to another, using the `as` operator.
  442. fn as_(self) -> T;
  443. }
  444. macro_rules! impl_as_primitive {
  445. ($T: ty => $( $U: ty ),* ) => {
  446. $(
  447. impl AsPrimitive<$U> for $T {
  448. #[inline] fn as_(self) -> $U { self as $U }
  449. }
  450. )*
  451. };
  452. }
  453. impl_as_primitive!(u8 => char, u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  454. impl_as_primitive!(i8 => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  455. impl_as_primitive!(u16 => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  456. impl_as_primitive!(i16 => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  457. impl_as_primitive!(u32 => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  458. impl_as_primitive!(i32 => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  459. impl_as_primitive!(u64 => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  460. impl_as_primitive!(i64 => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  461. impl_as_primitive!(usize => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  462. impl_as_primitive!(isize => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  463. impl_as_primitive!(f32 => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  464. impl_as_primitive!(f64 => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64, f32, f64);
  465. impl_as_primitive!(char => char, u8, i8, u16, i16, u32, i32, u64, isize, usize, i64);
  466. impl_as_primitive!(bool => u8, i8, u16, i16, u32, i32, u64, isize, usize, i64);
  467. #[test]
  468. fn to_primitive_float() {
  469. use core::f32;
  470. use core::f64;
  471. let f32_toolarge = 1e39f64;
  472. assert_eq!(f32_toolarge.to_f32(), None);
  473. assert_eq!((f32::MAX as f64).to_f32(), Some(f32::MAX));
  474. assert_eq!((-f32::MAX as f64).to_f32(), Some(-f32::MAX));
  475. assert_eq!(f64::INFINITY.to_f32(), Some(f32::INFINITY));
  476. assert_eq!((f64::NEG_INFINITY).to_f32(), Some(f32::NEG_INFINITY));
  477. assert!((f64::NAN).to_f32().map_or(false, |f| f.is_nan()));
  478. }
  479. #[test]
  480. fn wrapping_to_primitive() {
  481. macro_rules! test_wrapping_to_primitive {
  482. ($($t:ty)+) => {
  483. $({
  484. let i: $t = 0;
  485. let w = Wrapping(i);
  486. assert_eq!(i.to_u8(), w.to_u8());
  487. assert_eq!(i.to_u16(), w.to_u16());
  488. assert_eq!(i.to_u32(), w.to_u32());
  489. assert_eq!(i.to_u64(), w.to_u64());
  490. assert_eq!(i.to_usize(), w.to_usize());
  491. assert_eq!(i.to_i8(), w.to_i8());
  492. assert_eq!(i.to_i16(), w.to_i16());
  493. assert_eq!(i.to_i32(), w.to_i32());
  494. assert_eq!(i.to_i64(), w.to_i64());
  495. assert_eq!(i.to_isize(), w.to_isize());
  496. assert_eq!(i.to_f32(), w.to_f32());
  497. assert_eq!(i.to_f64(), w.to_f64());
  498. })+
  499. };
  500. }
  501. test_wrapping_to_primitive!(usize u8 u16 u32 u64 isize i8 i16 i32 i64);
  502. }
  503. #[test]
  504. fn wrapping_is_toprimitive() {
  505. fn require_toprimitive<T: ToPrimitive>(_: &T) {}
  506. require_toprimitive(&Wrapping(42));
  507. }
  508. #[test]
  509. fn wrapping_is_fromprimitive() {
  510. fn require_fromprimitive<T: FromPrimitive>(_: &T) {}
  511. require_fromprimitive(&Wrapping(42));
  512. }
  513. #[test]
  514. fn wrapping_is_numcast() {
  515. fn require_numcast<T: NumCast>(_: &T) {}
  516. require_numcast(&Wrapping(42));
  517. }
  518. #[test]
  519. fn as_primitive() {
  520. let x: f32 = (1.625f64).as_();
  521. assert_eq!(x, 1.625f32);
  522. let x: f32 = (3.14159265358979323846f64).as_();
  523. assert_eq!(x, 3.1415927f32);
  524. let x: u8 = (768i16).as_();
  525. assert_eq!(x, 0);
  526. }