1#![warn(clippy::all, clippy::pedantic, clippy::undocumented_unsafe_blocks)]
2#![allow(
3 clippy::let_underscore_untyped,
4 reason = "https://github.com/rust-lang/rust-clippy/pull/10442#issuecomment-1516570154"
5)]
6#![allow(
7 clippy::question_mark,
8 reason = "https://github.com/rust-lang/rust-clippy/issues/8281"
9)]
10#![allow(clippy::manual_let_else, reason = "manual_let_else was very buggy on release")]
11#![allow(clippy::missing_errors_doc, reason = "A lot of existing code fails this lint")]
12#![allow(
13 clippy::unnecessary_lazy_evaluations,
14 reason = "https://github.com/rust-lang/rust-clippy/issues/8109"
15)]
16#![cfg_attr(
17 test,
18 allow(clippy::non_ascii_literal, reason = "tests sometimes require UTF-8 string content")
19)]
20#![allow(unknown_lints)]
21#![warn(
22 missing_copy_implementations,
23 missing_debug_implementations,
24 missing_docs,
25 rust_2024_compatibility,
26 trivial_casts,
27 trivial_numeric_casts,
28 unused_qualifications,
29 variant_size_differences
30)]
31#![forbid(unsafe_code)]
32#![cfg_attr(docsrs, feature(doc_cfg))]
37#![cfg_attr(docsrs, feature(doc_alias))]
38
39#![no_std]
98
99#[cfg(doctest)]
101#[doc = include_str!("../README.md")]
102mod readme {}
103
104use core::time::Duration;
105
106pub use range::{RUBY_FIXNUM_MAX, RUBY_FIXNUM_MIN};
107
108mod range {
109 pub const RUBY_FIXNUM_MAX: i64 = i64::MAX / 2;
119
120 pub const RUBY_FIXNUM_MIN: i64 = i64::MIN / 2;
130
131 pub(crate) mod u64 {
132 pub(crate) const RUBY_FIXNUM_MAX: u64 = super::RUBY_FIXNUM_MAX as u64;
133 }
134
135 pub(crate) mod u128 {
136 pub(crate) const RUBY_FIXNUM_MAX: u128 = super::RUBY_FIXNUM_MAX as u128;
137 }
138
139 #[cfg(test)]
140 mod tests {
141 use super::RUBY_FIXNUM_MAX;
142
143 #[test]
144 fn casts_in_const_context_are_safe() {
145 assert_eq!(super::u64::RUBY_FIXNUM_MAX, u64::try_from(RUBY_FIXNUM_MAX).unwrap());
146 assert_eq!(super::u128::RUBY_FIXNUM_MAX, u128::try_from(RUBY_FIXNUM_MAX).unwrap());
147 }
148 }
149}
150
151mod private {
152 pub trait Sealed {}
153
154 impl Sealed for i8 {}
155 impl Sealed for i16 {}
156 impl Sealed for i32 {}
157 impl Sealed for i64 {}
158 impl Sealed for i128 {}
159
160 impl Sealed for u8 {}
161 impl Sealed for u16 {}
162 impl Sealed for u32 {}
163 impl Sealed for u64 {}
164 impl Sealed for u128 {}
165
166 impl Sealed for f32 {}
167 impl Sealed for f64 {}
168}
169
170pub trait Fixable: private::Sealed + Sized {
181 #[must_use]
185 fn to_fix(self) -> Option<i64>;
186
187 #[must_use]
189 #[expect(
190 clippy::wrong_self_convention,
191 reason = "implemented on numeric types which are smaller than a reference, to use value receivers for primitives like `f64::is_nan`"
192 )]
193 fn is_fixable(self) -> bool {
194 self.to_fix().is_some()
195 }
196}
197
198impl Fixable for i8 {
199 fn to_fix(self) -> Option<i64> {
204 Some(self.into())
205 }
206
207 fn is_fixable(self) -> bool {
212 true
213 }
214}
215
216impl Fixable for i16 {
217 fn to_fix(self) -> Option<i64> {
222 Some(self.into())
223 }
224
225 fn is_fixable(self) -> bool {
230 true
231 }
232}
233
234impl Fixable for i32 {
235 fn to_fix(self) -> Option<i64> {
240 Some(self.into())
241 }
242
243 fn is_fixable(self) -> bool {
248 true
249 }
250}
251
252impl Fixable for i64 {
253 fn to_fix(self) -> Option<i64> {
260 if self > RUBY_FIXNUM_MAX {
261 return None;
262 }
263 if self < RUBY_FIXNUM_MIN {
264 return None;
265 }
266 Some(self)
267 }
268
269 fn is_fixable(self) -> bool {
274 (RUBY_FIXNUM_MIN..=RUBY_FIXNUM_MAX).contains(&self)
275 }
276}
277
278impl Fixable for i128 {
279 fn to_fix(self) -> Option<i64> {
286 let x = i64::try_from(self).ok()?;
287 x.to_fix()
288 }
289
290 fn is_fixable(self) -> bool {
295 (RUBY_FIXNUM_MIN.into()..=RUBY_FIXNUM_MAX.into()).contains(&self)
296 }
297}
298
299impl Fixable for u8 {
300 fn to_fix(self) -> Option<i64> {
305 Some(self.into())
306 }
307
308 fn is_fixable(self) -> bool {
313 true
314 }
315}
316
317impl Fixable for u16 {
318 fn to_fix(self) -> Option<i64> {
323 Some(self.into())
324 }
325
326 fn is_fixable(self) -> bool {
331 true
332 }
333}
334
335impl Fixable for u32 {
336 fn to_fix(self) -> Option<i64> {
341 Some(self.into())
342 }
343
344 fn is_fixable(self) -> bool {
349 true
350 }
351}
352
353impl Fixable for u64 {
354 fn to_fix(self) -> Option<i64> {
360 let x = i64::try_from(self).ok()?;
361 if x > RUBY_FIXNUM_MAX {
362 return None;
363 }
364 Some(x)
366 }
367
368 fn is_fixable(self) -> bool {
373 use crate::range::u64::RUBY_FIXNUM_MAX;
374
375 (..=RUBY_FIXNUM_MAX).contains(&self)
376 }
377}
378
379impl Fixable for u128 {
380 fn to_fix(self) -> Option<i64> {
386 let x = i64::try_from(self).ok()?;
387 if x > RUBY_FIXNUM_MAX {
388 return None;
389 }
390 Some(x)
392 }
393
394 fn is_fixable(self) -> bool {
399 use crate::range::u128::RUBY_FIXNUM_MAX;
400
401 (..=RUBY_FIXNUM_MAX).contains(&self)
402 }
403}
404
405impl Fixable for f32 {
406 fn to_fix(self) -> Option<i64> {
424 if let Ok(d) = Duration::try_from_secs_f32(self) {
425 let x = d.as_secs();
426 return x.to_fix();
427 }
428 if let Ok(d) = Duration::try_from_secs_f32(-self) {
429 let x = d.as_secs();
430 let x = i64::try_from(x).ok()?;
431 let x = x.checked_neg()?;
432 return x.to_fix();
433 }
434 None
435 }
436}
437
438impl Fixable for f64 {
439 fn to_fix(self) -> Option<i64> {
457 if let Ok(d) = Duration::try_from_secs_f64(self) {
458 let x = d.as_secs();
459 return x.to_fix();
460 }
461 if let Ok(d) = Duration::try_from_secs_f64(-self) {
462 let x = d.as_secs();
463 let x = i64::try_from(x).ok()?;
464 let x = x.checked_neg()?;
465 return x.to_fix();
466 }
467 None
468 }
469}
470
471#[must_use]
520#[allow(non_snake_case)] pub fn RB_FIXABLE<T: Fixable>(x: T) -> bool {
522 x.is_fixable()
523}
524
525#[cfg(test)]
526mod tests {
527 use super::*;
528
529 #[test]
530 fn test_all_i8_are_fixable() {
531 for x in i8::MIN..=i8::MAX {
532 assert_eq!(x.to_fix(), Some(x.into()), "{x} should be its own fixnum");
533 assert!(x.is_fixable(), "{x} should be fixable");
534 assert!(RB_FIXABLE(x), "{x} should be fixable");
535 }
536 }
537
538 #[test]
539 fn test_all_i16_are_fixable() {
540 for x in i16::MIN..=i16::MAX {
541 assert_eq!(x.to_fix(), Some(x.into()), "{x} should be its own fixnum");
542 assert!(x.is_fixable(), "{x} should be fixable");
543 assert!(RB_FIXABLE(x), "{x} should be fixable");
544 }
545 }
546
547 #[test]
548 fn test_i32_are_fixable() {
549 let test_cases = [i32::MIN, -1, 0, i32::MAX];
550 for x in test_cases {
551 assert_eq!(x.to_fix(), Some(x.into()), "{x} did not fix correctly");
552 assert!(x.is_fixable(), "{x} did not is_fixable correctly");
553 assert!(RB_FIXABLE(x), "{x} did not RB_FIXABLE correctly");
554 }
555 }
556
557 #[test]
558 fn test_i64_are_fixable() {
559 let test_cases = [
560 (i64::MIN, None),
561 (RUBY_FIXNUM_MIN - 1, None),
562 (RUBY_FIXNUM_MIN, Some(RUBY_FIXNUM_MIN)),
563 (RUBY_FIXNUM_MIN + 1, Some(RUBY_FIXNUM_MIN + 1)),
564 (-4_611_686_018_427_387_904 - 1, None),
569 (-4_611_686_018_427_387_904, Some(-4_611_686_018_427_387_904)),
570 (-4_611_686_018_427_387_904 + 1, Some(-4_611_686_018_427_387_903)),
571 (-1024, Some(-1024)),
572 (-10, Some(-10)),
573 (-1, Some(-1)),
574 (0_i64, Some(0)),
575 (1, Some(1)),
576 (10, Some(10)),
577 (1024, Some(1024)),
578 (4_611_686_018_427_387_903 - 1, Some(4_611_686_018_427_387_902)),
583 (4_611_686_018_427_387_903, Some(4_611_686_018_427_387_903)),
584 (4_611_686_018_427_387_903 + 1, None),
585 (RUBY_FIXNUM_MAX - 1, Some(RUBY_FIXNUM_MAX - 1)),
586 (RUBY_FIXNUM_MAX, Some(RUBY_FIXNUM_MAX)),
587 (RUBY_FIXNUM_MAX + 1, None),
588 (i64::MAX, None),
589 ];
590 for (x, fixed) in test_cases {
591 assert_eq!(x.to_fix(), fixed, "{x} did not fix correctly");
592 assert_eq!(x.is_fixable(), fixed.is_some(), "{x} did not is_fixable correctly");
593 assert_eq!(RB_FIXABLE(x), fixed.is_some(), "{x} did not RB_FIXABLE correctly");
594 }
595 }
596
597 #[test]
598 fn test_i128_are_fixable() {
599 let test_cases = [
600 (i128::MIN, None),
601 (i64::MIN.into(), None),
602 (i128::from(RUBY_FIXNUM_MIN) - 1, None),
603 (i128::from(RUBY_FIXNUM_MIN), Some(RUBY_FIXNUM_MIN)),
604 (i128::from(RUBY_FIXNUM_MIN) + 1, Some(RUBY_FIXNUM_MIN + 1)),
605 (-4_611_686_018_427_387_904 - 1, None),
610 (-4_611_686_018_427_387_904, Some(-4_611_686_018_427_387_904)),
611 (-4_611_686_018_427_387_904 + 1, Some(-4_611_686_018_427_387_903)),
612 (-1024, Some(-1024)),
613 (-10, Some(-10)),
614 (-1, Some(-1)),
615 (0_i128, Some(0)),
616 (1, Some(1)),
617 (10, Some(10)),
618 (1024, Some(1024)),
619 (4_611_686_018_427_387_903 - 1, Some(4_611_686_018_427_387_902)),
624 (4_611_686_018_427_387_903, Some(4_611_686_018_427_387_903)),
625 (4_611_686_018_427_387_903 + 1, None),
626 (i128::from(RUBY_FIXNUM_MAX) - 1, Some(RUBY_FIXNUM_MAX - 1)),
627 (i128::from(RUBY_FIXNUM_MAX), Some(RUBY_FIXNUM_MAX)),
628 (i128::from(RUBY_FIXNUM_MAX) + 1, None),
629 (i64::MAX.into(), None),
630 (i128::MAX, None),
631 ];
632 for (x, fixed) in test_cases {
633 assert_eq!(x.to_fix(), fixed, "{x} did not fix correctly");
634 assert_eq!(x.is_fixable(), fixed.is_some(), "{x} did not is_fixable correctly");
635 assert_eq!(RB_FIXABLE(x), fixed.is_some(), "{x} did not RB_FIXABLE correctly");
636 }
637 }
638
639 #[test]
640 fn all_u8_are_fixable() {
641 for x in u8::MIN..=u8::MAX {
642 assert_eq!(x.to_fix(), Some(x.into()), "{x} should be its own fixnum");
643 assert!(x.is_fixable(), "{x} should be fixable");
644 assert!(RB_FIXABLE(x), "{x} should be fixable");
645 }
646 }
647
648 #[test]
649 fn all_u16_are_fixable() {
650 for x in u16::MIN..=u16::MAX {
651 assert_eq!(x.to_fix(), Some(x.into()), "{x} should be its own fixnum");
652 assert!(x.is_fixable(), "{x} should be fixable");
653 assert!(RB_FIXABLE(x), "{x} should be fixable");
654 }
655 }
656
657 #[test]
658 fn test_u32_are_fixable() {
659 let test_cases = [0, u32::MAX / 2, u32::MAX];
660 for x in test_cases {
661 assert_eq!(x.to_fix(), Some(x.into()), "{x} did not fix correctly");
662 assert!(x.is_fixable(), "{x} did not is_fixable correctly");
663 assert!(RB_FIXABLE(x), "{x} did not RB_FIXABLE correctly");
664 }
665 }
666
667 #[test]
668 fn test_u64_are_fixable() {
669 let test_cases = [
670 (0_u64, Some(0_i64)), (1_u64, Some(1)), (4_611_686_018_427_387_903_u64, Some(4_611_686_018_427_387_903)), (4_611_686_018_427_387_904_u64, None), (4_611_686_018_427_387_905_u64, None), (9_223_372_036_854_775_806_u64, None), (9_223_372_036_854_775_807_u64, None), (18_446_744_073_709_551_614_u64, None), (18_446_744_073_709_551_615_u64, None), (9_223_372_036_854_775_809_u64, None), ];
681 for (x, fixed) in test_cases {
682 assert_eq!(x.to_fix(), fixed, "{x} did not fix correctly");
683 assert_eq!(x.is_fixable(), fixed.is_some(), "{x} did not is_fixable correctly");
684 assert_eq!(RB_FIXABLE(x), fixed.is_some(), "{x} did not RB_FIXABLE correctly");
685 }
686 }
687
688 #[test]
689 fn test_u128_are_fixable() {
690 let test_cases = [
691 (u128::MIN, Some(0)),
692 (0_u128, Some(0)),
693 (1, Some(1)),
694 (10, Some(10)),
695 (1024, Some(1024)),
696 (4_611_686_018_427_387_903 - 1, Some(4_611_686_018_427_387_902)),
701 (4_611_686_018_427_387_903, Some(4_611_686_018_427_387_903)),
702 (4_611_686_018_427_387_903 + 1, None),
703 (u128::try_from(RUBY_FIXNUM_MAX).unwrap() - 1, Some(RUBY_FIXNUM_MAX - 1)),
704 (u128::try_from(RUBY_FIXNUM_MAX).unwrap(), Some(RUBY_FIXNUM_MAX)),
705 (u128::try_from(RUBY_FIXNUM_MAX).unwrap() + 1, None),
706 (i64::MAX.try_into().unwrap(), None),
707 (u128::MAX, None),
708 ];
709 for (x, fixed) in test_cases {
710 assert_eq!(x.to_fix(), fixed, "{x} did not fix correctly");
711 assert_eq!(x.is_fixable(), fixed.is_some(), "{x} did not is_fixable correctly");
712 assert_eq!(RB_FIXABLE(x), fixed.is_some(), "{x} did not RB_FIXABLE correctly");
713 }
714 }
715
716 #[test]
717 #[expect(clippy::approx_constant, clippy::cast_precision_loss, reason = "test values")]
718 fn test_f32_are_fixable() {
719 let test_cases = [
720 (0.0, Some(0)),
722 (123.45, Some(123)),
723 (-987.65, Some(-987)),
724 (1e38, None), (-1e38, None), (1.234e-18, Some(0)), (-0.0, Some(0)), (f32::NAN, None), (f32::INFINITY, None), (f32::EPSILON, Some(0)), (f32::NEG_INFINITY, None), (f32::MIN_POSITIVE, Some(0)), (f32::MAX, None), (f32::MIN, None), (i64::MIN as _, None), (-4.611_686e18, Some(-4_611_686_018_427_387_904)), (i64::MAX as _, None), (4.611_685_5e18, Some(4_611_685_468_671_574_016)), (1.99, Some(1)), (3.14159, Some(3)), (-2.71828, Some(-2)), ];
747 for (x, fixed) in test_cases {
748 assert_eq!(x.to_fix(), fixed, "{x} did not fix correctly");
749 assert_eq!(x.is_fixable(), fixed.is_some(), "{x} did not is_fixable correctly");
750 assert_eq!(RB_FIXABLE(x), fixed.is_some(), "{x} did not RB_FIXABLE correctly");
751 }
752 }
753
754 #[test]
755 #[expect(clippy::approx_constant, clippy::cast_precision_loss, reason = "test values")]
756 fn test_f64_are_fixable() {
757 let test_cases = [
758 (0.0, Some(0)),
760 (123.45, Some(123)),
761 (-987.65, Some(-987)),
762 (1e38, None), (-1e38, None), (1.234e-18, Some(0)), (-0.0, Some(0)), (f64::NAN, None), (f64::INFINITY, None), (f64::EPSILON, Some(0)), (f64::NEG_INFINITY, None), (f64::MIN_POSITIVE, Some(0)), (f64::MAX, None), (f64::MIN, None), (i64::MIN as _, None), (-4.611_686_018_427_387e18, Some(-4_611_686_018_427_386_880)), (i64::MAX as _, None), (4.611_686_018_427_387e18, Some(4_611_686_018_427_386_880)), (1.99, Some(1)), (3.14159, Some(3)), (-2.71828, Some(-2)), ];
785 for (x, fixed) in test_cases {
786 assert_eq!(x.to_fix(), fixed, "{x} did not fix correctly");
787 assert_eq!(x.is_fixable(), fixed.is_some(), "{x} did not is_fixable correctly");
788 assert_eq!(RB_FIXABLE(x), fixed.is_some(), "{x} did not RB_FIXABLE correctly");
789 }
790 }
791
792 #[test]
793 fn test_fixable_boundary_values_u8() {
794 assert!(0_u8.is_fixable());
795 assert!(255_u8.is_fixable());
796 }
797
798 #[test]
799 fn test_to_fix_boundary_values_u8() {
800 assert_eq!(0_u8.to_fix(), Some(0));
801 assert_eq!(255_u8.to_fix(), Some(255));
802 }
803
804 #[test]
805 fn test_fixable_boundary_values_i8() {
806 assert!(0_i8.is_fixable());
807 assert!(127_i8.is_fixable());
808 assert!((-128_i8).is_fixable());
809 }
810
811 #[test]
812 fn test_to_fix_boundary_values_i8() {
813 assert_eq!(0_i8.to_fix(), Some(0));
814 assert_eq!(127_i8.to_fix(), Some(127));
815 assert_eq!((-128_i8).to_fix(), Some(-128));
816 }
817
818 #[test]
819 fn test_fixable_boundary_values_u16() {
820 assert!(0_u16.is_fixable());
821 assert!(65_535_u16.is_fixable());
822 }
823
824 #[test]
825 fn test_to_fix_boundary_values_u16() {
826 assert_eq!(0_u16.to_fix(), Some(0));
827 assert_eq!(65_535_u16.to_fix(), Some(65_535));
828 }
829
830 #[test]
831 fn test_fixable_boundary_values_i16() {
832 assert!(0_i16.is_fixable());
833 assert!(32_767_i16.is_fixable());
834 assert!((-32_768_i16).is_fixable());
835 }
836
837 #[test]
838 fn test_to_fix_boundary_values_i16() {
839 assert_eq!(0_i16.to_fix(), Some(0));
840 assert_eq!(32_767_i16.to_fix(), Some(32_767));
841 assert_eq!((-32_768_i16).to_fix(), Some(-32_768));
842 }
843
844 #[test]
845 fn test_fixable_boundary_values_u32() {
846 assert!(0_u32.is_fixable());
847 assert!(4_294_967_295_u32.is_fixable());
848 }
849
850 #[test]
851 fn test_to_fix_boundary_values_u32() {
852 assert_eq!(0_u32.to_fix(), Some(0));
853 assert_eq!(4_294_967_295_u32.to_fix(), Some(4_294_967_295));
854 }
855
856 #[test]
857 fn test_fixable_boundary_values_i32() {
858 assert!(0_i32.is_fixable());
859 assert!(2_147_483_647_i32.is_fixable());
860 assert!((-2_147_483_648_i32).is_fixable());
861 }
862
863 #[test]
864 fn test_to_fix_boundary_values_i32() {
865 assert_eq!(0_i32.to_fix(), Some(0));
866 assert_eq!(2_147_483_647_i32.to_fix(), Some(2_147_483_647));
867 assert_eq!((-2_147_483_648_i32).to_fix(), Some(-2_147_483_648));
868 }
869
870 #[test]
871 fn test_fixable_boundary_values_u64() {
872 assert!(0_u64.is_fixable());
873 assert!((u64::MAX >> 2).is_fixable());
874 assert!(!(u64::MAX >> 1).is_fixable());
875 assert!(!u64::MAX.is_fixable());
876 }
877
878 #[test]
879 fn test_to_fix_boundary_values_u64() {
880 assert_eq!(0_u64.to_fix(), Some(0));
881 assert_eq!((u64::MAX >> 2).to_fix(), Some(i64::MAX / 2));
882 assert_eq!((u64::MAX >> 1).to_fix(), None);
883 assert_eq!(u64::MAX.to_fix(), None);
884 }
885
886 #[test]
887 fn test_fixable_boundary_values_i64() {
888 assert!(0_i64.is_fixable());
889 assert!((i64::MIN >> 1).is_fixable());
890 assert!((i64::MAX >> 1).is_fixable());
891 assert!(!i64::MIN.is_fixable());
892 assert!(!i64::MAX.is_fixable());
893 }
894
895 #[test]
896 fn test_to_fix_boundary_values_i64() {
897 assert_eq!(0_i64.to_fix(), Some(0));
898 assert_eq!((i64::MAX >> 1).to_fix(), Some(i64::MAX / 2));
899 assert_eq!((i64::MIN >> 1).to_fix(), Some(i64::MIN / 2));
900 assert_eq!(i64::MIN.to_fix(), None);
901 assert_eq!(i64::MAX.to_fix(), None);
902 }
903
904 #[test]
905 fn test_fixable_boundary_values_u128() {
906 assert!(0_u64.is_fixable());
907 assert!((u128::MAX >> 66).is_fixable());
908 assert!(!(u128::MAX >> 65).is_fixable());
909 assert!(!(u128::MAX >> 10).is_fixable());
910 assert!(!(u128::MAX >> 2).is_fixable());
911 assert!(!(u128::MAX >> 1).is_fixable());
912 assert!(!u128::MAX.is_fixable());
913 }
914
915 #[test]
916 fn test_to_fix_boundary_values_u128() {
917 assert_eq!(0_u64.to_fix(), Some(0));
918 assert_eq!((u128::MAX >> 66).to_fix(), Some(i64::MAX / 2));
919 assert_eq!((u128::MAX >> 65).to_fix(), None);
920 assert_eq!((u128::MAX >> 10).to_fix(), None);
921 assert_eq!((u128::MAX >> 2).to_fix(), None);
922 assert_eq!((u128::MAX >> 1).to_fix(), None);
923 assert_eq!(u128::MAX.to_fix(), None);
924 }
925
926 #[test]
927 fn test_fixable_boundary_values_i128() {
928 assert!(0_i128.is_fixable());
929 assert!((i128::MIN >> 65).is_fixable());
930 assert!((i128::MAX >> 65).is_fixable());
931 assert!(!(i128::MIN >> 1).is_fixable());
932 assert!(!(i128::MAX >> 1).is_fixable());
933 assert!(!i128::MIN.is_fixable());
934 assert!(!i128::MAX.is_fixable());
935 }
936
937 #[test]
938 fn test_to_fix_boundary_values_i128() {
939 assert_eq!(0_i128.to_fix(), Some(0));
940 assert_eq!((i128::MAX >> 65).to_fix(), Some(i64::MAX / 2));
941 assert_eq!((i128::MIN >> 65).to_fix(), Some(i64::MIN / 2));
942 assert_eq!((i128::MAX >> 1).to_fix(), None);
943 assert_eq!((i128::MIN >> 1).to_fix(), None);
944 assert_eq!(i128::MIN.to_fix(), None);
945 assert_eq!(i128::MAX.to_fix(), None);
946 }
947}