autd3-driver 38.1.0

AUTD3 driver
Documentation
use std::time::Duration;

use super::SilencerControlFlags;
use crate::{
    datagram::FixedCompletionTime,
    error::AUTDDriverError,
    firmware::{
        operation::{Operation, OperationGenerator, implement::null::NullOp},
        tag::TypeTag,
    },
};

use autd3_core::{common::ULTRASOUND_FREQ, geometry::Device};

#[repr(C, align(2))]
struct SilencerFixedCompletionTime {
    tag: TypeTag,
    flag: SilencerControlFlags,
    value_intensity: u16,
    value_phase: u16,
}

pub struct FixedCompletionTimeOp {
    is_done: bool,
    intensity: Duration,
    phase: Duration,
    strict: bool,
}

impl FixedCompletionTimeOp {
    pub(crate) const fn new(intensity: Duration, phase: Duration, strict: bool) -> Self {
        Self {
            is_done: false,
            intensity,
            phase,
            strict,
        }
    }
}

impl Operation<'_> for FixedCompletionTimeOp {
    type Error = AUTDDriverError;

    fn pack(&mut self, _: &Device, tx: &mut [u8]) -> Result<usize, AUTDDriverError> {
        let validate = |value: Duration| {
            const NANOSEC: u128 = 1_000_000_000;
            let v = value.as_nanos() * ULTRASOUND_FREQ.hz() as u128;
            let v = if v.is_multiple_of(NANOSEC) {
                v / NANOSEC
            } else {
                return Err(AUTDDriverError::InvalidSilencerCompletionTime(value));
            };
            if v == 0 || v > u16::MAX as u128 {
                return Err(AUTDDriverError::SilencerCompletionTimeOutOfRange(value));
            }
            Ok(v as u16)
        };
        let step_intensity = validate(self.intensity)?;
        let step_phase = validate(self.phase)?;

        crate::firmware::operation::write_to_tx(
            tx,
            SilencerFixedCompletionTime {
                tag: TypeTag::Silencer,
                flag: if self.strict {
                    SilencerControlFlags::STRICT_MODE
                } else {
                    SilencerControlFlags::NONE
                },
                value_intensity: step_intensity,
                value_phase: step_phase,
            },
        );

        self.is_done = true;
        Ok(std::mem::size_of::<SilencerFixedCompletionTime>())
    }

    fn required_size(&self, _: &Device) -> usize {
        std::mem::size_of::<SilencerFixedCompletionTime>()
    }

    fn is_done(&self) -> bool {
        self.is_done
    }
}

impl OperationGenerator<'_> for FixedCompletionTime {
    type O1 = FixedCompletionTimeOp;
    type O2 = NullOp;

    fn generate(&mut self, _: &Device) -> Option<(Self::O1, Self::O2)> {
        Some((
            Self::O1::new(self.intensity, self.phase, self.strict),
            Self::O2 {},
        ))
    }
}

#[cfg(test)]
mod tests {
    use std::mem::size_of;

    use super::*;
    use crate::common::ULTRASOUND_PERIOD;

    #[rstest::rstest]
    #[case(SilencerControlFlags::STRICT_MODE.0, true)]
    #[case(0x00, false)]
    fn test(#[case] value: u8, #[case] strict: bool) {
        let device = crate::tests::create_device();

        let mut tx = [0x00u8; size_of::<SilencerFixedCompletionTime>()];

        let mut op =
            FixedCompletionTimeOp::new(ULTRASOUND_PERIOD * 0x12, ULTRASOUND_PERIOD * 0x34, strict);

        assert_eq!(
            op.required_size(&device),
            size_of::<SilencerFixedCompletionTime>()
        );
        assert!(!op.is_done());

        assert!(op.pack(&device, &mut tx).is_ok());

        assert!(op.is_done());

        assert_eq!(tx[0], TypeTag::Silencer as u8);
        assert_eq!(tx[1], value);
        assert_eq!(tx[2], 0x12);
        assert_eq!(tx[3], 0x00);
        assert_eq!(tx[4], 0x34);
        assert_eq!(tx[5], 0x00);
    }

    #[rstest::rstest]
    #[case(
        AUTDDriverError::SilencerCompletionTimeOutOfRange(Duration::from_micros(0)),
        Duration::from_micros(0),
        Duration::from_micros(25)
    )]
    #[case(
        AUTDDriverError::SilencerCompletionTimeOutOfRange(Duration::from_micros(25 * 65536)),
        Duration::from_micros(25 * 65536),
        Duration::from_micros(25)
    )]
    #[case(
        AUTDDriverError::SilencerCompletionTimeOutOfRange(Duration::from_micros(0)),
        Duration::from_micros(25),
        Duration::from_micros(0)
    )]
    #[case(
        AUTDDriverError::SilencerCompletionTimeOutOfRange(Duration::from_micros(25 * 65536)),
        Duration::from_micros(25),
        Duration::from_micros(25 * 65536),
    )]
    #[case(
        AUTDDriverError::InvalidSilencerCompletionTime(Duration::from_micros(26)),
        Duration::from_micros(26),
        Duration::from_micros(50)
    )]
    #[case(
        AUTDDriverError::InvalidSilencerCompletionTime(Duration::from_micros(51)),
        Duration::from_micros(25),
        Duration::from_micros(51)
    )]
    fn invalid_time(
        #[case] expected: AUTDDriverError,
        #[case] time_intensity: Duration,
        #[case] time_phase: Duration,
    ) {
        let device = crate::tests::create_device();

        let mut tx = [0x00u8; size_of::<SilencerFixedCompletionTime>()];

        let mut op = FixedCompletionTimeOp::new(time_intensity, time_phase, true);

        assert_eq!(expected, op.pack(&device, &mut tx).unwrap_err());
    }
}