1 | use core::mem::size_of; |
2 | use core::num::Wrapping; |
3 | use core::{f32, f64}; |
4 | use core::{i128, i16, i32, i64, i8, isize}; |
5 | use core::{u128, u16, u32, u64, u8, usize}; |
6 | |
7 | /// A generic trait for converting a value to a number. |
8 | /// |
9 | /// A value can be represented by the target type when it lies within |
10 | /// the range of scalars supported by the target type. |
11 | /// For example, a negative integer cannot be represented by an unsigned |
12 | /// integer type, and an `i64` with a very high magnitude might not be |
13 | /// convertible to an `i32`. |
14 | /// On the other hand, conversions with possible precision loss or truncation |
15 | /// are admitted, like an `f32` with a decimal part to an integer type, or |
16 | /// even a large `f64` saturating to `f32` infinity. |
17 | pub trait ToPrimitive { |
18 | /// Converts the value of `self` to an `isize`. If the value cannot be |
19 | /// represented by an `isize`, then `None` is returned. |
20 | #[inline ] |
21 | fn to_isize(&self) -> Option<isize> { |
22 | self.to_i64().as_ref().and_then(ToPrimitive::to_isize) |
23 | } |
24 | |
25 | /// Converts the value of `self` to an `i8`. If the value cannot be |
26 | /// represented by an `i8`, then `None` is returned. |
27 | #[inline ] |
28 | fn to_i8(&self) -> Option<i8> { |
29 | self.to_i64().as_ref().and_then(ToPrimitive::to_i8) |
30 | } |
31 | |
32 | /// Converts the value of `self` to an `i16`. If the value cannot be |
33 | /// represented by an `i16`, then `None` is returned. |
34 | #[inline ] |
35 | fn to_i16(&self) -> Option<i16> { |
36 | self.to_i64().as_ref().and_then(ToPrimitive::to_i16) |
37 | } |
38 | |
39 | /// Converts the value of `self` to an `i32`. If the value cannot be |
40 | /// represented by an `i32`, then `None` is returned. |
41 | #[inline ] |
42 | fn to_i32(&self) -> Option<i32> { |
43 | self.to_i64().as_ref().and_then(ToPrimitive::to_i32) |
44 | } |
45 | |
46 | /// Converts the value of `self` to an `i64`. If the value cannot be |
47 | /// represented by an `i64`, then `None` is returned. |
48 | fn to_i64(&self) -> Option<i64>; |
49 | |
50 | /// Converts the value of `self` to an `i128`. If the value cannot be |
51 | /// represented by an `i128` (`i64` under the default implementation), then |
52 | /// `None` is returned. |
53 | /// |
54 | /// The default implementation converts through `to_i64()`. Types implementing |
55 | /// this trait should override this method if they can represent a greater range. |
56 | #[inline ] |
57 | fn to_i128(&self) -> Option<i128> { |
58 | self.to_i64().map(From::from) |
59 | } |
60 | |
61 | /// Converts the value of `self` to a `usize`. If the value cannot be |
62 | /// represented by a `usize`, then `None` is returned. |
63 | #[inline ] |
64 | fn to_usize(&self) -> Option<usize> { |
65 | self.to_u64().as_ref().and_then(ToPrimitive::to_usize) |
66 | } |
67 | |
68 | /// Converts the value of `self` to a `u8`. If the value cannot be |
69 | /// represented by a `u8`, then `None` is returned. |
70 | #[inline ] |
71 | fn to_u8(&self) -> Option<u8> { |
72 | self.to_u64().as_ref().and_then(ToPrimitive::to_u8) |
73 | } |
74 | |
75 | /// Converts the value of `self` to a `u16`. If the value cannot be |
76 | /// represented by a `u16`, then `None` is returned. |
77 | #[inline ] |
78 | fn to_u16(&self) -> Option<u16> { |
79 | self.to_u64().as_ref().and_then(ToPrimitive::to_u16) |
80 | } |
81 | |
82 | /// Converts the value of `self` to a `u32`. If the value cannot be |
83 | /// represented by a `u32`, then `None` is returned. |
84 | #[inline ] |
85 | fn to_u32(&self) -> Option<u32> { |
86 | self.to_u64().as_ref().and_then(ToPrimitive::to_u32) |
87 | } |
88 | |
89 | /// Converts the value of `self` to a `u64`. If the value cannot be |
90 | /// represented by a `u64`, then `None` is returned. |
91 | fn to_u64(&self) -> Option<u64>; |
92 | |
93 | /// Converts the value of `self` to a `u128`. If the value cannot be |
94 | /// represented by a `u128` (`u64` under the default implementation), then |
95 | /// `None` is returned. |
96 | /// |
97 | /// The default implementation converts through `to_u64()`. Types implementing |
98 | /// this trait should override this method if they can represent a greater range. |
99 | #[inline ] |
100 | fn to_u128(&self) -> Option<u128> { |
101 | self.to_u64().map(From::from) |
102 | } |
103 | |
104 | /// Converts the value of `self` to an `f32`. Overflows may map to positive |
105 | /// or negative inifinity, otherwise `None` is returned if the value cannot |
106 | /// be represented by an `f32`. |
107 | #[inline ] |
108 | fn to_f32(&self) -> Option<f32> { |
109 | self.to_f64().as_ref().and_then(ToPrimitive::to_f32) |
110 | } |
111 | |
112 | /// Converts the value of `self` to an `f64`. Overflows may map to positive |
113 | /// or negative inifinity, otherwise `None` is returned if the value cannot |
114 | /// be represented by an `f64`. |
115 | /// |
116 | /// The default implementation tries to convert through `to_i64()`, and |
117 | /// failing that through `to_u64()`. Types implementing this trait should |
118 | /// override this method if they can represent a greater range. |
119 | #[inline ] |
120 | fn to_f64(&self) -> Option<f64> { |
121 | match self.to_i64() { |
122 | Some(i) => i.to_f64(), |
123 | None => self.to_u64().as_ref().and_then(ToPrimitive::to_f64), |
124 | } |
125 | } |
126 | } |
127 | |
128 | macro_rules! impl_to_primitive_int_to_int { |
129 | ($SrcT:ident : $( $(#[$cfg:meta])* fn $method:ident -> $DstT:ident ; )*) => {$( |
130 | #[inline] |
131 | $(#[$cfg])* |
132 | fn $method(&self) -> Option<$DstT> { |
133 | let min = $DstT::MIN as $SrcT; |
134 | let max = $DstT::MAX as $SrcT; |
135 | if size_of::<$SrcT>() <= size_of::<$DstT>() || (min <= *self && *self <= max) { |
136 | Some(*self as $DstT) |
137 | } else { |
138 | None |
139 | } |
140 | } |
141 | )*} |
142 | } |
143 | |
144 | macro_rules! impl_to_primitive_int_to_uint { |
145 | ($SrcT:ident : $( $(#[$cfg:meta])* fn $method:ident -> $DstT:ident ; )*) => {$( |
146 | #[inline] |
147 | $(#[$cfg])* |
148 | fn $method(&self) -> Option<$DstT> { |
149 | let max = $DstT::MAX as $SrcT; |
150 | if 0 <= *self && (size_of::<$SrcT>() <= size_of::<$DstT>() || *self <= max) { |
151 | Some(*self as $DstT) |
152 | } else { |
153 | None |
154 | } |
155 | } |
156 | )*} |
157 | } |
158 | |
159 | macro_rules! impl_to_primitive_int { |
160 | ($T:ident) => { |
161 | impl ToPrimitive for $T { |
162 | impl_to_primitive_int_to_int! { $T: |
163 | fn to_isize -> isize; |
164 | fn to_i8 -> i8; |
165 | fn to_i16 -> i16; |
166 | fn to_i32 -> i32; |
167 | fn to_i64 -> i64; |
168 | fn to_i128 -> i128; |
169 | } |
170 | |
171 | impl_to_primitive_int_to_uint! { $T: |
172 | fn to_usize -> usize; |
173 | fn to_u8 -> u8; |
174 | fn to_u16 -> u16; |
175 | fn to_u32 -> u32; |
176 | fn to_u64 -> u64; |
177 | fn to_u128 -> u128; |
178 | } |
179 | |
180 | #[inline] |
181 | fn to_f32(&self) -> Option<f32> { |
182 | Some(*self as f32) |
183 | } |
184 | #[inline] |
185 | fn to_f64(&self) -> Option<f64> { |
186 | Some(*self as f64) |
187 | } |
188 | } |
189 | }; |
190 | } |
191 | |
192 | impl_to_primitive_int!(isize); |
193 | impl_to_primitive_int!(i8); |
194 | impl_to_primitive_int!(i16); |
195 | impl_to_primitive_int!(i32); |
196 | impl_to_primitive_int!(i64); |
197 | impl_to_primitive_int!(i128); |
198 | |
199 | macro_rules! impl_to_primitive_uint_to_int { |
200 | ($SrcT:ident : $( $(#[$cfg:meta])* fn $method:ident -> $DstT:ident ; )*) => {$( |
201 | #[inline] |
202 | $(#[$cfg])* |
203 | fn $method(&self) -> Option<$DstT> { |
204 | let max = $DstT::MAX as $SrcT; |
205 | if size_of::<$SrcT>() < size_of::<$DstT>() || *self <= max { |
206 | Some(*self as $DstT) |
207 | } else { |
208 | None |
209 | } |
210 | } |
211 | )*} |
212 | } |
213 | |
214 | macro_rules! impl_to_primitive_uint_to_uint { |
215 | ($SrcT:ident : $( $(#[$cfg:meta])* fn $method:ident -> $DstT:ident ; )*) => {$( |
216 | #[inline] |
217 | $(#[$cfg])* |
218 | fn $method(&self) -> Option<$DstT> { |
219 | let max = $DstT::MAX as $SrcT; |
220 | if size_of::<$SrcT>() <= size_of::<$DstT>() || *self <= max { |
221 | Some(*self as $DstT) |
222 | } else { |
223 | None |
224 | } |
225 | } |
226 | )*} |
227 | } |
228 | |
229 | macro_rules! impl_to_primitive_uint { |
230 | ($T:ident) => { |
231 | impl ToPrimitive for $T { |
232 | impl_to_primitive_uint_to_int! { $T: |
233 | fn to_isize -> isize; |
234 | fn to_i8 -> i8; |
235 | fn to_i16 -> i16; |
236 | fn to_i32 -> i32; |
237 | fn to_i64 -> i64; |
238 | fn to_i128 -> i128; |
239 | } |
240 | |
241 | impl_to_primitive_uint_to_uint! { $T: |
242 | fn to_usize -> usize; |
243 | fn to_u8 -> u8; |
244 | fn to_u16 -> u16; |
245 | fn to_u32 -> u32; |
246 | fn to_u64 -> u64; |
247 | fn to_u128 -> u128; |
248 | } |
249 | |
250 | #[inline] |
251 | fn to_f32(&self) -> Option<f32> { |
252 | Some(*self as f32) |
253 | } |
254 | #[inline] |
255 | fn to_f64(&self) -> Option<f64> { |
256 | Some(*self as f64) |
257 | } |
258 | } |
259 | }; |
260 | } |
261 | |
262 | impl_to_primitive_uint!(usize); |
263 | impl_to_primitive_uint!(u8); |
264 | impl_to_primitive_uint!(u16); |
265 | impl_to_primitive_uint!(u32); |
266 | impl_to_primitive_uint!(u64); |
267 | impl_to_primitive_uint!(u128); |
268 | |
269 | macro_rules! impl_to_primitive_float_to_float { |
270 | ($SrcT:ident : $( fn $method:ident -> $DstT:ident ; )*) => {$( |
271 | #[inline] |
272 | fn $method(&self) -> Option<$DstT> { |
273 | // We can safely cast all values, whether NaN, +-inf, or finite. |
274 | // Finite values that are reducing size may saturate to +-inf. |
275 | Some(*self as $DstT) |
276 | } |
277 | )*} |
278 | } |
279 | |
280 | macro_rules! float_to_int_unchecked { |
281 | // SAFETY: Must not be NaN or infinite; must be representable as the integer after truncating. |
282 | // We already checked that the float is in the exclusive range `(MIN-1, MAX+1)`. |
283 | ($float:expr => $int:ty) => { |
284 | unsafe { $float.to_int_unchecked::<$int>() } |
285 | }; |
286 | } |
287 | |
288 | macro_rules! impl_to_primitive_float_to_signed_int { |
289 | ($f:ident : $( $(#[$cfg:meta])* fn $method:ident -> $i:ident ; )*) => {$( |
290 | #[inline] |
291 | $(#[$cfg])* |
292 | fn $method(&self) -> Option<$i> { |
293 | // Float as int truncates toward zero, so we want to allow values |
294 | // in the exclusive range `(MIN-1, MAX+1)`. |
295 | if size_of::<$f>() > size_of::<$i>() { |
296 | // With a larger size, we can represent the range exactly. |
297 | const MIN_M1: $f = $i::MIN as $f - 1.0; |
298 | const MAX_P1: $f = $i::MAX as $f + 1.0; |
299 | if *self > MIN_M1 && *self < MAX_P1 { |
300 | return Some(float_to_int_unchecked!(*self => $i)); |
301 | } |
302 | } else { |
303 | // We can't represent `MIN-1` exactly, but there's no fractional part |
304 | // at this magnitude, so we can just use a `MIN` inclusive boundary. |
305 | const MIN: $f = $i::MIN as $f; |
306 | // We can't represent `MAX` exactly, but it will round up to exactly |
307 | // `MAX+1` (a power of two) when we cast it. |
308 | const MAX_P1: $f = $i::MAX as $f; |
309 | if *self >= MIN && *self < MAX_P1 { |
310 | return Some(float_to_int_unchecked!(*self => $i)); |
311 | } |
312 | } |
313 | None |
314 | } |
315 | )*} |
316 | } |
317 | |
318 | macro_rules! impl_to_primitive_float_to_unsigned_int { |
319 | ($f:ident : $( $(#[$cfg:meta])* fn $method:ident -> $u:ident ; )*) => {$( |
320 | #[inline] |
321 | $(#[$cfg])* |
322 | fn $method(&self) -> Option<$u> { |
323 | // Float as int truncates toward zero, so we want to allow values |
324 | // in the exclusive range `(-1, MAX+1)`. |
325 | if size_of::<$f>() > size_of::<$u>() { |
326 | // With a larger size, we can represent the range exactly. |
327 | const MAX_P1: $f = $u::MAX as $f + 1.0; |
328 | if *self > -1.0 && *self < MAX_P1 { |
329 | return Some(float_to_int_unchecked!(*self => $u)); |
330 | } |
331 | } else { |
332 | // We can't represent `MAX` exactly, but it will round up to exactly |
333 | // `MAX+1` (a power of two) when we cast it. |
334 | // (`u128::MAX as f32` is infinity, but this is still ok.) |
335 | const MAX_P1: $f = $u::MAX as $f; |
336 | if *self > -1.0 && *self < MAX_P1 { |
337 | return Some(float_to_int_unchecked!(*self => $u)); |
338 | } |
339 | } |
340 | None |
341 | } |
342 | )*} |
343 | } |
344 | |
345 | macro_rules! impl_to_primitive_float { |
346 | ($T:ident) => { |
347 | impl ToPrimitive for $T { |
348 | impl_to_primitive_float_to_signed_int! { $T: |
349 | fn to_isize -> isize; |
350 | fn to_i8 -> i8; |
351 | fn to_i16 -> i16; |
352 | fn to_i32 -> i32; |
353 | fn to_i64 -> i64; |
354 | fn to_i128 -> i128; |
355 | } |
356 | |
357 | impl_to_primitive_float_to_unsigned_int! { $T: |
358 | fn to_usize -> usize; |
359 | fn to_u8 -> u8; |
360 | fn to_u16 -> u16; |
361 | fn to_u32 -> u32; |
362 | fn to_u64 -> u64; |
363 | fn to_u128 -> u128; |
364 | } |
365 | |
366 | impl_to_primitive_float_to_float! { $T: |
367 | fn to_f32 -> f32; |
368 | fn to_f64 -> f64; |
369 | } |
370 | } |
371 | }; |
372 | } |
373 | |
374 | impl_to_primitive_float!(f32); |
375 | impl_to_primitive_float!(f64); |
376 | |
377 | /// A generic trait for converting a number to a value. |
378 | /// |
379 | /// A value can be represented by the target type when it lies within |
380 | /// the range of scalars supported by the target type. |
381 | /// For example, a negative integer cannot be represented by an unsigned |
382 | /// integer type, and an `i64` with a very high magnitude might not be |
383 | /// convertible to an `i32`. |
384 | /// On the other hand, conversions with possible precision loss or truncation |
385 | /// are admitted, like an `f32` with a decimal part to an integer type, or |
386 | /// even a large `f64` saturating to `f32` infinity. |
387 | pub trait FromPrimitive: Sized { |
388 | /// Converts an `isize` to return an optional value of this type. If the |
389 | /// value cannot be represented by this type, then `None` is returned. |
390 | #[inline ] |
391 | fn from_isize(n: isize) -> Option<Self> { |
392 | n.to_i64().and_then(FromPrimitive::from_i64) |
393 | } |
394 | |
395 | /// Converts an `i8` to return an optional value of this type. If the |
396 | /// value cannot be represented by this type, then `None` is returned. |
397 | #[inline ] |
398 | fn from_i8(n: i8) -> Option<Self> { |
399 | FromPrimitive::from_i64(From::from(n)) |
400 | } |
401 | |
402 | /// Converts an `i16` to return an optional value of this type. If the |
403 | /// value cannot be represented by this type, then `None` is returned. |
404 | #[inline ] |
405 | fn from_i16(n: i16) -> Option<Self> { |
406 | FromPrimitive::from_i64(From::from(n)) |
407 | } |
408 | |
409 | /// Converts an `i32` to return an optional value of this type. If the |
410 | /// value cannot be represented by this type, then `None` is returned. |
411 | #[inline ] |
412 | fn from_i32(n: i32) -> Option<Self> { |
413 | FromPrimitive::from_i64(From::from(n)) |
414 | } |
415 | |
416 | /// Converts an `i64` to return an optional value of this type. If the |
417 | /// value cannot be represented by this type, then `None` is returned. |
418 | fn from_i64(n: i64) -> Option<Self>; |
419 | |
420 | /// Converts an `i128` to return an optional value of this type. If the |
421 | /// value cannot be represented by this type, then `None` is returned. |
422 | /// |
423 | /// The default implementation converts through `from_i64()`. Types implementing |
424 | /// this trait should override this method if they can represent a greater range. |
425 | #[inline ] |
426 | fn from_i128(n: i128) -> Option<Self> { |
427 | n.to_i64().and_then(FromPrimitive::from_i64) |
428 | } |
429 | |
430 | /// Converts a `usize` to return an optional value of this type. If the |
431 | /// value cannot be represented by this type, then `None` is returned. |
432 | #[inline ] |
433 | fn from_usize(n: usize) -> Option<Self> { |
434 | n.to_u64().and_then(FromPrimitive::from_u64) |
435 | } |
436 | |
437 | /// Converts an `u8` to return an optional value of this type. If the |
438 | /// value cannot be represented by this type, then `None` is returned. |
439 | #[inline ] |
440 | fn from_u8(n: u8) -> Option<Self> { |
441 | FromPrimitive::from_u64(From::from(n)) |
442 | } |
443 | |
444 | /// Converts an `u16` to return an optional value of this type. If the |
445 | /// value cannot be represented by this type, then `None` is returned. |
446 | #[inline ] |
447 | fn from_u16(n: u16) -> Option<Self> { |
448 | FromPrimitive::from_u64(From::from(n)) |
449 | } |
450 | |
451 | /// Converts an `u32` to return an optional value of this type. If the |
452 | /// value cannot be represented by this type, then `None` is returned. |
453 | #[inline ] |
454 | fn from_u32(n: u32) -> Option<Self> { |
455 | FromPrimitive::from_u64(From::from(n)) |
456 | } |
457 | |
458 | /// Converts an `u64` to return an optional value of this type. If the |
459 | /// value cannot be represented by this type, then `None` is returned. |
460 | fn from_u64(n: u64) -> Option<Self>; |
461 | |
462 | /// Converts an `u128` to return an optional value of this type. If the |
463 | /// value cannot be represented by this type, then `None` is returned. |
464 | /// |
465 | /// The default implementation converts through `from_u64()`. Types implementing |
466 | /// this trait should override this method if they can represent a greater range. |
467 | #[inline ] |
468 | fn from_u128(n: u128) -> Option<Self> { |
469 | n.to_u64().and_then(FromPrimitive::from_u64) |
470 | } |
471 | |
472 | /// Converts a `f32` to return an optional value of this type. If the |
473 | /// value cannot be represented by this type, then `None` is returned. |
474 | #[inline ] |
475 | fn from_f32(n: f32) -> Option<Self> { |
476 | FromPrimitive::from_f64(From::from(n)) |
477 | } |
478 | |
479 | /// Converts a `f64` to return an optional value of this type. If the |
480 | /// value cannot be represented by this type, then `None` is returned. |
481 | /// |
482 | /// The default implementation tries to convert through `from_i64()`, and |
483 | /// failing that through `from_u64()`. Types implementing this trait should |
484 | /// override this method if they can represent a greater range. |
485 | #[inline ] |
486 | fn from_f64(n: f64) -> Option<Self> { |
487 | match n.to_i64() { |
488 | Some(i) => FromPrimitive::from_i64(i), |
489 | None => n.to_u64().and_then(FromPrimitive::from_u64), |
490 | } |
491 | } |
492 | } |
493 | |
494 | macro_rules! impl_from_primitive { |
495 | ($T:ty, $to_ty:ident) => { |
496 | #[allow(deprecated)] |
497 | impl FromPrimitive for $T { |
498 | #[inline] |
499 | fn from_isize(n: isize) -> Option<$T> { |
500 | n.$to_ty() |
501 | } |
502 | #[inline] |
503 | fn from_i8(n: i8) -> Option<$T> { |
504 | n.$to_ty() |
505 | } |
506 | #[inline] |
507 | fn from_i16(n: i16) -> Option<$T> { |
508 | n.$to_ty() |
509 | } |
510 | #[inline] |
511 | fn from_i32(n: i32) -> Option<$T> { |
512 | n.$to_ty() |
513 | } |
514 | #[inline] |
515 | fn from_i64(n: i64) -> Option<$T> { |
516 | n.$to_ty() |
517 | } |
518 | #[inline] |
519 | fn from_i128(n: i128) -> Option<$T> { |
520 | n.$to_ty() |
521 | } |
522 | |
523 | #[inline] |
524 | fn from_usize(n: usize) -> Option<$T> { |
525 | n.$to_ty() |
526 | } |
527 | #[inline] |
528 | fn from_u8(n: u8) -> Option<$T> { |
529 | n.$to_ty() |
530 | } |
531 | #[inline] |
532 | fn from_u16(n: u16) -> Option<$T> { |
533 | n.$to_ty() |
534 | } |
535 | #[inline] |
536 | fn from_u32(n: u32) -> Option<$T> { |
537 | n.$to_ty() |
538 | } |
539 | #[inline] |
540 | fn from_u64(n: u64) -> Option<$T> { |
541 | n.$to_ty() |
542 | } |
543 | #[inline] |
544 | fn from_u128(n: u128) -> Option<$T> { |
545 | n.$to_ty() |
546 | } |
547 | |
548 | #[inline] |
549 | fn from_f32(n: f32) -> Option<$T> { |
550 | n.$to_ty() |
551 | } |
552 | #[inline] |
553 | fn from_f64(n: f64) -> Option<$T> { |
554 | n.$to_ty() |
555 | } |
556 | } |
557 | }; |
558 | } |
559 | |
560 | impl_from_primitive!(isize, to_isize); |
561 | impl_from_primitive!(i8, to_i8); |
562 | impl_from_primitive!(i16, to_i16); |
563 | impl_from_primitive!(i32, to_i32); |
564 | impl_from_primitive!(i64, to_i64); |
565 | impl_from_primitive!(i128, to_i128); |
566 | impl_from_primitive!(usize, to_usize); |
567 | impl_from_primitive!(u8, to_u8); |
568 | impl_from_primitive!(u16, to_u16); |
569 | impl_from_primitive!(u32, to_u32); |
570 | impl_from_primitive!(u64, to_u64); |
571 | impl_from_primitive!(u128, to_u128); |
572 | impl_from_primitive!(f32, to_f32); |
573 | impl_from_primitive!(f64, to_f64); |
574 | |
575 | macro_rules! impl_to_primitive_wrapping { |
576 | ($( $(#[$cfg:meta])* fn $method:ident -> $i:ident ; )*) => {$( |
577 | #[inline] |
578 | $(#[$cfg])* |
579 | fn $method(&self) -> Option<$i> { |
580 | (self.0).$method() |
581 | } |
582 | )*} |
583 | } |
584 | |
585 | impl<T: ToPrimitive> ToPrimitive for Wrapping<T> { |
586 | impl_to_primitive_wrapping! { |
587 | fn to_isize -> isize; |
588 | fn to_i8 -> i8; |
589 | fn to_i16 -> i16; |
590 | fn to_i32 -> i32; |
591 | fn to_i64 -> i64; |
592 | fn to_i128 -> i128; |
593 | |
594 | fn to_usize -> usize; |
595 | fn to_u8 -> u8; |
596 | fn to_u16 -> u16; |
597 | fn to_u32 -> u32; |
598 | fn to_u64 -> u64; |
599 | fn to_u128 -> u128; |
600 | |
601 | fn to_f32 -> f32; |
602 | fn to_f64 -> f64; |
603 | } |
604 | } |
605 | |
606 | macro_rules! impl_from_primitive_wrapping { |
607 | ($( $(#[$cfg:meta])* fn $method:ident ( $i:ident ); )*) => {$( |
608 | #[inline] |
609 | $(#[$cfg])* |
610 | fn $method(n: $i) -> Option<Self> { |
611 | T::$method(n).map(Wrapping) |
612 | } |
613 | )*} |
614 | } |
615 | |
616 | impl<T: FromPrimitive> FromPrimitive for Wrapping<T> { |
617 | impl_from_primitive_wrapping! { |
618 | fn from_isize(isize); |
619 | fn from_i8(i8); |
620 | fn from_i16(i16); |
621 | fn from_i32(i32); |
622 | fn from_i64(i64); |
623 | fn from_i128(i128); |
624 | |
625 | fn from_usize(usize); |
626 | fn from_u8(u8); |
627 | fn from_u16(u16); |
628 | fn from_u32(u32); |
629 | fn from_u64(u64); |
630 | fn from_u128(u128); |
631 | |
632 | fn from_f32(f32); |
633 | fn from_f64(f64); |
634 | } |
635 | } |
636 | |
637 | /// Cast from one machine scalar to another. |
638 | /// |
639 | /// # Examples |
640 | /// |
641 | /// ``` |
642 | /// # use num_traits as num; |
643 | /// let twenty: f32 = num::cast(0x14).unwrap(); |
644 | /// assert_eq!(twenty, 20f32); |
645 | /// ``` |
646 | /// |
647 | #[inline ] |
648 | pub fn cast<T: NumCast, U: NumCast>(n: T) -> Option<U> { |
649 | NumCast::from(n) |
650 | } |
651 | |
652 | /// An interface for casting between machine scalars. |
653 | pub trait NumCast: Sized + ToPrimitive { |
654 | /// Creates a number from another value that can be converted into |
655 | /// a primitive via the `ToPrimitive` trait. If the source value cannot be |
656 | /// represented by the target type, then `None` is returned. |
657 | /// |
658 | /// A value can be represented by the target type when it lies within |
659 | /// the range of scalars supported by the target type. |
660 | /// For example, a negative integer cannot be represented by an unsigned |
661 | /// integer type, and an `i64` with a very high magnitude might not be |
662 | /// convertible to an `i32`. |
663 | /// On the other hand, conversions with possible precision loss or truncation |
664 | /// are admitted, like an `f32` with a decimal part to an integer type, or |
665 | /// even a large `f64` saturating to `f32` infinity. |
666 | fn from<T: ToPrimitive>(n: T) -> Option<Self>; |
667 | } |
668 | |
669 | macro_rules! impl_num_cast { |
670 | ($T:ty, $conv:ident) => { |
671 | impl NumCast for $T { |
672 | #[inline] |
673 | #[allow(deprecated)] |
674 | fn from<N: ToPrimitive>(n: N) -> Option<$T> { |
675 | // `$conv` could be generated using `concat_idents!`, but that |
676 | // macro seems to be broken at the moment |
677 | n.$conv() |
678 | } |
679 | } |
680 | }; |
681 | } |
682 | |
683 | impl_num_cast!(u8, to_u8); |
684 | impl_num_cast!(u16, to_u16); |
685 | impl_num_cast!(u32, to_u32); |
686 | impl_num_cast!(u64, to_u64); |
687 | impl_num_cast!(u128, to_u128); |
688 | impl_num_cast!(usize, to_usize); |
689 | impl_num_cast!(i8, to_i8); |
690 | impl_num_cast!(i16, to_i16); |
691 | impl_num_cast!(i32, to_i32); |
692 | impl_num_cast!(i64, to_i64); |
693 | impl_num_cast!(i128, to_i128); |
694 | impl_num_cast!(isize, to_isize); |
695 | impl_num_cast!(f32, to_f32); |
696 | impl_num_cast!(f64, to_f64); |
697 | |
698 | impl<T: NumCast> NumCast for Wrapping<T> { |
699 | fn from<U: ToPrimitive>(n: U) -> Option<Self> { |
700 | T::from(n).map(Wrapping) |
701 | } |
702 | } |
703 | |
704 | /// A generic interface for casting between machine scalars with the |
705 | /// `as` operator, which admits narrowing and precision loss. |
706 | /// Implementers of this trait `AsPrimitive` should behave like a primitive |
707 | /// numeric type (e.g. a newtype around another primitive), and the |
708 | /// intended conversion must never fail. |
709 | /// |
710 | /// # Examples |
711 | /// |
712 | /// ``` |
713 | /// # use num_traits::AsPrimitive; |
714 | /// let three: i32 = (3.14159265f32).as_(); |
715 | /// assert_eq!(three, 3); |
716 | /// ``` |
717 | /// |
718 | /// # Safety |
719 | /// |
720 | /// **In Rust versions before 1.45.0**, some uses of the `as` operator were not entirely safe. |
721 | /// In particular, it was undefined behavior if |
722 | /// a truncated floating point value could not fit in the target integer |
723 | /// type ([#10184](https://github.com/rust-lang/rust/issues/10184)). |
724 | /// |
725 | /// ```ignore |
726 | /// # use num_traits::AsPrimitive; |
727 | /// let x: u8 = (1.04E+17).as_(); // UB |
728 | /// ``` |
729 | /// |
730 | pub trait AsPrimitive<T>: 'static + Copy |
731 | where |
732 | T: 'static + Copy, |
733 | { |
734 | /// Convert a value to another, using the `as` operator. |
735 | fn as_(self) -> T; |
736 | } |
737 | |
738 | macro_rules! impl_as_primitive { |
739 | (@ $T: ty => $(#[$cfg:meta])* impl $U: ty ) => { |
740 | $(#[$cfg])* |
741 | impl AsPrimitive<$U> for $T { |
742 | #[inline] fn as_(self) -> $U { self as $U } |
743 | } |
744 | }; |
745 | (@ $T: ty => { $( $U: ty ),* } ) => {$( |
746 | impl_as_primitive!(@ $T => impl $U); |
747 | )*}; |
748 | ($T: ty => { $( $U: ty ),* } ) => { |
749 | impl_as_primitive!(@ $T => { $( $U ),* }); |
750 | impl_as_primitive!(@ $T => { u8, u16, u32, u64, u128, usize }); |
751 | impl_as_primitive!(@ $T => { i8, i16, i32, i64, i128, isize }); |
752 | }; |
753 | } |
754 | |
755 | impl_as_primitive!(u8 => { char, f32, f64 }); |
756 | impl_as_primitive!(i8 => { f32, f64 }); |
757 | impl_as_primitive!(u16 => { f32, f64 }); |
758 | impl_as_primitive!(i16 => { f32, f64 }); |
759 | impl_as_primitive!(u32 => { f32, f64 }); |
760 | impl_as_primitive!(i32 => { f32, f64 }); |
761 | impl_as_primitive!(u64 => { f32, f64 }); |
762 | impl_as_primitive!(i64 => { f32, f64 }); |
763 | impl_as_primitive!(u128 => { f32, f64 }); |
764 | impl_as_primitive!(i128 => { f32, f64 }); |
765 | impl_as_primitive!(usize => { f32, f64 }); |
766 | impl_as_primitive!(isize => { f32, f64 }); |
767 | impl_as_primitive!(f32 => { f32, f64 }); |
768 | impl_as_primitive!(f64 => { f32, f64 }); |
769 | impl_as_primitive!(char => { char }); |
770 | impl_as_primitive!(bool => {}); |
771 | |