use spi registers instead of rom functions

This commit is contained in:
Robin Appelman 2020-08-30 02:33:49 +02:00
commit b28e24a0b6
4 changed files with 54 additions and 40 deletions

View file

@ -6,14 +6,14 @@ edition = "2018"
description = "A driver for the esp8266 onboard flash"
[dependencies]
xtensa-lx106-rt = "0.0.1"
spi-memory = "0.2.0"
esp8266 = "0.0.6"
esp8266 = "0.1.1"
embedded-hal = "0.2.4"
void = { version = "1.0.2", default-features = false }
[dev-dependencies]
esp8266-hal = "0.2.0"
esp8266-hal = "0.3.2"
xtensa-lx106-rt = "0.1.0"
[profile.dev]
lto = true

View file

@ -14,15 +14,10 @@ use spi_memory::prelude::*;
#[entry]
fn main() -> ! {
// The default clock source is the onboard crystal
// In most cases 40mhz (but can be as low as 2mhz depending on the board)
// Clock speed is then doubled from the crystal frequency
let clock_frequency = 80.mhz();
let dp = unsafe { Peripherals::steal() };
let pins = dp.GPIO.split();
let mut led = pins.gpio2.into_push_pull_output();
let (mut timer1, _) = dp.TIMER.timers(clock_frequency);
let (mut timer1, _) = dp.TIMER.timers();
let mut serial = dp
.UART0

View file

@ -1,10 +0,0 @@
/* Linker script for the ESP8266 */
MEMORY
{
/* All .data/.bss/heap are in this segment. Reserve 1KB for old boot or ROM boot */
dram_seg : org = 0x3FFE8000, len = 0x18000 - 0x400
/* Functions which are critical should be put in this segment. */
iram_seg : org = 0x40100000, len = 0xFC00
}

View file

@ -5,9 +5,6 @@ use embedded_hal::digital::v2::OutputPin;
use esp8266::SPI0;
use spi_memory::{BlockDevice, Error, Read};
use void::Void;
use xtensa_lx106_rt::rom::{SPIEraseChip, SPIEraseSector, SPIRead, SPIWrite};
const SECTOR_SIZE: u32 = 4096;
pub struct FlashSpi(SPI0);
@ -45,16 +42,6 @@ pub enum ESPFlashError {
Timeout = 2,
}
impl ESPFlashError {
fn from(result: u32) -> Result<(), Self> {
match result {
0 => Ok(()),
2 => Err(ESPFlashError::Timeout),
_ => Err(ESPFlashError::Err),
}
}
}
impl From<ESPFlashError> for Error<FlashSpi, DummyCS> {
fn from(err: ESPFlashError) -> Self {
Error::Spi(err)
@ -63,6 +50,8 @@ impl From<ESPFlashError> for Error<FlashSpi, DummyCS> {
impl ESPFlash {
pub fn new(spi: SPI0) -> Self {
// take ownership of SPI0 to ensure nobody else can mess with the spi
ESPFlash { spi: FlashSpi(spi) }
}
@ -70,26 +59,60 @@ impl ESPFlash {
pub fn decompose(self) -> SPI0 {
self.spi.0
}
fn write_enable(&mut self) {
self.spi.0.spi_addr.write(|w| unsafe { w.bits(0) });
self.spi.0.spi_cmd.write(|w| w.spi_write_enable().set_bit());
while self.spi.0.spi_cmd.read().bits() > 0 {}
}
fn get_status(&mut self) -> u32 {
self.spi.0.spi_addr.write(|w| unsafe { w.bits(0) });
self.spi.0.spi_cmd.write(|w| w.spi_read_sr().set_bit());
while self.spi.0.spi_cmd.read().bits() > 0 {}
self.spi.0.spi_rd_status.read().bits()
}
}
impl BlockDevice<u32, FlashSpi, DummyCS> for ESPFlash {
/// Erase 4K sectors
fn erase_sectors(&mut self, addr: u32, amount: usize) -> Result<(), Error<FlashSpi, DummyCS>> {
let start_sector = addr / SECTOR_SIZE;
for i in 0..(amount as u32) {
ESPFlashError::from(unsafe { SPIEraseSector(start_sector + i) })?;
self.write_enable();
for i in 0..amount {
self.spi.0.spi_addr.write(|w| unsafe { w.address().bits(addr + i as u32) });
self.spi.0.spi_cmd.write(|w| w.spi_se().set_bit());
while self.spi.0.spi_cmd.read().bits() > 0 {}
while self.get_status() & 1 > 0 {}
}
Ok(())
}
fn erase_all(&mut self) -> Result<(), Error<FlashSpi, DummyCS>> {
ESPFlashError::from(unsafe { SPIEraseChip() })?;
self.spi.0.spi_cmd.write(|w| w.spi_ce().set_bit());
while self.spi.0.spi_cmd.read().bits() > 0 {}
Ok(())
}
fn write_bytes(&mut self, addr: u32, data: &mut [u8]) -> Result<(), Error<FlashSpi, DummyCS>> {
ESPFlashError::from(unsafe { SPIWrite(addr, data.as_ptr(), data.len() as u32) })?;
self.write_enable();
// todo 64 byte chunks
for (i, byte) in data.iter().enumerate() {
self.spi.0.spi_addr.write(|w| unsafe { w.address().bits(addr + i as u32).size().bits(1) });
self.spi.0.spi_w0.write(|w| unsafe { w.bits(*byte as u32) });
self.spi.0.spi_cmd.write(|w| w.spi_pp().set_bit());
while self.spi.0.spi_cmd.read().bits() > 0 {}
while self.get_status() & 1 > 0 {}
}
Ok(())
}
@ -97,9 +120,15 @@ impl BlockDevice<u32, FlashSpi, DummyCS> for ESPFlash {
impl Read<u32, FlashSpi, DummyCS> for ESPFlash {
fn read(&mut self, addr: u32, buf: &mut [u8]) -> Result<(), Error<FlashSpi, DummyCS>> {
ESPFlashError::from(unsafe {
SPIRead(addr, buf.as_mut_ptr() as *mut _, buf.len() as u32)
})?;
// todo 64 byte chunks
for (i, byte) in buf.iter_mut().enumerate() {
self.spi.0.spi_addr.write(|w| unsafe { w.address().bits(addr + i as u32).size().bits(1) });
self.spi.0.spi_cmd.write(|w| w.spi_read().set_bit());
while self.spi.0.spi_cmd.read().bits() > 0 {}
*byte = self.spi.0.spi_w0.read().bits() as u8;
}
Ok(())
}