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
/*
 * File: bundle.rs
 * Project: link
 * Created Date: 14/06/2023
 * Author: Shun Suzuki
 * -----
 * Last Modified: 14/06/2023
 * Modified By: Shun Suzuki (suzuki@hapis.k.u-tokyo.ac.jp)
 * -----
 * Copyright (c) 2023 Shun Suzuki. All rights reserved.
 *
 */

use std::{marker::PhantomData, time::Duration};

use autd3_core::{
    error::AUTDInternalError,
    geometry::{Geometry, Transducer},
    link::Link,
    RxDatagram, TxDatagram,
};

pub struct Bundle<T: Transducer, L1: Link<T>, L2: Link<T>> {
    main_link: L1,
    sub_link: L2,
    _t: PhantomData<T>,
}

impl<T: Transducer, L1: Link<T>, L2: Link<T>> Bundle<T, L1, L2> {
    pub fn new(main_link: L1, sub_link: L2) -> Self {
        Self {
            main_link,
            sub_link,
            _t: PhantomData,
        }
    }
}

impl<T: Transducer, L1: Link<T>, L2: Link<T>> Link<T> for Bundle<T, L1, L2> {
    fn open(&mut self, geometry: &Geometry<T>) -> Result<(), AUTDInternalError> {
        if self.is_open() {
            return Ok(());
        }

        self.main_link.open(geometry)?;
        self.sub_link.open(geometry)?;

        Ok(())
    }

    fn close(&mut self) -> Result<(), AUTDInternalError> {
        if !self.is_open() {
            return Ok(());
        }

        self.main_link.close()?;
        self.sub_link.close()?;

        Ok(())
    }

    fn send(&mut self, tx: &TxDatagram) -> Result<bool, AUTDInternalError> {
        if !self.is_open() {
            return Ok(false);
        }

        self.main_link.send(tx)?;
        self.sub_link.send(tx)?;

        Ok(true)
    }

    fn receive(&mut self, rx: &mut RxDatagram) -> Result<bool, AUTDInternalError> {
        if !self.is_open() {
            return Ok(false);
        }

        self.main_link.receive(rx)?;

        Ok(true)
    }

    fn is_open(&self) -> bool {
        self.main_link.is_open() && self.sub_link.is_open()
    }

    fn timeout(&self) -> Duration {
        self.main_link.timeout().max(self.sub_link.timeout())
    }
}