diff --git a/src/sfs.rs b/src/sfs.rs index 85b45de..14c2e9e 100644 --- a/src/sfs.rs +++ b/src/sfs.rs @@ -62,7 +62,7 @@ impl Debug for INode { impl INode { /// Map file block id to disk block id - fn disk_block_id(&self, file_block_id: BlockId) -> Option { + fn get_disk_block_id(&self, file_block_id: BlockId) -> Option { match file_block_id { id if id >= self.disk_inode.blocks as BlockId => None, @@ -81,6 +81,27 @@ impl INode { id => unimplemented!("double indirect blocks is not supported"), } } + fn set_disk_block_id(&mut self, file_block_id: BlockId, disk_block_id: BlockId) -> Result<(),()> { + match file_block_id { + id if id >= self.disk_inode.blocks as BlockId => + Err(()), + id if id < NDIRECT => { + self.disk_inode.direct[id] = disk_block_id as u32; + Ok(()) + }, + id if id < NDIRECT + BLK_NENTRY => { + let disk_block_id = disk_block_id as u32; + let fs = self.fs.upgrade().unwrap(); + fs.borrow_mut().device.write_block( + self.disk_inode.indirect as usize, + ENTRY_SIZE * (id - NDIRECT), + disk_block_id.as_buf(), + ).unwrap(); + Ok(()) + } + id => unimplemented!("double indirect blocks is not supported"), + } + } } impl vfs::INode for INode { @@ -102,7 +123,7 @@ impl vfs::INode for INode { // Read for each block let mut buf_offset = 0usize; for BlockRange { block, begin, end } in iter { - if let Some(disk_block_id) = self.disk_block_id(block) { + if let Some(disk_block_id) = self.get_disk_block_id(block) { let len = end - begin; fs.borrow_mut().device.read_block(disk_block_id, begin, &mut buf[buf_offset..buf_offset + len]).unwrap(); buf_offset += len; @@ -124,7 +145,7 @@ impl vfs::INode for INode { // Read for each block let mut buf_offset = 0usize; for BlockRange { block, begin, end } in iter { - if let Some(disk_block_id) = self.disk_block_id(block) { + if let Some(disk_block_id) = self.get_disk_block_id(block) { let len = end - begin; fs.borrow_mut().device.write_block(disk_block_id, begin, &buf[buf_offset..buf_offset + len]).unwrap(); buf_offset += len; @@ -156,6 +177,37 @@ impl vfs::INode for INode { fn type_(&self) -> Result { Ok(self.disk_inode.type_.clone() as u32) } + fn resize(&mut self, len: usize) -> Result<(), ()> { + if self.disk_inode.type_ != FileType::File || len > MAX_FILE_SIZE { + return Err(()); + } + let blocks = ((len + BLKSIZE - 1) / BLKSIZE) as u32; + use core::cmp::{Ord, Ordering}; + match blocks.cmp(&self.disk_inode.blocks) { + Ordering::Equal => {}, // Do nothing + Ordering::Greater => { + let fs = self.fs.upgrade().unwrap(); + let old_blocks = self.disk_inode.blocks; + self.disk_inode.blocks = blocks; + // allocate extra blocks + for i in old_blocks .. blocks { + let disk_block_id = fs.borrow_mut().alloc_block().expect("no more space"); + self.set_disk_block_id(i as usize, disk_block_id).unwrap(); + } + }, + Ordering::Less => { + let fs = self.fs.upgrade().unwrap(); + // free extra blocks + for i in blocks .. self.disk_inode.blocks { + let disk_block_id = self.get_disk_block_id(i as usize).unwrap(); + fs.borrow_mut().free_block(disk_block_id); + } + self.disk_inode.blocks = blocks; + }, + } + self.disk_inode.size = len as u32; + Ok(()) + } } /// Given a range and iterate sub-range for each block diff --git a/src/vfs.rs b/src/vfs.rs index bb6d7ce..74b91e5 100644 --- a/src/vfs.rs +++ b/src/vfs.rs @@ -13,7 +13,7 @@ pub trait INode { // fn reclaim(&mut self) -> Result<(), ()>; fn type_(&self) -> Result; // fn try_seek(&mut self, offset: u64) -> Result<(), ()>; -// fn truncate(&mut self, len: u64) -> Result<(), ()>; + fn resize(&mut self, len: usize) -> Result<(), ()>; // fn create(&mut self, name: &'static str, excl: bool) -> Result<(), ()>; // fn loopup(&mut self, path: &'static str) -> Result<(), ()>; // fn io_ctrl(&mut self, op: u32, data: &[u8]) -> Result<(), ()>;