Commit 877907d3 authored by Jeremy Kerr's avatar Jeremy Kerr Committed by Paul Mackerras

[POWERPC] spufs: Fix gang destroy leaks

Previously, closing a SPE gang that still has contexts would trigger
a WARN_ON, and leak the allocated gang.

This change fixes the problem by using the gang's reference counts to
destroy the gang instead. The gangs will persist until their last
reference (be it context or open file handle) is gone.

Also, avoid using statements with side-effects in a WARN_ON().
Signed-off-by: default avatarJeremy Kerr <jk@ozlabs.org>
Signed-off-by: default avatarArnd Bergmann <arnd.bergmann@de.ibm.com>
Signed-off-by: default avatarPaul Mackerras <paulus@samba.org>
parent ce92987b
...@@ -349,37 +349,6 @@ static int spufs_create_context(struct inode *inode, ...@@ -349,37 +349,6 @@ static int spufs_create_context(struct inode *inode,
return ret; return ret;
} }
static int spufs_rmgang(struct inode *root, struct dentry *dir)
{
/* FIXME: this fails if the dir is not empty,
which causes a leak of gangs. */
return simple_rmdir(root, dir);
}
static int spufs_gang_close(struct inode *inode, struct file *file)
{
struct inode *parent;
struct dentry *dir;
int ret;
dir = file->f_path.dentry;
parent = dir->d_parent->d_inode;
ret = spufs_rmgang(parent, dir);
WARN_ON(ret);
return dcache_dir_close(inode, file);
}
const struct file_operations spufs_gang_fops = {
.open = dcache_dir_open,
.release = spufs_gang_close,
.llseek = dcache_dir_lseek,
.read = generic_read_dir,
.readdir = dcache_readdir,
.fsync = simple_sync_file,
};
static int static int
spufs_mkgang(struct inode *dir, struct dentry *dentry, int mode) spufs_mkgang(struct inode *dir, struct dentry *dentry, int mode)
{ {
...@@ -407,7 +376,6 @@ spufs_mkgang(struct inode *dir, struct dentry *dentry, int mode) ...@@ -407,7 +376,6 @@ spufs_mkgang(struct inode *dir, struct dentry *dentry, int mode)
inode->i_fop = &simple_dir_operations; inode->i_fop = &simple_dir_operations;
d_instantiate(dentry, inode); d_instantiate(dentry, inode);
dget(dentry);
dir->i_nlink++; dir->i_nlink++;
dentry->d_inode->i_nlink++; dentry->d_inode->i_nlink++;
return ret; return ret;
...@@ -437,7 +405,7 @@ static int spufs_gang_open(struct dentry *dentry, struct vfsmount *mnt) ...@@ -437,7 +405,7 @@ static int spufs_gang_open(struct dentry *dentry, struct vfsmount *mnt)
goto out; goto out;
} }
filp->f_op = &spufs_gang_fops; filp->f_op = &simple_dir_operations;
fd_install(ret, filp); fd_install(ret, filp);
out: out:
return ret; return ret;
...@@ -458,8 +426,10 @@ static int spufs_create_gang(struct inode *inode, ...@@ -458,8 +426,10 @@ static int spufs_create_gang(struct inode *inode,
* in error path of *_open(). * in error path of *_open().
*/ */
ret = spufs_gang_open(dget(dentry), mntget(mnt)); ret = spufs_gang_open(dget(dentry), mntget(mnt));
if (ret < 0) if (ret < 0) {
WARN_ON(spufs_rmgang(inode, dentry)); int err = simple_rmdir(inode, dentry);
WARN_ON(err);
}
out: out:
mutex_unlock(&inode->i_mutex); mutex_unlock(&inode->i_mutex);
......
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