From 602a63acf643dbd60cdeaabf0dbdad240421de17 Mon Sep 17 00:00:00 2001 From: Jeremy Soller <jeremy@system76.com> Date: Tue, 23 Jul 2019 21:05:54 -0600 Subject: [PATCH] Return EXDEV to tell kernel to resolve cross-scheme links --- src/mount/redox/scheme.rs | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/src/mount/redox/scheme.rs b/src/mount/redox/scheme.rs index f8e4e30..6be8e1f 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::{Map, Stat, StatVfs, TimeSpec}; -use syscall::error::{Error, Result, EACCES, EEXIST, EISDIR, ENOTDIR, ENOTEMPTY, EPERM, ENOENT, EBADF, ELOOP, EINVAL}; +use syscall::error::{Error, Result, EACCES, EEXIST, EISDIR, ENOTDIR, ENOTEMPTY, EPERM, ENOENT, EBADF, ELOOP, EINVAL, EXDEV}; use syscall::flag::{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; @@ -37,7 +37,7 @@ impl<D: Disk> FileScheme<D> { fn resolve_symlink(&self, fs: &mut FileSystem<D>, uid: u32, gid: u32, url: &[u8], node: (u64, Node), nodes: &mut Vec<(u64, Node)>) -> Result<Vec<u8>> { let mut node = node; - for _ in 1..10 { // XXX What should the limit be? + for _ in 0..32 { // XXX What should the limit be? let mut buf = [0; 4096]; let count = fs.read_node(node.0, 0, &mut buf)?; let scheme = format!("{}:", &self.name); @@ -50,8 +50,7 @@ impl<D: Disk> FileScheme<D> { nodes.push(next_node); return Ok(canon[scheme.len()..].to_vec()); } else { - // TODO: Find way to support symlink to another scheme - return Err(Error::new(ENOENT)); + return Err(Error::new(EXDEV)); } } node = next_node; @@ -218,7 +217,7 @@ impl<D: Disk> Scheme for FileScheme<D> { } else { Box::new(DirResource::new(path.to_string(), node.0, None, uid)) } - } else if node.1.is_symlink() && !(flags & O_STAT == O_STAT && flags & O_NOFOLLOW == O_NOFOLLOW) && flags & O_SYMLINK != O_SYMLINK { + } else if node.1.is_symlink() && !(flags & O_STAT == O_STAT && flags & O_NOFOLLOW == O_NOFOLLOW) && flags & O_SYMLINK != O_SYMLINK { let mut resolve_nodes = Vec::new(); let resolved = self.resolve_symlink(&mut fs, uid, gid, url, node, &mut resolve_nodes)?; drop(fs); -- GitLab