add device inodes for SFS and device inode id for inodeimpl

master
gaotianyu1350 6 years ago
parent de6b914cd1
commit cef08da812

@ -19,7 +19,7 @@ use core::fmt::{Debug, Error, Formatter};
use core::mem::uninitialized; use core::mem::uninitialized;
use bitvec::BitVec; use bitvec::BitVec;
use spin::RwLock; use spin::{RwLock, RwLockReadGuard, RwLockWriteGuard};
use rcore_fs::dev::Device; use rcore_fs::dev::Device;
use rcore_fs::dirty::Dirty; use rcore_fs::dirty::Dirty;
@ -64,7 +64,7 @@ pub struct INodeImpl {
disk_inode: RwLock<Dirty<DiskINode>>, disk_inode: RwLock<Dirty<DiskINode>>,
/// Weak reference to SFS, used by almost all operations /// Weak reference to SFS, used by almost all operations
fs: Arc<SimpleFileSystem>, fs: Arc<SimpleFileSystem>,
device_inode: Option<Arc<DeviceINode>> device_inode_id: usize
} }
impl Debug for INodeImpl { impl Debug for INodeImpl {
@ -385,15 +385,17 @@ impl vfs::INode for INodeImpl {
FileType::File => self._read_at(offset, buf), FileType::File => self._read_at(offset, buf),
FileType::SymLink => self._read_at(offset, buf), FileType::SymLink => self._read_at(offset, buf),
FileType::CharDevice => { FileType::CharDevice => {
match &self.device_inode { let device_inodes = self.fs.device_inodes.read();
Some(x) => x.read_at(offset, buf), let device_inode = device_inodes.get(&self.device_inode_id);
None => Err(FsError::NoDevice) match device_inode {
Some(device) => device.read().read_at(offset, buf),
None => Err(FsError::DeviceError)
} }
}, },
_ => Err(FsError::NotFile) _ => Err(FsError::NotFile)
} }
} }
fn write_at(&self, offset: usize, buf: &[u8]) -> vfs::Result<usize> { fn write_at(&mut self, offset: usize, buf: &[u8]) -> vfs::Result<usize> {
let DiskINode { type_, size, .. } = **self.disk_inode.read(); let DiskINode { type_, size, .. } = **self.disk_inode.read();
match type_ { match type_ {
FileType::File | FileType::SymLink => { FileType::File | FileType::SymLink => {
@ -404,9 +406,11 @@ impl vfs::INode for INodeImpl {
self._write_at(offset, buf) self._write_at(offset, buf)
}, },
FileType::CharDevice => { FileType::CharDevice => {
match &self.device_inode { let device_inodes = self.fs.device_inodes.write();
Some(x) => x.write_at(offset, buf), let device_inode = device_inodes.get(&self.device_inode_id);
None => Err(FsError::NoDevice) match device_inode {
Some(device) => device.write().write_at(offset, buf),
None => Err(FsError::DeviceError)
} }
}, },
_ => Err(FsError::NotFile) _ => Err(FsError::NotFile)
@ -691,6 +695,8 @@ pub struct SimpleFileSystem {
device: Arc<Device>, device: Arc<Device>,
/// Pointer to self, used by INodes /// Pointer to self, used by INodes
self_ptr: Weak<SimpleFileSystem>, self_ptr: Weak<SimpleFileSystem>,
/// device inode
device_inodes: RwLock<BTreeMap<usize, RwLock<Box<DeviceINode>>>>
} }
impl SimpleFileSystem { impl SimpleFileSystem {
@ -711,6 +717,7 @@ impl SimpleFileSystem {
inodes: RwLock::new(BTreeMap::new()), inodes: RwLock::new(BTreeMap::new()),
device, device,
self_ptr: Weak::default(), self_ptr: Weak::default(),
device_inodes: RwLock::new(BTreeMap::new())
} }
.wrap()) .wrap())
} }
@ -742,6 +749,7 @@ impl SimpleFileSystem {
inodes: RwLock::new(BTreeMap::new()), inodes: RwLock::new(BTreeMap::new()),
device, device,
self_ptr: Weak::default(), self_ptr: Weak::default(),
device_inodes: RwLock::new(BTreeMap::new())
} }
.wrap(); .wrap();
@ -802,7 +810,18 @@ impl SimpleFileSystem {
id, id,
disk_inode: RwLock::new(disk_inode), disk_inode: RwLock::new(disk_inode),
fs: self.self_ptr.upgrade().unwrap(), fs: self.self_ptr.upgrade().unwrap(),
device_inode: None device_inode_id: NODEVICE
});
self.inodes.write().insert(id, Arc::downgrade(&inode));
inode
}
fn _new_inode_with_device(&self, id: INodeId, disk_inode: Dirty<DiskINode>, device_inode_id: usize) -> Arc<INodeImpl> {
let inode = Arc::new(INodeImpl {
id,
disk_inode: RwLock::new(disk_inode),
fs: self.self_ptr.upgrade().unwrap(),
device_inode_id: device_inode_id
}); });
self.inodes.write().insert(id, Arc::downgrade(&inode)); self.inodes.write().insert(id, Arc::downgrade(&inode));
inode inode
@ -842,6 +861,13 @@ impl SimpleFileSystem {
inode.init_dir_entry(parent)?; inode.init_dir_entry(parent)?;
Ok(inode) Ok(inode)
} }
/// Create a new INode chardevice
pub fn new_inode_chardevice(&self, device_inode_id: usize) -> vfs::Result<Arc<INodeImpl>> {
let id = self.alloc_block().ok_or(FsError::NoDeviceSpace)?;
let disk_inode = Dirty::new_dirty(DiskINode::new_chardevice());
let new_inode = self._new_inode_with_device(id, disk_inode, device_inode_id);
Ok(new_inode)
}
fn flush_weak_inodes(&self) { fn flush_weak_inodes(&self) {
let mut inodes = self.inodes.write(); let mut inodes = self.inodes.write();
let remove_ids: Vec<_> = inodes let remove_ids: Vec<_> = inodes

@ -158,6 +158,17 @@ impl DiskINode {
db_indirect: 0, db_indirect: 0,
} }
} }
pub const fn new_chardevice() -> Self {
DiskINode {
size: 0,
type_: FileType::CharDevice,
nlinks: 0,
blocks: 0,
direct: [0; NDIRECT],
indirect: 0,
db_indirect: 0,
}
}
} }
/// Convert structs to [u8] slice /// Convert structs to [u8] slice
@ -185,6 +196,8 @@ impl AsBuf for u32 {}
pub type BlockId = usize; pub type BlockId = usize;
pub type INodeId = BlockId; pub type INodeId = BlockId;
pub const NODEVICE: usize = 100;
/// magic number for sfs /// magic number for sfs
pub const MAGIC: u32 = 0x2f8dbe2a; pub const MAGIC: u32 = 0x2f8dbe2a;
/// size of block /// size of block

Loading…
Cancel
Save