use crate::{
gpio::{AlternateH, AnyPin, Pin, PA08, PA09, PA10, PA11, PB10, PB11},
pac::qspi::instrframe,
pac::{self, Mclk},
};
use core::marker::PhantomData;
#[derive(Debug, Clone, Copy, Eq, PartialEq)]
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
pub enum Error {
CommandFunctionMismatch,
}
pub struct OneShot;
pub struct XIP;
pub struct Qspi<MODE> {
qspi: pac::Qspi,
_sck: Pin<PB10, AlternateH>,
_cs: Pin<PB11, AlternateH>,
_io0: Pin<PA08, AlternateH>,
_io1: Pin<PA09, AlternateH>,
_io2: Pin<PA10, AlternateH>,
_io3: Pin<PA11, AlternateH>,
_mode: PhantomData<MODE>,
}
impl Qspi<OneShot> {
#[allow(clippy::too_many_arguments)]
pub fn new(
mclk: &mut Mclk,
qspi: pac::Qspi,
_sck: impl AnyPin<Id = PB10>,
_cs: impl AnyPin<Id = PB11>,
_io0: impl AnyPin<Id = PA08>,
_io1: impl AnyPin<Id = PA09>,
_io2: impl AnyPin<Id = PA10>,
_io3: impl AnyPin<Id = PA11>,
) -> Qspi<OneShot> {
mclk.apbcmask().modify(|_, w| w.qspi_().set_bit());
mclk.ahbmask().modify(|_, w| {
w.qspi_().set_bit();
w.qspi_2x_().clear_bit()
});
let _sck = _sck.into().into_alternate();
let _cs = _cs.into().into_alternate();
let _io0 = _io0.into().into_alternate();
let _io1 = _io1.into().into_alternate();
let _io2 = _io2.into().into_alternate();
let _io3 = _io3.into().into_alternate();
qspi.ctrla().write(|w| w.swrst().set_bit());
qspi.baud().write(|w| unsafe {
w.baud().bits(29); w.cpol().clear_bit();
w.cpha().clear_bit()
});
qspi.ctrlb().write(|w| {
w.mode().memory();
w.csmode().noreload();
w.csmode().lastxfer();
w.datalen()._8bits()
});
qspi.ctrla().modify(|_, w| w.enable().set_bit());
Self {
qspi,
_sck,
_cs,
_io0,
_io1,
_io2,
_io3,
_mode: PhantomData,
}
}
pub fn run_command(&self, command: Command) -> Result<(), Error> {
match command {
Command::WriteEnable
| Command::WriteDisable
| Command::Reset
| Command::EnableReset => (),
_ => return Err(Error::CommandFunctionMismatch),
}
let tfm = TransferMode {
instruction_enable: true,
..TransferMode::default()
};
unsafe {
self.run_read_instruction(command, tfm, 0, &mut [], true);
}
Ok(())
}
pub fn read_command(&self, command: Command, response: &mut [u8]) -> Result<(), Error> {
match command {
Command::Read
| Command::QuadRead
| Command::ReadId
| Command::ReadStatus
| Command::ReadStatus2 => (),
_ => return Err(Error::CommandFunctionMismatch),
}
let tfm = TransferMode {
data_enable: true,
instruction_enable: true,
..TransferMode::default()
};
unsafe {
self.run_read_instruction(command, tfm, 0, response, true);
}
Ok(())
}
pub fn write_command(&self, command: Command, data: &[u8]) -> Result<(), Error> {
match command {
Command::PageProgram
| Command::QuadPageProgram
| Command::WriteStatus
| Command::WriteStatus2 => (),
_ => return Err(Error::CommandFunctionMismatch),
}
let tfm = TransferMode {
data_enable: !data.is_empty(),
instruction_enable: true,
..TransferMode::default()
};
unsafe {
self.run_write_instruction(command, tfm, 0, data);
}
Ok(())
}
pub fn erase_command(&self, command: Command, address: u32) -> Result<(), Error> {
match command {
Command::EraseSector | Command::EraseBlock => {
let tfm = TransferMode {
address_enable: true,
instruction_enable: true,
..TransferMode::default()
};
unsafe {
self.run_write_instruction(command, tfm, address, &[]);
}
}
Command::EraseChip => {
let tfm = TransferMode {
instruction_enable: true,
..TransferMode::default()
};
unsafe {
self.run_read_instruction(command, tfm, 0, &mut [], true);
}
}
_ => return Err(Error::CommandFunctionMismatch),
}
Ok(())
}
pub fn read_memory(&mut self, addr: u32, buf: &mut [u8]) {
let tfm = TransferMode {
quad_width: true,
address_enable: true,
data_enable: true,
instruction_enable: true,
dummy_cycles: 8,
..TransferMode::default()
};
unsafe { self.run_read_instruction(Command::QuadRead, tfm, addr, buf, true) };
}
pub fn write_memory(&mut self, addr: u32, buf: &[u8]) {
let tfm = TransferMode {
quad_width: true,
address_enable: true,
data_enable: true,
instruction_enable: true,
..TransferMode::default()
};
unsafe { self.run_write_instruction(Command::QuadPageProgram, tfm, addr, buf) };
}
pub fn into_xip(self) -> Qspi<XIP> {
let tfm = TransferMode {
quad_width: true,
address_enable: true,
data_enable: true,
instruction_enable: true,
dummy_cycles: 8,
..TransferMode::default()
};
unsafe {
self.run_read_instruction(Command::QuadRead, tfm, 0, &mut [], false);
}
Qspi::<XIP> {
qspi: self.qspi,
_sck: self._sck,
_cs: self._cs,
_io0: self._io0,
_io1: self._io1,
_io2: self._io2,
_io3: self._io3,
_mode: PhantomData,
}
}
#[allow(clippy::type_complexity)]
pub fn free(
self,
) -> (
pac::Qspi,
Pin<PB10, AlternateH>,
Pin<PB11, AlternateH>,
Pin<PA08, AlternateH>,
Pin<PA09, AlternateH>,
Pin<PA10, AlternateH>,
Pin<PA11, AlternateH>,
) {
(
self.qspi, self._sck, self._cs, self._io0, self._io1, self._io2, self._io3,
)
}
}
impl Qspi<XIP> {
pub fn into_oneshot(self) -> Qspi<OneShot> {
unsafe { self.finalize() };
Qspi::<OneShot> {
qspi: self.qspi,
_sck: self._sck,
_cs: self._cs,
_io0: self._io0,
_io1: self._io1,
_io2: self._io2,
_io3: self._io3,
_mode: PhantomData,
}
}
}
impl<MODE> Qspi<MODE> {
unsafe fn finalize(&self) {
self.qspi.ctrla().write(|w| {
w.enable().set_bit();
w.lastxfer().set_bit()
});
while self.qspi.intflag().read().instrend().bit_is_clear() {}
self.qspi.intflag().write(|w| w.instrend().set_bit());
while self.qspi.intflag().read().csrise().bit_is_clear() {}
self.qspi.intflag().write(|w| w.csrise().set_bit());
}
unsafe fn run_write_instruction(
&self,
command: Command,
tfm: TransferMode,
addr: u32,
buf: &[u8],
) {
if command == Command::EraseSector || command == Command::EraseBlock {
self.qspi.instraddr().write(|w| w.addr().bits(addr));
}
self.qspi
.instrctrl()
.modify(|_, w| w.instr().bits(command.bits()));
self.qspi.instrframe().write(|w| {
tfm.instrframe(
w,
if command == Command::QuadPageProgram {
instrframe::Tfrtypeselect::Writememory
} else {
instrframe::Tfrtypeselect::Write
},
)
});
self.qspi.instrframe().read().bits();
if !buf.is_empty() {
core::ptr::copy(buf.as_ptr(), (QSPI_AHB + addr) as *mut u8, buf.len());
}
self.finalize();
}
unsafe fn run_read_instruction(
&self,
command: Command,
tfm: TransferMode,
addr: u32,
buf: &mut [u8],
finalize: bool,
) {
self.qspi
.instrctrl()
.modify(|_, w| w.instr().bits(command.bits()));
self.qspi.instrframe().write(|w| {
tfm.instrframe(
w,
if command == Command::QuadRead {
instrframe::Tfrtypeselect::Readmemory
} else {
instrframe::Tfrtypeselect::Read
},
)
});
self.qspi.instrframe().read().bits();
if !buf.is_empty() {
core::ptr::copy((QSPI_AHB + addr) as *mut u8, buf.as_mut_ptr(), buf.len());
}
if finalize {
self.finalize();
}
}
pub fn set_clk_divider(&mut self, value: u8) {
self.qspi
.baud()
.write(|w| unsafe { w.baud().bits(value.saturating_sub(1)) });
}
}
#[derive(Default, Debug, Copy, Clone)]
struct TransferMode {
quad_width: bool,
data_enable: bool,
opcode_enable: bool,
address_enable: bool,
instruction_enable: bool,
dummy_cycles: u8,
}
impl TransferMode {
unsafe fn instrframe(
self,
instrframe: &mut instrframe::W,
tfrtype: instrframe::Tfrtypeselect,
) -> &mut instrframe::W {
if self.quad_width {
instrframe.width().quad_output();
} else {
instrframe.width().single_bit_spi();
}
if self.data_enable {
instrframe.dataen().set_bit();
}
if self.opcode_enable {
instrframe.dataen().set_bit();
}
if self.address_enable {
instrframe.addren().set_bit();
}
if self.instruction_enable {
instrframe.instren().set_bit();
}
if self.dummy_cycles > 0 {
instrframe.dummylen().bits(self.dummy_cycles);
}
instrframe.addrlen()._24bits();
instrframe.optcodeen().clear_bit();
instrframe.tfrtype().variant(tfrtype);
instrframe
}
}
#[repr(u8)]
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
pub enum Command {
Read = 0x03,
QuadRead = 0x6B,
ReadId = 0x9F,
PageProgram = 0x02,
QuadPageProgram = 0x32,
ReadStatus = 0x05,
ReadStatus2 = 0x35,
WriteStatus = 0x01,
WriteStatus2 = 0x31,
EnableReset = 0x66,
Reset = 0x99,
WriteEnable = 0x06,
WriteDisable = 0x04,
EraseSector = 0x20,
EraseBlock = 0xD8,
EraseChip = 0xC7,
}
impl Command {
fn bits(self) -> u8 {
self as u8
}
}
const QSPI_AHB: u32 = 0x04000000;