atsamd_hal/sercom/spi/impl_ehal/
mod.rs
use super::*;
use crate::ehal::spi::{self, SpiBus};
#[allow(unused_imports)]
use crate::ehal_02::{blocking, serial};
use crate::ehal_nb;
use num_traits::PrimInt;
#[cfg(feature = "dma")]
mod dma;
mod panic_on;
#[hal_module(
any("sercom0-d11", "sercom0-d21") => "thumbv6m.rs",
"sercom0-d5x" => "thumbv7em.rs")]
pub mod impl_ehal_02 {}
impl spi::Error for Error {
#[allow(unreachable_patterns)]
#[inline]
fn kind(&self) -> crate::ehal::spi::ErrorKind {
match self {
Error::Overflow => crate::ehal::spi::ErrorKind::Overrun,
Error::LengthError => crate::ehal::spi::ErrorKind::Other,
_ => crate::ehal::spi::ErrorKind::Other,
}
}
}
impl<C, D, R, T> ehal::spi::ErrorType for Spi<C, D, R, T>
where
C: ValidConfig,
D: Capability,
{
type Error = Error;
}
impl embedded_io::Error for Error {
fn kind(&self) -> embedded_io::ErrorKind {
embedded_io::ErrorKind::Other
}
}
impl<C, D, R, T> embedded_io::ErrorType for Spi<C, D, R, T>
where
C: ValidConfig,
D: Capability,
{
type Error = Error;
}
impl ehal_nb::serial::Error for Error {
#[allow(unreachable_patterns)]
fn kind(&self) -> ehal_nb::serial::ErrorKind {
match self {
Error::Overflow => ehal_nb::serial::ErrorKind::Overrun,
Error::LengthError => ehal_nb::serial::ErrorKind::Other,
_ => ehal_nb::serial::ErrorKind::Other,
}
}
}
impl<C, D, R, T> ehal_nb::serial::ErrorType for Spi<C, D, R, T>
where
C: ValidConfig,
D: Capability,
{
type Error = Error;
}
impl<P, M, C, D, R, T> Spi<Config<P, M, C>, D, R, T>
where
Config<P, M, C>: ValidConfig,
P: ValidPads,
M: OpMode,
C: Size + 'static,
C::Word: PrimInt + AsPrimitive<DataWidth>,
DataWidth: AsPrimitive<C::Word>,
D: Capability,
{
#[inline]
fn transfer_word_in_place(&mut self, word: C::Word) -> Result<C::Word, Error> {
self.block_on_flags(Flags::DRE)?;
unsafe {
self.write_data(word.as_());
}
self.flush_rx()?;
let word = unsafe { self.read_data().as_() };
Ok(word)
}
#[inline]
fn transfer_word_by_word(
&mut self,
read: &mut [C::Word],
write: &[C::Word],
) -> Result<(), Error> {
let nop_word = self.config.nop_word;
for (r, w) in read
.iter_mut()
.zip(write.iter().chain(core::iter::repeat(&nop_word.as_())))
{
*r = self.transfer_word_in_place(*w)?;
}
Ok(())
}
#[inline]
fn flush_tx(&mut self) {
let _ = self.block_on_flags(Flags::TXC);
}
#[inline]
fn flush_rx(&mut self) -> Result<(), Error> {
self.block_on_flags(Flags::RXC)
}
}
impl<P, M, C, D> Spi<Config<P, M, C>, D>
where
Config<P, M, C>: ValidConfig,
P: ValidPads,
M: MasterMode,
C: Size + 'static,
C::Word: PrimInt + AsPrimitive<DataWidth> + Copy,
D: Capability,
DataWidth: AsPrimitive<C::Word>,
{
#[inline]
fn read_word_by_word(&mut self, words: &mut [Word<C>]) -> Result<(), Error> {
for word in words.iter_mut() {
*word = self.transfer_word_in_place(self.config.nop_word.as_())?;
}
Ok(())
}
#[inline]
fn write_word_by_word(&mut self, words: &[Word<C>]) -> Result<(), Error> {
self.config.as_mut().regs.rx_disable();
for word in words {
self.block_on_flags(Flags::DRE)?;
unsafe {
self.write_data(word.as_());
}
}
if D::RX_ENABLE {
self.config.as_mut().regs.rx_enable();
}
Ok(())
}
}
impl<P, M, C> SpiBus<Word<C>> for Spi<Config<P, M, C>, Duplex>
where
Config<P, M, C>: ValidConfig,
P: ValidPads,
M: MasterMode,
C: Size + 'static,
C::Word: PrimInt + AsPrimitive<DataWidth> + Copy,
DataWidth: AsPrimitive<C::Word>,
{
#[inline]
fn read(&mut self, words: &mut [Word<C>]) -> Result<(), Self::Error> {
self.read_word_by_word(words)
}
#[inline]
fn write(&mut self, words: &[Word<C>]) -> Result<(), Self::Error> {
self.write_word_by_word(words)
}
#[inline]
fn transfer(&mut self, read: &mut [Word<C>], write: &[Word<C>]) -> Result<(), Self::Error> {
self.transfer_word_by_word(read, write)
}
#[inline]
fn transfer_in_place(&mut self, words: &mut [Word<C>]) -> Result<(), Self::Error> {
for word in words {
let read = self.transfer_word_in_place(*word)?;
*word = read;
}
Ok(())
}
#[inline]
fn flush(&mut self) -> Result<(), Error> {
self.flush_tx();
Ok(())
}
}