2022-03-16 17:52:27 +01:00
|
|
|
#![no_std]
|
|
|
|
#![no_main]
|
|
|
|
#![feature(type_alias_impl_trait)]
|
|
|
|
|
2022-04-02 04:35:06 +02:00
|
|
|
use defmt::*;
|
2022-08-17 23:40:16 +02:00
|
|
|
use embassy_executor::Spawner;
|
2023-02-18 00:35:35 +01:00
|
|
|
use embassy_stm32::sdmmc::{DataBlock, Sdmmc};
|
2022-07-11 00:36:10 +02:00
|
|
|
use embassy_stm32::time::mhz;
|
2022-08-17 18:49:55 +02:00
|
|
|
use embassy_stm32::{interrupt, Config};
|
2022-06-12 22:15:44 +02:00
|
|
|
use {defmt_rtt as _, panic_probe as _};
|
2022-03-16 17:52:27 +01:00
|
|
|
|
2023-02-18 00:35:35 +01:00
|
|
|
/// This is a safeguard to not overwrite any data on the SD card.
|
|
|
|
/// If you don't care about SD card contents, set this to `true` to test writes.
|
|
|
|
const ALLOW_WRITES: bool = false;
|
|
|
|
|
2022-08-17 18:49:55 +02:00
|
|
|
#[embassy_executor::main]
|
2023-03-08 03:08:59 +01:00
|
|
|
async fn main(_spawner: Spawner) {
|
2022-08-17 22:25:58 +02:00
|
|
|
let mut config = Config::default();
|
|
|
|
config.rcc.sys_ck = Some(mhz(48));
|
2023-02-23 15:57:21 +01:00
|
|
|
config.rcc.pll48 = true;
|
2022-08-17 22:25:58 +02:00
|
|
|
let p = embassy_stm32::init(config);
|
2022-03-16 19:20:39 +01:00
|
|
|
info!("Hello World!");
|
2022-03-16 17:52:27 +01:00
|
|
|
|
|
|
|
let irq = interrupt::take!(SDIO);
|
|
|
|
|
2022-07-23 01:29:35 +02:00
|
|
|
let mut sdmmc = Sdmmc::new_4bit(
|
2022-03-16 19:20:39 +01:00
|
|
|
p.SDIO,
|
|
|
|
irq,
|
|
|
|
p.DMA2_CH3,
|
2022-07-23 01:29:35 +02:00
|
|
|
p.PC12,
|
|
|
|
p.PD2,
|
|
|
|
p.PC8,
|
|
|
|
p.PC9,
|
|
|
|
p.PC10,
|
|
|
|
p.PC11,
|
|
|
|
Default::default(),
|
2022-03-16 19:20:39 +01:00
|
|
|
);
|
2022-03-16 17:52:27 +01:00
|
|
|
|
2022-03-16 22:55:07 +01:00
|
|
|
// Should print 400kHz for initialization
|
|
|
|
info!("Configured clock: {}", sdmmc.clock().0);
|
2022-03-16 17:52:27 +01:00
|
|
|
|
2023-04-17 21:48:47 +02:00
|
|
|
let mut err = None;
|
|
|
|
loop {
|
|
|
|
match sdmmc.init_card(mhz(24)).await {
|
|
|
|
Ok(_) => break,
|
|
|
|
Err(e) => {
|
|
|
|
if err != Some(e) {
|
|
|
|
info!("waiting for card error, retrying: {:?}", e);
|
|
|
|
err = Some(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-03-16 17:52:27 +01:00
|
|
|
|
|
|
|
let card = unwrap!(sdmmc.card());
|
|
|
|
|
|
|
|
info!("Card: {:#?}", Debug2Format(card));
|
2023-02-18 00:35:35 +01:00
|
|
|
info!("Clock: {}", sdmmc.clock());
|
|
|
|
|
|
|
|
// Arbitrary block index
|
|
|
|
let block_idx = 16;
|
|
|
|
|
|
|
|
// SDMMC uses `DataBlock` instead of `&[u8]` to ensure 4 byte alignment required by the hardware.
|
|
|
|
let mut block = DataBlock([0u8; 512]);
|
|
|
|
|
|
|
|
sdmmc.read_block(block_idx, &mut block).await.unwrap();
|
|
|
|
info!("Read: {=[u8]:X}...{=[u8]:X}", block[..8], block[512 - 8..]);
|
|
|
|
|
|
|
|
if !ALLOW_WRITES {
|
|
|
|
info!("Writing is disabled.");
|
|
|
|
loop {}
|
|
|
|
}
|
|
|
|
|
|
|
|
info!("Filling block with 0x55");
|
|
|
|
block.fill(0x55);
|
|
|
|
sdmmc.write_block(block_idx, &block).await.unwrap();
|
|
|
|
info!("Write done");
|
|
|
|
|
|
|
|
sdmmc.read_block(block_idx, &mut block).await.unwrap();
|
|
|
|
info!("Read: {=[u8]:X}...{=[u8]:X}", block[..8], block[512 - 8..]);
|
|
|
|
|
|
|
|
info!("Filling block with 0xAA");
|
|
|
|
block.fill(0xAA);
|
|
|
|
sdmmc.write_block(block_idx, &block).await.unwrap();
|
|
|
|
info!("Write done");
|
|
|
|
|
|
|
|
sdmmc.read_block(block_idx, &mut block).await.unwrap();
|
|
|
|
info!("Read: {=[u8]:X}...{=[u8]:X}", block[..8], block[512 - 8..]);
|
2022-03-16 17:52:27 +01:00
|
|
|
}
|