diff --git a/src/filesystem.rs b/src/filesystem.rs index 406fed5fca13ab06215a5d7957e721ed1c3f109b..a5810e9d35b0b220b8b28a03908cc303e57b0a3e 100644 --- a/src/filesystem.rs +++ b/src/filesystem.rs @@ -48,11 +48,11 @@ impl<D: Disk> FileSystem<D> { let block_offset = (reserved.len() as u64 + BLOCK_SIZE - 1)/BLOCK_SIZE; if size >= (block_offset + 4) * BLOCK_SIZE { - let mut free = (2, Node::new(Node::MODE_FILE, "free", 0, ctime, ctime_nsec)); + let mut free = (2, Node::new(Node::MODE_FILE, "free", 0, ctime, ctime_nsec)?); free.1.extents[0] = Extent::new(4, size - (block_offset + 4) * BLOCK_SIZE); disk.write_at(block_offset + free.0, &free.1)?; - let root = (1, Node::new(Node::MODE_DIR | 0o755, "root", 0, ctime, ctime_nsec)); + let root = (1, Node::new(Node::MODE_DIR | 0o755, "root", 0, ctime, ctime_nsec)?); disk.write_at(block_offset + root.0, &root.1)?; let header = (0, Header::new(size, root.0, free.0)); @@ -171,7 +171,8 @@ impl<D: Disk> FileSystem<D> { self.find_node(name, parent.1.next) } - fn insert_blocks(&mut self, block: u64, length: u64, parent_block: u64) -> Result<()> { + //TODO: Accept length in units of BLOCK_SIZE to match remove_blocks + pub fn insert_blocks(&mut self, block: u64, length: u64, parent_block: u64) -> Result<()> { if parent_block == 0 { return Err(Error::new(ENOSPC)); } @@ -222,7 +223,8 @@ impl<D: Disk> FileSystem<D> { if self.find_node(name, parent_block).is_ok() { Err(Error::new(EEXIST)) } else { - let node = (self.allocate(1)?, Node::new(mode, name, parent_block, ctime, ctime_nsec)); + let node_data = Node::new(mode, name, parent_block, ctime, ctime_nsec)?; + let node = (self.allocate(1)?, node_data); self.write_at(node.0, &node.1)?; self.insert_blocks(node.0, BLOCK_SIZE, parent_block)?; @@ -231,7 +233,7 @@ impl<D: Disk> FileSystem<D> { } } - fn remove_blocks(&mut self, block: u64, length: u64, parent_block: u64) -> Result<()> { + pub fn remove_blocks(&mut self, block: u64, length: u64, parent_block: u64) -> Result<()> { if parent_block == 0 { return Err(Error::new(ENOENT)); } @@ -270,8 +272,6 @@ impl<D: Disk> FileSystem<D> { self.insert_blocks(replace.block, replace.length, parent_block)?; } - self.deallocate(block, BLOCK_SIZE)?; - Ok(()) } else { self.remove_blocks(block, length, parent.1.next) @@ -292,6 +292,7 @@ impl<D: Disk> FileSystem<D> { self.node_set_len(node.0, 0)?; self.remove_blocks(node.0, 1, parent_block)?; self.write_at(node.0, &Node::default())?; + self.deallocate(node.0, BLOCK_SIZE)?; Ok(()) } else if node.1.is_dir() { diff --git a/src/mount/redox/resource.rs b/src/mount/redox/resource.rs index 59b9a25e4eb35b4348df0b1703a4f094e02febd3..7d812bcce3532629c947c72d9aa5ae22c28197c9 100644 --- a/src/mount/redox/resource.rs +++ b/src/mount/redox/resource.rs @@ -10,6 +10,7 @@ use disk::Disk; use filesystem::FileSystem; pub trait Resource<D: Disk> { + fn block(&self) -> u64; fn dup(&self) -> Result<Box<Resource<D>>>; fn read(&mut self, buf: &mut [u8], fs: &mut FileSystem<D>) -> Result<usize>; fn write(&mut self, buf: &[u8], fs: &mut FileSystem<D>) -> Result<usize>; @@ -45,6 +46,10 @@ impl DirResource { } impl<D: Disk> Resource<D> for DirResource { + fn block(&self) -> u64 { + self.block + } + fn dup(&self) -> Result<Box<Resource<D>>> { Ok(Box::new(DirResource { path: self.path.clone(), @@ -187,6 +192,10 @@ impl FileResource { } impl<D: Disk> Resource<D> for FileResource { + fn block(&self) -> u64 { + self.block + } + fn dup(&self) -> Result<Box<Resource<D>>> { Ok(Box::new(FileResource { path: self.path.clone(), diff --git a/src/mount/redox/scheme.rs b/src/mount/redox/scheme.rs index 447e226f06499bb859442b1bb957d3a7bf5ae15a..966b56501a68b6d0b5198fe2270a62728809d9d8 100644 --- a/src/mount/redox/scheme.rs +++ b/src/mount/redox/scheme.rs @@ -5,7 +5,7 @@ use std::sync::atomic::{AtomicUsize, Ordering}; use std::time::{SystemTime, UNIX_EPOCH}; use syscall::data::{Stat, StatVfs, TimeSpec}; -use syscall::error::{Error, Result, EACCES, EEXIST, EISDIR, ENOTDIR, EPERM, ENOENT, EBADF, ELOOP, EINVAL}; +use syscall::error::{Error, Result, EACCES, EEXIST, EISDIR, ENOTDIR, ENOTEMPTY, EPERM, ENOENT, EBADF, ELOOP, EINVAL}; use syscall::flag::{O_APPEND, O_CREAT, O_DIRECTORY, O_STAT, O_EXCL, O_TRUNC, O_ACCMODE, O_RDONLY, O_WRONLY, O_RDWR, MODE_PERM, O_SYMLINK, O_NOFOLLOW}; use syscall::scheme::Scheme; @@ -522,11 +522,97 @@ impl<D: Disk> Scheme for FileScheme<D> { fn frename(&self, id: usize, url: &[u8], uid: u32, gid: u32) -> Result<usize> { let path = str::from_utf8(url).unwrap_or("").trim_matches('/'); - println!("Frename {}, {} from {}, {}", id, path, uid, gid); + // println!("Frename {}, {} from {}, {}", id, path, uid, gid); + let files = self.files.lock(); - if let Some(_file) = files.get(&id) { - // TODO - Err(Error::new(EPERM)) + if let Some(file) = files.get(&id) { + //TODO: Check for EINVAL + // The new pathname contained a path prefix of the old, or, more generally, + // an attempt was made to make a directory a subdirectory of itself. + + let mut last_part = String::new(); + for part in path.split('/') { + if ! part.is_empty() { + last_part = part.to_string(); + } + } + if last_part.is_empty() { + return Err(Error::new(EPERM)); + } + + let mut fs = self.fs.borrow_mut(); + + let mut orig = fs.node(file.block())?; + + if ! orig.1.owner(uid) { + // println!("orig not owned by caller {}", uid); + return Err(Error::new(EACCES)); + } + + let mut nodes = Vec::new(); + let node_opt = self.path_nodes(&mut fs, path, uid, gid, &mut nodes)?; + + if let Some(parent) = nodes.last() { + if ! parent.1.owner(uid) { + // println!("parent not owned by caller {}", uid); + return Err(Error::new(EACCES)); + } + + if let Some(ref node) = node_opt { + if ! node.1.owner(uid) { + // println!("new dir not owned by caller {}", uid); + return Err(Error::new(EACCES)); + } + + if node.1.is_dir() { + if ! orig.1.is_dir() { + // println!("orig is file, new is dir"); + return Err(Error::new(EACCES)); + } + + let mut children = Vec::new(); + fs.child_nodes(&mut children, node.0)?; + + if ! children.is_empty() { + // println!("new dir not empty"); + return Err(Error::new(ENOTEMPTY)); + } + } else { + if orig.1.is_dir() { + // println!("orig is dir, new is file"); + return Err(Error::new(ENOTDIR)); + } + } + } + + let orig_parent = orig.1.parent; + + orig.1.set_name(&last_part)?; + orig.1.parent = parent.0; + + if parent.0 != orig_parent { + fs.remove_blocks(orig.0, 1, orig_parent)?; + } + + fs.write_at(orig.0, &orig.1)?; + + if let Some(node) = node_opt { + if node.0 != orig.0 { + fs.node_set_len(node.0, 0)?; + fs.remove_blocks(node.0, 1, parent.0)?; + fs.write_at(node.0, &Node::default())?; + fs.deallocate(node.0, BLOCK_SIZE)?; + } + } + + if parent.0 != orig_parent { + fs.insert_blocks(orig.0, BLOCK_SIZE, parent.0)?; + } + + Ok(0) + } else { + Err(Error::new(EPERM)) + } } else { Err(Error::new(EBADF)) } diff --git a/src/node.rs b/src/node.rs index 22887fdda52d62766864934d551a7a4ad94ca0b5..1ae66c1ed77d439d7b30955ffc1e1862ee16c1dc 100644 --- a/src/node.rs +++ b/src/node.rs @@ -2,6 +2,7 @@ use std::{fmt, mem, ops, slice, str}; use BLOCK_SIZE; use super::Extent; +use syscall; /// A file/folder node #[repr(packed)] @@ -46,13 +47,16 @@ impl Node { } } - pub fn new(mode: u16, name: &str, parent: u64, ctime: u64, ctime_nsec: u32) -> Node { + pub fn new(mode: u16, name: &str, parent: u64, ctime: u64, ctime_nsec: u32) -> syscall::Result<Node> { let mut bytes = [0; 222]; + if name.len() > bytes.len() { + return Err(syscall::Error::new(syscall::ENAMETOOLONG)); + } for (b, c) in bytes.iter_mut().zip(name.bytes()) { *b = c; } - Node { + Ok(Node { mode: mode, uid: 0, gid: 0, @@ -64,7 +68,7 @@ impl Node { parent: parent, next: 0, extents: [Extent::default(); (BLOCK_SIZE as usize - 272)/16], - } + }) } pub fn name(&self) -> Result<&str, str::Utf8Error> { @@ -80,6 +84,20 @@ impl Node { str::from_utf8(&self.name[..len]) } + pub fn set_name(&mut self, name: &str) -> syscall::Result<()> { + let mut bytes = [0; 222]; + if name.len() > bytes.len() { + return Err(syscall::Error::new(syscall::ENAMETOOLONG)); + } + for (b, c) in bytes.iter_mut().zip(name.bytes()) { + *b = c; + } + + self.name = bytes; + + Ok(()) + } + pub fn is_dir(&self) -> bool { self.mode & Node::MODE_TYPE == Node::MODE_DIR } @@ -92,6 +110,10 @@ impl Node { self.mode & Node::MODE_TYPE == Node::MODE_SYMLINK } + pub fn owner(&self, uid: u32) -> bool { + uid == 0 || self.uid == uid + } + pub fn permission(&self, uid: u32, gid: u32, op: u16) -> bool { let mut perm = self.mode & 0o7; if self.uid == uid {