use core::mem; use alloc::Vec; use error::Error; use buffer::{Buffer, BufferSlice}; use sys::superblock::Superblock; use sys::block_group::BlockGroupDescriptor; use sys::inode::Inode; struct Struct { pub inner: T, pub offset: usize, } impl From<(T, usize)> for Struct { #[inline] fn from((inner, offset): (T, usize)) -> Struct { Struct { inner, offset } } } /// Safe wrapper for raw sys structs pub struct Ext2> { buffer: B, superblock: Struct, block_groups: Struct>, } impl> Ext2 where Error: From, { pub fn new(buffer: B) -> Result, Error> { let superblock = unsafe { Struct::from(Superblock::find(&buffer)?) }; let block_size = superblock.inner.block_size(); let block_groups_offset = (superblock.inner.first_data_block as usize + 1) * block_size; let block_groups_count = superblock .inner .block_group_count() .map(|count| count as usize) .map_err(|(a, b)| Error::BadBlockGroupCount(a, b))?; let block_groups = unsafe { BlockGroupDescriptor::find_descriptor_table( &buffer, block_groups_offset, block_groups_count, )? }; let block_groups = Struct::from(block_groups); Ok(Ext2 { buffer, superblock, block_groups, }) } pub fn update_global(&mut self) -> Result<(), Error> { // superblock { let slice = BufferSlice::from_cast( &self.superblock.inner, self.superblock.offset, ); let commit = slice.commit(); self.buffer.commit(commit).map_err(|err| Error::from(err))?; } // block group descriptors let mut offset = self.block_groups.offset; for descr in &self.block_groups.inner { let slice = BufferSlice::from_cast(descr, offset); let commit = slice.commit(); self.buffer.commit(commit).map_err(|err| Error::from(err))?; offset += mem::size_of::(); } Ok(()) } fn superblock(&self) -> &Superblock { &self.superblock.inner } fn superblock_mut(&mut self) -> &mut Superblock { &mut self.superblock.inner } pub fn version(&self) -> (u32, u16) { (self.superblock().rev_major, self.superblock().rev_minor) } pub fn inode_size(&self) -> usize { if self.version().0 == 0 { mem::size_of::() } else { // note: inodes bigger than 128 are not supported self.superblock().inode_size as usize } } pub fn block_group_count(&self) -> Result { self.superblock() .block_group_count() .map(|count| count as usize) .map_err(|(a, b)| Error::BadBlockGroupCount(a, b)) } pub fn total_block_count(&self) -> usize { self.superblock().blocks_count as _ } pub fn free_block_count(&self) -> usize { self.superblock().free_blocks_count as _ } pub fn block_size(&self) -> usize { self.superblock().block_size() } } #[cfg(test)] mod tests { use std::fs::File; use std::cell::RefCell; use buffer::Buffer; use super::Ext2; #[test] fn file_len() { let file = RefCell::new(File::open("ext2.bin").unwrap()); assert_eq!(unsafe { file.slice_unchecked(1024..2048).len() }, 1024); } #[test] fn file() { let file = RefCell::new(File::open("ext2.bin").unwrap()); let fs = Ext2::new(file); assert!( fs.is_ok(), "Err({:?})", fs.err().unwrap_or_else(|| unreachable!()), ); let fs = fs.unwrap(); let vers = fs.version(); println!("version: {}.{}", vers.0, vers.1); assert_eq!(128, fs.inode_size()); } }