-
Notifications
You must be signed in to change notification settings - Fork 102
/
sdmmc_fat.rs
146 lines (121 loc) · 4.21 KB
/
sdmmc_fat.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
#![no_main]
#![no_std]
use {
embedded_sdmmc::{Controller, Mode, VolumeIdx},
stm32h7xx_hal::sdmmc::{SdCard, Sdmmc},
stm32h7xx_hal::{pac, prelude::*, rcc},
};
#[macro_use]
mod utilities;
// This is just a placeholder TimeSource. In a real world application
// one would probably use the RTC to provide time.
pub struct TimeSource;
impl embedded_sdmmc::TimeSource for TimeSource {
fn get_timestamp(&self) -> embedded_sdmmc::Timestamp {
embedded_sdmmc::Timestamp {
year_since_1970: 0,
zero_indexed_month: 0,
zero_indexed_day: 0,
hours: 0,
minutes: 0,
seconds: 0,
}
}
}
#[cortex_m_rt::entry]
unsafe fn main() -> ! {
utilities::logger::init();
// Get peripherals
let cp = cortex_m::Peripherals::take().unwrap();
let dp = pac::Peripherals::take().unwrap();
// Constrain and Freeze power
let pwr = dp.PWR.constrain();
let pwrcfg = example_power!(pwr).freeze();
// Constrain and Freeze clock
let ccdr = dp
.RCC
.constrain()
.sys_ck(200.MHz())
.pll1_strategy(rcc::PllConfigStrategy::Iterative)
.pll1_q_ck(100.MHz())
.pll2_strategy(rcc::PllConfigStrategy::Iterative)
.pll3_strategy(rcc::PllConfigStrategy::Iterative)
.freeze(pwrcfg, &dp.SYSCFG);
// Get the delay provider.
let mut delay = cp.SYST.delay(ccdr.clocks);
let gpiob = dp.GPIOB.split(ccdr.peripheral.GPIOB);
let gpiod = dp.GPIOD.split(ccdr.peripheral.GPIOD);
let mut sd: Sdmmc<_, SdCard> = dp.SDMMC2.sdmmc(
(
gpiod.pd6.into_alternate(),
gpiod.pd7.into_alternate(),
gpiob.pb14.into_alternate(),
gpiob.pb15.into_alternate(),
gpiob.pb3.into_alternate(),
gpiob.pb4.into_alternate(),
),
ccdr.peripheral.SDMMC2,
&ccdr.clocks,
);
// Loop until we have a card
loop {
// On most development boards this can be increased up to 50MHz. We choose a
// lower frequency here so that it should work even with flying leads
// connected to a SD card breakout.
match sd.init(2.MHz()) {
Ok(_) => break,
Err(err) => {
log::info!("Init err: {:?}", err);
}
}
log::info!("Waiting for card...");
delay.delay_ms(1000u32);
}
// See https://github.com/rust-embedded-community/embedded-sdmmc-rs for docs
// and more examples
log::info!("Initialize file system manager");
let mut sd_fatfs = Controller::new(sd.sdmmc_block_device(), TimeSource);
let mut sd_fatfs_volume = sd_fatfs.get_volume(VolumeIdx(0)).unwrap();
let sd_fatfs_root_dir = sd_fatfs.open_root_dir(&sd_fatfs_volume).unwrap();
log::info!("List all the directories and their info");
sd_fatfs
.iterate_dir(&sd_fatfs_volume, &sd_fatfs_root_dir, |entry| {
log::info!("{:?}", entry);
})
.unwrap();
const WRITE_BUFFER: [u8; 8 * 1024] = [b'B'; 8 * 1024];
for (filename, length) in
&[("small.txt", 8), ("big.txt", WRITE_BUFFER.len())]
{
log::info!("Open file {:?}", filename);
let mut file = sd_fatfs
.open_file_in_dir(
&mut sd_fatfs_volume,
&sd_fatfs_root_dir,
filename,
Mode::ReadWriteCreateOrTruncate,
)
.unwrap();
log::info!("Write {:?} characters in it", length);
sd_fatfs
.write(&mut sd_fatfs_volume, &mut file, &WRITE_BUFFER[0..*length])
.unwrap();
log::info!("Read it back and confirm it contains the expected content");
file.seek_from_start(0).unwrap();
while !file.eof() {
let mut buffer = [0u8; 1024];
let num_read = sd_fatfs
.read(&sd_fatfs_volume, &mut file, &mut buffer)
.unwrap();
for b in &buffer[0..num_read] {
assert_eq!(*b as char, 'B');
}
}
sd_fatfs.close_file(&sd_fatfs_volume, file).unwrap();
}
log::info!("Test successfully finished");
sd_fatfs.close_dir(&sd_fatfs_volume, sd_fatfs_root_dir);
loop {
cortex_m::asm::nop()
}
}