fuse: prepare lookup for nfs export

Use d_splice_alias() instead of d_add() in fuse lookup code, to allow NFS
exporting.

Signed-off-by: Miklos Szeredi <mszeredi@suse.cz>
Cc: Christoph Hellwig <hch@lst.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
Miklos Szeredi 2008-07-25 01:48:59 -07:00 committed by Linus Torvalds
parent 764c76b371
commit 0de6256daa

View File

@ -239,18 +239,20 @@ int fuse_valid_type(int m)
* Add a directory inode to a dentry, ensuring that no other dentry * Add a directory inode to a dentry, ensuring that no other dentry
* refers to this inode. Called with fc->inst_mutex. * refers to this inode. Called with fc->inst_mutex.
*/ */
static int fuse_d_add_directory(struct dentry *entry, struct inode *inode) static struct dentry *fuse_d_add_directory(struct dentry *entry,
struct inode *inode)
{ {
struct dentry *alias = d_find_alias(inode); struct dentry *alias = d_find_alias(inode);
if (alias) { if (alias && !(alias->d_flags & DCACHE_DISCONNECTED)) {
/* This tries to shrink the subtree below alias */ /* This tries to shrink the subtree below alias */
fuse_invalidate_entry(alias); fuse_invalidate_entry(alias);
dput(alias); dput(alias);
if (!list_empty(&inode->i_dentry)) if (!list_empty(&inode->i_dentry))
return -EBUSY; return ERR_PTR(-EBUSY);
} else {
dput(alias);
} }
d_add(entry, inode); return d_splice_alias(inode, entry);
return 0;
} }
static struct dentry *fuse_lookup(struct inode *dir, struct dentry *entry, static struct dentry *fuse_lookup(struct inode *dir, struct dentry *entry,
@ -259,6 +261,7 @@ static struct dentry *fuse_lookup(struct inode *dir, struct dentry *entry,
int err; int err;
struct fuse_entry_out outarg; struct fuse_entry_out outarg;
struct inode *inode = NULL; struct inode *inode = NULL;
struct dentry *newent;
struct fuse_conn *fc = get_fuse_conn(dir); 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;
@ -303,21 +306,22 @@ static struct dentry *fuse_lookup(struct inode *dir, struct dentry *entry,
if (inode && S_ISDIR(inode->i_mode)) { if (inode && S_ISDIR(inode->i_mode)) {
mutex_lock(&fc->inst_mutex); mutex_lock(&fc->inst_mutex);
err = fuse_d_add_directory(entry, inode); newent = fuse_d_add_directory(entry, inode);
mutex_unlock(&fc->inst_mutex); mutex_unlock(&fc->inst_mutex);
if (err) { if (IS_ERR(newent)) {
iput(inode); iput(inode);
return ERR_PTR(err); return newent;
} }
} else } else
d_add(entry, inode); newent = d_splice_alias(inode, entry);
entry = newent ? newent : entry;
entry->d_op = &fuse_dentry_operations; entry->d_op = &fuse_dentry_operations;
if (!err) if (!err)
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 NULL; return newent;
} }
/* /*