Commit c180eebe authored by Miklos Szeredi's avatar Miklos Szeredi Committed by Linus Torvalds

fuse: add fuse_lookup_name() helper

Add a new helper function which sends a LOOKUP request with the supplied
name.  This will be used by the next patch to send special LOOKUP requests
with "." and ".." as the name.
Signed-off-by: default avatarMiklos Szeredi <mszeredi@suse.cz>
Cc: Christoph Hellwig <hch@lst.de>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent dbd561d2
...@@ -112,18 +112,16 @@ static void fuse_invalidate_entry(struct dentry *entry) ...@@ -112,18 +112,16 @@ static void fuse_invalidate_entry(struct dentry *entry)
fuse_invalidate_entry_cache(entry); fuse_invalidate_entry_cache(entry);
} }
static void fuse_lookup_init(struct fuse_req *req, struct inode *dir, static void fuse_lookup_init(struct fuse_conn *fc, struct fuse_req *req,
struct dentry *entry, u64 nodeid, struct qstr *name,
struct fuse_entry_out *outarg) struct fuse_entry_out *outarg)
{ {
struct fuse_conn *fc = get_fuse_conn(dir);
memset(outarg, 0, sizeof(struct fuse_entry_out)); memset(outarg, 0, sizeof(struct fuse_entry_out));
req->in.h.opcode = FUSE_LOOKUP; req->in.h.opcode = FUSE_LOOKUP;
req->in.h.nodeid = get_node_id(dir); req->in.h.nodeid = nodeid;
req->in.numargs = 1; req->in.numargs = 1;
req->in.args[0].size = entry->d_name.len + 1; req->in.args[0].size = name->len + 1;
req->in.args[0].value = entry->d_name.name; req->in.args[0].value = name->name;
req->out.numargs = 1; req->out.numargs = 1;
if (fc->minor < 9) if (fc->minor < 9)
req->out.args[0].size = FUSE_COMPAT_ENTRY_OUT_SIZE; req->out.args[0].size = FUSE_COMPAT_ENTRY_OUT_SIZE;
...@@ -189,7 +187,8 @@ static int fuse_dentry_revalidate(struct dentry *entry, struct nameidata *nd) ...@@ -189,7 +187,8 @@ static int fuse_dentry_revalidate(struct dentry *entry, struct nameidata *nd)
attr_version = fuse_get_attr_version(fc); attr_version = fuse_get_attr_version(fc);
parent = dget_parent(entry); parent = dget_parent(entry);
fuse_lookup_init(req, parent->d_inode, entry, &outarg); fuse_lookup_init(fc, req, get_node_id(parent->d_inode),
&entry->d_name, &outarg);
request_send(fc, req); request_send(fc, req);
dput(parent); dput(parent);
err = req->out.h.error; err = req->out.h.error;
...@@ -255,73 +254,111 @@ static struct dentry *fuse_d_add_directory(struct dentry *entry, ...@@ -255,73 +254,111 @@ static struct dentry *fuse_d_add_directory(struct dentry *entry,
return d_splice_alias(inode, entry); return d_splice_alias(inode, entry);
} }
static struct dentry *fuse_lookup(struct inode *dir, struct dentry *entry, int fuse_lookup_name(struct super_block *sb, u64 nodeid, struct qstr *name,
struct nameidata *nd) struct fuse_entry_out *outarg, struct inode **inode)
{ {
int err; struct fuse_conn *fc = get_fuse_conn_super(sb);
struct fuse_entry_out outarg;
struct inode *inode = NULL;
struct dentry *newent;
struct fuse_conn *fc = get_fuse_conn(dir);
struct fuse_req *req; struct fuse_req *req;
struct fuse_req *forget_req; struct fuse_req *forget_req;
u64 attr_version; u64 attr_version;
int err;
if (entry->d_name.len > FUSE_NAME_MAX) *inode = NULL;
return ERR_PTR(-ENAMETOOLONG); err = -ENAMETOOLONG;
if (name->len > FUSE_NAME_MAX)
goto out;
req = fuse_get_req(fc); req = fuse_get_req(fc);
err = PTR_ERR(req);
if (IS_ERR(req)) if (IS_ERR(req))
return ERR_CAST(req); goto out;
forget_req = fuse_get_req(fc); forget_req = fuse_get_req(fc);
err = PTR_ERR(forget_req);
if (IS_ERR(forget_req)) { if (IS_ERR(forget_req)) {
fuse_put_request(fc, req); fuse_put_request(fc, req);
return ERR_CAST(forget_req); goto out;
} }
attr_version = fuse_get_attr_version(fc); attr_version = fuse_get_attr_version(fc);
fuse_lookup_init(req, dir, entry, &outarg); fuse_lookup_init(fc, req, nodeid, name, outarg);
request_send(fc, req); request_send(fc, req);
err = req->out.h.error; err = req->out.h.error;
fuse_put_request(fc, req); fuse_put_request(fc, req);
/* Zero nodeid is same as -ENOENT, but with valid timeout */ /* Zero nodeid is same as -ENOENT, but with valid timeout */
if (!err && outarg.nodeid && if (err || !outarg->nodeid)
(invalid_nodeid(outarg.nodeid) || goto out_put_forget;
!fuse_valid_type(outarg.attr.mode)))
err = -EIO; err = -EIO;
if (!err && outarg.nodeid) { if (!outarg->nodeid)
inode = fuse_iget(dir->i_sb, outarg.nodeid, outarg.generation, goto out_put_forget;
&outarg.attr, entry_attr_timeout(&outarg), if (!fuse_valid_type(outarg->attr.mode))
attr_version); goto out_put_forget;
if (!inode) {
fuse_send_forget(fc, forget_req, outarg.nodeid, 1); *inode = fuse_iget(sb, outarg->nodeid, outarg->generation,
return ERR_PTR(-ENOMEM); &outarg->attr, entry_attr_timeout(outarg),
} attr_version);
err = -ENOMEM;
if (!*inode) {
fuse_send_forget(fc, forget_req, outarg->nodeid, 1);
goto out;
} }
err = 0;
out_put_forget:
fuse_put_request(fc, forget_req); fuse_put_request(fc, forget_req);
if (err && err != -ENOENT) out:
return ERR_PTR(err); return err;
}
static struct dentry *fuse_lookup(struct inode *dir, struct dentry *entry,
struct nameidata *nd)
{
int err;
struct fuse_entry_out outarg;
struct inode *inode;
struct dentry *newent;
struct fuse_conn *fc = get_fuse_conn(dir);
bool outarg_valid = true;
err = fuse_lookup_name(dir->i_sb, get_node_id(dir), &entry->d_name,
&outarg, &inode);
if (err == -ENOENT) {
outarg_valid = false;
err = 0;
}
if (err)
goto out_err;
err = -EIO;
if (inode && get_node_id(inode) == FUSE_ROOT_ID)
goto out_iput;
if (inode && S_ISDIR(inode->i_mode)) { if (inode && S_ISDIR(inode->i_mode)) {
mutex_lock(&fc->inst_mutex); mutex_lock(&fc->inst_mutex);
newent = fuse_d_add_directory(entry, inode); newent = fuse_d_add_directory(entry, inode);
mutex_unlock(&fc->inst_mutex); mutex_unlock(&fc->inst_mutex);
if (IS_ERR(newent)) { err = PTR_ERR(newent);
iput(inode); if (IS_ERR(newent))
return newent; goto out_iput;
} } else {
} else
newent = d_splice_alias(inode, entry); newent = d_splice_alias(inode, entry);
}
entry = newent ? newent : entry; entry = newent ? newent : entry;
entry->d_op = &fuse_dentry_operations; entry->d_op = &fuse_dentry_operations;
if (!err) if (outarg_valid)
fuse_change_entry_timeout(entry, &outarg); fuse_change_entry_timeout(entry, &outarg);
else else
fuse_invalidate_entry_cache(entry); fuse_invalidate_entry_cache(entry);
return newent; return newent;
out_iput:
iput(inode);
out_err:
return ERR_PTR(err);
} }
/* /*
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment