1use core::cmp;
5use core::marker::PhantomData;
6
7use embedded_hal::delay::DelayNs;
8
9use crate::fmc::{FmcBank, FmcRegisters};
10use crate::FmcPeripheral;
11
12use crate::ral::{fmc, modify_reg};
13
14pub mod device;
15
16#[cfg_attr(feature = "defmt", derive(defmt::Format))]
19#[derive(Clone, Copy, Debug, PartialEq)]
20pub struct NandConfiguration {
21 pub data_width: u8,
23 pub column_bits: u8,
25}
26
27#[cfg_attr(feature = "defmt", derive(defmt::Format))]
30#[derive(Clone, Copy, Debug, PartialEq)]
31pub struct NandTiming {
32 pub nce_setup_time: i32,
34 pub data_setup_time: i32,
36 pub ale_hold_time: i32,
38 pub cle_hold_time: i32,
40 pub ale_to_nre_delay: i32,
42 pub cle_to_nre_delay: i32,
44 pub nre_pulse_width_ns: i32,
46 pub nwe_pulse_width_ns: i32,
48 pub read_cycle_time_ns: i32,
50 pub write_cycle_time_ns: i32,
52 pub nwe_high_to_busy_ns: i32,
54}
55
56pub trait NandChip {
58 const CONFIG: NandConfiguration;
60 const TIMING: NandTiming;
62}
63
64#[allow(missing_debug_implementations)]
66pub struct Nand<FMC, IC> {
67 _chip: PhantomData<IC>,
69 fmc: FMC,
71 regs: FmcRegisters,
73}
74
75pub trait PinsNand {
77 const N_DATA: usize;
79}
80
81impl<IC: NandChip, FMC: FmcPeripheral> Nand<FMC, IC> {
82 pub fn new<PINS>(fmc: FMC, _pins: PINS, _chip: IC) -> Self
92 where
93 PINS: PinsNand,
94 {
95 assert!(
96 PINS::N_DATA == IC::CONFIG.data_width as usize,
97 "NAND Data Bus Width mismatch between IC and controller"
98 );
99
100 Nand {
101 _chip: PhantomData,
102 fmc,
103 regs: FmcRegisters::new::<FMC>(),
104 }
105 }
106
107 pub unsafe fn new_unchecked(fmc: FMC, _chip: IC) -> Self {
120 Nand {
121 _chip: PhantomData,
122 fmc,
123 regs: FmcRegisters::new::<FMC>(),
124 }
125 }
126
127 pub fn init<D>(&mut self, delay: &mut D) -> device::NandDevice
138 where
139 D: DelayNs,
140 {
141 let fmc_source_ck_hz = self.fmc.source_clock_hz();
143 let ker_clk_period_ns = 1_000_000_000u32 / fmc_source_ck_hz;
144
145 self.fmc.enable();
147
148 self.set_features_timings(IC::CONFIG, IC::TIMING, ker_clk_period_ns);
150
151 self.fmc.memory_controller_enable();
153 delay.delay_us(1);
154
155 unsafe {
158 let ptr = FmcBank::Bank3.ptr() as *mut u8;
160 device::NandDevice::init(ptr, IC::CONFIG.column_bits as usize)
161 }
162 }
163
164 #[allow(non_snake_case)]
168 fn set_features_timings(
169 &mut self,
170 config: NandConfiguration,
171 timing: NandTiming,
172 period_ns: u32,
173 ) {
174 let period_ns = period_ns as i32;
175 let n_clock_periods = |time_ns: i32| {
176 (time_ns + period_ns - 1) / period_ns };
178 let t_CS = timing.nce_setup_time;
179 let t_DS = timing.data_setup_time;
180 let t_ALH = timing.ale_hold_time;
181 let t_CLH = timing.cle_hold_time;
182 let t_AR = timing.ale_to_nre_delay;
183 let t_CLR = timing.cle_to_nre_delay;
184 let t_RP = timing.nre_pulse_width_ns;
185 let t_WP = timing.nwe_pulse_width_ns;
186 let t_RC = timing.read_cycle_time_ns;
187 let t_WC = timing.write_cycle_time_ns;
188 let t_WB = timing.nwe_high_to_busy_ns;
189
190 let setup_time = cmp::max(t_CS, cmp::max(t_AR, t_CLR));
192 let set = cmp::max(n_clock_periods(setup_time - t_WP), 1) - 1;
193 assert!(set < 255, "FMC ker clock too fast"); let wait = cmp::max(n_clock_periods(cmp::max(t_RP, t_WP)), 2) - 1;
197 assert!(wait < 255, "FMC ker clock too fast"); let mut hold = cmp::max(n_clock_periods(cmp::max(t_ALH, t_CLH)), 1);
201 let cycle_time = n_clock_periods(cmp::max(t_RC, t_WC));
203 while wait + 1 + hold + set + 1 < cycle_time {
204 hold += 1;
205 }
206 assert!(hold < 255, "FMC ker clock too fast"); let atthold = cmp::max(n_clock_periods(t_WB), 2) - 1;
210 let atthold = cmp::max(atthold, hold);
211 assert!(atthold < 255, "FMC ker clock too fast"); let hiz = cmp::max(n_clock_periods(t_CS + t_WP - t_DS), 0);
215 assert!(hiz < 255, "FMC ker clock too fast"); let ale_to_nre = n_clock_periods(t_AR);
219 let tar = cmp::max(ale_to_nre - set - 2, 0);
220 assert!(tar < 16, "FMC ker clock too fast");
221
222 let clr_to_nre = n_clock_periods(t_CLR);
224 let tclr = cmp::max(clr_to_nre - set - 2, 0);
225 assert!(tclr < 16, "FMC ker clock too fast");
226
227 let data_width = match config.data_width {
228 8 => 0,
229 16 => 1,
230 _ => panic!("not possible"),
231 };
232
233 #[rustfmt::skip]
235 modify_reg!(fmc, self.regs.global(), PCR,
236 TAR: tar as u32,
237 TCLR: tclr as u32,
238 ECCPS: 1, ECCEN: 0, PWID: data_width,
241 PTYP: 1, PWAITEN: 1 );
244
245 #[rustfmt::skip]
247 modify_reg!(fmc, self.regs.global(), PMEM,
248 MEMHIZ: hiz as u32,
249 MEMHOLD: hold as u32,
250 MEMWAIT: wait as u32,
251 MEMSET: set as u32);
252
253 #[rustfmt::skip]
255 modify_reg!(fmc, self.regs.global(), PATT,
256 ATTHIZ: hiz as u32,
257 ATTHOLD: atthold as u32,
258 ATTWAIT: wait as u32,
259 ATTSET: set as u32);
260
261 #[rustfmt::skip]
263 modify_reg!(fmc, self.regs.global(), PCR,
264 PBKEN: 1);
265 }
266}