1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
use *;

/// Minimum loop.
pub struct Min<I, T>(pub Secret<I, T>);

macro_rules! min_impl {
    ($usize:ty , $f32:ty , $nan:expr) => {
        impl Lup<$usize, $f32> for Min<$usize, $f32> {
            type Inner = Secret<$usize, $f32>;
            fn start() -> Self {Min(Secret {evidence: None, value: $nan})}
            fn it(&mut self, ind: $usize, val: $f32) -> bool {
                if self.0.value.is_nan() || val < self.0.value {
                    self.0 = Secret {evidence: Some(ind), value: val};
                }
                true
            }
            fn unwrap(self) -> Self::Inner {self.0}
        }

        impl Lup<$usize, Secret<$usize, $f32>> for Min<($usize, $usize), $f32> {
            type Inner = Secret<($usize, $usize), $f32>;
            fn start() -> Self {Min(Secret {evidence: None, value: $nan})}
            fn it(&mut self, ind: $usize, val: Secret<$usize, $f32>) -> bool {
                if self.0.value.is_nan() || val.value < self.0.value {
                    if let Some(ind2) = val.evidence {
                        self.0 = Secret {evidence: Some((ind, ind2)), value: val.value}
                    }
                }
                true
            }
            fn unwrap(self) -> Self::Inner {self.0}
        }

        impl Lup<$usize, Secret<($usize, $usize), $f32>> for Min<($usize, $usize, $usize), $f32> {
            type Inner = Secret<($usize, $usize, $usize), $f32>;
            fn start() -> Self {Min(Secret {evidence: None, value: $nan})}
            fn it(&mut self, ind: $usize, val: Secret<($usize, $usize), $f32>) -> bool {
                if self.0.value.is_nan() || val.value < self.0.value {
                    if let Some((a, b)) = val.evidence {
                        self.0 = Secret {evidence: Some((ind, a, b)), value: val.value}
                    }
                }
                true
            }
            fn unwrap(self) -> Self::Inner {self.0}
        }
    }
}

min_impl!{usize, f32, ::std::f32::NAN}

min_impl!{usize, f64, ::std::f64::NAN}