2024-09-18 15:07:29 +02:00
|
|
|
//! API for the SPI peripheral.
|
|
|
|
//!
|
|
|
|
//! The main abstraction provided by this module are the [Spi] and the [SpiBase] structure.
|
|
|
|
//! These provide the [embedded_hal::spi] traits, but also offer a low level interface
|
|
|
|
//! via the [SpiLowLevel] trait.
|
2024-06-16 16:16:45 +02:00
|
|
|
//!
|
|
|
|
//! ## Examples
|
|
|
|
//!
|
2024-06-23 22:05:11 +02:00
|
|
|
//! - [Blocking SPI example](https://egit.irs.uni-stuttgart.de/rust/va108xx-rs/src/branch/main/examples/simple/examples/spi.rs)
|
2024-09-18 15:07:29 +02:00
|
|
|
//! - [REB1 ADC example](https://egit.irs.uni-stuttgart.de/rust/va108xx-rs/src/branch/main/vorago-reb1/examples/max11519-adc.rs)
|
|
|
|
//! - [REB1 EEPROM library](https://egit.irs.uni-stuttgart.de/rust/va108xx-rs/src/branch/main/vorago-reb1/src/m95m01.rs)
|
2024-06-16 16:16:45 +02:00
|
|
|
use crate::{
|
2024-07-04 17:10:01 +02:00
|
|
|
clock::enable_peripheral_clock,
|
|
|
|
gpio::pin::{
|
2024-06-16 16:16:45 +02:00
|
|
|
AltFunc1, AltFunc2, AltFunc3, Pin, PA10, PA11, PA12, PA13, PA14, PA15, PA16, PA17, PA18,
|
|
|
|
PA19, PA20, PA21, PA22, PA23, PA24, PA25, PA26, PA27, PA28, PA29, PA30, PA31, PB0, PB1,
|
|
|
|
PB10, PB11, PB12, PB13, PB14, PB15, PB16, PB17, PB18, PB19, PB2, PB22, PB23, PB3, PB4, PB5,
|
|
|
|
PB6, PB7, PB8, PB9,
|
|
|
|
},
|
|
|
|
pac,
|
|
|
|
time::Hertz,
|
|
|
|
typelevel::Sealed,
|
2024-07-04 17:10:01 +02:00
|
|
|
PeripheralSelect,
|
2024-06-16 16:16:45 +02:00
|
|
|
};
|
2024-07-04 17:10:01 +02:00
|
|
|
use core::{convert::Infallible, fmt::Debug, marker::PhantomData, ops::Deref};
|
2024-09-18 15:07:29 +02:00
|
|
|
use embedded_hal::spi::{Mode, MODE_0};
|
2024-06-16 16:16:45 +02:00
|
|
|
|
|
|
|
//==================================================================================================
|
|
|
|
// Defintions
|
|
|
|
//==================================================================================================
|
|
|
|
|
|
|
|
// FIFO has a depth of 16.
|
|
|
|
const FILL_DEPTH: usize = 12;
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
pub const BMSTART_BMSTOP_MASK: u32 = 1 << 31;
|
|
|
|
|
|
|
|
pub const DEFAULT_CLK_DIV: u16 = 2;
|
|
|
|
|
2024-06-16 16:16:45 +02:00
|
|
|
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
|
|
|
|
pub enum HwChipSelectId {
|
|
|
|
Id0 = 0,
|
|
|
|
Id1 = 1,
|
|
|
|
Id2 = 2,
|
|
|
|
Id3 = 3,
|
|
|
|
Id4 = 4,
|
|
|
|
Id5 = 5,
|
|
|
|
Id6 = 6,
|
|
|
|
Id7 = 7,
|
|
|
|
Invalid = 0xff,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum SpiPort {
|
|
|
|
Porta = 0,
|
|
|
|
Portb = 1,
|
|
|
|
Portc = 2,
|
|
|
|
Invalid = 3,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, PartialEq, Eq, Copy, Clone)]
|
|
|
|
pub enum WordSize {
|
|
|
|
OneBit = 0x00,
|
|
|
|
FourBits = 0x03,
|
|
|
|
EightBits = 0x07,
|
|
|
|
SixteenBits = 0x0f,
|
|
|
|
}
|
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
pub type SpiRegBlock = pac::spia::RegisterBlock;
|
|
|
|
|
|
|
|
/// Common trait implemented by all PAC peripheral access structures. The register block
|
|
|
|
/// format is the same for all SPI blocks.
|
|
|
|
pub trait Instance: Deref<Target = SpiRegBlock> {
|
|
|
|
const IDX: u8;
|
|
|
|
const PERIPH_SEL: PeripheralSelect;
|
|
|
|
|
|
|
|
fn ptr() -> *const SpiRegBlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Instance for pac::Spia {
|
|
|
|
const IDX: u8 = 0;
|
|
|
|
const PERIPH_SEL: PeripheralSelect = PeripheralSelect::Spi0;
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn ptr() -> *const SpiRegBlock {
|
|
|
|
Self::ptr()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Instance for pac::Spib {
|
|
|
|
const IDX: u8 = 1;
|
|
|
|
const PERIPH_SEL: PeripheralSelect = PeripheralSelect::Spi1;
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn ptr() -> *const SpiRegBlock {
|
|
|
|
Self::ptr()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Instance for pac::Spic {
|
|
|
|
const IDX: u8 = 2;
|
|
|
|
const PERIPH_SEL: PeripheralSelect = PeripheralSelect::Spi2;
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn ptr() -> *const SpiRegBlock {
|
|
|
|
Self::ptr()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-16 16:16:45 +02:00
|
|
|
//==================================================================================================
|
|
|
|
// Pin type definitions
|
|
|
|
//==================================================================================================
|
|
|
|
|
|
|
|
pub trait PinSck<SPI>: Sealed {}
|
|
|
|
pub trait PinMosi<SPI>: Sealed {}
|
|
|
|
pub trait PinMiso<SPI>: Sealed {}
|
|
|
|
|
|
|
|
pub trait HwCsProvider: Sealed {
|
|
|
|
const CS_ID: HwChipSelectId;
|
|
|
|
const SPI_PORT: SpiPort;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub trait OptionalHwCs<Spi>: HwCsProvider + Sealed {}
|
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
macro_rules! hw_cs_pins {
|
|
|
|
($SPIx:path, $portId: path:
|
|
|
|
$(
|
|
|
|
($PXx:ident, $AFx:ident, $HwCsIdent:path, $typedef:ident),
|
|
|
|
)+
|
|
|
|
) => {
|
|
|
|
$(
|
|
|
|
impl HwCsProvider for Pin<$PXx, $AFx> {
|
|
|
|
const CS_ID: HwChipSelectId = $HwCsIdent;
|
|
|
|
const SPI_PORT: SpiPort = $portId;
|
|
|
|
}
|
|
|
|
impl OptionalHwCs<$SPIx> for Pin<$PXx, $AFx> {}
|
|
|
|
pub type $typedef = Pin<$PXx, $AFx>;
|
|
|
|
)+
|
2024-06-16 16:16:45 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HwCsProvider for NoneT {
|
|
|
|
const CS_ID: HwChipSelectId = HwChipSelectId::Invalid;
|
|
|
|
const SPI_PORT: SpiPort = SpiPort::Invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl OptionalHwCs<pac::Spia> for NoneT {}
|
|
|
|
impl OptionalHwCs<pac::Spib> for NoneT {}
|
|
|
|
|
|
|
|
// SPIA
|
|
|
|
|
|
|
|
impl PinSck<pac::Spia> for Pin<PA31, AltFunc1> {}
|
|
|
|
impl PinMosi<pac::Spia> for Pin<PA30, AltFunc1> {}
|
|
|
|
impl PinMiso<pac::Spia> for Pin<PA29, AltFunc1> {}
|
|
|
|
|
|
|
|
pub type SpiAPortASck = Pin<PA31, AltFunc1>;
|
|
|
|
pub type SpiAPortAMosi = Pin<PA30, AltFunc1>;
|
|
|
|
pub type SpiAPortAMiso = Pin<PA29, AltFunc1>;
|
|
|
|
|
|
|
|
impl PinSck<pac::Spia> for Pin<PB9, AltFunc2> {}
|
|
|
|
impl PinMosi<pac::Spia> for Pin<PB8, AltFunc2> {}
|
|
|
|
impl PinMiso<pac::Spia> for Pin<PB7, AltFunc2> {}
|
|
|
|
|
|
|
|
pub type SpiAPortBSck = Pin<PB9, AltFunc2>;
|
|
|
|
pub type SpiAPortBMosi = Pin<PB8, AltFunc2>;
|
|
|
|
pub type SpiAPortBMiso = Pin<PB7, AltFunc2>;
|
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
hw_cs_pins!(
|
|
|
|
pac::Spia, SpiPort::Porta:
|
|
|
|
(PA28, AltFunc1, HwChipSelectId::Id0, HwCs0SpiAPortA),
|
|
|
|
(PA27, AltFunc1, HwChipSelectId::Id1, HwCs1SpiAPortA),
|
|
|
|
(PA26, AltFunc1, HwChipSelectId::Id2, HwCs2SpiAPortA),
|
|
|
|
(PA25, AltFunc1, HwChipSelectId::Id3, HwCs3SpiAPortA),
|
|
|
|
(PA24, AltFunc1, HwChipSelectId::Id4, HwCs4SpiAPortA),
|
|
|
|
(PA23, AltFunc1, HwChipSelectId::Id5, HwCs5SpiAPortA),
|
|
|
|
(PA22, AltFunc1, HwChipSelectId::Id6, HwCs6SpiAPortA),
|
|
|
|
(PA21, AltFunc1, HwChipSelectId::Id7, HwCs7SpiAPortA),
|
|
|
|
(PB6, AltFunc2, HwChipSelectId::Id0, HwCs0SpiAPortB),
|
|
|
|
(PB5, AltFunc2, HwChipSelectId::Id6, HwCs6SpiAPortB),
|
|
|
|
(PB4, AltFunc2, HwChipSelectId::Id5, HwCs5SpiAPortB),
|
|
|
|
(PB3, AltFunc2, HwChipSelectId::Id4, HwCs4SpiAPortB),
|
|
|
|
(PB2, AltFunc2, HwChipSelectId::Id3, HwCs3SpiAPortB),
|
|
|
|
(PB1, AltFunc2, HwChipSelectId::Id2, HwCs2SpiAPortB),
|
|
|
|
(PB0, AltFunc2, HwChipSelectId::Id1, HwCs1SpiAPortB),
|
2024-06-16 16:16:45 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
// SPIB
|
|
|
|
|
|
|
|
impl PinSck<pac::Spib> for Pin<PA20, AltFunc2> {}
|
|
|
|
impl PinMosi<pac::Spib> for Pin<PA19, AltFunc2> {}
|
|
|
|
impl PinMiso<pac::Spib> for Pin<PA18, AltFunc2> {}
|
|
|
|
|
|
|
|
pub type SpiBPortASck = Pin<PA20, AltFunc2>;
|
|
|
|
pub type SpiBPortAMosi = Pin<PA19, AltFunc2>;
|
|
|
|
pub type SpiBPortAMiso = Pin<PA18, AltFunc2>;
|
|
|
|
|
|
|
|
impl PinSck<pac::Spib> for Pin<PB19, AltFunc1> {}
|
|
|
|
impl PinMosi<pac::Spib> for Pin<PB18, AltFunc1> {}
|
|
|
|
impl PinMiso<pac::Spib> for Pin<PB17, AltFunc1> {}
|
|
|
|
|
|
|
|
impl PinSck<pac::Spib> for Pin<PB5, AltFunc1> {}
|
|
|
|
impl PinMosi<pac::Spib> for Pin<PB4, AltFunc1> {}
|
|
|
|
impl PinMiso<pac::Spib> for Pin<PB3, AltFunc1> {}
|
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
hw_cs_pins!(
|
|
|
|
pac::Spib, SpiPort::Portb:
|
|
|
|
(PB16, AltFunc1, HwChipSelectId::Id0, HwCs0SpiBPortB0),
|
|
|
|
(PB15, AltFunc1, HwChipSelectId::Id1, HwCs1SpiBPortB0),
|
|
|
|
(PB14, AltFunc1, HwChipSelectId::Id2, HwCs2SpiBPortB0),
|
|
|
|
(PB13, AltFunc1, HwChipSelectId::Id3, HwCs3SpiBPortB),
|
|
|
|
(PB12, AltFunc1, HwChipSelectId::Id4, HwCs4SpiBPortB),
|
|
|
|
(PB11, AltFunc1, HwChipSelectId::Id5, HwCs5SpiBPortB),
|
|
|
|
(PB12, AltFunc2, HwChipSelectId::Id0, HwCs0SpiBPortB2),
|
|
|
|
(PB11, AltFunc2, HwChipSelectId::Id1, HwCs1SpiBPortB2),
|
|
|
|
(PB10, AltFunc1, HwChipSelectId::Id6, HwCs6SpiBPortB),
|
|
|
|
(PB10, AltFunc2, HwChipSelectId::Id2, HwCs2SpiBPortB2),
|
|
|
|
(PB2, AltFunc1, HwChipSelectId::Id0, HwCs0SpiBPortB1),
|
|
|
|
(PB1, AltFunc1, HwChipSelectId::Id1, HwCs1SpiBPortB1),
|
|
|
|
(PB0, AltFunc1, HwChipSelectId::Id2, HwCs2SpiBPortB1),
|
|
|
|
(PA17, AltFunc2, HwChipSelectId::Id0, HwCs0SpiBPortA),
|
|
|
|
(PA16, AltFunc2, HwChipSelectId::Id1, HwCs1SpiBPortA),
|
|
|
|
(PA15, AltFunc2, HwChipSelectId::Id2, HwCs2SpiBPortA),
|
|
|
|
(PA14, AltFunc2, HwChipSelectId::Id3, HwCs3SpiBPortA),
|
|
|
|
(PA13, AltFunc2, HwChipSelectId::Id4, HwCs4SpiBPortA),
|
|
|
|
(PA12, AltFunc2, HwChipSelectId::Id5, HwCs5SpiBPortA0),
|
|
|
|
(PA11, AltFunc2, HwChipSelectId::Id6, HwCs6SpiBPortA0),
|
|
|
|
(PA10, AltFunc2, HwChipSelectId::Id7, HwCs7SpiBPortA0),
|
|
|
|
(PA23, AltFunc2, HwChipSelectId::Id5, HwCs5SpiBPortA1),
|
|
|
|
(PA22, AltFunc2, HwChipSelectId::Id6, HwCs6SpiBPortA1),
|
|
|
|
(PA21, AltFunc2, HwChipSelectId::Id7, HwCs7SpiBPortA1),
|
2024-06-16 16:16:45 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
// SPIC
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
// Dummy pin defintion for the ROM SCK.
|
|
|
|
pub struct RomSck;
|
|
|
|
// Dummy pin defintion for the ROM MOSI.
|
|
|
|
pub struct RomMosi;
|
|
|
|
// Dummy pin defintion for the ROM MISO.
|
|
|
|
pub struct RomMiso;
|
|
|
|
// Dummy pin defintion for the ROM chip select.
|
|
|
|
pub struct RomCs;
|
|
|
|
|
|
|
|
impl Sealed for RomSck {}
|
|
|
|
impl PinSck<pac::Spic> for RomSck {}
|
|
|
|
impl Sealed for RomMosi {}
|
|
|
|
impl PinMosi<pac::Spic> for RomMosi {}
|
|
|
|
impl Sealed for RomMiso {}
|
|
|
|
impl PinMiso<pac::Spic> for RomMiso {}
|
|
|
|
impl Sealed for RomCs {}
|
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
hw_cs_pins!(
|
|
|
|
pac::Spic, SpiPort::Portc:
|
|
|
|
(PB9, AltFunc3, HwChipSelectId::Id1, HwCs1SpiCPortB0),
|
|
|
|
(PB8, AltFunc3, HwChipSelectId::Id2, HwCs2SpiCPortB0),
|
|
|
|
(PB7, AltFunc3, HwChipSelectId::Id3, HwCs3SpiCPortB),
|
|
|
|
(PB23, AltFunc3, HwChipSelectId::Id2, HwCs2SpiCPortB1),
|
|
|
|
(PB22, AltFunc3, HwChipSelectId::Id1, HwCs1SpiCPortB1),
|
|
|
|
(PA20, AltFunc1, HwChipSelectId::Id1, HwCs1SpiCPortA0),
|
|
|
|
(PA19, AltFunc1, HwChipSelectId::Id2, HwCs2SpiCPortA0),
|
|
|
|
(PB18, AltFunc1, HwChipSelectId::Id3, HwCs3SpiCPortA0),
|
|
|
|
(PA23, AltFunc3, HwChipSelectId::Id1, HwCs1SpiCPortA1),
|
|
|
|
(PA22, AltFunc3, HwChipSelectId::Id2, HwCs2SpiCPortA1),
|
|
|
|
(PA21, AltFunc3, HwChipSelectId::Id3, HwCs3SpiCPortA1),
|
|
|
|
(PA20, AltFunc3, HwChipSelectId::Id4, HwCs4SpiCPortA),
|
2024-06-16 16:16:45 +02:00
|
|
|
);
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
impl HwCsProvider for RomCs {
|
|
|
|
const CS_ID: HwChipSelectId = HwChipSelectId::Id0;
|
|
|
|
const SPI_PORT: SpiPort = SpiPort::Portc;
|
|
|
|
}
|
|
|
|
impl OptionalHwCs<pac::Spic> for RomCs {}
|
|
|
|
|
2024-06-16 16:16:45 +02:00
|
|
|
//==================================================================================================
|
|
|
|
// Config
|
|
|
|
//==================================================================================================
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
pub trait TransferConfigProvider {
|
2024-06-16 16:16:45 +02:00
|
|
|
fn sod(&mut self, sod: bool);
|
|
|
|
fn blockmode(&mut self, blockmode: bool);
|
|
|
|
fn mode(&mut self, mode: Mode);
|
2024-09-18 15:07:29 +02:00
|
|
|
fn clk_cfg(&mut self, clk_cfg: SpiClkConfig);
|
2024-06-16 16:16:45 +02:00
|
|
|
fn hw_cs_id(&self) -> u8;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This struct contains all configuration parameter which are transfer specific
|
|
|
|
/// and might change for transfers to different SPI slaves
|
2024-09-18 15:07:29 +02:00
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
|
|
pub struct TransferConfigWithHwcs<HwCs> {
|
2024-06-16 16:16:45 +02:00
|
|
|
pub hw_cs: Option<HwCs>,
|
2024-09-18 15:07:29 +02:00
|
|
|
pub cfg: TransferConfig,
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Type erased variant of the transfer configuration. This is required to avoid generics in
|
|
|
|
/// the SPI constructor.
|
2024-09-18 15:07:29 +02:00
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
|
|
pub struct TransferConfig {
|
|
|
|
pub clk_cfg: Option<SpiClkConfig>,
|
|
|
|
pub mode: Option<Mode>,
|
2024-06-16 16:16:45 +02:00
|
|
|
pub sod: bool,
|
|
|
|
/// If this is enabled, all data in the FIFO is transmitted in a single frame unless
|
|
|
|
/// the BMSTOP bit is set on a dataword. A frame is defined as CSn being active for the
|
|
|
|
/// duration of multiple data words
|
|
|
|
pub blockmode: bool,
|
2024-09-20 11:41:03 +02:00
|
|
|
/// Only used when blockmode is used. The SCK will be stalled until an explicit stop bit
|
|
|
|
/// is set on a written word.
|
|
|
|
pub bmstall: bool,
|
2024-06-16 16:16:45 +02:00
|
|
|
pub hw_cs: HwChipSelectId,
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
impl TransferConfigWithHwcs<NoneT> {
|
|
|
|
pub fn new_no_hw_cs(
|
|
|
|
clk_cfg: Option<SpiClkConfig>,
|
|
|
|
mode: Option<Mode>,
|
|
|
|
blockmode: bool,
|
2024-09-20 11:41:03 +02:00
|
|
|
bmstall: bool,
|
2024-09-18 15:07:29 +02:00
|
|
|
sod: bool,
|
|
|
|
) -> Self {
|
|
|
|
TransferConfigWithHwcs {
|
2024-06-16 16:16:45 +02:00
|
|
|
hw_cs: None,
|
2024-09-18 15:07:29 +02:00
|
|
|
cfg: TransferConfig {
|
|
|
|
clk_cfg,
|
|
|
|
mode,
|
|
|
|
sod,
|
|
|
|
blockmode,
|
2024-09-20 11:41:03 +02:00
|
|
|
bmstall,
|
2024-09-18 15:07:29 +02:00
|
|
|
hw_cs: HwChipSelectId::Invalid,
|
|
|
|
},
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
impl<HwCs: HwCsProvider> TransferConfigWithHwcs<HwCs> {
|
2024-06-16 16:16:45 +02:00
|
|
|
pub fn new(
|
2024-09-18 15:07:29 +02:00
|
|
|
clk_cfg: Option<SpiClkConfig>,
|
|
|
|
mode: Option<Mode>,
|
2024-06-16 16:16:45 +02:00
|
|
|
hw_cs: Option<HwCs>,
|
|
|
|
blockmode: bool,
|
2024-09-20 11:41:03 +02:00
|
|
|
bmstall: bool,
|
2024-06-16 16:16:45 +02:00
|
|
|
sod: bool,
|
|
|
|
) -> Self {
|
2024-09-18 15:07:29 +02:00
|
|
|
TransferConfigWithHwcs {
|
2024-06-16 16:16:45 +02:00
|
|
|
hw_cs,
|
2024-09-18 15:07:29 +02:00
|
|
|
cfg: TransferConfig {
|
|
|
|
clk_cfg,
|
|
|
|
mode,
|
|
|
|
sod,
|
|
|
|
blockmode,
|
2024-09-20 11:41:03 +02:00
|
|
|
bmstall,
|
2024-09-18 15:07:29 +02:00
|
|
|
hw_cs: HwCs::CS_ID,
|
|
|
|
},
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
pub fn downgrade(self) -> TransferConfig {
|
|
|
|
self.cfg
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
impl<HwCs: HwCsProvider> TransferConfigProvider for TransferConfigWithHwcs<HwCs> {
|
2024-06-16 16:16:45 +02:00
|
|
|
/// Slave Output Disable
|
|
|
|
fn sod(&mut self, sod: bool) {
|
2024-09-18 15:07:29 +02:00
|
|
|
self.cfg.sod = sod;
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn blockmode(&mut self, blockmode: bool) {
|
2024-09-18 15:07:29 +02:00
|
|
|
self.cfg.blockmode = blockmode;
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn mode(&mut self, mode: Mode) {
|
2024-09-18 15:07:29 +02:00
|
|
|
self.cfg.mode = Some(mode);
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
fn clk_cfg(&mut self, clk_cfg: SpiClkConfig) {
|
|
|
|
self.cfg.clk_cfg = Some(clk_cfg);
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn hw_cs_id(&self) -> u8 {
|
|
|
|
HwCs::CS_ID as u8
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Configuration options for the whole SPI bus. See Programmer Guide p.92 for more details
|
|
|
|
pub struct SpiConfig {
|
2024-09-18 15:07:29 +02:00
|
|
|
clk: SpiClkConfig,
|
|
|
|
// SPI mode configuration
|
|
|
|
pub init_mode: Mode,
|
|
|
|
/// If this is enabled, all data in the FIFO is transmitted in a single frame unless
|
|
|
|
/// the BMSTOP bit is set on a dataword. A frame is defined as CSn being active for the
|
|
|
|
/// duration of multiple data words. Defaults to true.
|
|
|
|
pub blockmode: bool,
|
|
|
|
/// This enables the stalling of the SPI SCK if in blockmode and the FIFO is empty.
|
|
|
|
/// Currently enabled by default.
|
|
|
|
pub bmstall: bool,
|
2024-06-16 16:16:45 +02:00
|
|
|
/// By default, configure SPI for master mode (ms == false)
|
|
|
|
ms: bool,
|
|
|
|
/// Slave output disable. Useful if separate GPIO pins or decoders are used for CS control
|
2024-09-18 15:07:29 +02:00
|
|
|
pub slave_output_disable: bool,
|
2024-06-16 16:16:45 +02:00
|
|
|
/// Loopback mode. If you use this, don't connect MISO to MOSI, they will be tied internally
|
2024-09-18 15:07:29 +02:00
|
|
|
pub loopback_mode: bool,
|
2024-06-16 16:16:45 +02:00
|
|
|
/// Enable Master Delayer Capture Mode. See Programmers Guide p.92 for more details
|
2024-09-18 15:07:29 +02:00
|
|
|
pub master_delayer_capture: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for SpiConfig {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self {
|
|
|
|
init_mode: MODE_0,
|
|
|
|
blockmode: true,
|
|
|
|
bmstall: true,
|
|
|
|
// Default value is definitely valid.
|
|
|
|
clk: SpiClkConfig::from_div(DEFAULT_CLK_DIV).unwrap(),
|
|
|
|
ms: Default::default(),
|
|
|
|
slave_output_disable: Default::default(),
|
|
|
|
loopback_mode: Default::default(),
|
|
|
|
master_delayer_capture: Default::default(),
|
|
|
|
}
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl SpiConfig {
|
|
|
|
pub fn loopback(mut self, enable: bool) -> Self {
|
2024-09-18 15:07:29 +02:00
|
|
|
self.loopback_mode = enable;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn blockmode(mut self, enable: bool) -> Self {
|
|
|
|
self.blockmode = enable;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2024-09-20 11:41:03 +02:00
|
|
|
pub fn bmstall(mut self, enable: bool) -> Self {
|
|
|
|
self.bmstall = enable;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
pub fn mode(mut self, mode: Mode) -> Self {
|
|
|
|
self.init_mode = mode;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn clk_cfg(mut self, clk_cfg: SpiClkConfig) -> Self {
|
|
|
|
self.clk = clk_cfg;
|
2024-06-16 16:16:45 +02:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn master_mode(mut self, master: bool) -> Self {
|
|
|
|
self.ms = !master;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn slave_output_disable(mut self, sod: bool) -> Self {
|
2024-09-18 15:07:29 +02:00
|
|
|
self.slave_output_disable = sod;
|
2024-06-16 16:16:45 +02:00
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==================================================================================================
|
|
|
|
// Word Size
|
|
|
|
//==================================================================================================
|
|
|
|
|
|
|
|
/// Configuration trait for the Word Size
|
|
|
|
/// used by the SPI peripheral
|
2024-07-04 17:10:01 +02:00
|
|
|
pub trait WordProvider: Copy + Default + Into<u32> + TryFrom<u32> + 'static {
|
|
|
|
const MASK: u32;
|
2024-06-16 16:16:45 +02:00
|
|
|
fn word_reg() -> u8;
|
|
|
|
}
|
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
impl WordProvider for u8 {
|
|
|
|
const MASK: u32 = 0xff;
|
2024-06-16 16:16:45 +02:00
|
|
|
fn word_reg() -> u8 {
|
|
|
|
0x07
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
impl WordProvider for u16 {
|
|
|
|
const MASK: u32 = 0xffff;
|
2024-06-16 16:16:45 +02:00
|
|
|
fn word_reg() -> u8 {
|
|
|
|
0x0f
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//==================================================================================================
|
|
|
|
// Spi
|
|
|
|
//==================================================================================================
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
/// Low level access trait for the SPI peripheral.
|
|
|
|
pub trait SpiLowLevel {
|
|
|
|
/// Low level function to write a word to the SPI FIFO but also checks whether
|
|
|
|
/// there is actually data in the FIFO.
|
|
|
|
///
|
|
|
|
/// Uses the [nb] API to allow usage in blocking and non-blocking contexts.
|
|
|
|
fn write_fifo(&self, data: u32) -> nb::Result<(), Infallible>;
|
|
|
|
|
|
|
|
/// Low level function to write a word to the SPI FIFO without checking whether
|
|
|
|
/// there FIFO is full.
|
|
|
|
///
|
|
|
|
/// This does not necesarily mean there is a space in the FIFO available.
|
|
|
|
/// Use [Self::write_fifo] function to write a word into the FIFO reliably.
|
|
|
|
fn write_fifo_unchecked(&self, data: u32);
|
|
|
|
|
|
|
|
/// Low level function to read a word from the SPI FIFO. Must be preceeded by a
|
|
|
|
/// [Self::write_fifo] call.
|
|
|
|
///
|
|
|
|
/// Uses the [nb] API to allow usage in blocking and non-blocking contexts.
|
|
|
|
fn read_fifo(&self) -> nb::Result<u32, Infallible>;
|
|
|
|
|
|
|
|
/// Low level function to read a word from from the SPI FIFO.
|
|
|
|
///
|
|
|
|
/// This does not necesarily mean there is a word in the FIFO available.
|
|
|
|
/// Use the [Self::read_fifo] function to read a word from the FIFO reliably using the [nb]
|
|
|
|
/// API.
|
|
|
|
/// You might also need to mask the value to ignore the BMSTART/BMSTOP bit.
|
|
|
|
fn read_fifo_unchecked(&self) -> u32;
|
|
|
|
}
|
|
|
|
|
2024-06-16 16:16:45 +02:00
|
|
|
pub struct SpiBase<SpiInstance, Word = u8> {
|
|
|
|
spi: SpiInstance,
|
|
|
|
cfg: SpiConfig,
|
|
|
|
sys_clk: Hertz,
|
|
|
|
/// Fill word for read-only SPI transactions.
|
|
|
|
pub fill_word: Word,
|
|
|
|
blockmode: bool,
|
2024-09-18 15:07:29 +02:00
|
|
|
bmstall: bool,
|
2024-07-04 17:10:01 +02:00
|
|
|
word: PhantomData<Word>,
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Spi<SpiInstance, Pins, Word = u8> {
|
|
|
|
inner: SpiBase<SpiInstance, Word>,
|
|
|
|
pins: Pins,
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
pub fn mode_to_cpo_cph_bit(mode: embedded_hal::spi::Mode) -> (bool, bool) {
|
|
|
|
match mode {
|
|
|
|
embedded_hal::spi::MODE_0 => (false, false),
|
|
|
|
embedded_hal::spi::MODE_1 => (false, true),
|
|
|
|
embedded_hal::spi::MODE_2 => (true, false),
|
|
|
|
embedded_hal::spi::MODE_3 => (true, true),
|
|
|
|
}
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
|
|
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
|
|
pub struct SpiClkConfig {
|
|
|
|
prescale_val: u16,
|
|
|
|
scrdv: u8,
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
impl SpiClkConfig {
|
|
|
|
pub fn prescale_val(&self) -> u16 {
|
|
|
|
self.prescale_val
|
|
|
|
}
|
|
|
|
pub fn scrdv(&self) -> u8 {
|
|
|
|
self.scrdv
|
|
|
|
}
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
impl SpiClkConfig {
|
|
|
|
pub fn new(prescale_val: u16, scrdv: u8) -> Self {
|
|
|
|
Self {
|
|
|
|
prescale_val,
|
|
|
|
scrdv,
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
pub fn from_div(div: u16) -> Result<Self, SpiClkConfigError> {
|
|
|
|
spi_clk_config_from_div(div)
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
pub fn from_clk(sys_clk: impl Into<Hertz>, spi_clk: impl Into<Hertz>) -> Option<Self> {
|
|
|
|
clk_div_for_target_clock(sys_clk, spi_clk).map(|div| spi_clk_config_from_div(div).unwrap())
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum SpiClkConfigError {
|
|
|
|
DivIsZero,
|
|
|
|
DivideValueNotEven,
|
|
|
|
ScrdvValueTooLarge,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn spi_clk_config_from_div(mut div: u16) -> Result<SpiClkConfig, SpiClkConfigError> {
|
|
|
|
if div == 0 {
|
|
|
|
return Err(SpiClkConfigError::DivIsZero);
|
|
|
|
}
|
|
|
|
if div % 2 != 0 {
|
|
|
|
return Err(SpiClkConfigError::DivideValueNotEven);
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
let mut prescale_val = 0;
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
// find largest (even) prescale value that divides into div
|
|
|
|
for i in (2..=0xfe).rev().step_by(2) {
|
|
|
|
if div % i == 0 {
|
|
|
|
prescale_val = i;
|
|
|
|
break;
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
if prescale_val == 0 {
|
|
|
|
return Err(SpiClkConfigError::DivideValueNotEven);
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
div /= prescale_val;
|
|
|
|
if div > u8::MAX as u16 + 1 {
|
|
|
|
return Err(SpiClkConfigError::ScrdvValueTooLarge);
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
Ok(SpiClkConfig {
|
|
|
|
prescale_val,
|
|
|
|
scrdv: (div - 1) as u8,
|
|
|
|
})
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
#[inline]
|
|
|
|
pub fn clk_div_for_target_clock(
|
|
|
|
sys_clk: impl Into<Hertz>,
|
|
|
|
spi_clk: impl Into<Hertz>,
|
|
|
|
) -> Option<u16> {
|
|
|
|
let spi_clk = spi_clk.into();
|
|
|
|
let sys_clk = sys_clk.into();
|
|
|
|
if spi_clk > sys_clk {
|
|
|
|
return None;
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
// Step 1: Calculate raw divider.
|
|
|
|
let raw_div = sys_clk.raw() / spi_clk.raw();
|
|
|
|
let remainder = sys_clk.raw() % spi_clk.raw();
|
|
|
|
|
|
|
|
// Step 2: Round up if necessary.
|
|
|
|
let mut rounded_div = if remainder * 2 >= spi_clk.raw() {
|
|
|
|
raw_div + 1
|
|
|
|
} else {
|
|
|
|
raw_div
|
|
|
|
};
|
|
|
|
|
|
|
|
if rounded_div % 2 != 0 {
|
|
|
|
// Take slower clock conservatively.
|
|
|
|
rounded_div += 1;
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
if rounded_div > u16::MAX as u32 {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
Some(rounded_div as u16)
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
// Re-export this so it can be used for the constructor
|
|
|
|
pub use crate::typelevel::NoneT;
|
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
impl<SpiInstance: Instance, Word: WordProvider> SpiBase<SpiInstance, Word>
|
|
|
|
where
|
|
|
|
<Word as TryFrom<u32>>::Error: core::fmt::Debug,
|
|
|
|
{
|
|
|
|
#[inline]
|
2024-09-18 15:07:29 +02:00
|
|
|
pub fn spi(&self) -> &SpiInstance {
|
|
|
|
&self.spi
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn cfg_clock(&mut self, cfg: SpiClkConfig) {
|
|
|
|
self.spi
|
|
|
|
.ctrl0()
|
|
|
|
.modify(|_, w| unsafe { w.scrdv().bits(cfg.scrdv) });
|
2024-07-04 17:10:01 +02:00
|
|
|
self.spi
|
|
|
|
.clkprescale()
|
2024-09-18 15:07:29 +02:00
|
|
|
.write(|w| unsafe { w.bits(cfg.prescale_val as u32) });
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn cfg_clock_from_div(&mut self, div: u16) -> Result<(), SpiClkConfigError> {
|
|
|
|
let val = spi_clk_config_from_div(div)?;
|
|
|
|
self.cfg_clock(val);
|
|
|
|
Ok(())
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn cfg_mode(&mut self, mode: Mode) {
|
2024-09-18 15:07:29 +02:00
|
|
|
let (cpo_bit, cph_bit) = mode_to_cpo_cph_bit(mode);
|
2024-07-04 17:10:01 +02:00
|
|
|
self.spi.ctrl0().modify(|_, w| {
|
|
|
|
w.spo().bit(cpo_bit);
|
|
|
|
w.sph().bit(cph_bit)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
#[inline]
|
|
|
|
pub fn fill_word(&self) -> Word {
|
|
|
|
self.fill_word
|
|
|
|
}
|
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
#[inline]
|
|
|
|
pub fn clear_tx_fifo(&self) {
|
|
|
|
self.spi.fifo_clr().write(|w| w.txfifo().set_bit());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn clear_rx_fifo(&self) {
|
|
|
|
self.spi.fifo_clr().write(|w| w.rxfifo().set_bit());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn perid(&self) -> u32 {
|
|
|
|
self.spi.perid().read().bits()
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
/// Configure the hardware chip select given a hardware chip select ID.
|
2024-07-04 17:10:01 +02:00
|
|
|
#[inline]
|
|
|
|
pub fn cfg_hw_cs(&mut self, hw_cs: HwChipSelectId) {
|
|
|
|
if hw_cs == HwChipSelectId::Invalid {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
self.spi.ctrl1().modify(|_, w| {
|
|
|
|
w.sod().clear_bit();
|
|
|
|
unsafe {
|
|
|
|
w.ss().bits(hw_cs as u8);
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
w
|
|
|
|
});
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
/// Configure the hardware chip select given a physical hardware CS pin.
|
2024-07-04 17:10:01 +02:00
|
|
|
#[inline]
|
|
|
|
pub fn cfg_hw_cs_with_pin<HwCs: OptionalHwCs<SpiInstance>>(&mut self, _: &HwCs) {
|
|
|
|
self.cfg_hw_cs(HwCs::CS_ID);
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
/// Disables the hardware chip select functionality. This can be used when performing
|
|
|
|
/// external chip select handling, for example with GPIO pins.
|
|
|
|
#[inline]
|
2024-07-04 17:10:01 +02:00
|
|
|
pub fn cfg_hw_cs_disable(&mut self) {
|
|
|
|
self.spi.ctrl1().modify(|_, w| {
|
|
|
|
w.sod().set_bit();
|
|
|
|
w
|
|
|
|
});
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
/// Utility function to configure all relevant transfer parameters in one go.
|
|
|
|
/// This is useful if multiple devices with different clock and mode configurations
|
|
|
|
/// are connected to one bus.
|
2024-07-04 17:10:01 +02:00
|
|
|
pub fn cfg_transfer<HwCs: OptionalHwCs<SpiInstance>>(
|
|
|
|
&mut self,
|
2024-09-18 15:07:29 +02:00
|
|
|
transfer_cfg: &TransferConfigWithHwcs<HwCs>,
|
2024-07-04 17:10:01 +02:00
|
|
|
) {
|
2024-09-18 15:07:29 +02:00
|
|
|
if let Some(trans_clk_div) = transfer_cfg.cfg.clk_cfg {
|
|
|
|
self.cfg_clock(trans_clk_div);
|
|
|
|
}
|
|
|
|
if let Some(mode) = transfer_cfg.cfg.mode {
|
|
|
|
self.cfg_mode(mode);
|
|
|
|
}
|
|
|
|
self.blockmode = transfer_cfg.cfg.blockmode;
|
2024-07-04 17:10:01 +02:00
|
|
|
self.spi.ctrl1().modify(|_, w| {
|
2024-09-18 15:07:29 +02:00
|
|
|
if transfer_cfg.cfg.sod {
|
2024-07-04 17:10:01 +02:00
|
|
|
w.sod().set_bit();
|
|
|
|
} else if transfer_cfg.hw_cs.is_some() {
|
|
|
|
w.sod().clear_bit();
|
|
|
|
unsafe {
|
|
|
|
w.ss().bits(HwCs::CS_ID as u8);
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
} else {
|
|
|
|
w.sod().clear_bit();
|
|
|
|
}
|
2024-09-20 11:41:03 +02:00
|
|
|
w.blockmode().bit(transfer_cfg.cfg.blockmode);
|
|
|
|
w.bmstall().bit(transfer_cfg.cfg.bmstall)
|
2024-07-04 17:10:01 +02:00
|
|
|
});
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
fn flush_internal(&self) {
|
|
|
|
let mut status_reg = self.spi.status().read();
|
|
|
|
while status_reg.tfe().bit_is_clear()
|
|
|
|
|| status_reg.rne().bit_is_set()
|
|
|
|
|| status_reg.busy().bit_is_set()
|
|
|
|
{
|
|
|
|
if status_reg.rne().bit_is_set() {
|
|
|
|
self.read_fifo_unchecked();
|
|
|
|
}
|
|
|
|
status_reg = self.spi.status().read();
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
fn transfer_preparation(&self, words: &[Word]) -> Result<(), Infallible> {
|
|
|
|
if words.is_empty() {
|
|
|
|
return Ok(());
|
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
self.flush_internal();
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
// The FIFO can hold a guaranteed amount of data, so we can pump it on transfer
|
|
|
|
// initialization. Returns the amount of written bytes.
|
|
|
|
fn initial_send_fifo_pumping_with_words(&self, words: &[Word]) -> usize {
|
|
|
|
if self.blockmode {
|
|
|
|
self.spi.ctrl1().modify(|_, w| w.mtxpause().set_bit())
|
|
|
|
}
|
|
|
|
// Fill the first half of the write FIFO
|
|
|
|
let mut current_write_idx = 0;
|
|
|
|
let smaller_idx = core::cmp::min(FILL_DEPTH, words.len());
|
|
|
|
for _ in 0..smaller_idx {
|
|
|
|
if current_write_idx == smaller_idx.saturating_sub(1) && self.bmstall {
|
|
|
|
self.write_fifo_unchecked(words[current_write_idx].into() | BMSTART_BMSTOP_MASK);
|
|
|
|
} else {
|
|
|
|
self.write_fifo_unchecked(words[current_write_idx].into());
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
current_write_idx += 1;
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
if self.blockmode {
|
|
|
|
self.spi.ctrl1().modify(|_, w| w.mtxpause().clear_bit())
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
current_write_idx
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
// The FIFO can hold a guaranteed amount of data, so we can pump it on transfer
|
|
|
|
// initialization.
|
|
|
|
fn initial_send_fifo_pumping_with_fill_words(&self, send_len: usize) -> usize {
|
2024-07-04 17:10:01 +02:00
|
|
|
if self.blockmode {
|
|
|
|
self.spi.ctrl1().modify(|_, w| w.mtxpause().set_bit())
|
|
|
|
}
|
|
|
|
// Fill the first half of the write FIFO
|
|
|
|
let mut current_write_idx = 0;
|
2024-09-18 15:07:29 +02:00
|
|
|
let smaller_idx = core::cmp::min(FILL_DEPTH, send_len);
|
|
|
|
for _ in 0..smaller_idx {
|
|
|
|
if current_write_idx == smaller_idx.saturating_sub(1) && self.bmstall {
|
|
|
|
self.write_fifo_unchecked(self.fill_word.into() | BMSTART_BMSTOP_MASK);
|
|
|
|
} else {
|
|
|
|
self.write_fifo_unchecked(self.fill_word.into());
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
current_write_idx += 1;
|
|
|
|
}
|
|
|
|
if self.blockmode {
|
|
|
|
self.spi.ctrl1().modify(|_, w| w.mtxpause().clear_bit())
|
|
|
|
}
|
|
|
|
current_write_idx
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
impl<SpiInstance: Instance, Word: WordProvider> SpiLowLevel for SpiBase<SpiInstance, Word>
|
|
|
|
where
|
|
|
|
<Word as TryFrom<u32>>::Error: core::fmt::Debug,
|
2024-07-04 17:10:01 +02:00
|
|
|
{
|
2024-09-18 15:07:29 +02:00
|
|
|
#[inline(always)]
|
|
|
|
fn write_fifo(&self, data: u32) -> nb::Result<(), Infallible> {
|
|
|
|
if self.spi.status().read().tnf().bit_is_clear() {
|
|
|
|
return Err(nb::Error::WouldBlock);
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
self.write_fifo_unchecked(data);
|
|
|
|
Ok(())
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
#[inline(always)]
|
|
|
|
fn write_fifo_unchecked(&self, data: u32) {
|
|
|
|
self.spi.data().write(|w| unsafe { w.bits(data) });
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn read_fifo(&self) -> nb::Result<u32, Infallible> {
|
|
|
|
if self.spi.status().read().rne().bit_is_clear() {
|
|
|
|
return Err(nb::Error::WouldBlock);
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
Ok(self.read_fifo_unchecked())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn read_fifo_unchecked(&self) -> u32 {
|
|
|
|
self.spi.data().read().bits()
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<SpiI: Instance, Word: WordProvider> embedded_hal::spi::ErrorType for SpiBase<SpiI, Word> {
|
|
|
|
type Error = Infallible;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<SpiI: Instance, Word: WordProvider> embedded_hal::spi::SpiBus<Word> for SpiBase<SpiI, Word>
|
|
|
|
where
|
|
|
|
<Word as TryFrom<u32>>::Error: core::fmt::Debug,
|
|
|
|
{
|
|
|
|
fn read(&mut self, words: &mut [Word]) -> Result<(), Self::Error> {
|
|
|
|
self.transfer_preparation(words)?;
|
|
|
|
let mut current_read_idx = 0;
|
2024-09-18 15:07:29 +02:00
|
|
|
let mut current_write_idx = self.initial_send_fifo_pumping_with_fill_words(words.len());
|
2024-07-04 17:10:01 +02:00
|
|
|
loop {
|
|
|
|
if current_read_idx < words.len() {
|
2024-09-18 15:07:29 +02:00
|
|
|
words[current_read_idx] = (nb::block!(self.read_fifo())? & Word::MASK)
|
|
|
|
.try_into()
|
|
|
|
.unwrap();
|
2024-07-04 17:10:01 +02:00
|
|
|
current_read_idx += 1;
|
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
if current_write_idx < words.len() {
|
|
|
|
if current_write_idx == words.len() - 1 && self.bmstall {
|
|
|
|
nb::block!(self.write_fifo(self.fill_word.into() | BMSTART_BMSTOP_MASK))?;
|
|
|
|
} else {
|
|
|
|
nb::block!(self.write_fifo(self.fill_word.into()))?;
|
|
|
|
}
|
|
|
|
current_write_idx += 1;
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
if current_read_idx >= words.len() && current_write_idx >= words.len() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
fn write(&mut self, words: &[Word]) -> Result<(), Self::Error> {
|
|
|
|
self.transfer_preparation(words)?;
|
2024-09-18 15:07:29 +02:00
|
|
|
let mut current_write_idx = self.initial_send_fifo_pumping_with_words(words);
|
2024-07-04 17:10:01 +02:00
|
|
|
while current_write_idx < words.len() {
|
2024-09-18 15:07:29 +02:00
|
|
|
if current_write_idx == words.len() - 1 && self.bmstall {
|
|
|
|
nb::block!(self.write_fifo(words[current_write_idx].into() | BMSTART_BMSTOP_MASK))?;
|
|
|
|
} else {
|
|
|
|
nb::block!(self.write_fifo(words[current_write_idx].into()))?;
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
current_write_idx += 1;
|
|
|
|
// Ignore received words.
|
|
|
|
if self.spi.status().read().rne().bit_is_set() {
|
|
|
|
self.clear_rx_fifo();
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
fn transfer(&mut self, read: &mut [Word], write: &[Word]) -> Result<(), Self::Error> {
|
|
|
|
self.transfer_preparation(write)?;
|
|
|
|
let mut current_read_idx = 0;
|
2024-09-18 15:07:29 +02:00
|
|
|
let mut current_write_idx = self.initial_send_fifo_pumping_with_words(write);
|
2024-07-04 17:10:01 +02:00
|
|
|
while current_read_idx < read.len() || current_write_idx < write.len() {
|
|
|
|
if current_write_idx < write.len() {
|
2024-09-18 15:07:29 +02:00
|
|
|
if current_write_idx == write.len() - 1 && self.bmstall {
|
|
|
|
nb::block!(
|
|
|
|
self.write_fifo(write[current_write_idx].into() | BMSTART_BMSTOP_MASK)
|
|
|
|
)?;
|
|
|
|
} else {
|
|
|
|
nb::block!(self.write_fifo(write[current_write_idx].into()))?;
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
current_write_idx += 1;
|
|
|
|
}
|
|
|
|
if current_read_idx < read.len() {
|
2024-09-18 15:07:29 +02:00
|
|
|
read[current_read_idx] = (nb::block!(self.read_fifo())? & Word::MASK)
|
|
|
|
.try_into()
|
|
|
|
.unwrap();
|
2024-07-04 17:10:01 +02:00
|
|
|
current_read_idx += 1;
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
Ok(())
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
fn transfer_in_place(&mut self, words: &mut [Word]) -> Result<(), Self::Error> {
|
|
|
|
self.transfer_preparation(words)?;
|
|
|
|
let mut current_read_idx = 0;
|
2024-09-18 15:07:29 +02:00
|
|
|
let mut current_write_idx = self.initial_send_fifo_pumping_with_words(words);
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
while current_read_idx < words.len() || current_write_idx < words.len() {
|
|
|
|
if current_write_idx < words.len() {
|
2024-09-18 15:07:29 +02:00
|
|
|
if current_write_idx == words.len() - 1 && self.bmstall {
|
|
|
|
nb::block!(
|
|
|
|
self.write_fifo(words[current_write_idx].into() | BMSTART_BMSTOP_MASK)
|
|
|
|
)?;
|
|
|
|
} else {
|
|
|
|
nb::block!(self.write_fifo(words[current_write_idx].into()))?;
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
current_write_idx += 1;
|
|
|
|
}
|
|
|
|
if current_read_idx < words.len() && current_read_idx < current_write_idx {
|
2024-09-18 15:07:29 +02:00
|
|
|
words[current_read_idx] = (nb::block!(self.read_fifo())? & Word::MASK)
|
|
|
|
.try_into()
|
|
|
|
.unwrap();
|
2024-07-04 17:10:01 +02:00
|
|
|
current_read_idx += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
fn flush(&mut self) -> Result<(), Self::Error> {
|
2024-09-18 15:07:29 +02:00
|
|
|
self.flush_internal();
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<
|
|
|
|
SpiI: Instance,
|
|
|
|
Sck: PinSck<SpiI>,
|
|
|
|
Miso: PinMiso<SpiI>,
|
|
|
|
Mosi: PinMosi<SpiI>,
|
|
|
|
Word: WordProvider,
|
|
|
|
> Spi<SpiI, (Sck, Miso, Mosi), Word>
|
|
|
|
where
|
|
|
|
<Word as TryFrom<u32>>::Error: core::fmt::Debug,
|
|
|
|
{
|
|
|
|
/// Create a new SPI struct
|
|
|
|
///
|
|
|
|
/// You can delete the pin type information by calling the
|
|
|
|
/// [`downgrade`](Self::downgrade) function
|
|
|
|
///
|
|
|
|
/// ## Arguments
|
|
|
|
/// * `syscfg` - Can be passed optionally to enable the peripheral clock
|
|
|
|
/// * `sys_clk` - System clock
|
|
|
|
/// * `spi` - SPI bus to use
|
|
|
|
/// * `pins` - Pins to be used for SPI transactions. These pins are consumed
|
|
|
|
/// to ensure the pins can not be used for other purposes anymore
|
|
|
|
/// * `spi_cfg` - Configuration specific to the SPI bus
|
|
|
|
pub fn new(
|
|
|
|
syscfg: &mut pac::Sysconfig,
|
|
|
|
sys_clk: impl Into<Hertz>,
|
|
|
|
spi: SpiI,
|
|
|
|
pins: (Sck, Miso, Mosi),
|
|
|
|
spi_cfg: SpiConfig,
|
|
|
|
) -> Self {
|
|
|
|
enable_peripheral_clock(syscfg, SpiI::PERIPH_SEL);
|
|
|
|
let SpiConfig {
|
|
|
|
clk,
|
|
|
|
init_mode,
|
|
|
|
blockmode,
|
|
|
|
bmstall,
|
|
|
|
ms,
|
|
|
|
slave_output_disable,
|
|
|
|
loopback_mode,
|
|
|
|
master_delayer_capture,
|
|
|
|
} = spi_cfg;
|
|
|
|
|
|
|
|
let (cpo_bit, cph_bit) = mode_to_cpo_cph_bit(init_mode);
|
|
|
|
spi.ctrl0().write(|w| {
|
|
|
|
unsafe {
|
|
|
|
w.size().bits(Word::word_reg());
|
|
|
|
w.scrdv().bits(clk.scrdv);
|
|
|
|
// Clear clock phase and polarity. Will be set to correct value for each
|
|
|
|
// transfer
|
|
|
|
w.spo().bit(cpo_bit);
|
|
|
|
w.sph().bit(cph_bit)
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
spi.ctrl1().write(|w| {
|
|
|
|
w.lbm().bit(loopback_mode);
|
|
|
|
w.sod().bit(slave_output_disable);
|
|
|
|
w.ms().bit(ms);
|
|
|
|
w.mdlycap().bit(master_delayer_capture);
|
|
|
|
w.blockmode().bit(blockmode);
|
|
|
|
w.bmstall().bit(bmstall);
|
|
|
|
unsafe { w.ss().bits(0) }
|
|
|
|
});
|
|
|
|
spi.clkprescale()
|
|
|
|
.write(|w| unsafe { w.bits(clk.prescale_val as u32) });
|
|
|
|
|
|
|
|
spi.fifo_clr().write(|w| {
|
|
|
|
w.rxfifo().set_bit();
|
|
|
|
w.txfifo().set_bit()
|
|
|
|
});
|
|
|
|
// Enable the peripheral as the last step as recommended in the
|
|
|
|
// programmers guide
|
|
|
|
spi.ctrl1().modify(|_, w| w.enable().set_bit());
|
|
|
|
Spi {
|
|
|
|
inner: SpiBase {
|
|
|
|
spi,
|
|
|
|
cfg: spi_cfg,
|
|
|
|
sys_clk: sys_clk.into(),
|
|
|
|
fill_word: Default::default(),
|
|
|
|
bmstall,
|
|
|
|
blockmode,
|
|
|
|
word: PhantomData,
|
|
|
|
},
|
|
|
|
pins,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
delegate::delegate! {
|
|
|
|
to self.inner {
|
|
|
|
#[inline]
|
|
|
|
pub fn cfg_clock(&mut self, cfg: SpiClkConfig);
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn cfg_clock_from_div(&mut self, div: u16) -> Result<(), SpiClkConfigError>;
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn cfg_mode(&mut self, mode: Mode);
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn perid(&self) -> u32;
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn fill_word(&self) -> Word;
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn spi(&self) -> &SpiI;
|
|
|
|
|
|
|
|
/// Configure the hardware chip select given a hardware chip select ID.
|
|
|
|
#[inline]
|
|
|
|
pub fn cfg_hw_cs(&mut self, hw_cs: HwChipSelectId);
|
|
|
|
|
|
|
|
/// Configure the hardware chip select given a physical hardware CS pin.
|
|
|
|
#[inline]
|
|
|
|
pub fn cfg_hw_cs_with_pin<HwCs: OptionalHwCs<SpiI>>(&mut self, _hwcs: &HwCs);
|
|
|
|
|
|
|
|
/// Disables the hardware chip select functionality. This can be used when performing
|
|
|
|
/// external chip select handling, for example with GPIO pins.
|
|
|
|
#[inline]
|
|
|
|
pub fn cfg_hw_cs_disable(&mut self);
|
|
|
|
|
|
|
|
/// Utility function to configure all relevant transfer parameters in one go.
|
|
|
|
/// This is useful if multiple devices with different clock and mode configurations
|
|
|
|
/// are connected to one bus.
|
|
|
|
pub fn cfg_transfer<HwCs: OptionalHwCs<SpiI>>(
|
|
|
|
&mut self, transfer_cfg: &TransferConfigWithHwcs<HwCs>
|
|
|
|
);
|
|
|
|
|
|
|
|
/// Low level function to write a word to the SPI FIFO but also checks whether
|
|
|
|
/// there is actually data in the FIFO.
|
|
|
|
///
|
|
|
|
/// Uses the [nb] API to allow usage in blocking and non-blocking contexts.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn write_fifo(&self, data: u32) -> nb::Result<(), Infallible>;
|
|
|
|
|
|
|
|
/// Low level function to write a word to the SPI FIFO.
|
|
|
|
///
|
|
|
|
/// This does not necesarily mean there is a space in the FIFO available.
|
|
|
|
/// Use [Self::write_fifo] function to write a word into the FIFO reliably using the
|
|
|
|
/// [nb] API.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn write_fifo_unchecked(&self, data: u32);
|
|
|
|
|
|
|
|
/// Low level function to read a word from the SPI FIFO. Must be preceeded by a
|
|
|
|
/// [Self::write_fifo] call.
|
|
|
|
///
|
|
|
|
/// Uses the [nb] API to allow usage in blocking and non-blocking contexts.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn read_fifo(&self) -> nb::Result<u32, Infallible>;
|
|
|
|
|
|
|
|
/// Low level function to read a word from from the SPI FIFO.
|
|
|
|
///
|
|
|
|
/// This does not necesarily mean there is a word in the FIFO available.
|
|
|
|
/// Use the [Self::read_fifo] function to read a word from the FIFO reliably using the [nb]
|
|
|
|
/// API.
|
|
|
|
/// You might also need to mask the value to ignore the BMSTART/BMSTOP bit.
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn read_fifo_unchecked(&self) -> u32;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn set_fill_word(&mut self, fill_word: Word) {
|
|
|
|
self.inner.fill_word = fill_word;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Releases the SPI peripheral and associated pins
|
|
|
|
pub fn release(self) -> (SpiI, (Sck, Miso, Mosi), SpiConfig) {
|
|
|
|
(self.inner.spi, self.pins, self.inner.cfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn downgrade(self) -> SpiBase<SpiI, Word> {
|
|
|
|
self.inner
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<
|
|
|
|
SpiI: Instance,
|
|
|
|
Sck: PinSck<SpiI>,
|
|
|
|
Miso: PinMiso<SpiI>,
|
|
|
|
Mosi: PinMosi<SpiI>,
|
|
|
|
Word: WordProvider,
|
|
|
|
> SpiLowLevel for Spi<SpiI, (Sck, Miso, Mosi), Word>
|
|
|
|
where
|
|
|
|
<Word as TryFrom<u32>>::Error: core::fmt::Debug,
|
|
|
|
{
|
|
|
|
delegate::delegate! {
|
|
|
|
to self.inner {
|
|
|
|
fn write_fifo(&self, data: u32) -> nb::Result<(), Infallible>;
|
|
|
|
fn write_fifo_unchecked(&self, data: u32);
|
|
|
|
fn read_fifo(&self) -> nb::Result<u32, Infallible>;
|
|
|
|
fn read_fifo_unchecked(&self) -> u32;
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<
|
|
|
|
SpiI: Instance,
|
|
|
|
Word: WordProvider,
|
|
|
|
Sck: PinSck<SpiI>,
|
|
|
|
Miso: PinMiso<SpiI>,
|
|
|
|
Mosi: PinMosi<SpiI>,
|
|
|
|
> embedded_hal::spi::ErrorType for Spi<SpiI, (Sck, Miso, Mosi), Word>
|
|
|
|
{
|
|
|
|
type Error = Infallible;
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-07-04 17:10:01 +02:00
|
|
|
impl<
|
|
|
|
SpiI: Instance,
|
|
|
|
Word: WordProvider,
|
|
|
|
Sck: PinSck<SpiI>,
|
|
|
|
Miso: PinMiso<SpiI>,
|
|
|
|
Mosi: PinMosi<SpiI>,
|
|
|
|
> embedded_hal::spi::SpiBus<Word> for Spi<SpiI, (Sck, Miso, Mosi), Word>
|
|
|
|
where
|
|
|
|
<Word as TryFrom<u32>>::Error: core::fmt::Debug,
|
|
|
|
{
|
2024-09-18 15:07:29 +02:00
|
|
|
delegate::delegate! {
|
|
|
|
to self.inner {
|
|
|
|
fn read(&mut self, words: &mut [Word]) -> Result<(), Self::Error>;
|
|
|
|
fn write(&mut self, words: &[Word]) -> Result<(), Self::Error>;
|
|
|
|
fn transfer(&mut self, read: &mut [Word], write: &[Word]) -> Result<(), Self::Error>;
|
|
|
|
fn transfer_in_place(&mut self, words: &mut [Word]) -> Result<(), Self::Error>;
|
|
|
|
fn flush(&mut self) -> Result<(), Self::Error>;
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
/// Changing the word size also requires a type conversion
|
|
|
|
impl<SpiI: Instance, Sck: PinSck<SpiI>, Miso: PinMiso<SpiI>, Mosi: PinMosi<SpiI>>
|
|
|
|
From<Spi<SpiI, (Sck, Miso, Mosi), u8>> for Spi<SpiI, (Sck, Miso, Mosi), u16>
|
|
|
|
{
|
|
|
|
fn from(old_spi: Spi<SpiI, (Sck, Miso, Mosi), u8>) -> Self {
|
|
|
|
old_spi
|
|
|
|
.inner
|
|
|
|
.spi
|
|
|
|
.ctrl0()
|
|
|
|
.modify(|_, w| unsafe { w.size().bits(WordSize::SixteenBits as u8) });
|
|
|
|
Spi {
|
|
|
|
inner: SpiBase {
|
|
|
|
spi: old_spi.inner.spi,
|
|
|
|
cfg: old_spi.inner.cfg,
|
|
|
|
blockmode: old_spi.inner.blockmode,
|
|
|
|
fill_word: Default::default(),
|
|
|
|
bmstall: old_spi.inner.bmstall,
|
|
|
|
sys_clk: old_spi.inner.sys_clk,
|
|
|
|
word: PhantomData,
|
|
|
|
},
|
|
|
|
pins: old_spi.pins,
|
|
|
|
}
|
2024-07-04 17:10:01 +02:00
|
|
|
}
|
2024-09-18 15:07:29 +02:00
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
|
2024-09-18 15:07:29 +02:00
|
|
|
/// Changing the word size also requires a type conversion
|
|
|
|
impl<SpiI: Instance, Sck: PinSck<SpiI>, Miso: PinMiso<SpiI>, Mosi: PinMosi<SpiI>>
|
|
|
|
From<Spi<SpiI, (Sck, Miso, Mosi), u16>> for Spi<SpiI, (Sck, Miso, Mosi), u8>
|
|
|
|
{
|
|
|
|
fn from(old_spi: Spi<SpiI, (Sck, Miso, Mosi), u16>) -> Self {
|
|
|
|
old_spi
|
|
|
|
.inner
|
|
|
|
.spi
|
|
|
|
.ctrl0()
|
|
|
|
.modify(|_, w| unsafe { w.size().bits(WordSize::EightBits as u8) });
|
|
|
|
Spi {
|
|
|
|
inner: SpiBase {
|
|
|
|
spi: old_spi.inner.spi,
|
|
|
|
cfg: old_spi.inner.cfg,
|
|
|
|
blockmode: old_spi.inner.blockmode,
|
|
|
|
bmstall: old_spi.inner.bmstall,
|
|
|
|
sys_clk: old_spi.inner.sys_clk,
|
|
|
|
fill_word: Default::default(),
|
|
|
|
word: PhantomData,
|
|
|
|
},
|
|
|
|
pins: old_spi.pins,
|
|
|
|
}
|
2024-06-16 16:16:45 +02:00
|
|
|
}
|
|
|
|
}
|