use embedded_hal::blocking::spi::{Transfer, Write};
use embedded_hal::digital::v2::{OutputPin, InputPin};
use crate::{Transaction, Transactional, PinState, Error};
#[derive(Debug, Clone, PartialEq)]
pub struct Wrapper<Spi, SpiError, OutputPin, InputPin, PinError> {
spi: Spi,
cs: OutputPin,
busy: Option<InputPin>,
pub(crate) err: Option<Error<SpiError, PinError>>,
}
impl <Spi, SpiError, Output, Input, PinError> Wrapper<Spi, SpiError, Output, Input, PinError>
where
Spi: Transfer<u8, Error = SpiError> + Write<u8, Error = SpiError>,
Output: OutputPin<Error = PinError>,
Input: InputPin<Error = PinError>,
{
pub fn new(spi: Spi, cs: Output) -> Self {
Self{spi, cs, err: None, busy: None}
}
pub fn with_busy(s: &mut Self, busy: Input) -> &mut Self {
s.busy = Some(busy);
s
}
pub fn pin_write<P>(&mut self, pin: &mut P, value: bool) -> i32
where P: OutputPin<Error = PinError>
{
let r = match value {
true => pin.set_high(),
false => pin.set_low(),
};
match r {
Ok(_) => 0,
Err(e) => {
self.err = Some(Error::Pin(e));
-1
}
}
}
pub fn pin_read<P>(&mut self, pin: &mut P) -> i32
where P: InputPin<Error = PinError>
{
let r = pin.is_high();
match r {
Ok(true) => 1,
Ok(false) => 0,
Err(e) => {
self.err = Some(Error::Pin(e));
-1
}
}
}
pub fn check_error(&mut self) -> Result<(), Error<SpiError, PinError>> {
match self.err.take() {
Some(e) => Err(e),
None => Ok(())
}
}
}
impl <Spi, SpiError, Output, Input, PinError> Transactional for Wrapper<Spi, SpiError, Output, Input, PinError>
where
Spi: Transfer<u8, Error = SpiError> + Write<u8, Error = SpiError>,
Output: OutputPin<Error = PinError>,
Input: InputPin<Error = PinError>,
{
type Error = Error<SpiError, PinError>;
fn spi_read<'a>(&mut self, prefix: &[u8], mut data: &'a mut [u8]) -> Result<(), Self::Error> {
self.cs.set_low().map_err(|e| { Error::Pin(e) })?;
let mut res = self.spi.write(&prefix);
if res.is_ok() {
res = self.spi.transfer(&mut data).map(|_r| () );
}
self.cs.set_high().map_err(|e| { Error::Pin(e) })?;
match res {
Err(e) => Err( Error::Spi(e) ),
Ok(_) => Ok(()),
}
}
fn spi_write(&mut self, prefix: &[u8], data: &[u8]) -> Result<(), Self::Error> {
self.cs.set_low().map_err(|e| { Error::Pin(e) })?;
let mut res = self.spi.write(&prefix);
if res.is_ok() {
res = self.spi.write(&data);
}
self.cs.set_high().map_err(|e| { Error::Pin(e) })?;
match res {
Err(e) => Err( Error::Spi(e) ),
Ok(_) => Ok(()),
}
}
fn spi_exec(&mut self, transactions: &mut [Transaction]) -> Result<(), Self::Error> {
let mut res = Ok(());
self.cs.set_low().map_err(|e| { Error::Pin(e) })?;
for i in 0..transactions.len() {
let mut t = &mut transactions[i];
res = match &mut t {
Transaction::Write(d) => self.spi.write(d),
Transaction::Read(d) => self.spi.transfer(d).map(|_r| () ),
}.map_err(|e| Error::Spi(e) );
if res.is_err() {
break;
}
}
self.cs.set_low().map_err(|e| { Error::Pin(e) })?;
res
}
fn spi_busy(&mut self) -> Result<PinState, Self::Error> {
match &self.busy {
None => Ok(PinState::Low),
Some(b) => {
let v = b.is_high().map_err(|e| Error::Pin(e) )?;
match v {
true => Ok(PinState::High),
false => Ok(PinState::Low),
}
}
}
}
}
impl <Spi, SpiError, Output, Input, PinError> Transfer<u8> for Wrapper<Spi, SpiError, Output, Input, PinError>
where
Spi: Transfer<u8, Error = SpiError> + Write<u8, Error = SpiError>,
Output: OutputPin<Error = PinError>,
Input: InputPin<Error = PinError>,
{
type Error = SpiError;
fn transfer<'w>(&mut self, buffer: &'w mut [u8]) -> Result<&'w [u8], Self::Error> {
Transfer::transfer(&mut self.spi, buffer)
}
}
impl <Spi, SpiError, Output, Input, PinError> Write<u8> for Wrapper<Spi, SpiError, Output, Input, PinError>
where
Spi: Transfer<u8, Error = SpiError> + Write<u8, Error = SpiError>,
Output: OutputPin<Error = PinError>,
Input: InputPin<Error = PinError>,
{
type Error = SpiError;
fn write<'w>(&mut self, buffer: &[u8]) -> Result<(), Self::Error> {
Write::write(&mut self.spi, buffer)
}
}