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
117
118
119
120
121
122
123
124
125
use crate::common::Reg;
use embedded_hal::blocking::{delay, i2c};

/// Blanket trait for something that implements I2C bus operations, with a
/// combined Error associated type
pub trait I2cDriver: i2c::Write + i2c::WriteRead + i2c::Read {
    type I2cError: From<<Self as i2c::Write>::Error>
        + From<<Self as i2c::WriteRead>::Error>
        + From<<Self as i2c::Read>::Error>;
}

impl<T, E> I2cDriver for T
where
    T: i2c::Write<Error = E> + i2c::WriteRead<Error = E> + i2c::Read<Error = E>,
{
    type I2cError = E;
}

pub trait Driver: I2cDriver + delay::DelayUs<u32> {}
impl<T> Driver for T where T: I2cDriver + delay::DelayUs<u32> {}

pub trait DriverExt {
    type Error;

    fn register_read<const N: usize>(
        &mut self,
        addr: i2c::SevenBitAddress,
        reg: &Reg,
    ) -> Result<[u8; N], Self::Error>;

    fn register_write<const N: usize>(
        &mut self,
        addr: i2c::SevenBitAddress,
        reg: &Reg,
        bytes: &[u8; N],
    ) -> Result<(), Self::Error>
    where
        [(); N + 2]: Sized;

    fn read_u8(&mut self, addr: i2c::SevenBitAddress, reg: &Reg) -> Result<u8, Self::Error> {
        self.register_read::<1>(addr, reg).map(|buf| buf[0])
    }

    fn read_i32(&mut self, addr: i2c::SevenBitAddress, reg: &Reg) -> Result<i32, Self::Error> {
        self.register_read::<4>(addr, reg).map(i32::from_be_bytes)
    }

    fn read_u16(&mut self, addr: i2c::SevenBitAddress, reg: &Reg) -> Result<u16, Self::Error> {
        self.register_read::<2>(addr, reg).map(u16::from_be_bytes)
    }

    fn read_u32(&mut self, addr: i2c::SevenBitAddress, reg: &Reg) -> Result<u32, Self::Error> {
        self.register_read::<4>(addr, reg).map(u32::from_be_bytes)
    }

    fn write_u8(
        &mut self,
        addr: i2c::SevenBitAddress,
        reg: &Reg,
        value: u8,
    ) -> Result<(), Self::Error> {
        self.register_write(addr, reg, &[value])
    }

    fn write_u16(
        &mut self,
        addr: i2c::SevenBitAddress,
        reg: &Reg,
        value: u16,
    ) -> Result<(), Self::Error> {
        self.register_write(addr, reg, &u16::to_be_bytes(value))
    }

    fn write_i32(
        &mut self,
        addr: i2c::SevenBitAddress,
        reg: &Reg,
        value: i32,
    ) -> Result<(), Self::Error> {
        self.register_write(addr, reg, &i32::to_be_bytes(value))
    }

    fn write_u32(
        &mut self,
        addr: i2c::SevenBitAddress,
        reg: &Reg,
        value: u32,
    ) -> Result<(), Self::Error> {
        self.register_write(addr, reg, &u32::to_be_bytes(value))
    }
}

impl<T: Driver> DriverExt for T {
    type Error = T::I2cError;

    fn register_read<const N: usize>(
        &mut self,
        addr: i2c::SevenBitAddress,
        reg: &Reg,
    ) -> Result<[u8; N], Self::Error> {
        let mut buffer = [0u8; N];
        self.write(addr, reg)?;
        self.delay_us(crate::DELAY_TIME);
        self.write_read(addr, &[], &mut buffer)?;
        Ok(buffer)
    }

    fn register_write<const N: usize>(
        &mut self,
        addr: i2c::SevenBitAddress,
        reg: &Reg,
        bytes: &[u8; N],
    ) -> Result<(), Self::Error>
    where
        [(); N + 2]: Sized,
    {
        let mut buffer = [0u8; N + 2];
        buffer[0..2].copy_from_slice(reg);
        buffer[2..].copy_from_slice(bytes);

        self.write(addr, &buffer)?;
        self.delay_us(crate::DELAY_TIME);
        Ok(())
    }
}