Commit 62e1761c authored by Trond Myklebust's avatar Trond Myklebust

SUNRPC: Clean up rpc_pipefs.

Add a dentry_ops with a d_delete() method in order to ensure that dentries
are removed as soon as the last reference is gone.

Clean up rpc_depopulate() so that it only removes files that were created
via rpc_populate().
Signed-off-by: default avatarTrond Myklebust <Trond.Myklebust@netapp.com>
parent 34f30896
...@@ -448,6 +448,15 @@ void rpc_put_mount(void) ...@@ -448,6 +448,15 @@ void rpc_put_mount(void)
simple_release_fs(&rpc_mount, &rpc_mount_count); simple_release_fs(&rpc_mount, &rpc_mount_count);
} }
static int rpc_delete_dentry(struct dentry *dentry)
{
return 1;
}
static struct dentry_operations rpc_dentry_operations = {
.d_delete = rpc_delete_dentry,
};
static int static int
rpc_lookup_parent(char *path, struct nameidata *nd) rpc_lookup_parent(char *path, struct nameidata *nd)
{ {
...@@ -506,7 +515,7 @@ rpc_get_inode(struct super_block *sb, int mode) ...@@ -506,7 +515,7 @@ rpc_get_inode(struct super_block *sb, int mode)
* FIXME: This probably has races. * FIXME: This probably has races.
*/ */
static void static void
rpc_depopulate(struct dentry *parent) rpc_depopulate(struct dentry *parent, int start, int eof)
{ {
struct inode *dir = parent->d_inode; struct inode *dir = parent->d_inode;
struct list_head *pos, *next; struct list_head *pos, *next;
...@@ -518,6 +527,10 @@ repeat: ...@@ -518,6 +527,10 @@ repeat:
spin_lock(&dcache_lock); spin_lock(&dcache_lock);
list_for_each_safe(pos, next, &parent->d_subdirs) { list_for_each_safe(pos, next, &parent->d_subdirs) {
dentry = list_entry(pos, struct dentry, d_u.d_child); dentry = list_entry(pos, struct dentry, d_u.d_child);
if (!dentry->d_inode ||
dentry->d_inode->i_ino < start ||
dentry->d_inode->i_ino >= eof)
continue;
spin_lock(&dentry->d_lock); spin_lock(&dentry->d_lock);
if (!d_unhashed(dentry)) { if (!d_unhashed(dentry)) {
dget_locked(dentry); dget_locked(dentry);
...@@ -533,11 +546,11 @@ repeat: ...@@ -533,11 +546,11 @@ repeat:
if (n) { if (n) {
do { do {
dentry = dvec[--n]; dentry = dvec[--n];
if (dentry->d_inode) { if (S_ISREG(dentry->d_inode->i_mode))
rpc_close_pipes(dentry->d_inode);
simple_unlink(dir, dentry); simple_unlink(dir, dentry);
} else if (S_ISDIR(dentry->d_inode->i_mode))
inode_dir_notify(dir, DN_DELETE); simple_rmdir(dir, dentry);
d_delete(dentry);
dput(dentry); dput(dentry);
} while (n); } while (n);
goto repeat; goto repeat;
...@@ -560,6 +573,7 @@ rpc_populate(struct dentry *parent, ...@@ -560,6 +573,7 @@ rpc_populate(struct dentry *parent,
dentry = d_alloc_name(parent, files[i].name); dentry = d_alloc_name(parent, files[i].name);
if (!dentry) if (!dentry)
goto out_bad; goto out_bad;
dentry->d_op = &rpc_dentry_operations;
mode = files[i].mode; mode = files[i].mode;
inode = rpc_get_inode(dir->i_sb, mode); inode = rpc_get_inode(dir->i_sb, mode);
if (!inode) { if (!inode) {
...@@ -607,17 +621,10 @@ static int ...@@ -607,17 +621,10 @@ static int
__rpc_rmdir(struct inode *dir, struct dentry *dentry) __rpc_rmdir(struct inode *dir, struct dentry *dentry)
{ {
int error; int error;
error = simple_rmdir(dir, dentry);
shrink_dcache_parent(dentry); if (!error)
if (d_unhashed(dentry)) d_delete(dentry);
return 0; return error;
if ((error = simple_rmdir(dir, dentry)) != 0)
return error;
if (!error) {
inode_dir_notify(dir, DN_DELETE);
d_drop(dentry);
}
return 0;
} }
static struct dentry * static struct dentry *
...@@ -630,7 +637,9 @@ rpc_lookup_create(struct dentry *parent, const char *name, int len, int exclusiv ...@@ -630,7 +637,9 @@ rpc_lookup_create(struct dentry *parent, const char *name, int len, int exclusiv
dentry = lookup_one_len(name, parent, len); dentry = lookup_one_len(name, parent, len);
if (IS_ERR(dentry)) if (IS_ERR(dentry))
goto out_err; goto out_err;
if (dentry->d_inode && exclusive) { if (!dentry->d_inode)
dentry->d_op = &rpc_dentry_operations;
else if (exclusive) {
dput(dentry); dput(dentry);
dentry = ERR_PTR(-EEXIST); dentry = ERR_PTR(-EEXIST);
goto out_err; goto out_err;
...@@ -681,7 +690,7 @@ out: ...@@ -681,7 +690,7 @@ out:
rpc_release_path(&nd); rpc_release_path(&nd);
return dentry; return dentry;
err_depopulate: err_depopulate:
rpc_depopulate(dentry); rpc_depopulate(dentry, RPCAUTH_info, RPCAUTH_EOF);
__rpc_rmdir(dir, dentry); __rpc_rmdir(dir, dentry);
err_dput: err_dput:
dput(dentry); dput(dentry);
...@@ -701,7 +710,7 @@ rpc_rmdir(struct dentry *dentry) ...@@ -701,7 +710,7 @@ rpc_rmdir(struct dentry *dentry)
parent = dget_parent(dentry); parent = dget_parent(dentry);
dir = parent->d_inode; dir = parent->d_inode;
mutex_lock_nested(&dir->i_mutex, I_MUTEX_PARENT); mutex_lock_nested(&dir->i_mutex, I_MUTEX_PARENT);
rpc_depopulate(dentry); rpc_depopulate(dentry, RPCAUTH_info, RPCAUTH_EOF);
error = __rpc_rmdir(dir, dentry); error = __rpc_rmdir(dir, dentry);
dput(dentry); dput(dentry);
mutex_unlock(&dir->i_mutex); mutex_unlock(&dir->i_mutex);
...@@ -764,14 +773,10 @@ rpc_unlink(struct dentry *dentry) ...@@ -764,14 +773,10 @@ rpc_unlink(struct dentry *dentry)
parent = dget_parent(dentry); parent = dget_parent(dentry);
dir = parent->d_inode; dir = parent->d_inode;
mutex_lock_nested(&dir->i_mutex, I_MUTEX_PARENT); mutex_lock_nested(&dir->i_mutex, I_MUTEX_PARENT);
if (!d_unhashed(dentry)) { rpc_close_pipes(dentry->d_inode);
d_drop(dentry); error = simple_unlink(dir, dentry);
if (dentry->d_inode) { if (!error)
rpc_close_pipes(dentry->d_inode); d_delete(dentry);
error = simple_unlink(dir, dentry);
}
inode_dir_notify(dir, DN_DELETE);
}
dput(dentry); dput(dentry);
mutex_unlock(&dir->i_mutex); mutex_unlock(&dir->i_mutex);
dput(parent); dput(parent);
......
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