1use super::Ulps;
5
6pub trait ApproxEqUlps {
11 type Flt: Ulps;
12
13 fn approx_eq_ulps(&self, other: &Self, ulps: <Self::Flt as Ulps>::U) -> bool;
19
20 #[inline]
26 fn approx_ne_ulps(&self, other: &Self, ulps: <Self::Flt as Ulps>::U) -> bool {
27 !self.approx_eq_ulps(other, ulps)
28 }
29}
30
31impl ApproxEqUlps for f32 {
32 type Flt = f32;
33
34 fn approx_eq_ulps(&self, other: &f32, ulps: i32) -> bool {
35 if *self==*other { return true; }
38
39 if self.is_sign_positive() != other.is_sign_positive() { return false; }
43
44 let diff: i32 = self.ulps(other);
45 diff >= -ulps && diff <= ulps
46 }
47}
48
49#[test]
50fn f32_approx_eq_ulps_test1() {
51 let f: f32 = 0.1_f32;
52 let mut sum: f32 = 0.0_f32;
53 for _ in 0_isize..10_isize { sum += f; }
54 let product: f32 = f * 10.0_f32;
55 assert!(sum != product); assert!(sum.approx_eq_ulps(&product,1) == true); assert!(sum.approx_eq_ulps(&product,0) == false);
58}
59#[test]
60fn f32_approx_eq_ulps_test2() {
61 let x: f32 = 1000000_f32;
62 let y: f32 = 1000000.1_f32;
63 assert!(x != y); assert!(x.approx_eq_ulps(&y,2) == true);
65 assert!(x.approx_eq_ulps(&y,1) == false);
66}
67#[test]
68fn f32_approx_eq_ulps_test_zeroes() {
69 let x: f32 = 0.0_f32;
70 let y: f32 = -0.0_f32;
71 assert!(x.approx_eq_ulps(&y,0) == true);
72}
73
74impl ApproxEqUlps for f64 {
75 type Flt = f64;
76
77 fn approx_eq_ulps(&self, other: &f64, ulps: i64) -> bool {
78 if *self==*other { return true; }
81
82 if self.is_sign_positive() != other.is_sign_positive() { return false; }
86
87 let diff: i64 = self.ulps(other);
88 diff >= -ulps && diff <= ulps
89 }
90}
91
92#[test]
93fn f64_approx_eq_ulps_test1() {
94 let f: f64 = 0.1_f64;
95 let mut sum: f64 = 0.0_f64;
96 for _ in 0_isize..10_isize { sum += f; }
97 let product: f64 = f * 10.0_f64;
98 assert!(sum != product); assert!(sum.approx_eq_ulps(&product,1) == true); assert!(sum.approx_eq_ulps(&product,0) == false);
101}
102#[test]
103fn f64_approx_eq_ulps_test2() {
104 let x: f64 = 1000000_f64;
105 let y: f64 = 1000000.0000000003_f64;
106 assert!(x != y); assert!(x.approx_eq_ulps(&y,3) == true);
108 assert!(x.approx_eq_ulps(&y,2) == false);
109}
110#[test]
111fn f64_approx_eq_ulps_test_zeroes() {
112 let x: f64 = 0.0_f64;
113 let y: f64 = -0.0_f64;
114 assert!(x.approx_eq_ulps(&y,0) == true);
115}