adding docs

This commit is contained in:
okhsunrog 2025-05-21 08:41:04 +03:00
parent 1accd56005
commit bc265b98b7

View File

@ -4,6 +4,7 @@ use super::{family, Blocking, Error, Flash, EEPROM_BASE, EEPROM_SIZE};
impl<'d> Flash<'d, Blocking> { impl<'d> Flash<'d, Blocking> {
// --- Internal helpers --- // --- Internal helpers ---
/// Checks if the given offset and size are within the EEPROM bounds.
fn check_eeprom_offset(&self, offset: u32, size: u32) -> Result<(), Error> { fn check_eeprom_offset(&self, offset: u32, size: u32) -> Result<(), Error> {
if offset if offset
.checked_add(size) .checked_add(size)
@ -18,6 +19,10 @@ impl<'d> Flash<'d, Blocking> {
// --- Unlocked (unsafe, internal) functions --- // --- Unlocked (unsafe, internal) functions ---
/// Writes a slice of bytes to EEPROM at the given offset without locking.
///
/// # Safety
/// Caller must ensure EEPROM is unlocked and offset is valid.
unsafe fn eeprom_write_u8_slice_unlocked(&self, offset: u32, data: &[u8]) -> Result<(), Error> { unsafe fn eeprom_write_u8_slice_unlocked(&self, offset: u32, data: &[u8]) -> Result<(), Error> {
for (i, &byte) in data.iter().enumerate() { for (i, &byte) in data.iter().enumerate() {
let addr = EEPROM_BASE as u32 + offset + i as u32; let addr = EEPROM_BASE as u32 + offset + i as u32;
@ -28,6 +33,10 @@ impl<'d> Flash<'d, Blocking> {
Ok(()) Ok(())
} }
/// Writes a slice of u16 values to EEPROM at the given offset without locking.
///
/// # Safety
/// Caller must ensure EEPROM is unlocked and offset is valid and aligned.
unsafe fn eeprom_write_u16_slice_unlocked(&self, offset: u32, data: &[u16]) -> Result<(), Error> { unsafe fn eeprom_write_u16_slice_unlocked(&self, offset: u32, data: &[u16]) -> Result<(), Error> {
for (i, &value) in data.iter().enumerate() { for (i, &value) in data.iter().enumerate() {
let addr = EEPROM_BASE as u32 + offset + i as u32 * 2; let addr = EEPROM_BASE as u32 + offset + i as u32 * 2;
@ -38,6 +47,10 @@ impl<'d> Flash<'d, Blocking> {
Ok(()) Ok(())
} }
/// Writes a slice of u32 values to EEPROM at the given offset without locking.
///
/// # Safety
/// Caller must ensure EEPROM is unlocked and offset is valid and aligned.
unsafe fn eeprom_write_u32_slice_unlocked(&self, offset: u32, data: &[u32]) -> Result<(), Error> { unsafe fn eeprom_write_u32_slice_unlocked(&self, offset: u32, data: &[u32]) -> Result<(), Error> {
for (i, &value) in data.iter().enumerate() { for (i, &value) in data.iter().enumerate() {
let addr = EEPROM_BASE as u32 + offset + i as u32 * 4; let addr = EEPROM_BASE as u32 + offset + i as u32 * 4;
@ -50,20 +63,20 @@ impl<'d> Flash<'d, Blocking> {
// --- Public, safe API --- // --- Public, safe API ---
/// Writes a single byte to EEPROM at the given offset.
pub fn eeprom_write_u8(&mut self, offset: u32, value: u8) -> Result<(), Error> { pub fn eeprom_write_u8(&mut self, offset: u32, value: u8) -> Result<(), Error> {
self.check_eeprom_offset(offset, 1)?; self.check_eeprom_offset(offset, 1)?;
unsafe { unsafe {
family::unlock(); family::unlock();
}
unsafe {
self.eeprom_write_u8_slice_unlocked(offset, core::slice::from_ref(&value))?; self.eeprom_write_u8_slice_unlocked(offset, core::slice::from_ref(&value))?;
}
unsafe {
family::lock(); family::lock();
} }
Ok(()) Ok(())
} }
/// Writes a single 16-bit value to EEPROM at the given offset.
///
/// Returns an error if the offset is not 2-byte aligned.
pub fn eeprom_write_u16(&mut self, offset: u32, value: u16) -> Result<(), Error> { pub fn eeprom_write_u16(&mut self, offset: u32, value: u16) -> Result<(), Error> {
if offset % 2 != 0 { if offset % 2 != 0 {
return Err(Error::Unaligned); return Err(Error::Unaligned);
@ -71,16 +84,15 @@ impl<'d> Flash<'d, Blocking> {
self.check_eeprom_offset(offset, 2)?; self.check_eeprom_offset(offset, 2)?;
unsafe { unsafe {
family::unlock(); family::unlock();
}
unsafe {
self.eeprom_write_u16_slice_unlocked(offset, core::slice::from_ref(&value))?; self.eeprom_write_u16_slice_unlocked(offset, core::slice::from_ref(&value))?;
}
unsafe {
family::lock(); family::lock();
} }
Ok(()) Ok(())
} }
/// Writes a single 32-bit value to EEPROM at the given offset.
///
/// Returns an error if the offset is not 4-byte aligned.
pub fn eeprom_write_u32(&mut self, offset: u32, value: u32) -> Result<(), Error> { pub fn eeprom_write_u32(&mut self, offset: u32, value: u32) -> Result<(), Error> {
if offset % 4 != 0 { if offset % 4 != 0 {
return Err(Error::Unaligned); return Err(Error::Unaligned);
@ -88,30 +100,26 @@ impl<'d> Flash<'d, Blocking> {
self.check_eeprom_offset(offset, 4)?; self.check_eeprom_offset(offset, 4)?;
unsafe { unsafe {
family::unlock(); family::unlock();
}
unsafe {
self.eeprom_write_u32_slice_unlocked(offset, core::slice::from_ref(&value))?; self.eeprom_write_u32_slice_unlocked(offset, core::slice::from_ref(&value))?;
}
unsafe {
family::lock(); family::lock();
} }
Ok(()) Ok(())
} }
/// Writes a slice of bytes to EEPROM at the given offset.
pub fn eeprom_write_u8_slice(&mut self, offset: u32, data: &[u8]) -> Result<(), Error> { pub fn eeprom_write_u8_slice(&mut self, offset: u32, data: &[u8]) -> Result<(), Error> {
self.check_eeprom_offset(offset, data.len() as u32)?; self.check_eeprom_offset(offset, data.len() as u32)?;
unsafe { unsafe {
family::unlock(); family::unlock();
}
unsafe {
self.eeprom_write_u8_slice_unlocked(offset, data)?; self.eeprom_write_u8_slice_unlocked(offset, data)?;
}
unsafe {
family::lock(); family::lock();
} }
Ok(()) Ok(())
} }
/// Writes a slice of 16-bit values to EEPROM at the given offset.
///
/// Returns an error if the offset is not 2-byte aligned.
pub fn eeprom_write_u16_slice(&mut self, offset: u32, data: &[u16]) -> Result<(), Error> { pub fn eeprom_write_u16_slice(&mut self, offset: u32, data: &[u16]) -> Result<(), Error> {
if offset % 2 != 0 { if offset % 2 != 0 {
return Err(Error::Unaligned); return Err(Error::Unaligned);
@ -119,16 +127,15 @@ impl<'d> Flash<'d, Blocking> {
self.check_eeprom_offset(offset, data.len() as u32 * 2)?; self.check_eeprom_offset(offset, data.len() as u32 * 2)?;
unsafe { unsafe {
family::unlock(); family::unlock();
}
unsafe {
self.eeprom_write_u16_slice_unlocked(offset, data)?; self.eeprom_write_u16_slice_unlocked(offset, data)?;
}
unsafe {
family::lock(); family::lock();
} }
Ok(()) Ok(())
} }
/// Writes a slice of 32-bit values to EEPROM at the given offset.
///
/// Returns an error if the offset is not 4-byte aligned.
pub fn eeprom_write_u32_slice(&mut self, offset: u32, data: &[u32]) -> Result<(), Error> { pub fn eeprom_write_u32_slice(&mut self, offset: u32, data: &[u32]) -> Result<(), Error> {
if offset % 4 != 0 { if offset % 4 != 0 {
return Err(Error::Unaligned); return Err(Error::Unaligned);
@ -136,23 +143,18 @@ impl<'d> Flash<'d, Blocking> {
self.check_eeprom_offset(offset, data.len() as u32 * 4)?; self.check_eeprom_offset(offset, data.len() as u32 * 4)?;
unsafe { unsafe {
family::unlock(); family::unlock();
}
unsafe {
self.eeprom_write_u32_slice_unlocked(offset, data)?; self.eeprom_write_u32_slice_unlocked(offset, data)?;
}
unsafe {
family::lock(); family::lock();
} }
Ok(()) Ok(())
} }
/// Writes a byte slice to EEPROM at the given offset, handling alignment.
///
/// This method will write unaligned prefix and suffix as bytes, and aligned middle as u32.
pub fn eeprom_write(&mut self, offset: u32, data: &[u8]) -> Result<(), Error> { pub fn eeprom_write(&mut self, offset: u32, data: &[u8]) -> Result<(), Error> {
self.check_eeprom_offset(offset, data.len() as u32)?;
let start = offset; let start = offset;
let end = offset.checked_add(data.len() as u32).ok_or(Error::Size)?;
if end > EEPROM_SIZE as u32 {
return Err(Error::Size);
}
let misalign = (start % 4) as usize; let misalign = (start % 4) as usize;
let prefix_len = if misalign == 0 { let prefix_len = if misalign == 0 {
0 0
@ -165,37 +167,33 @@ impl<'d> Flash<'d, Blocking> {
unsafe { unsafe {
family::unlock(); family::unlock();
}
if !prefix.is_empty() { if !prefix.is_empty() {
unsafe {
self.eeprom_write_u8_slice_unlocked(start, prefix)?; self.eeprom_write_u8_slice_unlocked(start, prefix)?;
} }
}
if !aligned.is_empty() { if !aligned.is_empty() {
let aligned_offset = start + prefix_len as u32; let aligned_offset = start + prefix_len as u32;
let u32_data = unsafe { core::slice::from_raw_parts(aligned.as_ptr() as *const u32, aligned.len() / 4) }; let u32_data = core::slice::from_raw_parts(aligned.as_ptr() as *const u32, aligned.len() / 4);
unsafe {
self.eeprom_write_u32_slice_unlocked(aligned_offset, u32_data)?; self.eeprom_write_u32_slice_unlocked(aligned_offset, u32_data)?;
} }
}
if !suffix.is_empty() { if !suffix.is_empty() {
let suffix_offset = start + (prefix_len + aligned_len) as u32; let suffix_offset = start + (prefix_len + aligned_len) as u32;
unsafe {
self.eeprom_write_u8_slice_unlocked(suffix_offset, suffix)?; self.eeprom_write_u8_slice_unlocked(suffix_offset, suffix)?;
} }
}
unsafe {
family::lock(); family::lock();
} }
Ok(()) Ok(())
} }
/// Reads a single byte from EEPROM at the given offset.
pub fn eeprom_read_u8(&self, offset: u32) -> Result<u8, Error> { pub fn eeprom_read_u8(&self, offset: u32) -> Result<u8, Error> {
self.check_eeprom_offset(offset, 1)?; self.check_eeprom_offset(offset, 1)?;
let addr = EEPROM_BASE as u32 + offset; let addr = EEPROM_BASE as u32 + offset;
Ok(unsafe { core::ptr::read_volatile(addr as *const u8) }) Ok(unsafe { core::ptr::read_volatile(addr as *const u8) })
} }
/// Reads a single 16-bit value from EEPROM at the given offset.
///
/// Returns an error if the offset is not 2-byte aligned.
pub fn eeprom_read_u16(&self, offset: u32) -> Result<u16, Error> { pub fn eeprom_read_u16(&self, offset: u32) -> Result<u16, Error> {
if offset % 2 != 0 { if offset % 2 != 0 {
return Err(Error::Unaligned); return Err(Error::Unaligned);
@ -205,6 +203,9 @@ impl<'d> Flash<'d, Blocking> {
Ok(unsafe { core::ptr::read_volatile(addr as *const u16) }) Ok(unsafe { core::ptr::read_volatile(addr as *const u16) })
} }
/// Reads a single 32-bit value from EEPROM at the given offset.
///
/// Returns an error if the offset is not 4-byte aligned.
pub fn eeprom_read_u32(&self, offset: u32) -> Result<u32, Error> { pub fn eeprom_read_u32(&self, offset: u32) -> Result<u32, Error> {
if offset % 4 != 0 { if offset % 4 != 0 {
return Err(Error::Unaligned); return Err(Error::Unaligned);
@ -214,6 +215,7 @@ impl<'d> Flash<'d, Blocking> {
Ok(unsafe { core::ptr::read_volatile(addr as *const u32) }) Ok(unsafe { core::ptr::read_volatile(addr as *const u32) })
} }
/// Reads a slice of bytes from EEPROM at the given offset into the provided buffer.
pub fn eeprom_read_slice(&self, offset: u32, buf: &mut [u8]) -> Result<(), Error> { pub fn eeprom_read_slice(&self, offset: u32, buf: &mut [u8]) -> Result<(), Error> {
self.check_eeprom_offset(offset, buf.len() as u32)?; self.check_eeprom_offset(offset, buf.len() as u32)?;
let addr = EEPROM_BASE as u32 + offset; let addr = EEPROM_BASE as u32 + offset;