mathru/algebra/abstr/
relative_eq.rs1use crate::algebra::abstr::abs_diff_eq::AbsDiffEq;
2
3pub struct Relative<A, B = A>
21where
22 A: RelativeEq<B> + ?Sized,
23 B: ?Sized,
24{
25 pub epsilon: A::Epsilon,
27 pub max_relative: A::Epsilon,
29}
30
31impl<A, B> Default for Relative<A, B>
32where
33 A: RelativeEq<B> + ?Sized,
34 B: ?Sized,
35{
36 fn default() -> Relative<A, B> {
37 Relative {
38 epsilon: A::default_epsilon(),
39 max_relative: A::default_max_relative(),
40 }
41 }
42}
43
44impl<A, B> Relative<A, B>
45where
46 A: RelativeEq<B> + ?Sized,
47 B: ?Sized,
48{
49 pub fn epsilon(self, epsilon: A::Epsilon) -> Relative<A, B> {
51 Relative { epsilon, ..self }
52 }
53
54 pub fn max_relative(self, max_relative: A::Epsilon) -> Relative<A, B> {
56 Relative {
57 max_relative,
58 ..self
59 }
60 }
61
62 pub fn eq(self, lhs: &A, rhs: &B) -> bool {
64 A::relative_eq(lhs, rhs, self.epsilon, self.max_relative)
65 }
66
67 pub fn ne(self, lhs: &A, rhs: &B) -> bool {
69 A::relative_ne(lhs, rhs, self.epsilon, self.max_relative)
70 }
71}
72
73pub trait RelativeEq<Rhs = Self>: AbsDiffEq<Rhs>
76where
77 Rhs: ?Sized,
78{
79 fn default_max_relative() -> Self::Epsilon;
83
84 fn relative_eq(&self, other: &Rhs, epsilon: Self::Epsilon, max_relative: Self::Epsilon)
86 -> bool;
87
88 fn relative_ne(
90 &self,
91 other: &Rhs,
92 epsilon: Self::Epsilon,
93 max_relative: Self::Epsilon,
94 ) -> bool {
95 !Self::relative_eq(self, other, epsilon, max_relative)
96 }
97}
98
99macro_rules! impl_relative_eq_float {
106 ($T:ident, $U:ident) => {
107 impl RelativeEq for $T {
108 fn default_max_relative() -> $T {
109 $T::EPSILON
110 }
111
112 fn relative_eq(&self, other: &$T, epsilon: $T, max_relative: $T) -> bool {
113 if self == other {
115 return true;
116 }
117
118 if $T::is_infinite(*self) || $T::is_infinite(*other) {
120 return false;
121 }
122
123 let abs_diff = $T::abs(self - other);
124
125 if abs_diff <= epsilon {
127 return true;
128 }
129
130 let abs_self = $T::abs(*self);
131 let abs_other = $T::abs(*other);
132
133 let largest = if abs_other > abs_self {
134 abs_other
135 } else {
136 abs_self
137 };
138
139 abs_diff <= largest * max_relative
141 }
142 }
143 };
144}
145
146impl_relative_eq_float!(f32, i32);
147impl_relative_eq_float!(f64, i64);
148
149#[macro_export]
151macro_rules! relative_eq {
152 ($lhs:expr, $rhs:expr $(, $opt:ident = $val:expr)*) => {
153 $crate::algebra::abstr::Relative::default()$(.$opt($val))*.eq(&$lhs, &$rhs)
154 };
155 ($lhs:expr, $rhs:expr $(, $opt:ident = $val:expr)*,) => {
156 $crate::algebra::abstr::Relative::default()$(.$opt($val))*.eq(&$lhs, &$rhs)
157 };
158}
159
160#[macro_export]
162macro_rules! relative_ne {
163 ($lhs:expr, $rhs:expr $(, $opt:ident = $val:expr)*) => {
164 $crate::algebra::abstr::Relative::default()$(.$opt($val))*.ne(&$lhs, &$rhs)
165 };
166 ($lhs:expr, $rhs:expr $(, $opt:ident = $val:expr)*,) => {
167 $crate::algebra::abstr::Relative::default()$(.$opt($val))*.ne(&$lhs, &$rhs)
168 };
169}
170
171#[macro_export(local_inner_macros)]
173macro_rules! assert_relative_eq {
174 ($given:expr, $expected:expr $(, $opt:ident = $val:expr)*) => {
175 __assert_approx!(relative_eq, $given, $expected $(, $opt = $val)*)
176 };
177 ($given:expr, $expected:expr $(, $opt:ident = $val:expr)*,) => {
178 __assert_approx!(relative_eq, $given, $expected $(, $opt = $val)*)
179 };
180}
181
182#[macro_export(local_inner_macros)]
184macro_rules! assert_relative_ne {
185 ($given:expr, $expected:expr $(, $opt:ident = $val:expr)*) => {
186 __assert_approx!(relative_ne, $given, $expected $(, $opt = $val)*)
187 };
188 ($given:expr, $expected:expr $(, $opt:ident = $val:expr)*,) => {
189 __assert_approx!(relative_ne, $given, $expected $(, $opt = $val)*)
190 };
191}