2019-11-21 07:14:09 +08:00
|
|
|
//! Quad-SPI Flash Controller
|
|
|
|
|
2019-11-28 10:02:51 +08:00
|
|
|
use core::marker::PhantomData;
|
2019-11-30 09:48:39 +08:00
|
|
|
use crate::regs::{RegisterR, RegisterW, RegisterRW};
|
2019-11-21 07:14:09 +08:00
|
|
|
use super::slcr;
|
|
|
|
use super::clocks::CpuClocks;
|
|
|
|
|
2019-12-05 06:56:38 +08:00
|
|
|
mod regs;
|
2019-12-05 08:15:14 +08:00
|
|
|
mod bytes;
|
|
|
|
pub use bytes::{BytesTransferExt, BytesTransfer};
|
2019-11-21 07:14:09 +08:00
|
|
|
|
2019-11-23 08:59:24 +08:00
|
|
|
const FLASH_BAUD_RATE: u32 = 50_000_000;
|
2019-11-30 09:48:39 +08:00
|
|
|
const SINGLE_CAPACITY: u32 = 16 * 1024 * 1024;
|
2019-11-23 08:59:24 +08:00
|
|
|
|
2019-12-07 09:11:50 +08:00
|
|
|
///Instruction: Read Configure Register
|
|
|
|
const INST_RDCR: u8 = 0x3f;
|
|
|
|
/// Instruction Read Identification
|
|
|
|
const INST_RDID: u8 = 0x9F;
|
|
|
|
|
2019-11-28 10:02:51 +08:00
|
|
|
pub struct LinearAddressing;
|
2019-11-30 09:48:39 +08:00
|
|
|
pub struct Manual;
|
2019-11-28 10:02:51 +08:00
|
|
|
|
2019-11-21 07:14:09 +08:00
|
|
|
/// Flash Interface Driver
|
2019-11-23 08:59:24 +08:00
|
|
|
///
|
|
|
|
/// For 2x Spansion S25FL128SAGMFIR01
|
2019-11-28 10:02:51 +08:00
|
|
|
pub struct Flash<MODE> {
|
2019-11-21 07:14:09 +08:00
|
|
|
regs: &'static mut regs::RegisterBlock,
|
2019-11-28 10:02:51 +08:00
|
|
|
_mode: PhantomData<MODE>,
|
2019-11-21 07:14:09 +08:00
|
|
|
}
|
|
|
|
|
2019-11-30 06:48:08 +08:00
|
|
|
impl<MODE> Flash<MODE> {
|
|
|
|
fn transition<TO>(self) -> Flash<TO> {
|
|
|
|
Flash {
|
|
|
|
regs: self.regs,
|
|
|
|
_mode: PhantomData,
|
|
|
|
}
|
|
|
|
}
|
2019-12-05 06:56:38 +08:00
|
|
|
|
|
|
|
fn disable_interrupts(&mut self) {
|
|
|
|
self.regs.intr_dis.write(
|
|
|
|
regs::IntrDis::zeroed()
|
|
|
|
.rx_overflow(true)
|
|
|
|
.tx_fifo_not_full(true)
|
|
|
|
.tx_fifo_full(true)
|
|
|
|
.rx_fifo_not_empty(true)
|
|
|
|
.rx_fifo_full(true)
|
|
|
|
.tx_fifo_underflow(true)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn enable_interrupts(&mut self) {
|
|
|
|
self.regs.intr_en.write(
|
|
|
|
regs::IntrEn::zeroed()
|
|
|
|
.rx_overflow(true)
|
|
|
|
.tx_fifo_not_full(true)
|
|
|
|
.tx_fifo_full(true)
|
|
|
|
.rx_fifo_not_empty(true)
|
|
|
|
.rx_fifo_full(true)
|
|
|
|
.tx_fifo_underflow(true)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn clear_rx_fifo(&self) {
|
|
|
|
while self.regs.intr_status.read().rx_fifo_not_empty() {
|
|
|
|
let _ = self.regs.rx_data.read();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn clear_interrupt_status(&mut self) {
|
|
|
|
self.regs.intr_status.write(
|
|
|
|
regs::IntrStatus::zeroed()
|
|
|
|
.rx_overflow(true)
|
|
|
|
.tx_fifo_underflow(true)
|
|
|
|
);
|
|
|
|
}
|
2019-11-30 06:48:08 +08:00
|
|
|
}
|
|
|
|
|
2019-11-28 10:02:51 +08:00
|
|
|
impl Flash<()> {
|
2019-11-21 07:14:09 +08:00
|
|
|
pub fn new(clock: u32) -> Self {
|
|
|
|
Self::enable_clocks(clock);
|
|
|
|
Self::setup_signals();
|
|
|
|
Self::reset();
|
|
|
|
|
|
|
|
let regs = regs::RegisterBlock::qspi();
|
2019-11-28 10:02:51 +08:00
|
|
|
let mut flash = Flash { regs, _mode: PhantomData };
|
2019-11-23 08:59:24 +08:00
|
|
|
flash.configure((FLASH_BAUD_RATE - 1 + clock) / FLASH_BAUD_RATE);
|
2019-11-21 07:14:09 +08:00
|
|
|
flash
|
|
|
|
}
|
|
|
|
|
|
|
|
fn enable_clocks(clock: u32) {
|
|
|
|
let io_pll = CpuClocks::get().io;
|
|
|
|
let divisor = ((clock - 1 + io_pll) / clock)
|
|
|
|
.max(1).min(63) as u8;
|
|
|
|
|
|
|
|
slcr::RegisterBlock::unlocked(|slcr| {
|
|
|
|
slcr.lqspi_clk_ctrl.write(
|
|
|
|
slcr::LqspiClkCtrl::zeroed()
|
|
|
|
.src_sel(slcr::PllSource::IoPll)
|
|
|
|
.divisor(divisor)
|
|
|
|
.clkact(true)
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
fn setup_signals() {
|
2019-11-23 08:59:24 +08:00
|
|
|
slcr::RegisterBlock::unlocked(|slcr| {
|
|
|
|
// 1. Configure MIO pin 1 for chip select 0 output.
|
|
|
|
slcr.mio_pin_01.write(
|
|
|
|
slcr::MioPin01::zeroed()
|
|
|
|
.l0_sel(true)
|
|
|
|
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
|
|
.pullup(true)
|
|
|
|
);
|
|
|
|
|
|
|
|
// Configure MIO pins 2 through 5 for I/O.
|
|
|
|
slcr.mio_pin_02.write(
|
|
|
|
slcr::MioPin02::zeroed()
|
|
|
|
.l0_sel(true)
|
|
|
|
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
|
|
);
|
|
|
|
slcr.mio_pin_03.write(
|
|
|
|
slcr::MioPin03::zeroed()
|
|
|
|
.l0_sel(true)
|
|
|
|
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
|
|
);
|
|
|
|
slcr.mio_pin_04.write(
|
|
|
|
slcr::MioPin04::zeroed()
|
|
|
|
.l0_sel(true)
|
|
|
|
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
|
|
);
|
|
|
|
slcr.mio_pin_05.write(
|
|
|
|
slcr::MioPin05::zeroed()
|
|
|
|
.l0_sel(true)
|
|
|
|
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
|
|
);
|
|
|
|
|
|
|
|
// 3. Configure MIO pin 6 for serial clock 0 output.
|
|
|
|
slcr.mio_pin_06.write(
|
|
|
|
slcr::MioPin06::zeroed()
|
|
|
|
.l0_sel(true)
|
|
|
|
.io_type(slcr::IoBufferType::Lvcmos18)
|
|
|
|
);
|
|
|
|
|
2019-11-28 10:22:26 +08:00
|
|
|
// Option: Add Second Device Chip Select
|
|
|
|
// 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)
|
|
|
|
);
|
2019-11-23 08:59:24 +08:00
|
|
|
});
|
2019-11-21 07:14:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn reset() {
|
|
|
|
slcr::RegisterBlock::unlocked(|slcr| {
|
|
|
|
slcr.lqspi_rst_ctrl.write(
|
|
|
|
slcr::LqspiRstCtrl::zeroed()
|
|
|
|
.ref_rst(true)
|
|
|
|
.cpu1x_rst(true)
|
|
|
|
);
|
|
|
|
slcr.lqspi_rst_ctrl.write(
|
|
|
|
slcr::LqspiRstCtrl::zeroed()
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-11-23 08:59:24 +08:00
|
|
|
fn configure(&mut self, divider: u32) {
|
2019-12-05 06:56:38 +08:00
|
|
|
// Disable
|
|
|
|
self.regs.enable.write(
|
|
|
|
regs::Enable::zeroed()
|
|
|
|
);
|
2019-12-03 09:41:49 +08:00
|
|
|
self.disable_interrupts();
|
2019-12-05 06:56:38 +08:00
|
|
|
self.regs.lqspi_cfg.write(
|
|
|
|
regs::LqspiCfg::zeroed()
|
|
|
|
);
|
2019-12-03 09:41:49 +08:00
|
|
|
self.clear_rx_fifo();
|
2019-12-05 06:56:38 +08:00
|
|
|
self.clear_interrupt_status();
|
2019-12-03 09:41:49 +08:00
|
|
|
|
2019-11-23 08:59:24 +08:00
|
|
|
// for a baud_rate_div=1 LPBK_DLY_ADJ would be required
|
|
|
|
let mut baud_rate_div = 2u32;
|
|
|
|
while baud_rate_div < 7 && 2u32.pow(1 + baud_rate_div) < divider {
|
|
|
|
baud_rate_div += 1;
|
|
|
|
}
|
|
|
|
|
2019-11-28 10:02:51 +08:00
|
|
|
self.regs.config.write(regs::Config::zeroed()
|
2019-11-23 08:59:24 +08:00
|
|
|
.baud_rate_div(baud_rate_div as u8)
|
2019-11-21 07:14:09 +08:00
|
|
|
.mode_sel(true)
|
|
|
|
.leg_flsh(true)
|
2019-12-05 06:56:38 +08:00
|
|
|
.holdb_dr(true)
|
2019-12-03 09:41:49 +08:00
|
|
|
// 32 bits TX FIFO width
|
2019-11-21 07:14:09 +08:00
|
|
|
.fifo_width(0b11)
|
|
|
|
);
|
2019-12-03 09:41:49 +08:00
|
|
|
|
|
|
|
// Initialize RX/TX pipes thresholds
|
|
|
|
unsafe {
|
|
|
|
self.regs.rx_thres.write(32);
|
|
|
|
self.regs.tx_thres.write(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-28 10:02:51 +08:00
|
|
|
pub fn linear_addressing_mode(self) -> Flash<LinearAddressing> {
|
|
|
|
// Set manual start enable to auto mode.
|
|
|
|
// Assert the chip select.
|
|
|
|
self.regs.config.modify(|_, w| w
|
|
|
|
.man_start_en(false)
|
|
|
|
.pcs(false)
|
2019-12-05 06:56:38 +08:00
|
|
|
.manual_cs(false)
|
2019-11-28 10:02:51 +08:00
|
|
|
);
|
|
|
|
|
|
|
|
self.regs.lqspi_cfg.write(regs::LqspiCfg::zeroed()
|
2019-11-30 06:37:54 +08:00
|
|
|
// Quad I/O Fast Read
|
|
|
|
.inst_code(0xEB)
|
|
|
|
.mode_bits(0xFF)
|
|
|
|
.dummy_byte(0x2)
|
|
|
|
.mode_en(true)
|
|
|
|
// 2 devices
|
|
|
|
.two_mem(true)
|
2019-12-05 06:56:38 +08:00
|
|
|
.u_page(false)
|
2019-11-30 06:37:54 +08:00
|
|
|
// Linear Addressing Mode
|
2019-11-23 08:59:24 +08:00
|
|
|
.lq_mode(true)
|
|
|
|
);
|
2019-11-28 10:02:51 +08:00
|
|
|
|
2019-12-05 06:56:38 +08:00
|
|
|
self.regs.enable.write(
|
|
|
|
regs::Enable::zeroed()
|
|
|
|
.spi_en(true)
|
|
|
|
);
|
2019-11-28 10:02:51 +08:00
|
|
|
|
2019-11-30 06:48:08 +08:00
|
|
|
self.transition()
|
2019-11-23 08:59:24 +08:00
|
|
|
}
|
2019-11-30 09:48:39 +08:00
|
|
|
|
|
|
|
pub fn manual_mode(self, chip_index: usize) -> Flash<Manual> {
|
|
|
|
self.regs.config.modify(|_, w| w
|
|
|
|
.man_start_en(true)
|
|
|
|
.manual_cs(true)
|
|
|
|
);
|
|
|
|
|
|
|
|
self.regs.lqspi_cfg.write(regs::LqspiCfg::zeroed()
|
|
|
|
.mode_bits(0xFF)
|
|
|
|
.dummy_byte(0x2)
|
|
|
|
.mode_en(true)
|
|
|
|
// 2 devices
|
|
|
|
.two_mem(true)
|
2019-12-07 09:11:50 +08:00
|
|
|
// .sep_bus(true)
|
2019-11-30 09:48:39 +08:00
|
|
|
.u_page(chip_index != 0)
|
2019-12-05 06:56:38 +08:00
|
|
|
// Manual I/O mode
|
|
|
|
.lq_mode(false)
|
2019-11-30 09:48:39 +08:00
|
|
|
);
|
|
|
|
|
|
|
|
self.transition()
|
|
|
|
}
|
2019-11-28 10:02:51 +08:00
|
|
|
}
|
2019-11-23 08:59:24 +08:00
|
|
|
|
2019-11-28 10:02:51 +08:00
|
|
|
impl Flash<LinearAddressing> {
|
2019-11-30 06:48:08 +08:00
|
|
|
/// Stop linear addressing mode
|
|
|
|
pub fn stop(self) -> Flash<()> {
|
|
|
|
self.regs.enable.modify(|_, w| w.spi_en(false));
|
|
|
|
// De-assert chip select.
|
|
|
|
self.regs.config.modify(|_, w| w.pcs(true));
|
|
|
|
|
|
|
|
self.transition()
|
|
|
|
}
|
|
|
|
|
2019-11-23 08:59:24 +08:00
|
|
|
pub fn ptr<T>(&mut self) -> *mut T {
|
|
|
|
0xFC00_0000 as *mut _
|
|
|
|
}
|
2019-11-30 06:48:08 +08:00
|
|
|
|
|
|
|
pub fn size(&self) -> usize {
|
2019-11-30 09:48:39 +08:00
|
|
|
2 * (SINGLE_CAPACITY as usize)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Flash<Manual> {
|
|
|
|
pub fn stop(self) -> Flash<()> {
|
|
|
|
self.transition()
|
|
|
|
}
|
|
|
|
|
2019-12-07 09:11:50 +08:00
|
|
|
/// Read Configuration Register
|
2019-12-05 06:56:38 +08:00
|
|
|
pub fn rdcr(&mut self) -> u8 {
|
2019-12-07 09:11:50 +08:00
|
|
|
self.transfer(INST_RDCR, core::iter::empty())
|
2019-12-05 08:15:14 +08:00
|
|
|
.bytes_transfer().skip(1)
|
|
|
|
.next().unwrap() as u8
|
2019-12-05 06:56:38 +08:00
|
|
|
}
|
|
|
|
|
2019-12-07 09:11:50 +08:00
|
|
|
/// Read Identifiaction
|
2019-12-05 08:15:14 +08:00
|
|
|
pub fn rdid(&mut self) -> core::iter::Skip<BytesTransfer<Transfer<core::iter::Empty<u32>>>> {
|
2019-12-07 09:11:50 +08:00
|
|
|
self.transfer(INST_RDID, core::iter::empty())
|
2019-12-05 08:15:14 +08:00
|
|
|
.bytes_transfer().skip(1)
|
2019-12-05 06:56:38 +08:00
|
|
|
}
|
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
pub fn transfer<'s: 't, 't, Args>(&'s mut self, inst_code: u8, args: Args) -> Transfer<'t, Args>
|
|
|
|
where
|
|
|
|
Args: Iterator<Item = u32>,
|
|
|
|
{
|
|
|
|
Transfer::new(self, inst_code, args)
|
|
|
|
}
|
2019-12-05 06:56:38 +08:00
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
pub fn read(&mut self, offset: u32, len: usize) -> core::iter::Take<core::iter::Skip<BytesTransfer<Transfer<core::option::IntoIter<u32>>>>> {
|
2019-11-30 09:48:39 +08:00
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
// TODO:
|
|
|
|
let args = Some(0u32);
|
2019-12-07 09:11:50 +08:00
|
|
|
// Read
|
|
|
|
self.transfer(0x03, args.into_iter())
|
2019-12-05 08:15:14 +08:00
|
|
|
.bytes_transfer().skip(1).take(len)
|
2019-12-05 06:56:38 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
pub struct Transfer<'a, Args: Iterator<Item = u32>> {
|
|
|
|
flash: &'a mut Flash<Manual>,
|
2019-12-05 06:56:38 +08:00
|
|
|
args: Args,
|
|
|
|
}
|
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
impl<'a, Args: Iterator<Item = u32>> Transfer<'a, Args> {
|
|
|
|
pub fn new(flash: &'a mut Flash<Manual>, inst_code: u8, mut args: Args) -> Self
|
2019-12-05 06:56:38 +08:00
|
|
|
where
|
|
|
|
Args: Iterator<Item = u32>,
|
|
|
|
{
|
2019-12-07 09:11:50 +08:00
|
|
|
flash.regs.config.modify(|_, w| w.pcs(false));
|
|
|
|
flash.regs.enable.write(
|
|
|
|
regs::Enable::zeroed()
|
|
|
|
.spi_en(true)
|
|
|
|
);
|
2019-12-05 08:15:14 +08:00
|
|
|
while flash.regs.intr_status.read().rx_fifo_not_empty() {
|
|
|
|
flash.regs.rx_data.read();
|
|
|
|
}
|
|
|
|
|
2019-12-05 06:56:38 +08:00
|
|
|
unsafe {
|
2019-12-05 08:15:14 +08:00
|
|
|
flash.regs.txd1.write(inst_code.into());
|
2019-12-05 06:56:38 +08:00
|
|
|
}
|
2019-12-07 09:48:55 +08:00
|
|
|
flash.regs.config.modify(|_, w| w.man_start_com(true));
|
|
|
|
// Flush after `txd1` access
|
|
|
|
while !flash.regs.intr_status.read().tx_fifo_not_full() {}
|
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
while !flash.regs.intr_status.read().tx_fifo_full() {
|
2019-12-05 06:56:38 +08:00
|
|
|
let arg = args.next().unwrap_or(0);
|
|
|
|
unsafe {
|
2019-12-05 08:15:14 +08:00
|
|
|
flash.regs.txd0.write(arg);
|
2019-11-30 09:48:39 +08:00
|
|
|
}
|
|
|
|
}
|
2019-12-05 06:56:38 +08:00
|
|
|
|
2019-12-07 09:11:50 +08:00
|
|
|
flash.regs.config.modify(|_, w| w.man_start_com(true));
|
2019-12-05 08:15:14 +08:00
|
|
|
Transfer {
|
|
|
|
flash,
|
|
|
|
args,
|
2019-12-05 06:56:38 +08:00
|
|
|
}
|
2019-11-30 09:48:39 +08:00
|
|
|
}
|
2019-12-05 06:56:38 +08:00
|
|
|
}
|
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
impl<'a, Args: Iterator<Item = u32>> Drop for Transfer<'a, Args> {
|
2019-12-05 06:56:38 +08:00
|
|
|
fn drop(&mut self) {
|
2019-12-07 09:11:50 +08:00
|
|
|
// Discard remaining rx_data
|
|
|
|
while self.flash.regs.intr_status.read().rx_fifo_not_empty() {
|
|
|
|
self.flash.regs.rx_data.read();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stop
|
|
|
|
self.flash.regs.enable.write(
|
|
|
|
regs::Enable::zeroed()
|
|
|
|
.spi_en(false)
|
|
|
|
);
|
2019-12-05 08:15:14 +08:00
|
|
|
self.flash.regs.config.modify(|_, w| w
|
2019-12-07 09:11:50 +08:00
|
|
|
.pcs(true)
|
|
|
|
.man_start_com(false)
|
2019-12-05 06:56:38 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
impl<'a, Args: Iterator<Item = u32>> Iterator for Transfer<'a, Args> {
|
|
|
|
type Item = u32;
|
2019-12-05 06:56:38 +08:00
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
fn next<'s>(&'s mut self) -> Option<u32> {
|
|
|
|
while !self.flash.regs.intr_status.read().rx_fifo_not_empty() {}
|
|
|
|
let rx = self.flash.regs.rx_data.read();
|
2019-12-05 06:56:38 +08:00
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
let arg = self.args.next().unwrap_or(0);
|
|
|
|
unsafe {
|
|
|
|
self.flash.regs.txd0.write(arg);
|
|
|
|
}
|
2019-11-30 09:48:39 +08:00
|
|
|
|
2019-12-05 08:15:14 +08:00
|
|
|
Some(rx)
|
2019-11-30 06:48:08 +08:00
|
|
|
}
|
2019-11-21 07:14:09 +08:00
|
|
|
}
|