Merge pull request #1987 from tyler-gilbert/issue-1986-add-sai-receiver-driver
Issue 1986 add sai receiver driver
This commit is contained in:
commit
a7b1e51650
@ -4,7 +4,7 @@ use embassy_embedded_hal::SetConfig;
|
|||||||
use embassy_hal_internal::{into_ref, PeripheralRef};
|
use embassy_hal_internal::{into_ref, PeripheralRef};
|
||||||
|
|
||||||
pub use crate::dma::word;
|
pub use crate::dma::word;
|
||||||
use crate::dma::{ringbuffer, Channel, ReadableRingBuffer, TransferOptions, WritableRingBuffer};
|
use crate::dma::{ringbuffer, Channel, ReadableRingBuffer, Request, TransferOptions, WritableRingBuffer};
|
||||||
use crate::gpio::sealed::{AFType, Pin as _};
|
use crate::gpio::sealed::{AFType, Pin as _};
|
||||||
use crate::gpio::AnyPin;
|
use crate::gpio::AnyPin;
|
||||||
use crate::pac::sai::{vals, Sai as Regs};
|
use crate::pac::sai::{vals, Sai as Regs};
|
||||||
@ -48,8 +48,8 @@ pub enum Mode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone)]
|
||||||
enum TxRx {
|
pub enum TxRx {
|
||||||
Transmiter,
|
Transmitter,
|
||||||
Receiver,
|
Receiver,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -57,7 +57,7 @@ impl Mode {
|
|||||||
#[cfg(any(sai_v1, sai_v2, sai_v3, sai_v4))]
|
#[cfg(any(sai_v1, sai_v2, sai_v3, sai_v4))]
|
||||||
const fn mode(&self, tx_rx: TxRx) -> vals::Mode {
|
const fn mode(&self, tx_rx: TxRx) -> vals::Mode {
|
||||||
match tx_rx {
|
match tx_rx {
|
||||||
TxRx::Transmiter => match self {
|
TxRx::Transmitter => match self {
|
||||||
Mode::Master => vals::Mode::MASTERTX,
|
Mode::Master => vals::Mode::MASTERTX,
|
||||||
Mode::Slave => vals::Mode::SLAVETX,
|
Mode::Slave => vals::Mode::SLAVETX,
|
||||||
},
|
},
|
||||||
@ -206,12 +206,13 @@ impl Protocol {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone, PartialEq)]
|
||||||
pub enum SyncEnable {
|
pub enum SyncEnable {
|
||||||
Asynchronous,
|
Asynchronous,
|
||||||
/// Syncs with the other A/B sub-block within the SAI unit
|
/// Syncs with the other A/B sub-block within the SAI unit
|
||||||
Internal,
|
Internal,
|
||||||
/// Syncs with a sub-block in the other SAI unit - use set_sync_output() and set_sync_input()
|
/// Syncs with a sub-block in the other SAI unit - use set_sync_output() and set_sync_input()
|
||||||
|
#[cfg(any(sai_v4))]
|
||||||
External,
|
External,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -221,6 +222,7 @@ impl SyncEnable {
|
|||||||
match self {
|
match self {
|
||||||
SyncEnable::Asynchronous => vals::Syncen::ASYNCHRONOUS,
|
SyncEnable::Asynchronous => vals::Syncen::ASYNCHRONOUS,
|
||||||
SyncEnable::Internal => vals::Syncen::INTERNAL,
|
SyncEnable::Internal => vals::Syncen::INTERNAL,
|
||||||
|
#[cfg(any(sai_v4))]
|
||||||
SyncEnable::External => vals::Syncen::EXTERNAL,
|
SyncEnable::External => vals::Syncen::EXTERNAL,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -425,6 +427,7 @@ impl MasterClockDivider {
|
|||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone)]
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
pub mode: Mode,
|
pub mode: Mode,
|
||||||
|
pub tx_rx: TxRx,
|
||||||
pub sync_enable: SyncEnable,
|
pub sync_enable: SyncEnable,
|
||||||
pub is_sync_output: bool,
|
pub is_sync_output: bool,
|
||||||
pub protocol: Protocol,
|
pub protocol: Protocol,
|
||||||
@ -455,6 +458,7 @@ impl Default for Config {
|
|||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Self {
|
Self {
|
||||||
mode: Mode::Master,
|
mode: Mode::Master,
|
||||||
|
tx_rx: TxRx::Transmitter,
|
||||||
is_sync_output: false,
|
is_sync_output: false,
|
||||||
sync_enable: SyncEnable::Asynchronous,
|
sync_enable: SyncEnable::Asynchronous,
|
||||||
protocol: Protocol::Free,
|
protocol: Protocol::Free,
|
||||||
@ -498,42 +502,127 @@ impl Config {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone)]
|
||||||
pub enum SubBlock {
|
enum WhichSubBlock {
|
||||||
A = 0,
|
A = 0,
|
||||||
B = 1,
|
B = 1,
|
||||||
}
|
}
|
||||||
|
|
||||||
enum RingBuffer<'d, C: Channel, W: word::Word> {
|
enum RingBuffer<'d, C: Channel, W: word::Word> {
|
||||||
Writable(WritableRingBuffer<'d, C, W>),
|
Writable(WritableRingBuffer<'d, C, W>),
|
||||||
#[allow(dead_code)] // remove this after implementing new_* functions for receiver
|
|
||||||
Readable(ReadableRingBuffer<'d, C, W>),
|
Readable(ReadableRingBuffer<'d, C, W>),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(sai_v1, sai_v2, sai_v3, sai_v4))]
|
#[cfg(any(sai_v1, sai_v2, sai_v3, sai_v4))]
|
||||||
fn wdr<W: word::Word>(w: crate::pac::sai::Sai, sub_block: SubBlock) -> *mut W {
|
fn dr<W: word::Word>(w: crate::pac::sai::Sai, sub_block: WhichSubBlock) -> *mut W {
|
||||||
let ch = w.ch(sub_block as usize);
|
let ch = w.ch(sub_block as usize);
|
||||||
ch.dr().as_ptr() as _
|
ch.dr().as_ptr() as _
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Sai<'d, T: Instance, C: Channel, W: word::Word> {
|
pub struct SubBlock<'d, T: Instance, C: Channel, W: word::Word> {
|
||||||
_peri: PeripheralRef<'d, T>,
|
_peri: PeripheralRef<'d, T>,
|
||||||
sd: Option<PeripheralRef<'d, AnyPin>>,
|
sd: Option<PeripheralRef<'d, AnyPin>>,
|
||||||
fs: Option<PeripheralRef<'d, AnyPin>>,
|
fs: Option<PeripheralRef<'d, AnyPin>>,
|
||||||
sck: Option<PeripheralRef<'d, AnyPin>>,
|
sck: Option<PeripheralRef<'d, AnyPin>>,
|
||||||
mclk: Option<PeripheralRef<'d, AnyPin>>,
|
mclk: Option<PeripheralRef<'d, AnyPin>>,
|
||||||
ring_buffer: RingBuffer<'d, C, W>,
|
ring_buffer: RingBuffer<'d, C, W>,
|
||||||
sub_block: SubBlock,
|
sub_block: WhichSubBlock,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
pub struct SubBlockA {}
|
||||||
fn get_transmitter_af_types(mode: Mode) -> (AFType, AFType) {
|
pub struct SubBlockB {}
|
||||||
|
|
||||||
|
pub struct Sai<'d, T: Instance> {
|
||||||
|
_peri: PeripheralRef<'d, T>,
|
||||||
|
sub_block_a_peri: Option<PeripheralRef<'d, T>>,
|
||||||
|
sub_block_b_peri: Option<PeripheralRef<'d, T>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
// return the type for (sd, sck)
|
||||||
|
fn get_af_types(mode: Mode, tx_rx: TxRx) -> (AFType, AFType) {
|
||||||
|
(
|
||||||
|
//sd is defined by tx/rx mode
|
||||||
|
match tx_rx {
|
||||||
|
TxRx::Transmitter => AFType::OutputPushPull,
|
||||||
|
TxRx::Receiver => AFType::Input,
|
||||||
|
},
|
||||||
|
//clocks (mclk, sck and fs) are defined by master/slave
|
||||||
match mode {
|
match mode {
|
||||||
Mode::Master => (AFType::OutputPushPull, AFType::OutputPushPull),
|
Mode::Master => AFType::OutputPushPull,
|
||||||
Mode::Slave => (AFType::OutputPushPull, AFType::Input),
|
Mode::Slave => AFType::Input,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_ring_buffer<'d, T: Instance, C: Channel, W: word::Word>(
|
||||||
|
dma: impl Peripheral<P = C> + 'd,
|
||||||
|
dma_buf: &'d mut [W],
|
||||||
|
request: Request,
|
||||||
|
sub_block: WhichSubBlock,
|
||||||
|
tx_rx: TxRx,
|
||||||
|
) -> RingBuffer<'d, C, W> {
|
||||||
|
let opts = TransferOptions {
|
||||||
|
half_transfer_ir: true,
|
||||||
|
//the new_write() and new_read() always use circular mode
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
match tx_rx {
|
||||||
|
TxRx::Transmitter => RingBuffer::Writable(unsafe {
|
||||||
|
WritableRingBuffer::new_write(dma, request, dr(T::REGS, sub_block), dma_buf, opts)
|
||||||
|
}),
|
||||||
|
TxRx::Receiver => RingBuffer::Readable(unsafe {
|
||||||
|
ReadableRingBuffer::new_read(dma, request, dr(T::REGS, sub_block), dma_buf, opts)
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'d, T: Instance> Sai<'d, T> {
|
||||||
|
pub fn new(peri: impl Peripheral<P = T> + 'd) -> Self {
|
||||||
|
T::enable();
|
||||||
|
T::reset();
|
||||||
|
|
||||||
|
Self {
|
||||||
|
_peri: unsafe { peri.clone_unchecked().into_ref() },
|
||||||
|
sub_block_a_peri: Some(unsafe { peri.clone_unchecked().into_ref() }),
|
||||||
|
sub_block_b_peri: Some(peri.into_ref()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_asynchronous_transmitter_with_mclk_a(
|
pub fn take_sub_block_a(self: &mut Self) -> Option<PeripheralRef<'d, T>> {
|
||||||
|
if self.sub_block_a_peri.is_some() {
|
||||||
|
self.sub_block_a_peri.take()
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn take_sub_block_b(self: &mut Self) -> Option<PeripheralRef<'d, T>> {
|
||||||
|
if self.sub_block_b_peri.is_some() {
|
||||||
|
self.sub_block_b_peri.take()
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn update_synchronous_config(config: &mut Config) {
|
||||||
|
config.mode = Mode::Slave;
|
||||||
|
config.is_sync_output = false;
|
||||||
|
|
||||||
|
#[cfg(any(sai_v1, sai_v2, sai_v3))]
|
||||||
|
{
|
||||||
|
config.sync_enable = SyncEnable::Internal;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(any(sai_v4))]
|
||||||
|
{
|
||||||
|
//this must either be Internal or External
|
||||||
|
//The asynchronous sub-block on the same SAI needs to enable is_sync_output
|
||||||
|
assert!(config.sync_enable != SyncEnable::Asynchronous);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SubBlockA {
|
||||||
|
pub fn new_asynchronous_with_mclk<'d, T: Instance, C: Channel, W: word::Word>(
|
||||||
peri: impl Peripheral<P = T> + 'd,
|
peri: impl Peripheral<P = T> + 'd,
|
||||||
sck: impl Peripheral<P = impl SckAPin<T>> + 'd,
|
sck: impl Peripheral<P = impl SckAPin<T>> + 'd,
|
||||||
sd: impl Peripheral<P = impl SdAPin<T>> + 'd,
|
sd: impl Peripheral<P = impl SdAPin<T>> + 'd,
|
||||||
@ -542,23 +631,25 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
dma: impl Peripheral<P = C> + 'd,
|
dma: impl Peripheral<P = C> + 'd,
|
||||||
dma_buf: &'d mut [W],
|
dma_buf: &'d mut [W],
|
||||||
mut config: Config,
|
mut config: Config,
|
||||||
) -> Self
|
) -> SubBlock<T, C, W>
|
||||||
where
|
where
|
||||||
C: Channel + DmaA<T>,
|
C: Channel + DmaA<T>,
|
||||||
{
|
{
|
||||||
into_ref!(mclk);
|
into_ref!(mclk);
|
||||||
|
|
||||||
mclk.set_as_af(mclk.af_num(), AFType::OutputPushPull);
|
let (_sd_af_type, ck_af_type) = get_af_types(config.mode, config.tx_rx);
|
||||||
|
|
||||||
|
mclk.set_as_af(mclk.af_num(), ck_af_type);
|
||||||
mclk.set_speed(crate::gpio::Speed::VeryHigh);
|
mclk.set_speed(crate::gpio::Speed::VeryHigh);
|
||||||
|
|
||||||
if config.master_clock_divider == MasterClockDivider::MasterClockDisabled {
|
if config.master_clock_divider == MasterClockDivider::MasterClockDisabled {
|
||||||
config.master_clock_divider = MasterClockDivider::Div1;
|
config.master_clock_divider = MasterClockDivider::Div1;
|
||||||
}
|
}
|
||||||
|
|
||||||
Self::new_asynchronous_transmitter_a(peri, sck, sd, fs, dma, dma_buf, config)
|
Self::new_asynchronous(peri, sck, sd, fs, dma, dma_buf, config)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_asynchronous_transmitter_a(
|
pub fn new_asynchronous<'d, T: Instance, C: Channel, W: word::Word>(
|
||||||
peri: impl Peripheral<P = T> + 'd,
|
peri: impl Peripheral<P = T> + 'd,
|
||||||
sck: impl Peripheral<P = impl SckAPin<T>> + 'd,
|
sck: impl Peripheral<P = impl SckAPin<T>> + 'd,
|
||||||
sd: impl Peripheral<P = impl SdAPin<T>> + 'd,
|
sd: impl Peripheral<P = impl SdAPin<T>> + 'd,
|
||||||
@ -566,13 +657,13 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
dma: impl Peripheral<P = C> + 'd,
|
dma: impl Peripheral<P = C> + 'd,
|
||||||
dma_buf: &'d mut [W],
|
dma_buf: &'d mut [W],
|
||||||
config: Config,
|
config: Config,
|
||||||
) -> Self
|
) -> SubBlock<T, C, W>
|
||||||
where
|
where
|
||||||
C: Channel + DmaA<T>,
|
C: Channel + DmaA<T>,
|
||||||
{
|
{
|
||||||
into_ref!(peri, dma, sck, sd, fs);
|
into_ref!(peri, dma, sck, sd, fs);
|
||||||
|
|
||||||
let (sd_af_type, ck_af_type) = Self::get_transmitter_af_types(config.mode);
|
let (sd_af_type, ck_af_type) = get_af_types(config.mode, config.tx_rx);
|
||||||
sd.set_as_af(sd.af_num(), sd_af_type);
|
sd.set_as_af(sd.af_num(), sd_af_type);
|
||||||
sd.set_speed(crate::gpio::Speed::VeryHigh);
|
sd.set_speed(crate::gpio::Speed::VeryHigh);
|
||||||
|
|
||||||
@ -581,30 +672,58 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
fs.set_as_af(fs.af_num(), ck_af_type);
|
fs.set_as_af(fs.af_num(), ck_af_type);
|
||||||
fs.set_speed(crate::gpio::Speed::VeryHigh);
|
fs.set_speed(crate::gpio::Speed::VeryHigh);
|
||||||
|
|
||||||
|
let sub_block = WhichSubBlock::A;
|
||||||
let request = dma.request();
|
let request = dma.request();
|
||||||
let opts = TransferOptions {
|
|
||||||
half_transfer_ir: true,
|
|
||||||
circular: true,
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
let sub_block = SubBlock::A;
|
SubBlock::new_inner(
|
||||||
|
|
||||||
Self::new_inner(
|
|
||||||
peri,
|
peri,
|
||||||
sub_block,
|
sub_block,
|
||||||
Some(sck.map_into()),
|
Some(sck.map_into()),
|
||||||
None,
|
None,
|
||||||
Some(sd.map_into()),
|
Some(sd.map_into()),
|
||||||
Some(fs.map_into()),
|
Some(fs.map_into()),
|
||||||
RingBuffer::Writable(unsafe {
|
get_ring_buffer::<T, C, W>(dma, dma_buf, request, sub_block, config.tx_rx),
|
||||||
WritableRingBuffer::new_write(dma, request, wdr(T::REGS, sub_block), dma_buf, opts)
|
|
||||||
}),
|
|
||||||
config,
|
config,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_asynchronous_transmitter_with_mclk_b(
|
pub fn new_synchronous<'d, T: Instance, C: Channel, W: word::Word>(
|
||||||
|
peri: impl Peripheral<P = T> + 'd,
|
||||||
|
sd: impl Peripheral<P = impl SdAPin<T>> + 'd,
|
||||||
|
dma: impl Peripheral<P = C> + 'd,
|
||||||
|
dma_buf: &'d mut [W],
|
||||||
|
mut config: Config,
|
||||||
|
) -> SubBlock<T, C, W>
|
||||||
|
where
|
||||||
|
C: Channel + DmaA<T>,
|
||||||
|
{
|
||||||
|
update_synchronous_config(&mut config);
|
||||||
|
|
||||||
|
into_ref!(dma, peri, sd);
|
||||||
|
|
||||||
|
let (sd_af_type, _ck_af_type) = get_af_types(config.mode, config.tx_rx);
|
||||||
|
|
||||||
|
sd.set_as_af(sd.af_num(), sd_af_type);
|
||||||
|
sd.set_speed(crate::gpio::Speed::VeryHigh);
|
||||||
|
|
||||||
|
let sub_block = WhichSubBlock::A;
|
||||||
|
let request = dma.request();
|
||||||
|
|
||||||
|
SubBlock::new_inner(
|
||||||
|
peri,
|
||||||
|
sub_block,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
Some(sd.map_into()),
|
||||||
|
None,
|
||||||
|
get_ring_buffer::<T, C, W>(dma, dma_buf, request, sub_block, config.tx_rx),
|
||||||
|
config,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SubBlockB {
|
||||||
|
pub fn new_asynchronous_with_mclk<'d, T: Instance, C: Channel, W: word::Word>(
|
||||||
peri: impl Peripheral<P = T> + 'd,
|
peri: impl Peripheral<P = T> + 'd,
|
||||||
sck: impl Peripheral<P = impl SckBPin<T>> + 'd,
|
sck: impl Peripheral<P = impl SckBPin<T>> + 'd,
|
||||||
sd: impl Peripheral<P = impl SdBPin<T>> + 'd,
|
sd: impl Peripheral<P = impl SdBPin<T>> + 'd,
|
||||||
@ -613,23 +732,25 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
dma: impl Peripheral<P = C> + 'd,
|
dma: impl Peripheral<P = C> + 'd,
|
||||||
dma_buf: &'d mut [W],
|
dma_buf: &'d mut [W],
|
||||||
mut config: Config,
|
mut config: Config,
|
||||||
) -> Self
|
) -> SubBlock<T, C, W>
|
||||||
where
|
where
|
||||||
C: Channel + DmaB<T>,
|
C: Channel + DmaB<T>,
|
||||||
{
|
{
|
||||||
into_ref!(mclk);
|
into_ref!(mclk);
|
||||||
|
|
||||||
mclk.set_as_af(mclk.af_num(), AFType::OutputPushPull);
|
let (_sd_af_type, ck_af_type) = get_af_types(config.mode, config.tx_rx);
|
||||||
|
|
||||||
|
mclk.set_as_af(mclk.af_num(), ck_af_type);
|
||||||
mclk.set_speed(crate::gpio::Speed::VeryHigh);
|
mclk.set_speed(crate::gpio::Speed::VeryHigh);
|
||||||
|
|
||||||
if config.master_clock_divider == MasterClockDivider::MasterClockDisabled {
|
if config.master_clock_divider == MasterClockDivider::MasterClockDisabled {
|
||||||
config.master_clock_divider = MasterClockDivider::Div1;
|
config.master_clock_divider = MasterClockDivider::Div1;
|
||||||
}
|
}
|
||||||
|
|
||||||
Self::new_asynchronous_transmitter_b(peri, sck, sd, fs, dma, dma_buf, config)
|
Self::new_asynchronous(peri, sck, sd, fs, dma, dma_buf, config)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_asynchronous_transmitter_b(
|
pub fn new_asynchronous<'d, T: Instance, C: Channel, W: word::Word>(
|
||||||
peri: impl Peripheral<P = T> + 'd,
|
peri: impl Peripheral<P = T> + 'd,
|
||||||
sck: impl Peripheral<P = impl SckBPin<T>> + 'd,
|
sck: impl Peripheral<P = impl SckBPin<T>> + 'd,
|
||||||
sd: impl Peripheral<P = impl SdBPin<T>> + 'd,
|
sd: impl Peripheral<P = impl SdBPin<T>> + 'd,
|
||||||
@ -637,13 +758,13 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
dma: impl Peripheral<P = C> + 'd,
|
dma: impl Peripheral<P = C> + 'd,
|
||||||
dma_buf: &'d mut [W],
|
dma_buf: &'d mut [W],
|
||||||
config: Config,
|
config: Config,
|
||||||
) -> Self
|
) -> SubBlock<T, C, W>
|
||||||
where
|
where
|
||||||
C: Channel + DmaB<T>,
|
C: Channel + DmaB<T>,
|
||||||
{
|
{
|
||||||
into_ref!(dma, peri, sck, sd, fs);
|
into_ref!(dma, peri, sck, sd, fs);
|
||||||
|
|
||||||
let (sd_af_type, ck_af_type) = Self::get_transmitter_af_types(config.mode);
|
let (sd_af_type, ck_af_type) = get_af_types(config.mode, config.tx_rx);
|
||||||
|
|
||||||
sd.set_as_af(sd.af_num(), sd_af_type);
|
sd.set_as_af(sd.af_num(), sd_af_type);
|
||||||
sd.set_speed(crate::gpio::Speed::VeryHigh);
|
sd.set_speed(crate::gpio::Speed::VeryHigh);
|
||||||
@ -653,28 +774,57 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
fs.set_as_af(fs.af_num(), ck_af_type);
|
fs.set_as_af(fs.af_num(), ck_af_type);
|
||||||
fs.set_speed(crate::gpio::Speed::VeryHigh);
|
fs.set_speed(crate::gpio::Speed::VeryHigh);
|
||||||
|
|
||||||
|
let sub_block = WhichSubBlock::B;
|
||||||
let request = dma.request();
|
let request = dma.request();
|
||||||
let opts = TransferOptions {
|
|
||||||
half_transfer_ir: true,
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
let sub_block = SubBlock::B;
|
SubBlock::new_inner(
|
||||||
|
|
||||||
Self::new_inner(
|
|
||||||
peri,
|
peri,
|
||||||
sub_block,
|
sub_block,
|
||||||
Some(sck.map_into()),
|
Some(sck.map_into()),
|
||||||
None,
|
None,
|
||||||
Some(sd.map_into()),
|
Some(sd.map_into()),
|
||||||
Some(fs.map_into()),
|
Some(fs.map_into()),
|
||||||
RingBuffer::Writable(unsafe {
|
get_ring_buffer::<T, C, W>(dma, dma_buf, request, sub_block, config.tx_rx),
|
||||||
WritableRingBuffer::new_write(dma, request, wdr(T::REGS, sub_block), dma_buf, opts)
|
|
||||||
}),
|
|
||||||
config,
|
config,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn new_synchronous<'d, T: Instance, C: Channel, W: word::Word>(
|
||||||
|
peri: impl Peripheral<P = T> + 'd,
|
||||||
|
sd: impl Peripheral<P = impl SdBPin<T>> + 'd,
|
||||||
|
dma: impl Peripheral<P = C> + 'd,
|
||||||
|
dma_buf: &'d mut [W],
|
||||||
|
mut config: Config,
|
||||||
|
) -> SubBlock<T, C, W>
|
||||||
|
where
|
||||||
|
C: Channel + DmaB<T>,
|
||||||
|
{
|
||||||
|
update_synchronous_config(&mut config);
|
||||||
|
|
||||||
|
into_ref!(dma, peri, sd);
|
||||||
|
|
||||||
|
let (sd_af_type, _ck_af_type) = get_af_types(config.mode, config.tx_rx);
|
||||||
|
|
||||||
|
sd.set_as_af(sd.af_num(), sd_af_type);
|
||||||
|
sd.set_speed(crate::gpio::Speed::VeryHigh);
|
||||||
|
|
||||||
|
let sub_block = WhichSubBlock::B;
|
||||||
|
let request = dma.request();
|
||||||
|
|
||||||
|
SubBlock::new_inner(
|
||||||
|
peri,
|
||||||
|
sub_block,
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
Some(sd.map_into()),
|
||||||
|
None,
|
||||||
|
get_ring_buffer::<T, C, W>(dma, dma_buf, request, sub_block, config.tx_rx),
|
||||||
|
config,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'d, T: Instance, C: Channel, W: word::Word> SubBlock<'d, T, C, W> {
|
||||||
pub fn start(self: &mut Self) {
|
pub fn start(self: &mut Self) {
|
||||||
match self.ring_buffer {
|
match self.ring_buffer {
|
||||||
RingBuffer::Writable(ref mut rb) => {
|
RingBuffer::Writable(ref mut rb) => {
|
||||||
@ -695,7 +845,7 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
|
|
||||||
fn new_inner(
|
fn new_inner(
|
||||||
peri: impl Peripheral<P = T> + 'd,
|
peri: impl Peripheral<P = T> + 'd,
|
||||||
sub_block: SubBlock,
|
sub_block: WhichSubBlock,
|
||||||
sck: Option<PeripheralRef<'d, AnyPin>>,
|
sck: Option<PeripheralRef<'d, AnyPin>>,
|
||||||
mclk: Option<PeripheralRef<'d, AnyPin>>,
|
mclk: Option<PeripheralRef<'d, AnyPin>>,
|
||||||
sd: Option<PeripheralRef<'d, AnyPin>>,
|
sd: Option<PeripheralRef<'d, AnyPin>>,
|
||||||
@ -704,12 +854,21 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
config: Config,
|
config: Config,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
T::enable();
|
T::enable();
|
||||||
T::reset();
|
|
||||||
|
// can't reset here because the other sub-block might be in use
|
||||||
|
|
||||||
|
#[cfg(any(sai_v1, sai_v2, sai_v3, sai_v4))]
|
||||||
|
{
|
||||||
|
let ch = T::REGS.ch(sub_block as usize);
|
||||||
|
ch.cr1().modify(|w| w.set_saien(false));
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(any(sai_v4))]
|
#[cfg(any(sai_v4))]
|
||||||
{
|
{
|
||||||
// Not totally clear from the datasheet if this is right
|
// Not totally clear from the datasheet if this is right
|
||||||
// This is only used if using SyncEnable::External
|
// This is only used if using SyncEnable::External on the other SAI unit
|
||||||
|
// Syncing from SAIX subblock A to subblock B does not require this
|
||||||
|
// Only syncing from SAI1 subblock A/B to SAI2 subblock A/B
|
||||||
let value: u8 = if T::REGS.as_ptr() == stm32_metapac::SAI1.as_ptr() {
|
let value: u8 = if T::REGS.as_ptr() == stm32_metapac::SAI1.as_ptr() {
|
||||||
1 //this is SAI1, so sync with SAI2
|
1 //this is SAI1, so sync with SAI2
|
||||||
} else {
|
} else {
|
||||||
@ -721,8 +880,8 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
|
|
||||||
if config.is_sync_output {
|
if config.is_sync_output {
|
||||||
let syncout: u8 = match sub_block {
|
let syncout: u8 = match sub_block {
|
||||||
SubBlock::A => 0b01,
|
WhichSubBlock::A => 0b01,
|
||||||
SubBlock::B => 0b10,
|
WhichSubBlock::B => 0b10,
|
||||||
};
|
};
|
||||||
T::REGS.gcr().modify(|w| {
|
T::REGS.gcr().modify(|w| {
|
||||||
w.set_syncout(syncout);
|
w.set_syncout(syncout);
|
||||||
@ -735,7 +894,7 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
let ch = T::REGS.ch(sub_block as usize);
|
let ch = T::REGS.ch(sub_block as usize);
|
||||||
ch.cr1().modify(|w| {
|
ch.cr1().modify(|w| {
|
||||||
w.set_mode(config.mode.mode(if Self::is_transmitter(&ring_buffer) {
|
w.set_mode(config.mode.mode(if Self::is_transmitter(&ring_buffer) {
|
||||||
TxRx::Transmiter
|
TxRx::Transmitter
|
||||||
} else {
|
} else {
|
||||||
TxRx::Receiver
|
TxRx::Receiver
|
||||||
}));
|
}));
|
||||||
@ -770,7 +929,7 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
w.set_fsoff(config.frame_sync_offset.fsoff());
|
w.set_fsoff(config.frame_sync_offset.fsoff());
|
||||||
w.set_fspol(config.frame_sync_polarity.fspol());
|
w.set_fspol(config.frame_sync_polarity.fspol());
|
||||||
w.set_fsdef(config.frame_sync_definition.fsdef());
|
w.set_fsdef(config.frame_sync_definition.fsdef());
|
||||||
w.set_fsall(config.frame_sync_active_level_length.0 as u8);
|
w.set_fsall(config.frame_sync_active_level_length.0 as u8 - 1);
|
||||||
w.set_frl(config.frame_length - 1);
|
w.set_frl(config.frame_length - 1);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -782,6 +941,10 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
});
|
});
|
||||||
|
|
||||||
ch.cr1().modify(|w| w.set_saien(true));
|
ch.cr1().modify(|w| w.set_saien(true));
|
||||||
|
|
||||||
|
if ch.cr1().read().saien() == false {
|
||||||
|
panic!("SAI failed to enable. Check that config is valid (frame length, slot count, etc)");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
@ -795,6 +958,11 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn reset() {
|
||||||
|
T::enable();
|
||||||
|
T::reset();
|
||||||
|
}
|
||||||
|
|
||||||
pub fn flush(&mut self) {
|
pub fn flush(&mut self) {
|
||||||
let ch = T::REGS.ch(self.sub_block as usize);
|
let ch = T::REGS.ch(self.sub_block as usize);
|
||||||
ch.cr1().modify(|w| w.set_saien(false));
|
ch.cr1().modify(|w| w.set_saien(false));
|
||||||
@ -842,7 +1010,7 @@ impl<'d, T: Instance, C: Channel, W: word::Word> Sai<'d, T, C, W> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'d, T: Instance, C: Channel, W: word::Word> Drop for Sai<'d, T, C, W> {
|
impl<'d, T: Instance, C: Channel, W: word::Word> Drop for SubBlock<'d, T, C, W> {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
let ch = T::REGS.ch(self.sub_block as usize);
|
let ch = T::REGS.ch(self.sub_block as usize);
|
||||||
ch.cr1().modify(|w| w.set_saien(false));
|
ch.cr1().modify(|w| w.set_saien(false));
|
||||||
@ -886,9 +1054,9 @@ foreach_peripheral!(
|
|||||||
};
|
};
|
||||||
);
|
);
|
||||||
|
|
||||||
impl<'d, T: Instance, C: Channel, W: word::Word> SetConfig for Sai<'d, T, C, W> {
|
impl<'d, T: Instance> SetConfig for Sai<'d, T> {
|
||||||
type Config = Config;
|
type Config = Config;
|
||||||
fn set_config(&mut self, config: &Self::Config) {
|
fn set_config(&mut self, _config: &Self::Config) {
|
||||||
self.reconfigure(*config);
|
// self.reconfigure(*config);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user