diff options
author | Miklos Szeredi <miklos@szeredi.hu> | 2005-08-01 12:48:30 +0000 |
---|---|---|
committer | Miklos Szeredi <miklos@szeredi.hu> | 2005-08-01 12:48:30 +0000 |
commit | 7b28eaeac5f2d8d591b1b0c94e10b64644017869 (patch) | |
tree | b59344f5b9261da640b0e21ac398093755fdfbae /kernel/dir.c | |
parent | e77cc07e755c1e360455e707653f705d0bd25622 (diff) | |
download | libfuse-7b28eaeac5f2d8d591b1b0c94e10b64644017869.tar.gz |
fix
Diffstat (limited to 'kernel/dir.c')
-rw-r--r-- | kernel/dir.c | 45 |
1 files changed, 40 insertions, 5 deletions
diff --git a/kernel/dir.c b/kernel/dir.c index 2aa16e9..26d2dfc 100644 --- a/kernel/dir.c +++ b/kernel/dir.c @@ -476,17 +476,50 @@ static int fuse_revalidate(struct dentry *entry) return fuse_do_getattr(inode); } +static int fuse_access(struct inode *inode, int mask) +{ + struct fuse_conn *fc = get_fuse_conn(inode); + struct fuse_req *req; + struct fuse_access_in inarg; + int err; + + if (fc->no_access) + return 0; + + req = fuse_get_request(fc); + if (!req) + return -EINTR; + + memset(&inarg, 0, sizeof(inarg)); + inarg.mask = mask; + req->in.h.opcode = FUSE_ACCESS; + req->in.h.nodeid = get_node_id(inode); + req->inode = inode; + req->in.numargs = 1; + req->in.args[0].size = sizeof(inarg); + req->in.args[0].value = &inarg; + request_send(fc, req); + err = req->out.h.error; + fuse_put_request(fc, req); + if (err == -ENOSYS) { + fc->no_access = 1; + err = 0; + } + return err; +} + static int fuse_permission(struct inode *inode, int mask, struct nameidata *nd) { struct fuse_conn *fc = get_fuse_conn(inode); + int err; if (!fuse_allow_task(fc, current)) return -EACCES; else if (fc->flags & FUSE_DEFAULT_PERMISSIONS) { #ifdef KERNEL_2_6_10_PLUS - int err = generic_permission(inode, mask, NULL); + err = generic_permission(inode, mask, NULL); #else - int err = vfs_permission(inode, mask); + err = vfs_permission(inode, mask); #endif /* If permission is denied, try to refresh file @@ -510,8 +543,6 @@ static int fuse_permission(struct inode *inode, int mask, struct nameidata *nd) This is actually not so grave, since the user can simply keep access to the file/directory anyway by keeping it open... */ - - return err; } else { int mode = inode->i_mode; if ((mask & MAY_WRITE) && IS_RDONLY(inode) && @@ -519,8 +550,12 @@ static int fuse_permission(struct inode *inode, int mask, struct nameidata *nd) return -EROFS; if ((mask & MAY_EXEC) && !S_ISDIR(mode) && !(mode & S_IXUGO)) return -EACCES; - return 0; + + err = 0; + if (nd->flags & LOOKUP_ACCESS) + err = fuse_access(inode, mask); } + return err; } static int parse_dirfile(char *buf, size_t nbytes, struct file *file, |