diff --git a/scheme/scheme.rs b/scheme/scheme.rs
index 3774a65cedc874cc5b3c5555267255dc660b4b92..06d8406700f3a7aa2e8119e885ce963b298b50e3 100644
--- a/scheme/scheme.rs
+++ b/scheme/scheme.rs
@@ -9,7 +9,7 @@ use std::sync::atomic::{AtomicUsize, Ordering};
 
 use syscall::data::{Stat, StatVfs};
 use syscall::error::{Error, Result, EACCES, EEXIST, EISDIR, ENOTDIR, EPERM, ENOENT, EBADF};
-use syscall::flag::{O_CREAT, O_TRUNC, O_ACCMODE, O_RDONLY, O_WRONLY, O_RDWR, MODE_PERM};
+use syscall::flag::{O_CREAT, O_DIRECTORY, O_STAT, O_EXCL, O_TRUNC, O_ACCMODE, O_RDONLY, O_WRONLY, O_RDWR, MODE_PERM};
 use syscall::scheme::Scheme;
 
 pub struct FileScheme {
@@ -51,31 +51,46 @@ impl Scheme for FileScheme {
         }
 
         let resource: Box<Resource> = match node_result {
-            Ok(node) => if node.1.is_dir() {
-                if flags & O_ACCMODE != O_RDONLY {
-                    // println!("dir not opened with O_RDONLY");
-                    return Err(Error::new(EACCES));
+            Ok(node) => if flags & (O_CREAT | O_EXCL) == O_CREAT | O_EXCL {
+                return Err(Error::new(EEXIST));
+            } else if node.1.is_dir() {
+                if flags & O_STAT != O_STAT && flags & O_DIRECTORY != O_DIRECTORY {
+                    // println!("{:X} & {:X}: EISDIR {}", flags, O_DIRECTORY, path);
+                    return Err(Error::new(EISDIR));
                 }
 
-                if ! node.1.permission(uid, gid, Node::MODE_READ) {
-                    // println!("dir not readable {:o}", node.1.mode);
-                    return Err(Error::new(EACCES));
-                }
+                if flags & O_ACCMODE == O_RDONLY {
+                    if ! node.1.permission(uid, gid, Node::MODE_READ) {
+                        // println!("dir not readable {:o}", node.1.mode);
+                        return Err(Error::new(EACCES));
+                    }
 
-                let mut data = Vec::new();
-                let mut children = Vec::new();
-                try!(fs.child_nodes(&mut children, node.0));
-                for child in children.iter() {
-                    if let Ok(name) = child.1.name() {
-                        if ! data.is_empty() {
-                            data.push(b'\n');
+                    let mut children = Vec::new();
+                    try!(fs.child_nodes(&mut children, node.0));
+
+                    let mut data = Vec::new();
+                    for child in children.iter() {
+                        if let Ok(name) = child.1.name() {
+                            if ! data.is_empty() {
+                                data.push(b'\n');
+                            }
+                            data.extend_from_slice(&name.as_bytes());
                         }
-                        data.extend_from_slice(&name.as_bytes());
                     }
-                }
 
-                Box::new(DirResource::new(path.to_string(), node.0, data))
+                    Box::new(DirResource::new(path.to_string(), node.0, data))
+                } else if flags & O_STAT == O_STAT {
+                    Box::new(DirResource::new(path.to_string(), node.0, Vec::new()))
+                } else {
+                    // println!("dir not opened with O_RDONLY");
+                    return Err(Error::new(EACCES));
+                }
             } else {
+                if flags & O_DIRECTORY == O_DIRECTORY {
+                    // println!("{:X} & {:X}: ENOTDIR {}", flags, O_DIRECTORY, path);
+                    return Err(Error::new(ENOTDIR));
+                }
+
                 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));
@@ -111,7 +126,9 @@ impl Scheme for FileScheme {
                             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));
+                        let dir = flags & O_DIRECTORY == O_DIRECTORY;
+
+                        let mut node = try!(fs.create_node(if dir { Node::MODE_DIR } else { 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));
@@ -126,7 +143,11 @@ impl Scheme for FileScheme {
                             return Err(Error::new(EACCES));
                         }
 
-                        Box::new(FileResource::new(path.to_string(), node.0, flags))
+                        if dir {
+                            Box::new(DirResource::new(path.to_string(), node.0, Vec::new()))
+                        } else {
+                            Box::new(FileResource::new(path.to_string(), node.0, flags))
+                        }
                     } else {
                         return Err(Error::new(EPERM));
                     }
@@ -144,58 +165,6 @@ impl Scheme for FileScheme {
         Ok(id)
     }
 
-    fn mkdir(&self, url: &[u8], mode: u16, uid: u32, gid: u32) -> Result<usize> {
-        let path = str::from_utf8(url).unwrap_or("").trim_matches('/');
-
-        // println!("Mkdir '{}'", path);
-
-        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));
-            }
-            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 chmod(&self, url: &[u8], mode: u16, uid: u32, gid: u32) -> Result<usize> {
         let path = str::from_utf8(url).unwrap_or("").trim_matches('/');