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
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
use std::time::Duration;

use super::{Datagram, OperationGenerator};
use crate::{
    derive::*,
    firmware::{
        fpga::{Segment, TransitionMode},
        operation::Operation,
    },
};

use derive_more::Deref;

#[derive(Builder, Clone, Deref)]

pub struct DatagramWithSegmentTransition<D: DatagramST> {
    #[deref]
    datagram: D,
    #[get]
    segment: Segment,
    #[get]
    transition_mode: Option<TransitionMode>,
}

impl<D: DatagramST> Datagram for DatagramWithSegmentTransition<D> {
    type O1 = D::O1;
    type O2 = D::O2;
    type G = D::G;

    fn operation_generator(self, geometry: &Geometry) -> Result<Self::G, AUTDInternalError> {
        self.datagram
            .operation_generator_with_segment(geometry, self.segment, self.transition_mode)
    }

    fn timeout(&self) -> Option<Duration> {
        self.datagram.timeout()
    }

    fn parallel_threshold(&self) -> Option<usize> {
        self.datagram.parallel_threshold()
    }
}

impl<D: DatagramST> Datagram for D {
    type O1 = D::O1;
    type O2 = D::O2;
    type G = D::G;

    fn operation_generator(self, geometry: &Geometry) -> Result<Self::G, AUTDInternalError> {
        self.operation_generator_with_segment(
            geometry,
            Segment::S0,
            Some(TransitionMode::Immediate),
        )
    }

    fn timeout(&self) -> Option<Duration> {
        <Self as DatagramST>::timeout(self)
    }

    fn parallel_threshold(&self) -> Option<usize> {
        <Self as DatagramST>::parallel_threshold(self)
    }
}

pub trait DatagramST {
    type O1: Operation;
    type O2: Operation;
    type G: OperationGenerator<O1 = Self::O1, O2 = Self::O2>;

    fn operation_generator_with_segment(
        self,
        geometry: &Geometry,
        segment: Segment,
        transition_mode: Option<TransitionMode>,
    ) -> Result<Self::G, AUTDInternalError>;

    fn timeout(&self) -> Option<Duration>;

    fn parallel_threshold(&self) -> Option<usize> {
        None
    }
}

pub trait IntoDatagramWithSegmentTransition<D: DatagramST> {
    fn with_segment(
        self,
        segment: Segment,
        transition_mode: Option<TransitionMode>,
    ) -> DatagramWithSegmentTransition<D>;
}

impl<D: DatagramST> IntoDatagramWithSegmentTransition<D> for D {
    fn with_segment(
        self,
        segment: Segment,
        transition_mode: Option<TransitionMode>,
    ) -> DatagramWithSegmentTransition<D> {
        DatagramWithSegmentTransition {
            datagram: self,
            segment,
            transition_mode,
        }
    }
}

#[cfg(feature = "capi")]
impl<D: DatagramST + Default> Default for DatagramWithSegmentTransition<D> {
    fn default() -> Self {
        Self {
            datagram: D::default(),
            segment: Segment::default(),
            transition_mode: None,
        }
    }
}