Compare commits
No commits in common. "78caca1f04291a05930d76c7b2f49bb262789e3c" and "a199a5dc7d1253995fc2d70ee8f548be5286e3d4" have entirely different histories.
78caca1f04
...
a199a5dc7d
|
@ -1,6 +1,5 @@
|
||||||
//! Quad-SPI Flash Controller
|
//! Quad-SPI Flash Controller
|
||||||
|
|
||||||
use core::marker::PhantomData;
|
|
||||||
use crate::regs::{RegisterW, RegisterRW};
|
use crate::regs::{RegisterW, RegisterRW};
|
||||||
use super::slcr;
|
use super::slcr;
|
||||||
use super::clocks::CpuClocks;
|
use super::clocks::CpuClocks;
|
||||||
|
@ -9,25 +8,23 @@ pub mod regs;
|
||||||
|
|
||||||
const FLASH_BAUD_RATE: u32 = 50_000_000;
|
const FLASH_BAUD_RATE: u32 = 50_000_000;
|
||||||
|
|
||||||
pub struct LinearAddressing;
|
|
||||||
|
|
||||||
/// Flash Interface Driver
|
/// Flash Interface Driver
|
||||||
///
|
///
|
||||||
/// For 2x Spansion S25FL128SAGMFIR01
|
/// For 2x Spansion S25FL128SAGMFIR01
|
||||||
pub struct Flash<MODE> {
|
pub struct Flash {
|
||||||
regs: &'static mut regs::RegisterBlock,
|
regs: &'static mut regs::RegisterBlock,
|
||||||
_mode: PhantomData<MODE>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Flash<()> {
|
impl Flash {
|
||||||
pub fn new(clock: u32) -> Self {
|
pub fn new(clock: u32) -> Self {
|
||||||
Self::enable_clocks(clock);
|
Self::enable_clocks(clock);
|
||||||
Self::setup_signals();
|
Self::setup_signals();
|
||||||
Self::reset();
|
Self::reset();
|
||||||
|
|
||||||
let regs = regs::RegisterBlock::qspi();
|
let regs = regs::RegisterBlock::qspi();
|
||||||
let mut flash = Flash { regs, _mode: PhantomData };
|
let mut flash = Flash { regs };
|
||||||
flash.configure((FLASH_BAUD_RATE - 1 + clock) / FLASH_BAUD_RATE);
|
flash.configure((FLASH_BAUD_RATE - 1 + clock) / FLASH_BAUD_RATE);
|
||||||
|
flash.setup_linear_addressing_mode();
|
||||||
flash
|
flash
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -85,53 +82,7 @@ impl Flash<()> {
|
||||||
.io_type(slcr::IoBufferType::Lvcmos18)
|
.io_type(slcr::IoBufferType::Lvcmos18)
|
||||||
);
|
);
|
||||||
|
|
||||||
// Option: Add Second Device Chip Select
|
// TODO: optional 2nd chip setup
|
||||||
// 4. Configure MIO pin 0 for chip select 1 output.
|
|
||||||
slcr.mio_pin_00.write(
|
|
||||||
slcr::MioPin00::zeroed()
|
|
||||||
.l0_sel(true)
|
|
||||||
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
||||||
.pullup(true)
|
|
||||||
);
|
|
||||||
|
|
||||||
// Option: Add Second Serial Clock
|
|
||||||
// 5. Configure MIO pin 9 for serial clock 1 output.
|
|
||||||
slcr.mio_pin_09.write(
|
|
||||||
slcr::MioPin09::zeroed()
|
|
||||||
.l0_sel(true)
|
|
||||||
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
||||||
);
|
|
||||||
|
|
||||||
// Option: Add 4-bit Data
|
|
||||||
// 6. Configure MIO pins 10 through 13 for I/O.
|
|
||||||
slcr.mio_pin_10.write(
|
|
||||||
slcr::MioPin10::zeroed()
|
|
||||||
.l0_sel(true)
|
|
||||||
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
||||||
);
|
|
||||||
slcr.mio_pin_11.write(
|
|
||||||
slcr::MioPin11::zeroed()
|
|
||||||
.l0_sel(true)
|
|
||||||
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
||||||
);
|
|
||||||
slcr.mio_pin_12.write(
|
|
||||||
slcr::MioPin12::zeroed()
|
|
||||||
.l0_sel(true)
|
|
||||||
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
||||||
);
|
|
||||||
slcr.mio_pin_13.write(
|
|
||||||
slcr::MioPin13::zeroed()
|
|
||||||
.l0_sel(true)
|
|
||||||
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
||||||
);
|
|
||||||
|
|
||||||
// Option: Add Feedback Output Clock
|
|
||||||
// 7. Configure MIO pin 8 for feedback clock.
|
|
||||||
slcr.mio_pin_08.write(
|
|
||||||
slcr::MioPin08::zeroed()
|
|
||||||
.l0_sel(true)
|
|
||||||
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
||||||
);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -155,40 +106,25 @@ impl Flash<()> {
|
||||||
baud_rate_div += 1;
|
baud_rate_div += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
self.regs.config.write(regs::Config::zeroed()
|
self.regs.config.modify(|_, w| w
|
||||||
.baud_rate_div(baud_rate_div as u8)
|
.baud_rate_div(baud_rate_div as u8)
|
||||||
.mode_sel(true)
|
.mode_sel(true)
|
||||||
.leg_flsh(true)
|
.leg_flsh(true)
|
||||||
|
.endian(false)
|
||||||
.fifo_width(0b11)
|
.fifo_width(0b11)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn linear_addressing_mode(self) -> Flash<LinearAddressing> {
|
fn setup_linear_addressing_mode(&mut self) {
|
||||||
// Set manual start enable to auto mode.
|
self.regs.lqspi_cfg.modify(|_, w| w
|
||||||
// Assert the chip select.
|
|
||||||
self.regs.config.modify(|_, w| w
|
|
||||||
.man_start_en(false)
|
|
||||||
.pcs(false)
|
|
||||||
);
|
|
||||||
|
|
||||||
self.regs.lqspi_cfg.write(regs::LqspiCfg::zeroed()
|
|
||||||
.inst_code(0x3)
|
.inst_code(0x3)
|
||||||
.u_page(false)
|
.u_page(false)
|
||||||
.sep_bus(false)
|
.sep_bus(false)
|
||||||
.two_mem(false)
|
.two_mem(false)
|
||||||
.lq_mode(true)
|
.lq_mode(true)
|
||||||
);
|
);
|
||||||
|
|
||||||
self.regs.enable.modify(|_, w| w.spi_en(true));
|
|
||||||
|
|
||||||
Flash {
|
|
||||||
regs: self.regs,
|
|
||||||
_mode: PhantomData,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
impl Flash<LinearAddressing> {
|
|
||||||
pub fn ptr<T>(&mut self) -> *mut T {
|
pub fn ptr<T>(&mut self) -> *mut T {
|
||||||
0xFC00_0000 as *mut _
|
0xFC00_0000 as *mut _
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,7 +9,7 @@ pub struct RegisterBlock {
|
||||||
pub intr_en: RW<u32>,
|
pub intr_en: RW<u32>,
|
||||||
pub intr_dis: RW<u32>,
|
pub intr_dis: RW<u32>,
|
||||||
pub intr_mask: RO<u32>,
|
pub intr_mask: RO<u32>,
|
||||||
pub enable: Enable,
|
pub enable: RW<u32>,
|
||||||
pub delay: RW<u32>,
|
pub delay: RW<u32>,
|
||||||
pub txd0: WO<u32>,
|
pub txd0: WO<u32>,
|
||||||
pub rx_data: RO<u32>,
|
pub rx_data: RO<u32>,
|
||||||
|
@ -77,9 +77,6 @@ register_bit!(config,
|
||||||
/// false: legacy SPI mode, true: Flash memory interface mode
|
/// false: legacy SPI mode, true: Flash memory interface mode
|
||||||
leg_flsh, 31);
|
leg_flsh, 31);
|
||||||
|
|
||||||
register!(enable, Enable, RW, u32);
|
|
||||||
register_bit!(enable, spi_en, 0);
|
|
||||||
|
|
||||||
register!(lqspi_cfg, LqspiCfg, RW, u32);
|
register!(lqspi_cfg, LqspiCfg, RW, u32);
|
||||||
register_bits!(lqspi_cfg, inst_code, u8, 0, 7);
|
register_bits!(lqspi_cfg, inst_code, u8, 0, 7);
|
||||||
register_bits!(lqspi_cfg, dummy_byte, u8, 8, 10);
|
register_bits!(lqspi_cfg, dummy_byte, u8, 8, 10);
|
||||||
|
|
Loading…
Reference in New Issue