Skip to content

Commit

Permalink
Handle platforms where c_int is not i32
Browse files Browse the repository at this point in the history
Some functions assume that c_int is always i32.  This is not true for
all platforms.  This patch adapts those functions to handle those cases.

Some littlefs functions return an error code (c_int) from functions
returning i32.  In these cases, error codes are truncated to c_int::MIN.
  • Loading branch information
robin-nitrokey committed Jul 18, 2024
1 parent 2c9ecad commit 4b2ee2c
Show file tree
Hide file tree
Showing 3 changed files with 31 additions and 21 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ and this project adheres to [Semantic Versioning](http://semver.org/).
accessing `Storage`, `Filesystem` and `File` implementations for any storage.
- Added `Filesystem::mount_or_else` function ([#57][])
- Marked `Path::is_empty`, `Path::from_bytes_with_nul`, `Path::from_cstr`, `Path::from_cstr_unchecked`, `Path::as_str_ref_with_trailing_nul`, `Path::as_str`, and `PathBuf::new` as `const`.
- Support platforms where `c_int` is not `i32`.

### Fixed

Expand Down
42 changes: 22 additions & 20 deletions src/fs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -287,8 +287,9 @@ impl<Storage: driver::Storage> Filesystem<'_, Storage> {
/// by this method available, at any given time.
pub fn available_blocks(&self) -> Result<usize> {
let return_code = unsafe { ll::lfs_fs_size(&mut self.alloc.borrow_mut().state) };
io::result_from(return_code, return_code)
.map(|blocks| self.total_blocks() - blocks as usize)
io::u32_result(return_code)
.map(|blocks| usize::try_from(blocks).unwrap_or(usize::MAX))
.map(|blocks| self.total_blocks().saturating_sub(blocks))
}

/// Available number of unused bytes in the filesystem
Expand Down Expand Up @@ -463,17 +464,18 @@ impl<Storage: driver::Storage> Filesystem<'_, Storage> {
)
};

if return_code >= 0 {
attribute.size = cmp::min(attr_max, return_code as u32) as usize;
return Ok(Some(attribute));
}
if return_code == ll::lfs_error_LFS_ERR_NOATTR {
return Ok(None);
}

io::result_from((), return_code)?;
// TODO: get rid of this
unreachable!();
io::u32_result(return_code)
.map(|n| {
attribute.size = cmp::min(attr_max, n) as usize;
Some(attribute)
})
.or_else(|err| {
if err == Error::NO_ATTRIBUTE {
Ok(None)
} else {
Err(err)
}
})
}

/// Remove attribute.
Expand Down Expand Up @@ -549,7 +551,7 @@ impl<Storage: driver::Storage> Filesystem<'_, Storage> {

/// C callback interface used by LittleFS to sync data with the lower level interface below the
/// filesystem. Note that this function currently does nothing.
extern "C" fn lfs_config_sync(_c: *const ll::lfs_config) -> i32 {
extern "C" fn lfs_config_sync(_c: *const ll::lfs_config) -> c_int {
// println!("in lfs_config_sync");
// Do nothing; we presume that data is synchronized.
0
Expand Down Expand Up @@ -605,7 +607,7 @@ impl Attribute {
bitflags! {
/// Definition of file open flags which can be mixed and matched as appropriate. These definitions
/// are reminiscent of the ones defined by POSIX.
struct FileOpenFlags: u32 {
struct FileOpenFlags: c_int {
/// Open file in read only mode.
const READ = 0x1;
/// Open file in write only mode.
Expand Down Expand Up @@ -754,7 +756,7 @@ impl<'a, 'b, Storage: driver::Storage> File<'a, 'b, Storage> {
addr_of_mut!((*(*self.alloc.borrow_mut())).state),
)
};
io::result_from(return_code as usize, return_code)
io::u32_result(return_code).map(|n| n as usize)
}

pub fn is_empty(&self) -> Result<bool> {
Expand Down Expand Up @@ -851,7 +853,7 @@ impl OpenOptions {
&mut fs.alloc.borrow_mut().state,
addr_of_mut!(alloc.state),
path.as_ptr(),
self.0.bits() as i32,
self.0.bits(),
addr_of!(alloc.config),
);

Expand Down Expand Up @@ -955,7 +957,7 @@ impl<S: driver::Storage> io::Read for File<'_, '_, S> {
buf.len() as u32,
)
};
io::result_from(return_code as usize, return_code)
io::u32_result(return_code).map(|n| n as usize)
}
}

Expand All @@ -972,7 +974,7 @@ impl<S: driver::Storage> io::Seek for File<'_, '_, S> {
pos.whence(),
)
};
io::result_from(return_code as usize, return_code)
io::u32_result(return_code).map(|n| n as usize)
}
}

Expand All @@ -989,7 +991,7 @@ impl<S: driver::Storage> io::Write for File<'_, '_, S> {
buf.len() as u32,
)
};
io::result_from(return_code as usize, return_code)
io::u32_result(return_code).map(|n| n as usize)
}

fn flush(&self) -> Result<()> {
Expand Down
9 changes: 8 additions & 1 deletion src/io.rs
Original file line number Diff line number Diff line change
Expand Up @@ -78,7 +78,7 @@ impl SeekFrom {
}
}

pub(crate) fn whence(self) -> i32 {
pub(crate) fn whence(self) -> c_int {
match self {
SeekFrom::Start(_) => 0,
SeekFrom::End(_) => 2,
Expand Down Expand Up @@ -265,3 +265,10 @@ pub fn result_from<T>(return_value: T, error_code: ll::lfs_error) -> Result<T> {
Ok(return_value)
}
}

pub fn u32_result(return_value: i32) -> Result<u32> {
u32::try_from(return_value).map_err(|_| {
let error_code = c_int::try_from(return_value).unwrap_or(c_int::MIN);
Error::new(error_code).unwrap()
})
}

0 comments on commit 4b2ee2c

Please sign in to comment.