embassy/embassy-stm32/src/sdmmc/v2.rs

1616 lines
47 KiB
Rust
Raw Normal View History

#![macro_use]
use core::default::Default;
2021-05-02 00:19:14 +00:00
use core::marker::PhantomData;
2021-05-11 04:28:45 +00:00
use core::task::Poll;
2021-05-02 00:19:14 +00:00
2021-05-09 20:36:13 +00:00
use embassy::interrupt::InterruptExt;
2021-05-02 00:19:14 +00:00
use embassy::util::{AtomicWaker, OnDrop, Unborrow};
use embassy_extras::unborrow;
use futures::future::poll_fn;
use sdio_host::{BusWidth, CardCapacity, CardStatus, CurrentState, SDStatus, CID, CSD, OCR, SCR};
use crate::fmt::*;
2021-05-31 01:21:44 +00:00
use crate::interrupt::Interrupt;
2021-05-02 00:19:14 +00:00
use crate::pac;
use crate::pac::gpio::Gpio;
use crate::pac::sdmmc::Sdmmc as RegBlock;
2021-06-03 17:50:48 +00:00
use crate::peripherals;
2021-05-02 00:19:14 +00:00
use crate::time::Hertz;
/// The signalling scheme used on the SDMMC bus
#[non_exhaustive]
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
pub enum Signalling {
SDR12,
SDR25,
SDR50,
SDR104,
DDR50,
}
impl Default for Signalling {
fn default() -> Self {
Signalling::SDR12
}
}
2021-05-11 04:28:45 +00:00
#[repr(align(4))]
pub struct DataBlock([u8; 512]);
2021-05-02 00:19:14 +00:00
/// Errors
#[non_exhaustive]
#[derive(Debug, Copy, Clone)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
pub enum Error {
Timeout,
SoftwareTimeout,
UnsupportedCardVersion,
UnsupportedCardType,
Crc,
DataCrcFail,
RxOverFlow,
NoCard,
BadClock,
SignalingSwitchFailed,
PeripheralBusy,
}
/// A SD command
struct Cmd {
cmd: u8,
arg: u32,
resp: Response,
}
#[derive(Clone, Copy, Debug, Default)]
/// SD Card
pub struct Card {
/// The type of this card
pub card_type: CardCapacity,
/// Operation Conditions Register
pub ocr: OCR,
/// Relative Card Address
pub rca: u32,
/// Card ID
pub cid: CID,
/// Card Specific Data
pub csd: CSD,
/// SD CARD Configuration Register
pub scr: SCR,
/// SD Status
pub status: SDStatus,
}
impl Card {
/// Size in bytes
pub fn size(&self) -> u64 {
// SDHC / SDXC / SDUC
u64::from(self.csd.block_count()) * 512
}
}
/// Indicates transfer direction
enum Dir {
CardToHost,
HostToCard,
}
#[repr(u8)]
enum PowerCtrl {
Off = 0b00,
On = 0b11,
}
#[repr(u32)]
#[allow(dead_code)]
#[allow(non_camel_case_types)]
enum CmdAppOper {
VOLTAGE_WINDOW_SD = 0x8010_0000,
HIGH_CAPACITY = 0x4000_0000,
SDMMC_STD_CAPACITY = 0x0000_0000,
SDMMC_CHECK_PATTERN = 0x0000_01AA,
SD_SWITCH_1_8V_CAPACITY = 0x0100_0000,
}
#[derive(Eq, PartialEq, Copy, Clone)]
enum Response {
None = 0,
Short = 1,
Long = 3,
}
/// Calculate clock divisor. Returns a SDMMC_CK less than or equal to
/// `sdmmc_ck` in Hertz.
///
/// Returns `(clk_div, clk_f)`, where `clk_div` is the divisor register
/// value and `clk_f` is the resulting new clock frequency.
fn clk_div(ker_ck: Hertz, sdmmc_ck: u32) -> Result<(u16, Hertz), Error> {
match (ker_ck.0 + sdmmc_ck - 1) / sdmmc_ck {
0 | 1 => Ok((0, ker_ck)),
x @ 2..=2046 => {
let clk_div = ((x + 1) / 2) as u16;
let clk = Hertz(ker_ck.0 / (clk_div as u32 * 2));
Ok((clk_div, clk))
}
_ => Err(Error::BadClock),
}
}
/// SDMMC configuration
///
/// You should probably change the default clock values to match your configuration
///
/// Default values:
/// hclk = 400_000_000 Hz
/// kernel_clk: 100_000_000 Hz
/// data_transfer_timeout: 5_000_000
2021-05-15 02:34:16 +00:00
#[non_exhaustive]
pub struct Config {
/// AHB clock
pub hclk: Hertz,
/// SDMMC kernel clock
pub kernel_clk: Hertz,
/// The timeout to be set for data transfers, in card bus clock periods
pub data_transfer_timeout: u32,
}
impl Default for Config {
fn default() -> Self {
Self {
hclk: Hertz(400_000_000),
kernel_clk: Hertz(100_000_000),
data_transfer_timeout: 5_000_000,
}
}
}
2021-05-02 00:19:14 +00:00
/// Sdmmc device
2021-05-09 20:36:13 +00:00
pub struct Sdmmc<'d, T: Instance, P: Pins<T>> {
sdmmc: PhantomData<&'d mut T>,
pins: P,
irq: T::Interrupt,
2021-05-15 02:34:16 +00:00
config: Config,
2021-05-02 00:19:14 +00:00
/// Current clock to card
clock: Hertz,
/// Current signalling scheme to card
signalling: Signalling,
/// Card
card: Option<Card>,
}
2021-05-09 20:36:13 +00:00
impl<'d, T: Instance, P: Pins<T>> Sdmmc<'d, T, P> {
/// # Safety
///
/// Futures that borrow this type can't be leaked
#[inline(always)]
pub unsafe fn new(
_peripheral: impl Unborrow<Target = T> + 'd,
pins: impl Unborrow<Target = P> + 'd,
irq: impl Unborrow<Target = T::Interrupt>,
2021-05-15 02:34:16 +00:00
config: Config,
2021-05-09 20:36:13 +00:00
) -> Self {
unborrow!(irq, pins);
pins.configure();
let inner = T::inner();
2021-05-15 02:34:16 +00:00
let clock = inner.new_inner(config.kernel_clk);
2021-05-09 20:36:13 +00:00
irq.set_handler(Self::on_interrupt);
irq.unpend();
irq.enable();
Self {
sdmmc: PhantomData,
pins,
irq,
2021-05-15 02:34:16 +00:00
config,
2021-05-09 20:36:13 +00:00
clock,
signalling: Default::default(),
card: None,
}
}
2021-05-02 00:19:14 +00:00
#[inline(always)]
2021-05-09 20:36:13 +00:00
pub async fn init_card(&mut self, freq: impl Into<Hertz>) -> Result<(), Error> {
let inner = T::inner();
let freq = freq.into();
inner
.init_card(
freq,
P::BUSWIDTH,
&mut self.card,
&mut self.signalling,
2021-05-15 02:34:16 +00:00
self.config.hclk,
self.config.kernel_clk,
2021-05-09 20:36:13 +00:00
&mut self.clock,
T::state(),
2021-05-15 02:34:16 +00:00
self.config.data_transfer_timeout,
2021-05-09 20:36:13 +00:00
)
.await
2021-05-02 00:19:14 +00:00
}
2021-05-11 04:28:45 +00:00
#[inline(always)]
pub async fn read_block(
&mut self,
block_idx: u32,
buffer: &mut DataBlock,
) -> Result<(), Error> {
let card_capacity = self.card()?.card_type;
let inner = T::inner();
let state = T::state();
// NOTE(unsafe) DataBlock uses align 4
let buf = unsafe { &mut *((&mut buffer.0) as *mut [u8; 512] as *mut [u32; 128]) };
inner
.read_block(
block_idx,
buf,
card_capacity,
state,
2021-05-15 02:34:16 +00:00
self.config.data_transfer_timeout,
)
.await
2021-05-11 04:28:45 +00:00
}
pub async fn write_block(&mut self, block_idx: u32, buffer: &DataBlock) -> Result<(), Error> {
let card = self.card.as_mut().ok_or(Error::NoCard)?;
let inner = T::inner();
let state = T::state();
// NOTE(unsafe) DataBlock uses align 4
let buf = unsafe { &*((&buffer.0) as *const [u8; 512] as *const [u32; 128]) };
inner
2021-05-15 02:34:16 +00:00
.write_block(
block_idx,
buf,
card,
state,
self.config.data_transfer_timeout,
)
.await
2021-05-11 04:28:45 +00:00
}
2021-05-09 20:36:13 +00:00
/// Get a reference to the initialized card
2021-05-02 00:19:14 +00:00
///
2021-05-09 20:36:13 +00:00
/// # Errors
///
/// Returns Error::NoCard if [`init_card`](#method.init_card)
/// has not previously succeeded
2021-05-11 04:28:45 +00:00
#[inline(always)]
2021-05-09 20:36:13 +00:00
pub fn card(&self) -> Result<&Card, Error> {
self.card.as_ref().ok_or(Error::NoCard)
}
2021-05-11 04:28:45 +00:00
#[inline(always)]
2021-05-09 20:36:13 +00:00
fn on_interrupt(_: *mut ()) {
let regs = T::inner();
let state = T::state();
regs.data_interrupts(false);
state.wake();
}
}
impl<'d, T: Instance, P: Pins<T>> Drop for Sdmmc<'d, T, P> {
fn drop(&mut self) {
self.irq.disable();
let inner = T::inner();
unsafe { inner.on_drop() };
self.pins.deconfigure();
2021-05-02 00:19:14 +00:00
}
2021-05-09 20:36:13 +00:00
}
pub struct SdmmcInner(pub(crate) RegBlock);
2021-05-02 00:19:14 +00:00
2021-05-09 20:36:13 +00:00
impl SdmmcInner {
2021-05-02 00:19:14 +00:00
/// # Safety
///
/// Access to `regs` registers should be exclusive
2021-05-09 20:36:13 +00:00
unsafe fn new_inner(&self, kernel_clk: Hertz) -> Hertz {
let regs = self.0;
2021-05-02 00:19:14 +00:00
// While the SD/SDIO card or eMMC is in identification mode,
// the SDMMC_CK frequency must be less than 400 kHz.
let (clkdiv, clock) = unwrap!(clk_div(kernel_clk, 400_000));
regs.clkcr().write(|w| {
w.set_widbus(0);
w.set_clkdiv(clkdiv);
w.set_pwrsav(false);
w.set_negedge(false);
w.set_hwfc_en(true);
});
// Power off, writen 00: Clock to the card is stopped;
// D[7:0], CMD, and CK are driven high.
regs.power().modify(|w| w.set_pwrctrl(PowerCtrl::Off as u8));
clock
}
/// Initializes card (if present) and sets the bus at the
/// specified frequency.
2021-05-09 20:36:13 +00:00
#[allow(clippy::too_many_arguments)]
2021-05-11 04:28:45 +00:00
async fn init_card(
2021-05-09 20:36:13 +00:00
&self,
freq: Hertz,
bus_width: BusWidth,
old_card: &mut Option<Card>,
signalling: &mut Signalling,
hclk: Hertz,
ker_ck: Hertz,
clock: &mut Hertz,
waker_reg: &AtomicWaker,
data_transfer_timeout: u32,
2021-05-09 20:36:13 +00:00
) -> Result<(), Error> {
let regs = self.0;
2021-05-02 00:19:14 +00:00
// NOTE(unsafe) We have exclusive access to the peripheral
unsafe {
regs.power().modify(|w| w.set_pwrctrl(PowerCtrl::On as u8));
self.cmd(Cmd::idle(), false)?;
// Check if cards supports CMD8 (with pattern)
self.cmd(Cmd::hs_send_ext_csd(0x1AA), false)?;
let r1 = regs.respr(0).read().cardstatus1();
let mut card = if r1 == 0x1AA {
// Card echoed back the pattern. Must be at least v2
Card::default()
} else {
return Err(Error::UnsupportedCardVersion);
};
let ocr = loop {
// Signal that next command is a app command
self.cmd(Cmd::app_cmd(0), false)?; // CMD55
let arg = CmdAppOper::VOLTAGE_WINDOW_SD as u32
| CmdAppOper::HIGH_CAPACITY as u32
| CmdAppOper::SD_SWITCH_1_8V_CAPACITY as u32;
// Initialize card
match self.cmd(Cmd::app_op_cmd(arg), false) {
// ACMD41
Ok(_) => (),
Err(Error::Crc) => (),
Err(err) => return Err(err),
}
let ocr: OCR = regs.respr(0).read().cardstatus1().into();
if !ocr.is_busy() {
// Power up done
break ocr;
}
};
if ocr.high_capacity() {
// Card is SDHC or SDXC or SDUC
card.card_type = CardCapacity::SDHC;
} else {
card.card_type = CardCapacity::SDSC;
}
card.ocr = ocr;
self.cmd(Cmd::all_send_cid(), false)?; // CMD2
let cid0 = regs.respr(0).read().cardstatus1() as u128;
let cid1 = regs.respr(1).read().cardstatus1() as u128;
let cid2 = regs.respr(2).read().cardstatus1() as u128;
let cid3 = regs.respr(3).read().cardstatus1() as u128;
let cid = (cid0 << 96) | (cid1 << 64) | (cid2 << 32) | (cid3);
card.cid = cid.into();
self.cmd(Cmd::send_rel_addr(), false)?;
card.rca = regs.respr(0).read().cardstatus1() >> 16;
self.cmd(Cmd::send_csd(card.rca << 16), false)?;
let csd0 = regs.respr(0).read().cardstatus1() as u128;
let csd1 = regs.respr(1).read().cardstatus1() as u128;
let csd2 = regs.respr(2).read().cardstatus1() as u128;
let csd3 = regs.respr(3).read().cardstatus1() as u128;
let csd = (csd0 << 96) | (csd1 << 64) | (csd2 << 32) | (csd3);
card.csd = csd.into();
self.select_card(Some(&card))?;
self.get_scr(&mut card, waker_reg, data_transfer_timeout)
.await?;
2021-05-02 00:19:14 +00:00
// Set bus width
2021-05-09 20:36:13 +00:00
let (width, acmd_arg) = match bus_width {
2021-05-02 00:19:14 +00:00
BusWidth::Eight => unimplemented!(),
BusWidth::Four if card.scr.bus_width_four() => (BusWidth::Four, 2),
_ => (BusWidth::One, 0),
};
self.cmd(Cmd::app_cmd(card.rca << 16), false)?;
self.cmd(Cmd::cmd6(acmd_arg), false)?;
// CPSMACT and DPSMACT must be 0 to set WIDBUS
self.wait_idle();
regs.clkcr().modify(|w| {
w.set_widbus(match width {
BusWidth::One => 0,
BusWidth::Four => 1,
BusWidth::Eight => 2,
_ => self::panic!("Invalid Bus Width"),
})
});
// Set Clock
if freq.0 <= 25_000_000 {
// Final clock frequency
2021-05-09 20:36:13 +00:00
self.clkcr_set_clkdiv(freq.0, width, hclk, ker_ck, clock)?;
2021-05-02 00:19:14 +00:00
} else {
// Switch to max clock for SDR12
2021-05-09 20:36:13 +00:00
self.clkcr_set_clkdiv(25_000_000, width, hclk, ker_ck, clock)?;
2021-05-02 00:19:14 +00:00
}
// Read status
self.read_sd_status(&mut card, waker_reg, data_transfer_timeout)
.await?;
2021-05-02 00:19:14 +00:00
if freq.0 > 25_000_000 {
// Switch to SDR25
2021-05-09 20:36:13 +00:00
*signalling = self
.switch_signalling_mode(Signalling::SDR25, waker_reg, data_transfer_timeout)
2021-05-09 20:36:13 +00:00
.await?;
2021-05-02 00:19:14 +00:00
2021-05-09 20:36:13 +00:00
if *signalling == Signalling::SDR25 {
2021-05-02 00:19:14 +00:00
// Set final clock frequency
2021-05-09 20:36:13 +00:00
self.clkcr_set_clkdiv(freq.0, width, hclk, ker_ck, clock)?;
2021-05-02 00:19:14 +00:00
2021-05-09 20:36:13 +00:00
if self.read_status(&card)?.state() != CurrentState::Transfer {
2021-05-02 00:19:14 +00:00
return Err(Error::SignalingSwitchFailed);
}
}
}
// Read status after signalling change
self.read_sd_status(&mut card, waker_reg, data_transfer_timeout)
.await?;
2021-05-09 20:36:13 +00:00
old_card.replace(card);
2021-05-02 00:19:14 +00:00
}
2021-05-09 20:36:13 +00:00
Ok(())
2021-05-02 00:19:14 +00:00
}
2021-05-11 04:28:45 +00:00
async fn read_block(
&self,
block_idx: u32,
buffer: &mut [u32; 128],
capacity: CardCapacity,
waker_reg: &AtomicWaker,
data_transfer_timeout: u32,
2021-05-11 04:28:45 +00:00
) -> Result<(), Error> {
// Always read 1 block of 512 bytes
// SDSC cards are byte addressed hence the blockaddress is in multiples of 512 bytes
let address = match capacity {
CardCapacity::SDSC => block_idx * 512,
_ => block_idx,
};
self.cmd(Cmd::set_block_length(512), false)?; // CMD16
let regs = self.0;
let on_drop = OnDrop::new(|| unsafe { self.on_drop() });
let buf_addr = buffer as *mut [u32; 128] as u32;
unsafe {
self.prepare_datapath_transfer(
buf_addr,
512,
9,
Dir::CardToHost,
data_transfer_timeout,
);
2021-05-11 04:28:45 +00:00
self.data_interrupts(true);
}
self.cmd(Cmd::read_single_block(address), true)?;
let res = poll_fn(|cx| {
waker_reg.register(cx.waker());
let status = unsafe { regs.star().read() };
if status.dcrcfail() {
return Poll::Ready(Err(Error::Crc));
} else if status.dtimeout() {
return Poll::Ready(Err(Error::Timeout));
} else if status.dataend() {
return Poll::Ready(Ok(()));
}
Poll::Pending
})
.await;
self.clear_interrupt_flags();
if res.is_ok() {
on_drop.defuse();
unsafe {
regs.idmactrlr().modify(|w| w.set_idmaen(false));
}
}
res
}
async fn write_block(
&self,
block_idx: u32,
buffer: &[u32; 128],
card: &mut Card,
waker_reg: &AtomicWaker,
data_transfer_timeout: u32,
2021-05-11 04:28:45 +00:00
) -> Result<(), Error> {
// Always read 1 block of 512 bytes
// SDSC cards are byte addressed hence the blockaddress is in multiples of 512 bytes
let address = match card.card_type {
CardCapacity::SDSC => block_idx * 512,
_ => block_idx,
};
self.cmd(Cmd::set_block_length(512), false)?; // CMD16
let regs = self.0;
let on_drop = OnDrop::new(|| unsafe { self.on_drop() });
let buf_addr = buffer as *const [u32; 128] as u32;
unsafe {
self.prepare_datapath_transfer(
buf_addr,
512,
9,
Dir::HostToCard,
data_transfer_timeout,
);
2021-05-11 04:28:45 +00:00
self.data_interrupts(true);
}
self.cmd(Cmd::write_single_block(address), true)?;
let res = poll_fn(|cx| {
waker_reg.register(cx.waker());
let status = unsafe { regs.star().read() };
if status.dcrcfail() {
return Poll::Ready(Err(Error::Crc));
} else if status.dtimeout() {
return Poll::Ready(Err(Error::Timeout));
} else if status.dataend() {
return Poll::Ready(Ok(()));
}
Poll::Pending
})
.await;
self.clear_interrupt_flags();
match res {
Ok(_) => {
on_drop.defuse();
unsafe {
regs.idmactrlr().modify(|w| w.set_idmaen(false));
}
// TODO: Make this configurable
let mut timeout: u32 = 0x00FF_FFFF;
2021-05-11 04:28:45 +00:00
// Try to read card status (ACMD13)
while timeout > 0 {
match self
.read_sd_status(card, waker_reg, data_transfer_timeout)
.await
{
2021-05-11 04:28:45 +00:00
Ok(_) => return Ok(()),
Err(Error::Timeout) => (), // Try again
Err(e) => return Err(e),
}
timeout -= 1;
}
Err(Error::SoftwareTimeout)
}
Err(e) => Err(e),
}
2021-05-02 00:19:14 +00:00
}
/// Get the current SDMMC bus clock
2021-05-09 20:36:13 +00:00
//pub fn clock(&self) -> Hertz {
// self.clock
//}
2021-05-02 00:19:14 +00:00
/// Wait idle on DOSNACT and CPSMACT
#[inline(always)]
2021-05-09 20:36:13 +00:00
fn wait_idle(&self) {
let regs = self.0;
2021-05-02 00:19:14 +00:00
// NOTE(unsafe) Atomic read with no side-effects
unsafe {
while {
let status = regs.star().read();
status.dpsmact() || status.cpsmact()
} {}
}
}
/// # Safety
///
/// `buffer_addr` must be valid for the whole transfer and word aligned
unsafe fn prepare_datapath_transfer(
2021-05-09 20:36:13 +00:00
&self,
2021-05-02 00:19:14 +00:00
buffer_addr: u32,
length_bytes: u32,
block_size: u8,
direction: Dir,
data_transfer_timeout: u32,
2021-05-02 00:19:14 +00:00
) {
self::assert!(block_size <= 14, "Block size up to 2^14 bytes");
2021-05-09 20:36:13 +00:00
let regs = self.0;
2021-05-02 00:19:14 +00:00
let dtdir = match direction {
Dir::CardToHost => true,
Dir::HostToCard => false,
};
// Command AND Data state machines must be idle
self.wait_idle();
2021-05-09 20:36:13 +00:00
self.clear_interrupt_flags();
2021-05-02 00:19:14 +00:00
// NOTE(unsafe) We have exclusive access to the regisers
regs.dtimer()
.write(|w| w.set_datatime(data_transfer_timeout));
2021-05-02 00:19:14 +00:00
regs.dlenr().write(|w| w.set_datalength(length_bytes));
regs.idmabase0r().write(|w| w.set_idmabase0(buffer_addr));
regs.idmactrlr().modify(|w| w.set_idmaen(true));
regs.dctrl().modify(|w| {
w.set_dblocksize(block_size);
w.set_dtdir(dtdir);
});
}
/// Sets the CLKDIV field in CLKCR. Updates clock field in self
2021-05-09 20:36:13 +00:00
fn clkcr_set_clkdiv(
&self,
freq: u32,
width: BusWidth,
hclk: Hertz,
ker_ck: Hertz,
clock: &mut Hertz,
) -> Result<(), Error> {
let regs = self.0;
2021-05-02 00:19:14 +00:00
2021-05-09 20:36:13 +00:00
let (clkdiv, new_clock) = clk_div(ker_ck, freq)?;
2021-05-02 00:19:14 +00:00
// Enforce AHB and SDMMC_CK clock relation. See RM0433 Rev 7
// Section 55.5.8
let sdmmc_bus_bandwidth = new_clock.0 * (width as u32);
2021-05-09 20:36:13 +00:00
self::assert!(hclk.0 > 3 * sdmmc_bus_bandwidth / 32);
*clock = new_clock;
2021-05-02 00:19:14 +00:00
// NOTE(unsafe) We have exclusive access to the regblock
unsafe {
// CPSMACT and DPSMACT must be 0 to set CLKDIV
self.wait_idle();
regs.clkcr().modify(|w| w.set_clkdiv(clkdiv));
}
Ok(())
}
/// Switch mode using CMD6.
///
/// Attempt to set a new signalling mode. The selected
/// signalling mode is returned. Expects the current clock
/// frequency to be > 12.5MHz.
2021-05-09 05:19:00 +00:00
async fn switch_signalling_mode(
2021-05-09 20:36:13 +00:00
&self,
2021-05-09 05:19:00 +00:00
signalling: Signalling,
2021-05-09 20:36:13 +00:00
waker_reg: &AtomicWaker,
data_transfer_timeout: u32,
2021-05-09 05:19:00 +00:00
) -> Result<Signalling, Error> {
// NB PLSS v7_10 4.3.10.4: "the use of SET_BLK_LEN command is not
// necessary"
let set_function = 0x8000_0000
| match signalling {
// See PLSS v7_10 Table 4-11
Signalling::DDR50 => 0xFF_FF04,
Signalling::SDR104 => 0xFF_1F03,
Signalling::SDR50 => 0xFF_1F02,
Signalling::SDR25 => 0xFF_FF01,
Signalling::SDR12 => 0xFF_FF00,
};
let mut status = [0u32; 16];
let status_addr = &mut status as *mut [u32; 16] as u32;
// Arm `OnDrop` after the buffer, so it will be dropped first
2021-05-09 20:36:13 +00:00
let regs = self.0;
let on_drop = OnDrop::new(|| unsafe { self.on_drop() });
2021-05-09 05:19:00 +00:00
unsafe {
self.prepare_datapath_transfer(
status_addr,
64,
6,
Dir::CardToHost,
data_transfer_timeout,
);
2021-05-09 20:36:13 +00:00
self.data_interrupts(true);
2021-05-09 05:19:00 +00:00
}
self.cmd(Cmd::cmd6(set_function), true)?; // CMD6
let res = poll_fn(|cx| {
2021-05-09 20:36:13 +00:00
waker_reg.register(cx.waker());
2021-05-09 05:19:00 +00:00
let status = unsafe { regs.star().read() };
if status.dcrcfail() {
return Poll::Ready(Err(Error::Crc));
} else if status.dtimeout() {
return Poll::Ready(Err(Error::Timeout));
} else if status.dataend() {
return Poll::Ready(Ok(()));
}
Poll::Pending
})
.await;
2021-05-09 20:36:13 +00:00
self.clear_interrupt_flags();
2021-05-09 05:19:00 +00:00
// Host is allowed to use the new functions at least 8
// clocks after the end of the switch command
// transaction. We know the current clock period is < 80ns,
// so a total delay of 640ns is required here
for _ in 0..300 {
cortex_m::asm::nop();
}
match res {
Ok(_) => {
on_drop.defuse();
unsafe {
regs.idmactrlr().modify(|w| w.set_idmaen(false));
}
// Function Selection of Function Group 1
let selection = (u32::from_be(status[4]) >> 24) & 0xF;
match selection {
0 => Ok(Signalling::SDR12),
1 => Ok(Signalling::SDR25),
2 => Ok(Signalling::SDR50),
3 => Ok(Signalling::SDR104),
4 => Ok(Signalling::DDR50),
_ => Err(Error::UnsupportedCardType),
}
}
Err(e) => Err(e),
}
2021-05-02 00:19:14 +00:00
}
/// Query the card status (CMD13, returns R1)
///
2021-05-09 20:36:13 +00:00
fn read_status(&self, card: &Card) -> Result<CardStatus, Error> {
let regs = self.0;
let rca = card.rca;
2021-05-02 00:19:14 +00:00
self.cmd(Cmd::card_status(rca << 16), false)?; // CMD13
// NOTE(unsafe) Atomic read with no side-effects
let r1 = unsafe { regs.respr(0).read().cardstatus1() };
Ok(r1.into())
}
/// Reads the SD Status (ACMD13)
async fn read_sd_status(
&self,
card: &mut Card,
waker_reg: &AtomicWaker,
data_transfer_timeout: u32,
) -> Result<(), Error> {
2021-05-02 00:19:14 +00:00
let rca = card.rca;
self.cmd(Cmd::set_block_length(64), false)?; // CMD16
self.cmd(Cmd::app_cmd(rca << 16), false)?; // APP
let mut status = [0u32; 16];
let status_addr = &mut status as *mut [u32; 16] as u32;
// Arm `OnDrop` after the buffer, so it will be dropped first
2021-05-09 20:36:13 +00:00
let regs = self.0;
let on_drop = OnDrop::new(|| unsafe { self.on_drop() });
2021-05-02 00:19:14 +00:00
unsafe {
self.prepare_datapath_transfer(
status_addr,
64,
6,
Dir::CardToHost,
data_transfer_timeout,
);
2021-05-09 20:36:13 +00:00
self.data_interrupts(true);
2021-05-02 00:19:14 +00:00
}
self.cmd(Cmd::card_status(0), true)?;
let res = poll_fn(|cx| {
2021-05-09 20:36:13 +00:00
waker_reg.register(cx.waker());
2021-05-02 00:19:14 +00:00
let status = unsafe { regs.star().read() };
if status.dcrcfail() {
return Poll::Ready(Err(Error::Crc));
} else if status.dtimeout() {
return Poll::Ready(Err(Error::Timeout));
} else if status.dataend() {
return Poll::Ready(Ok(()));
}
Poll::Pending
})
.await;
2021-05-09 20:36:13 +00:00
self.clear_interrupt_flags();
2021-05-02 00:19:14 +00:00
2021-05-09 20:36:13 +00:00
if res.is_ok() {
2021-05-02 00:19:14 +00:00
on_drop.defuse();
unsafe {
regs.idmactrlr().modify(|w| w.set_idmaen(false));
}
for byte in status.iter_mut() {
*byte = u32::from_be(*byte);
}
2021-05-09 20:36:13 +00:00
card.status = status.into();
2021-05-02 00:19:14 +00:00
}
res
}
/// Select one card and place it into the _Tranfer State_
///
/// If `None` is specifed for `card`, all cards are put back into
/// _Stand-by State_
2021-05-09 20:36:13 +00:00
fn select_card(&self, card: Option<&Card>) -> Result<(), Error> {
2021-05-02 00:19:14 +00:00
// Determine Relative Card Address (RCA) of given card
let rca = card.map(|c| c.rca << 16).unwrap_or(0);
let r = self.cmd(Cmd::sel_desel_card(rca), false);
match (r, rca) {
(Err(Error::Timeout), 0) => Ok(()),
_ => r,
}
}
/// Clear flags in interrupt clear register
#[inline(always)]
2021-05-09 20:36:13 +00:00
fn clear_interrupt_flags(&self) {
let regs = self.0;
2021-05-02 00:19:14 +00:00
// NOTE(unsafe) Atomic write
unsafe {
regs.icr().write(|w| {
w.set_ccrcfailc(true);
w.set_dcrcfailc(true);
w.set_ctimeoutc(true);
w.set_dtimeoutc(true);
w.set_txunderrc(true);
w.set_rxoverrc(true);
2021-05-09 05:19:00 +00:00
w.set_cmdrendc(true);
2021-05-02 00:19:14 +00:00
w.set_cmdsentc(true);
w.set_dataendc(true);
w.set_dholdc(true);
w.set_dbckendc(true);
w.set_dabortc(true);
w.set_busyd0endc(true);
w.set_sdioitc(true);
w.set_ackfailc(true);
w.set_acktimeoutc(true);
w.set_vswendc(true);
w.set_ckstopc(true);
w.set_idmatec(true);
w.set_idmabtcc(true);
});
}
}
/// Enables the interrupts for data transfer
#[inline(always)]
2021-05-09 20:36:13 +00:00
fn data_interrupts(&self, enable: bool) {
let regs = self.0;
// NOTE(unsafe) Atomic write
unsafe {
regs.maskr().write(|w| {
w.set_dcrcfailie(enable);
w.set_dtimeoutie(enable);
w.set_dataendie(enable);
w.set_dabortie(enable);
});
}
2021-05-02 00:19:14 +00:00
}
async fn get_scr(
&self,
card: &mut Card,
waker_reg: &AtomicWaker,
data_transfer_timeout: u32,
) -> Result<(), Error> {
2021-05-02 00:19:14 +00:00
// Read the the 64-bit SCR register
self.cmd(Cmd::set_block_length(8), false)?; // CMD16
self.cmd(Cmd::app_cmd(card.rca << 16), false)?;
let mut scr = [0u32; 2];
let scr_addr = &mut scr as *mut u32 as u32;
// Arm `OnDrop` after the buffer, so it will be dropped first
2021-05-09 20:36:13 +00:00
let regs = self.0;
let on_drop = OnDrop::new(move || unsafe { self.on_drop() });
2021-05-02 00:19:14 +00:00
unsafe {
self.prepare_datapath_transfer(scr_addr, 8, 3, Dir::CardToHost, data_transfer_timeout);
2021-05-09 20:36:13 +00:00
self.data_interrupts(true);
2021-05-02 00:19:14 +00:00
}
self.cmd(Cmd::cmd51(), true)?;
let res = poll_fn(|cx| {
2021-05-09 20:36:13 +00:00
waker_reg.register(cx.waker());
2021-05-02 00:19:14 +00:00
let status = unsafe { regs.star().read() };
if status.dcrcfail() {
return Poll::Ready(Err(Error::Crc));
} else if status.dtimeout() {
return Poll::Ready(Err(Error::Timeout));
} else if status.dataend() {
return Poll::Ready(Ok(()));
}
Poll::Pending
})
.await;
2021-05-09 20:36:13 +00:00
self.clear_interrupt_flags();
2021-05-02 00:19:14 +00:00
2021-05-09 20:36:13 +00:00
if res.is_ok() {
2021-05-02 00:19:14 +00:00
on_drop.defuse();
unsafe {
regs.idmactrlr().modify(|w| w.set_idmaen(false));
let scr_bytes = &*(&scr as *const [u32; 2] as *const [u8; 8]);
card.scr = SCR(u64::from_be_bytes(*scr_bytes));
}
}
res
}
/// Send command to card
2021-05-09 20:36:13 +00:00
fn cmd(&self, cmd: Cmd, data: bool) -> Result<(), Error> {
let regs = self.0;
2021-05-02 00:19:14 +00:00
2021-05-09 20:36:13 +00:00
self.clear_interrupt_flags();
2021-05-02 00:19:14 +00:00
// NOTE(safety) Atomic operations
unsafe {
// CP state machine must be idle
while regs.star().read().cpsmact() {}
// Command arg
regs.argr().write(|w| w.set_cmdarg(cmd.arg));
// Special mode in CP State Machine
// CMD12: Stop Transmission
let cpsm_stop_transmission = cmd.cmd == 12;
// Command index and start CP State Machine
regs.cmdr().write(|w| {
w.set_waitint(false);
w.set_waitresp(cmd.resp as u8);
w.set_cmdstop(cpsm_stop_transmission);
w.set_cmdindex(cmd.cmd);
w.set_cpsmen(true);
w.set_cmdtrans(data);
});
let mut status;
if cmd.resp == Response::None {
// Wait for CMDSENT or a timeout
while {
status = regs.star().read();
!(status.ctimeout() || status.cmdsent())
} {}
2021-05-02 00:19:14 +00:00
} else {
// Wait for CMDREND or CCRCFAIL or a timeout
while {
status = regs.star().read();
!(status.ctimeout() || status.cmdrend() || status.ccrcfail())
} {}
2021-05-02 00:19:14 +00:00
}
if status.ctimeout() {
return Err(Error::Timeout);
} else if status.ccrcfail() {
return Err(Error::Crc);
}
Ok(())
}
}
/// # Safety
///
/// Ensure that `regs` has exclusive access to the regblocks
2021-05-09 20:36:13 +00:00
unsafe fn on_drop(&self) {
let regs = self.0;
2021-05-02 00:19:14 +00:00
if regs.star().read().dpsmact() {
2021-05-11 04:28:45 +00:00
self.clear_interrupt_flags();
2021-05-02 00:19:14 +00:00
// Send abort
// CP state machine must be idle
while regs.star().read().cpsmact() {}
// Command arg
regs.argr().write(|w| w.set_cmdarg(0));
// Command index and start CP State Machine
regs.cmdr().write(|w| {
w.set_waitint(false);
w.set_waitresp(Response::Short as u8);
w.set_cmdstop(true);
w.set_cmdindex(12);
w.set_cpsmen(true);
w.set_cmdtrans(false);
});
// Wait for the abort
while regs.star().read().dpsmact() {}
}
2021-05-09 20:36:13 +00:00
self.data_interrupts(false);
self.clear_interrupt_flags();
2021-05-02 00:19:14 +00:00
regs.idmactrlr().modify(|w| w.set_idmaen(false));
}
}
/// SD card Commands
impl Cmd {
const fn new(cmd: u8, arg: u32, resp: Response) -> Cmd {
Cmd { cmd, arg, resp }
}
/// CMD0: Idle
const fn idle() -> Cmd {
Cmd::new(0, 0, Response::None)
}
/// CMD2: Send CID
const fn all_send_cid() -> Cmd {
Cmd::new(2, 0, Response::Long)
}
/// CMD3: Send Relative Address
const fn send_rel_addr() -> Cmd {
Cmd::new(3, 0, Response::Short)
}
/// CMD6: Switch Function Command
/// ACMD6: Bus Width
const fn cmd6(arg: u32) -> Cmd {
Cmd::new(6, arg, Response::Short)
}
/// CMD7: Select one card and put it into the _Tranfer State_
const fn sel_desel_card(rca: u32) -> Cmd {
Cmd::new(7, rca, Response::Short)
}
/// CMD8:
const fn hs_send_ext_csd(arg: u32) -> Cmd {
Cmd::new(8, arg, Response::Short)
}
/// CMD9:
const fn send_csd(rca: u32) -> Cmd {
Cmd::new(9, rca, Response::Long)
}
/// CMD12:
2021-05-11 04:28:45 +00:00
//const fn stop_transmission() -> Cmd {
// Cmd::new(12, 0, Response::Short)
//}
2021-05-02 00:19:14 +00:00
/// CMD13: Ask card to send status register
/// ACMD13: SD Status
const fn card_status(rca: u32) -> Cmd {
Cmd::new(13, rca, Response::Short)
}
/// CMD16:
const fn set_block_length(blocklen: u32) -> Cmd {
Cmd::new(16, blocklen, Response::Short)
}
/// CMD17: Block Read
const fn read_single_block(addr: u32) -> Cmd {
Cmd::new(17, addr, Response::Short)
}
/// CMD18: Multiple Block Read
2021-05-11 04:28:45 +00:00
//const fn read_multiple_blocks(addr: u32) -> Cmd {
// Cmd::new(18, addr, Response::Short)
//}
2021-05-02 00:19:14 +00:00
/// CMD24: Block Write
const fn write_single_block(addr: u32) -> Cmd {
Cmd::new(24, addr, Response::Short)
}
const fn app_op_cmd(arg: u32) -> Cmd {
Cmd::new(41, arg, Response::Short)
}
const fn cmd51() -> Cmd {
Cmd::new(51, 0, Response::Short)
}
/// App Command. Indicates that next command will be a app command
const fn app_cmd(rca: u32) -> Cmd {
Cmd::new(55, rca, Response::Short)
}
}
//////////////////////////////////////////////////////
pub(crate) mod sealed {
use super::*;
use crate::gpio::Pin as GpioPin;
pub trait Instance {
type Interrupt: Interrupt;
2021-05-09 20:36:13 +00:00
fn inner() -> SdmmcInner;
fn state() -> &'static AtomicWaker;
2021-05-02 00:19:14 +00:00
}
pub trait CkPin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
pub trait CmdPin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
pub trait D0Pin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
pub trait D1Pin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
pub trait D2Pin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
pub trait D3Pin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
pub trait D4Pin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
pub trait D5Pin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
pub trait D6Pin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
pub trait D7Pin<T: Instance>: GpioPin {
const AF_NUM: u8;
}
2021-05-09 20:36:13 +00:00
pub trait Pins<T: Instance> {}
2021-05-02 00:19:14 +00:00
}
2021-05-09 20:36:13 +00:00
pub trait Instance: sealed::Instance + 'static {}
pub trait CkPin<T: Instance>: sealed::CkPin<T> + 'static {}
pub trait CmdPin<T: Instance>: sealed::CmdPin<T> + 'static {}
pub trait D0Pin<T: Instance>: sealed::D0Pin<T> + 'static {}
pub trait D1Pin<T: Instance>: sealed::D1Pin<T> + 'static {}
pub trait D2Pin<T: Instance>: sealed::D2Pin<T> + 'static {}
pub trait D3Pin<T: Instance>: sealed::D3Pin<T> + 'static {}
pub trait D4Pin<T: Instance>: sealed::D4Pin<T> + 'static {}
pub trait D5Pin<T: Instance>: sealed::D5Pin<T> + 'static {}
pub trait D6Pin<T: Instance>: sealed::D6Pin<T> + 'static {}
pub trait D7Pin<T: Instance>: sealed::D7Pin<T> + 'static {}
pub trait Pins<T: Instance>: sealed::Pins<T> + 'static {
const BUSWIDTH: BusWidth;
fn configure(&mut self);
fn deconfigure(&mut self);
}
impl<T, CLK, CMD, D0, D1, D2, D3> sealed::Pins<T> for (CLK, CMD, D0, D1, D2, D3)
where
T: Instance,
CLK: CkPin<T>,
CMD: CmdPin<T>,
D0: D0Pin<T>,
D1: D1Pin<T>,
D2: D2Pin<T>,
D3: D3Pin<T>,
{
}
impl<T, CLK, CMD, D0> sealed::Pins<T> for (CLK, CMD, D0)
where
T: Instance,
CLK: CkPin<T>,
CMD: CmdPin<T>,
D0: D0Pin<T>,
{
}
/// # Safety
///
/// Access to `block` registers should be exclusive
unsafe fn configure_pin(block: Gpio, n: usize, afr_num: u8, pup: bool) {
use pac::gpio::vals::{Afr, Moder, Ospeedr, Pupdr};
let (afr, n_af) = if n < 8 { (0, n) } else { (1, n - 8) };
block.afr(afr).modify(|w| w.set_afr(n_af, Afr(afr_num)));
block.moder().modify(|w| w.set_moder(n, Moder::ALTERNATE));
if pup {
block.pupdr().modify(|w| w.set_pupdr(n, Pupdr::PULLUP));
}
block
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::VERYHIGHSPEED));
}
impl<T, CLK, CMD, D0, D1, D2, D3> Pins<T> for (CLK, CMD, D0, D1, D2, D3)
where
T: Instance,
CLK: CkPin<T>,
CMD: CmdPin<T>,
D0: D0Pin<T>,
D1: D1Pin<T>,
D2: D2Pin<T>,
D3: D3Pin<T>,
{
const BUSWIDTH: BusWidth = BusWidth::Four;
fn configure(&mut self) {
let (clk_pin, cmd_pin, d0_pin, d1_pin, d2_pin, d3_pin) = self;
cortex_m::interrupt::free(|_| unsafe {
// clk
let block = clk_pin.block();
let n = clk_pin.pin() as usize;
let afr_num = CLK::AF_NUM;
configure_pin(block, n, afr_num, false);
// cmd
let block = cmd_pin.block();
let n = cmd_pin.pin() as usize;
let afr_num = CMD::AF_NUM;
configure_pin(block, n, afr_num, true);
// d0
let block = d0_pin.block();
let n = d0_pin.pin() as usize;
let afr_num = D0::AF_NUM;
configure_pin(block, n, afr_num, true);
// d1
let block = d1_pin.block();
let n = d1_pin.pin() as usize;
let afr_num = D1::AF_NUM;
configure_pin(block, n, afr_num, true);
// d2
let block = d2_pin.block();
let n = d2_pin.pin() as usize;
let afr_num = D2::AF_NUM;
configure_pin(block, n, afr_num, true);
// d3
let block = d3_pin.block();
let n = d3_pin.pin() as usize;
let afr_num = D3::AF_NUM;
configure_pin(block, n, afr_num, true);
});
}
fn deconfigure(&mut self) {
use pac::gpio::vals::{Moder, Ospeedr, Pupdr};
let (clk_pin, cmd_pin, d0_pin, d1_pin, d2_pin, d3_pin) = self;
cortex_m::interrupt::free(|_| unsafe {
// clk
let n = clk_pin.pin().into();
clk_pin
.block()
.moder()
.modify(|w| w.set_moder(n, Moder::ANALOG));
clk_pin
.block()
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::LOWSPEED));
// cmd
let n = cmd_pin.pin().into();
cmd_pin
.block()
.moder()
.modify(|w| w.set_moder(n, Moder::ANALOG));
cmd_pin
.block()
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::LOWSPEED));
cmd_pin
.block()
.pupdr()
.modify(|w| w.set_pupdr(n, Pupdr::FLOATING));
// d0
let n = d0_pin.pin().into();
d0_pin
.block()
.moder()
.modify(|w| w.set_moder(n, Moder::ANALOG));
d0_pin
.block()
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::LOWSPEED));
d0_pin
.block()
.pupdr()
.modify(|w| w.set_pupdr(n, Pupdr::FLOATING));
// d1
let n = d1_pin.pin().into();
d1_pin
.block()
.moder()
.modify(|w| w.set_moder(n, Moder::ANALOG));
d1_pin
.block()
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::LOWSPEED));
d1_pin
.block()
.pupdr()
.modify(|w| w.set_pupdr(n, Pupdr::FLOATING));
// d2
let n = d2_pin.pin().into();
d2_pin
.block()
.moder()
.modify(|w| w.set_moder(n, Moder::ANALOG));
d2_pin
.block()
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::LOWSPEED));
d2_pin
.block()
.pupdr()
.modify(|w| w.set_pupdr(n, Pupdr::FLOATING));
// d3
let n = d3_pin.pin().into();
d3_pin
.block()
.moder()
.modify(|w| w.set_moder(n, Moder::ANALOG));
d3_pin
.block()
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::LOWSPEED));
d3_pin
.block()
.pupdr()
.modify(|w| w.set_pupdr(n, Pupdr::FLOATING));
});
}
}
impl<T, CLK, CMD, D0> Pins<T> for (CLK, CMD, D0)
where
T: Instance,
CLK: CkPin<T>,
CMD: CmdPin<T>,
D0: D0Pin<T>,
{
const BUSWIDTH: BusWidth = BusWidth::One;
fn configure(&mut self) {
let (clk_pin, cmd_pin, d0_pin) = self;
cortex_m::interrupt::free(|_| unsafe {
// clk
let block = clk_pin.block();
let n = clk_pin.pin() as usize;
let afr_num = CLK::AF_NUM;
configure_pin(block, n, afr_num, false);
// cmd
let block = cmd_pin.block();
let n = cmd_pin.pin() as usize;
let afr_num = CMD::AF_NUM;
configure_pin(block, n, afr_num, true);
// d0
let block = d0_pin.block();
let n = d0_pin.pin() as usize;
let afr_num = D0::AF_NUM;
configure_pin(block, n, afr_num, true);
});
2021-05-09 20:36:13 +00:00
}
fn deconfigure(&mut self) {
use pac::gpio::vals::{Moder, Ospeedr, Pupdr};
let (clk_pin, cmd_pin, d0_pin) = self;
cortex_m::interrupt::free(|_| unsafe {
// clk
let n = clk_pin.pin().into();
clk_pin
.block()
.moder()
.modify(|w| w.set_moder(n, Moder::ANALOG));
clk_pin
.block()
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::LOWSPEED));
// cmd
let n = cmd_pin.pin().into();
cmd_pin
.block()
.moder()
.modify(|w| w.set_moder(n, Moder::ANALOG));
cmd_pin
.block()
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::LOWSPEED));
cmd_pin
.block()
.pupdr()
.modify(|w| w.set_pupdr(n, Pupdr::FLOATING));
// d0
let n = d0_pin.pin().into();
d0_pin
.block()
.moder()
.modify(|w| w.set_moder(n, Moder::ANALOG));
d0_pin
.block()
.ospeedr()
.modify(|w| w.set_ospeedr(n, Ospeedr::LOWSPEED));
d0_pin
.block()
.pupdr()
.modify(|w| w.set_pupdr(n, Pupdr::FLOATING));
});
2021-05-09 20:36:13 +00:00
}
}
2021-05-02 00:19:14 +00:00
2021-06-03 17:50:48 +00:00
crate::pac::peripherals!(
(sdmmc, $inst:ident) => {
impl sealed::Instance for peripherals::$inst {
2021-05-31 01:21:44 +00:00
type Interrupt = crate::interrupt::$inst;
2021-05-09 20:36:13 +00:00
2021-06-03 17:50:48 +00:00
fn inner() -> SdmmcInner {
2021-05-31 01:21:44 +00:00
const INNER: crate::sdmmc::SdmmcInner = crate::sdmmc::SdmmcInner(crate::pac::$inst);
2021-05-09 20:36:13 +00:00
INNER
}
fn state() -> &'static ::embassy::util::AtomicWaker {
static WAKER: ::embassy::util::AtomicWaker = ::embassy::util::AtomicWaker::new();
&WAKER
}
2021-05-02 00:19:14 +00:00
}
2021-06-03 17:50:48 +00:00
impl Instance for peripherals::$inst {}
2021-05-02 00:19:14 +00:00
};
2021-06-03 17:50:48 +00:00
);
2021-05-02 00:19:14 +00:00
2021-06-03 17:50:48 +00:00
macro_rules! impl_pin {
($inst:ident, $pin:ident, $signal:ident, $af:expr) => {
impl crate::sdmmc::sealed::$signal<peripherals::$inst> for peripherals::$pin {
const AF_NUM: u8 = $af;
2021-05-02 00:19:14 +00:00
}
2021-06-03 17:50:48 +00:00
impl crate::sdmmc::$signal<peripherals::$inst> for peripherals::$pin {}
2021-05-02 00:19:14 +00:00
};
}
2021-05-13 00:59:48 +00:00
2021-06-03 17:50:48 +00:00
crate::pac::peripheral_pins!(
($inst:ident, sdmmc, SDMMC, $pin:ident, CK, $af:expr) => {
impl_pin!($inst, $pin, CkPin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, CMD, $af:expr) => {
impl_pin!($inst, $pin, CmdPin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, D0, $af:expr) => {
impl_pin!($inst, $pin, D0Pin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, D1, $af:expr) => {
impl_pin!($inst, $pin, D1Pin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, D2, $af:expr) => {
impl_pin!($inst, $pin, D2Pin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, D3, $af:expr) => {
impl_pin!($inst, $pin, D3Pin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, D4, $af:expr) => {
impl_pin!($inst, $pin, D4Pin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, D5, $af:expr) => {
impl_pin!($inst, $pin, D5Pin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, D6, $af:expr) => {
impl_pin!($inst, $pin, D6Pin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, D6, $af:expr) => {
impl_pin!($inst, $pin, D7Pin, $af);
};
($inst:ident, sdmmc, SDMMC, $pin:ident, D8, $af:expr) => {
impl_pin!($inst, $pin, D8Pin, $af);
};
);
#[cfg(feature = "sdmmc-rs")]
mod sdmmc_rs {
use super::*;
2021-05-21 01:17:53 +00:00
use core::future::Future;
use embedded_sdmmc::{Block, BlockCount, BlockDevice, BlockIdx};
impl<'d, T: Instance, P: Pins<T>> BlockDevice for Sdmmc<'d, T, P> {
type Error = Error;
#[rustfmt::skip]
type ReadFuture<'a> where Self: 'a = impl Future<Output = Result<(), Self::Error>> + 'a;
#[rustfmt::skip]
type WriteFuture<'a> where Self: 'a = impl Future<Output = Result<(), Self::Error>> + 'a;
fn read<'a>(
&'a mut self,
blocks: &'a mut [Block],
start_block_idx: BlockIdx,
_reason: &str,
) -> Self::ReadFuture<'a> {
async move {
let card_capacity = self.card()?.card_type;
let inner = T::inner();
let state = T::state();
let mut address = start_block_idx.0;
for block in blocks.iter_mut() {
let block: &mut [u8; 512] = &mut block.contents;
// NOTE(unsafe) Block uses align(4)
let buf = unsafe { &mut *(block as *mut [u8; 512] as *mut [u32; 128]) };
inner
.read_block(
address,
buf,
card_capacity,
state,
2021-05-15 02:34:16 +00:00
self.config.data_transfer_timeout,
)
.await?;
address += 1;
}
Ok(())
2021-05-13 00:59:48 +00:00
}
}
fn write<'a>(
&'a mut self,
blocks: &'a [Block],
start_block_idx: BlockIdx,
) -> Self::WriteFuture<'a> {
async move {
let card = self.card.as_mut().ok_or(Error::NoCard)?;
let inner = T::inner();
let state = T::state();
let mut address = start_block_idx.0;
for block in blocks.iter() {
let block: &[u8; 512] = &block.contents;
// NOTE(unsafe) DataBlock uses align 4
let buf = unsafe { &*(block as *const [u8; 512] as *const [u32; 128]) };
inner
2021-05-15 02:34:16 +00:00
.write_block(address, buf, card, state, self.config.data_transfer_timeout)
.await?;
address += 1;
}
Ok(())
2021-05-13 00:59:48 +00:00
}
}
fn num_blocks(&self) -> Result<BlockCount, Self::Error> {
let card = self.card()?;
let count = card.csd.block_count();
Ok(BlockCount(count))
}
2021-05-13 00:59:48 +00:00
}
}