Skip to content
Snippets Groups Projects
scheme.rs 13.9 KiB
Newer Older
use resource::{Resource, DirResource, FileResource};
Jeremy Soller's avatar
Jeremy Soller committed

use redoxfs::{FileSystem, Node};
Jeremy Soller's avatar
Jeremy Soller committed
use spin::Mutex;
use std::cell::RefCell;
Jeremy Soller's avatar
Jeremy Soller committed
use std::collections::BTreeMap;
Jeremy Soller's avatar
Jeremy Soller committed
use std::str;
use std::sync::atomic::{AtomicUsize, Ordering};
Jeremy Soller's avatar
Jeremy Soller committed

Jeremy Soller's avatar
Jeremy Soller committed
use syscall::data::Stat;
use syscall::error::{Error, Result, EACCES, EEXIST, EISDIR, ENOTDIR, EPERM, ENOENT, EBADF};
Jeremy Soller's avatar
Jeremy Soller committed
use syscall::flag::{O_CREAT, O_TRUNC, O_ACCMODE, O_RDONLY, O_WRONLY, O_RDWR};
Jeremy Soller's avatar
Jeremy Soller committed
use syscall::scheme::Scheme;
Jeremy Soller's avatar
Jeremy Soller committed

pub struct FileScheme {
Jeremy Soller's avatar
Jeremy Soller committed
    name: &'static str,
Jeremy Soller's avatar
Jeremy Soller committed
    fs: RefCell<FileSystem>,
    next_id: AtomicUsize,
    files: Mutex<BTreeMap<usize, Box<Resource>>>
Jeremy Soller's avatar
Jeremy Soller committed
}

impl FileScheme {
Jeremy Soller's avatar
Jeremy Soller committed
    pub fn new(name: &'static str, fs: FileSystem) -> FileScheme {
Jeremy Soller's avatar
Jeremy Soller committed
        FileScheme {
Jeremy Soller's avatar
Jeremy Soller committed
            name: name,
Jeremy Soller's avatar
Jeremy Soller committed
            fs: RefCell::new(fs),
            next_id: AtomicUsize::new(1),
            files: Mutex::new(BTreeMap::new())
Jeremy Soller's avatar
Jeremy Soller committed
        }
    }
Jeremy Soller's avatar
Jeremy Soller committed

impl Scheme for FileScheme {
    fn open(&self, url: &[u8], flags: usize, uid: u32, gid: u32) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        let path = str::from_utf8(url).unwrap_or("").trim_matches('/');
Jeremy Soller's avatar
Jeremy Soller committed

Jeremy Soller's avatar
Jeremy Soller committed
        //println!("Open '{}' {:X}", path, flags);
Jeremy Soller's avatar
Jeremy Soller committed
        let mut fs = self.fs.borrow_mut();

Jeremy Soller's avatar
Jeremy Soller committed
        let mut nodes = Vec::new();
Jeremy Soller's avatar
Jeremy Soller committed
        let node_result = fs.path_nodes(path, &mut nodes);
        for node in nodes.iter() {
            if ! node.1.permission(uid, gid, Node::MODE_EXEC) {
                // println!("dir not executable {:o}", node.1.mode);
                return Err(Error::new(EACCES));
            }
Jeremy Soller's avatar
Jeremy Soller committed
            if ! node.1.is_dir() {
                return Err(Error::new(ENOTDIR));
            }
Jeremy Soller's avatar
Jeremy Soller committed

        let resource: Box<Resource> = match node_result {
Jeremy Soller's avatar
Jeremy Soller committed
            Ok(node) => if node.1.is_dir() {
Jeremy Soller's avatar
Jeremy Soller committed
                if flags & O_ACCMODE != O_RDONLY {
                    // println!("dir not opened with O_RDONLY");
                    return Err(Error::new(EACCES));
                }

                if ! node.1.permission(uid, gid, Node::MODE_READ) {
                    // println!("dir not readable {:o}", node.1.mode);
                    return Err(Error::new(EACCES));
                }

Jeremy Soller's avatar
Jeremy Soller committed
                let mut children = Vec::new();
Jeremy Soller's avatar
Jeremy Soller committed
                try!(fs.child_nodes(&mut children, node.0));
Jeremy Soller's avatar
Jeremy Soller committed
                for child in children.iter() {
                    if let Ok(name) = child.1.name() {
                        if ! data.is_empty() {
Jeremy Soller's avatar
Jeremy Soller committed
                        }
                        data.extend_from_slice(&name.as_bytes());
                    }
                }

                Box::new(DirResource::new(path.to_string(), node.0, data))
Jeremy Soller's avatar
Jeremy Soller committed
            } else {
Jeremy Soller's avatar
Jeremy Soller committed
                if (flags & O_ACCMODE == O_RDONLY || flags & O_ACCMODE == O_RDWR) && ! node.1.permission(uid, gid, Node::MODE_READ) {
                    // println!("file not readable {:o}", node.1.mode);
                    return Err(Error::new(EACCES));
                }

Jeremy Soller's avatar
Jeremy Soller committed
                if (flags & O_ACCMODE == O_WRONLY || flags & O_ACCMODE == O_RDWR) && ! node.1.permission(uid, gid, Node::MODE_WRITE) {
                    // println!("file not writable {:o}", node.1.mode);
                    return Err(Error::new(EACCES));
                }

Jeremy Soller's avatar
Jeremy Soller committed
                if flags & O_TRUNC == O_TRUNC {
                    if ! node.1.permission(uid, gid, Node::MODE_WRITE) {
                        // println!("file not writable {:o}", node.1.mode);
                        return Err(Error::new(EACCES));
                    }

Jeremy Soller's avatar
Jeremy Soller committed
                    try!(fs.node_set_len(node.0, 0));
Jeremy Soller's avatar
Jeremy Soller committed
                Box::new(FileResource::new(path.to_string(), node.0, flags))
Jeremy Soller's avatar
Jeremy Soller committed
            },
            Err(err) => if err.errno == ENOENT && flags & O_CREAT == O_CREAT {
                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() {
                    if let Some(parent) = nodes.last() {
                        if ! parent.1.permission(uid, gid, Node::MODE_WRITE) {
                            // println!("dir not writable {:o}", parent.1.mode);
                            return Err(Error::new(EACCES));
                        }

                        let mut node = try!(fs.create_node(Node::MODE_FILE | (flags as u16 & Node::MODE_PERM), &last_part, parent.0));
                        node.1.uid = uid;
                        node.1.gid = gid;
                        try!(fs.write_at(node.0, &node.1));
Jeremy Soller's avatar
Jeremy Soller committed

                        if (flags & O_ACCMODE == O_RDONLY || flags & O_ACCMODE == O_RDWR) && ! node.1.permission(uid, gid, Node::MODE_READ) {
                            // println!("file not readable {:o}", node.1.mode);
                            return Err(Error::new(EACCES));
                        }

                        if (flags & O_ACCMODE == O_WRONLY || flags & O_ACCMODE == O_RDWR) && ! node.1.permission(uid, gid, Node::MODE_WRITE) {
                            // println!("file not writable {:o}", node.1.mode);
                            return Err(Error::new(EACCES));
                        }

                        Box::new(FileResource::new(path.to_string(), node.0, flags))
Jeremy Soller's avatar
Jeremy Soller committed
                    } else {
                        return Err(Error::new(EPERM));
                    }
                } else {
                    return Err(Error::new(EPERM));
                }
            } else {
                return Err(err);
            }
Jeremy Soller's avatar
Jeremy Soller committed
        let id = self.next_id.fetch_add(1, Ordering::SeqCst);
        self.files.lock().insert(id, resource);
    fn mkdir(&self, url: &[u8], mode: u16, uid: u32, gid: u32) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        let path = str::from_utf8(url).unwrap_or("").trim_matches('/');
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Mkdir '{}'", path);
Jeremy Soller's avatar
Jeremy Soller committed
        let mut fs = self.fs.borrow_mut();

        let mut nodes = Vec::new();
        let node_result = fs.path_nodes(path, &mut nodes);
        for node in nodes.iter() {
            if ! node.1.permission(uid, gid, Node::MODE_EXEC) {
                // println!("dir not executable {:o}", node.1.mode);
                return Err(Error::new(EACCES));
            }
Jeremy Soller's avatar
Jeremy Soller committed
            if ! node.1.is_dir() {
                return Err(Error::new(ENOTDIR));
            }
        }

        match node_result {
            Ok(_node) => Err(Error::new(EEXIST)),
            Err(err) => if err.errno == ENOENT {
                let mut last_part = String::new();
                for part in path.split('/') {
                    if ! part.is_empty() {
                        last_part = part.to_owned();
                    }
                }
                if ! last_part.is_empty() {
                    if let Some(parent) = nodes.last() {
                        if ! parent.1.permission(uid, gid, Node::MODE_WRITE) {
                            // println!("dir not writable {:o}", parent.1.mode);
                            return Err(Error::new(EACCES));
                        }

                        let mut node = try!(fs.create_node(Node::MODE_DIR | (mode & Node::MODE_PERM), &last_part, parent.0));
                        node.1.uid = uid;
                        node.1.gid = gid;
                        try!(fs.write_at(node.0, &node.1));
                        Ok(0)
                    } else {
                        Err(Error::new(EPERM))
                    }
                } else {
                    Err(Error::new(EPERM))
                }
            } else {
                Err(err)
            }
        }
    fn rmdir(&self, url: &[u8], uid: u32, gid: u32) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        let path = str::from_utf8(url).unwrap_or("").trim_matches('/');
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Rmdir '{}'", path);
Jeremy Soller's avatar
Jeremy Soller committed
        let mut fs = self.fs.borrow_mut();

        let mut nodes = Vec::new();
Jeremy Soller's avatar
Jeremy Soller committed
        let child = try!(fs.path_nodes(path, &mut nodes));
        for node in nodes.iter() {
            if ! node.1.permission(uid, gid, Node::MODE_EXEC) {
                // println!("dir not executable {:o}", node.1.mode);
                return Err(Error::new(EACCES));
            }
Jeremy Soller's avatar
Jeremy Soller committed
            if ! node.1.is_dir() {
                return Err(Error::new(ENOTDIR));
            }
        if let Some(parent) = nodes.last() {
            if ! parent.1.permission(uid, gid, Node::MODE_WRITE) {
                // println!("dir not writable {:o}", parent.1.mode);
                return Err(Error::new(EACCES));
            }

            if child.1.is_dir() {
                if ! child.1.permission(uid, gid, Node::MODE_WRITE) {
                    // println!("dir not writable {:o}", parent.1.mode);
                    return Err(Error::new(EACCES));
                }

                if let Ok(child_name) = child.1.name() {
Jeremy Soller's avatar
Jeremy Soller committed
                    fs.remove_node(Node::MODE_DIR, child_name, parent.0).and(Ok(0))
                } else {
                    Err(Error::new(ENOENT))
                }
            } else {
                Err(Error::new(ENOTDIR))
            }
        } else {
            Err(Error::new(EPERM))
        }
    fn unlink(&self, url: &[u8], uid: u32, gid: u32) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        let path = str::from_utf8(url).unwrap_or("").trim_matches('/');
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Unlink '{}'", path);
Jeremy Soller's avatar
Jeremy Soller committed
        let mut fs = self.fs.borrow_mut();

        let mut nodes = Vec::new();
Jeremy Soller's avatar
Jeremy Soller committed
        let child = try!(fs.path_nodes(path, &mut nodes));
        for node in nodes.iter() {
            if ! node.1.permission(uid, gid, Node::MODE_EXEC) {
                // println!("dir not executable {:o}", node.1.mode);
                return Err(Error::new(EACCES));
            }
Jeremy Soller's avatar
Jeremy Soller committed
            if ! node.1.is_dir() {
                return Err(Error::new(ENOTDIR));
            }
        if let Some(parent) = nodes.last() {
            if ! parent.1.permission(uid, gid, Node::MODE_WRITE) {
                // println!("dir not writable {:o}", parent.1.mode);
                return Err(Error::new(EACCES));
            }

            if ! child.1.is_dir() {
                if ! child.1.permission(uid, gid, Node::MODE_WRITE) {
                    // println!("file not writable {:o}", parent.1.mode);
                    return Err(Error::new(EACCES));
                }

                if let Ok(child_name) = child.1.name() {
Jeremy Soller's avatar
Jeremy Soller committed
                    fs.remove_node(Node::MODE_FILE, child_name, parent.0).and(Ok(0))
                } else {
                    Err(Error::new(ENOENT))
                }
            } else {
                Err(Error::new(EISDIR))
            }
        } else {
            Err(Error::new(EPERM))
        }
Jeremy Soller's avatar
Jeremy Soller committed
    }

    /* Resource operations */
Jeremy Soller's avatar
Jeremy Soller committed
    #[allow(unused_variables)]
Jeremy Soller's avatar
Jeremy Soller committed
    fn dup(&self, old_id: usize, _buf: &[u8]) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Dup {}", old_id);
Jeremy Soller's avatar
Jeremy Soller committed

Jeremy Soller's avatar
Jeremy Soller committed
        let mut files = self.files.lock();
Jeremy Soller's avatar
Jeremy Soller committed
        let resource = if let Some(old_resource) = files.get(&old_id) {
            try!(old_resource.dup())
        } else {
            return Err(Error::new(EBADF));
        };
Jeremy Soller's avatar
Jeremy Soller committed

Jeremy Soller's avatar
Jeremy Soller committed
        let id = self.next_id.fetch_add(1, Ordering::SeqCst);
        files.insert(id, resource);
Jeremy Soller's avatar
Jeremy Soller committed

        Ok(id)
    }

Jeremy Soller's avatar
Jeremy Soller committed
    #[allow(unused_variables)]
Jeremy Soller's avatar
Jeremy Soller committed
    fn read(&self, id: usize, buf: &mut [u8]) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Read {}, {:X} {}", id, buf.as_ptr() as usize, buf.len());
Jeremy Soller's avatar
Jeremy Soller committed
        let mut files = self.files.lock();
        if let Some(mut file) = files.get_mut(&id) {
            file.read(buf, &mut self.fs.borrow_mut())
Jeremy Soller's avatar
Jeremy Soller committed
        } else {
            Err(Error::new(EBADF))
        }
    }

Jeremy Soller's avatar
Jeremy Soller committed
    fn write(&self, id: usize, buf: &[u8]) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Write {}, {:X} {}", id, buf.as_ptr() as usize, buf.len());
Jeremy Soller's avatar
Jeremy Soller committed
        let mut files = self.files.lock();
        if let Some(mut file) = files.get_mut(&id) {
            file.write(buf, &mut self.fs.borrow_mut())
Jeremy Soller's avatar
Jeremy Soller committed
        } else {
            Err(Error::new(EBADF))
        }
    }

Jeremy Soller's avatar
Jeremy Soller committed
    fn seek(&self, id: usize, pos: usize, whence: usize) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Seek {}, {} {}", id, pos, whence);
Jeremy Soller's avatar
Jeremy Soller committed
        let mut files = self.files.lock();
        if let Some(mut file) = files.get_mut(&id) {
            file.seek(pos, whence, &mut self.fs.borrow_mut())
Jeremy Soller's avatar
Jeremy Soller committed
        } else {
            Err(Error::new(EBADF))
        }
    }

    fn fpath(&self, id: usize, buf: &mut [u8]) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Fpath {}, {:X} {}", id, buf.as_ptr() as usize, buf.len());
Jeremy Soller's avatar
Jeremy Soller committed
        let files = self.files.lock();
        if let Some(file) = files.get(&id) {
Jeremy Soller's avatar
Jeremy Soller committed
            let name = self.name.as_bytes();
Jeremy Soller's avatar
Jeremy Soller committed
            let mut i = 0;
            while i < buf.len() && i < name.len() {
                buf[i] = name[i];
                i += 1;
            }
            if i < buf.len() {
                buf[i] = b':';
                i += 1;
            }
            file.path(&mut buf[i..]).map(|count| i + count)
Jeremy Soller's avatar
Jeremy Soller committed
        } else {
            Err(Error::new(EBADF))
        }
    }

    fn fstat(&self, id: usize, stat: &mut Stat) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Fstat {}, {:X}", id, stat as *mut Stat as usize);
Jeremy Soller's avatar
Jeremy Soller committed
        let files = self.files.lock();
        if let Some(file) = files.get(&id) {
            file.stat(stat, &mut self.fs.borrow_mut())
Jeremy Soller's avatar
Jeremy Soller committed
        } else {
            Err(Error::new(EBADF))
        }
    }

Jeremy Soller's avatar
Jeremy Soller committed
    fn fsync(&self, id: usize) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Fsync {}", id);
Jeremy Soller's avatar
Jeremy Soller committed
        let mut files = self.files.lock();
        if let Some(mut file) = files.get_mut(&id) {
Jeremy Soller's avatar
Jeremy Soller committed
            file.sync()
        } else {
            Err(Error::new(EBADF))
        }
    }

Jeremy Soller's avatar
Jeremy Soller committed
    fn ftruncate(&self, id: usize, len: usize) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Ftruncate {}, {}", id, len);
Jeremy Soller's avatar
Jeremy Soller committed
        let mut files = self.files.lock();
        if let Some(mut file) = files.get_mut(&id) {
            file.truncate(len, &mut self.fs.borrow_mut())
Jeremy Soller's avatar
Jeremy Soller committed
        } else {
            Err(Error::new(EBADF))
        }
    }

Jeremy Soller's avatar
Jeremy Soller committed
    fn close(&self, id: usize) -> Result<usize> {
Jeremy Soller's avatar
Jeremy Soller committed
        // println!("Close {}", id);
Jeremy Soller's avatar
Jeremy Soller committed
        let mut files = self.files.lock();
        if files.remove(&id).is_some() {
Jeremy Soller's avatar
Jeremy Soller committed
            Ok(0)
        } else {
            Err(Error::new(EBADF))
        }
    }
}