cmp.rs 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. #![allow(unused_macros)]
  2. use testcrate::*;
  3. macro_rules! cmp {
  4. ($x:ident, $y:ident, $($unordered_val:expr, $fn:ident);*;) => {
  5. $(
  6. let cmp0 = if $x.is_nan() || $y.is_nan() {
  7. $unordered_val
  8. } else if $x < $y {
  9. -1
  10. } else if $x == $y {
  11. 0
  12. } else {
  13. 1
  14. };
  15. let cmp1 = $fn($x, $y);
  16. if cmp0 != cmp1 {
  17. panic!("{}({}, {}): std: {}, builtins: {}", stringify!($fn_builtins), $x, $y, cmp0, cmp1);
  18. }
  19. )*
  20. };
  21. }
  22. #[test]
  23. fn float_comparisons() {
  24. use compiler_builtins::float::cmp::{
  25. __eqdf2, __eqsf2, __gedf2, __gesf2, __gtdf2, __gtsf2, __ledf2, __lesf2, __ltdf2, __ltsf2,
  26. __nedf2, __nesf2, __unorddf2, __unordsf2,
  27. };
  28. fuzz_float_2(N, |x: f32, y: f32| {
  29. assert_eq!(__unordsf2(x, y) != 0, x.is_nan() || y.is_nan());
  30. cmp!(x, y,
  31. 1, __ltsf2;
  32. 1, __lesf2;
  33. 1, __eqsf2;
  34. -1, __gesf2;
  35. -1, __gtsf2;
  36. 1, __nesf2;
  37. );
  38. });
  39. fuzz_float_2(N, |x: f64, y: f64| {
  40. assert_eq!(__unorddf2(x, y) != 0, x.is_nan() || y.is_nan());
  41. cmp!(x, y,
  42. 1, __ltdf2;
  43. 1, __ledf2;
  44. 1, __eqdf2;
  45. -1, __gedf2;
  46. -1, __gtdf2;
  47. 1, __nedf2;
  48. );
  49. });
  50. }
  51. macro_rules! cmp2 {
  52. ($x:ident, $y:ident, $($unordered_val:expr, $fn_std:expr, $fn_builtins:ident);*;) => {
  53. $(
  54. let cmp0: i32 = if $x.is_nan() || $y.is_nan() {
  55. $unordered_val
  56. } else {
  57. $fn_std as i32
  58. };
  59. let cmp1: i32 = $fn_builtins($x, $y);
  60. if cmp0 != cmp1 {
  61. panic!("{}({}, {}): std: {}, builtins: {}", stringify!($fn_builtins), $x, $y, cmp0, cmp1);
  62. }
  63. )*
  64. };
  65. }
  66. #[cfg(target_arch = "arm")]
  67. #[test]
  68. fn float_comparisons_arm() {
  69. use compiler_builtins::float::cmp::{
  70. __aeabi_dcmpeq, __aeabi_dcmpge, __aeabi_dcmpgt, __aeabi_dcmple, __aeabi_dcmplt,
  71. __aeabi_fcmpeq, __aeabi_fcmpge, __aeabi_fcmpgt, __aeabi_fcmple, __aeabi_fcmplt, __eqdf2vfp,
  72. __eqsf2vfp, __gedf2vfp, __gesf2vfp, __gtdf2vfp, __gtsf2vfp, __ledf2vfp, __lesf2vfp,
  73. __ltdf2vfp, __ltsf2vfp, __nedf2vfp, __nesf2vfp,
  74. };
  75. fuzz_float_2(N, |x: f32, y: f32| {
  76. cmp2!(x, y,
  77. 0, x < y, __aeabi_fcmplt;
  78. 0, x <= y, __aeabi_fcmple;
  79. 0, x == y, __aeabi_fcmpeq;
  80. 0, x >= y, __aeabi_fcmpge;
  81. 0, x > y, __aeabi_fcmpgt;
  82. 0, x < y, __ltsf2vfp;
  83. 0, x <= y, __lesf2vfp;
  84. 0, x == y, __eqsf2vfp;
  85. 0, x >= y, __gesf2vfp;
  86. 0, x > y, __gtsf2vfp;
  87. 1, x != y, __nesf2vfp;
  88. );
  89. });
  90. fuzz_float_2(N, |x: f64, y: f64| {
  91. cmp2!(x, y,
  92. 0, x < y, __aeabi_dcmplt;
  93. 0, x <= y, __aeabi_dcmple;
  94. 0, x == y, __aeabi_dcmpeq;
  95. 0, x >= y, __aeabi_dcmpge;
  96. 0, x > y, __aeabi_dcmpgt;
  97. 0, x < y, __ltdf2vfp;
  98. 0, x <= y, __ledf2vfp;
  99. 0, x == y, __eqdf2vfp;
  100. 0, x >= y, __gedf2vfp;
  101. 0, x > y, __gtdf2vfp;
  102. 1, x != y, __nedf2vfp;
  103. );
  104. });
  105. }