diff options
Diffstat (limited to 'fs/coda/inode.c')
| -rw-r--r-- | fs/coda/inode.c | 61 | 
1 files changed, 30 insertions, 31 deletions
| diff --git a/fs/coda/inode.c b/fs/coda/inode.c index bfe8179b1295..7993b96ca348 100644 --- a/fs/coda/inode.c +++ b/fs/coda/inode.c @@ -15,7 +15,8 @@  #include <linux/stat.h>  #include <linux/errno.h>  #include <linux/unistd.h> -#include <linux/smp_lock.h> +#include <linux/mutex.h> +#include <linux/spinlock.h>  #include <linux/file.h>  #include <linux/vfs.h>  #include <linux/slab.h> @@ -51,6 +52,7 @@ static struct inode *coda_alloc_inode(struct super_block *sb)  	ei->c_flags = 0;  	ei->c_uid = 0;  	ei->c_cached_perm = 0; +	spin_lock_init(&ei->c_lock);  	return &ei->vfs_inode;  } @@ -143,13 +145,11 @@ static int get_device_index(struct coda_mount_data *data)  static int coda_fill_super(struct super_block *sb, void *data, int silent)  {  	struct inode *root = NULL; -	struct venus_comm *vc = NULL; +	struct venus_comm *vc;  	struct CodaFid fid;  	int error;  	int idx; -	lock_kernel(); -  	idx = get_device_index((struct coda_mount_data *) data);  	/* Ignore errors in data, for backward compatibility */ @@ -159,23 +159,26 @@ static int coda_fill_super(struct super_block *sb, void *data, int silent)  	printk(KERN_INFO "coda_read_super: device index: %i\n", idx);  	vc = &coda_comms[idx]; +	mutex_lock(&vc->vc_mutex); +  	if (!vc->vc_inuse) {  		printk("coda_read_super: No pseudo device\n"); -		unlock_kernel(); -		return -EINVAL; +		error = -EINVAL; +		goto unlock_out;  	} -        if ( vc->vc_sb ) { +	if (vc->vc_sb) {  		printk("coda_read_super: Device already mounted\n"); -		unlock_kernel(); -		return -EBUSY; +		error = -EBUSY; +		goto unlock_out;  	}  	error = bdi_setup_and_register(&vc->bdi, "coda", BDI_CAP_MAP_COPY);  	if (error) -		goto bdi_err; +		goto unlock_out;  	vc->vc_sb = sb; +	mutex_unlock(&vc->vc_mutex);  	sb->s_fs_info = vc;  	sb->s_flags |= MS_NOATIME; @@ -204,28 +207,33 @@ static int coda_fill_super(struct super_block *sb, void *data, int silent)  	printk("coda_read_super: rootinode is %ld dev %s\n",   	       root->i_ino, root->i_sb->s_id);  	sb->s_root = d_alloc_root(root); -	if (!sb->s_root) +	if (!sb->s_root) { +		error = -EINVAL;  		goto error; -	unlock_kernel(); +	}  	return 0; - error: -	bdi_destroy(&vc->bdi); - bdi_err: +error:  	if (root)  		iput(root); -	if (vc) -		vc->vc_sb = NULL; -	unlock_kernel(); -	return -EINVAL; +	mutex_lock(&vc->vc_mutex); +	bdi_destroy(&vc->bdi); +	vc->vc_sb = NULL; +	sb->s_fs_info = NULL; +unlock_out: +	mutex_unlock(&vc->vc_mutex); +	return error;  }  static void coda_put_super(struct super_block *sb)  { -	bdi_destroy(&coda_vcp(sb)->bdi); -	coda_vcp(sb)->vc_sb = NULL; +	struct venus_comm *vcp = coda_vcp(sb); +	mutex_lock(&vcp->vc_mutex); +	bdi_destroy(&vcp->bdi); +	vcp->vc_sb = NULL;  	sb->s_fs_info = NULL; +	mutex_unlock(&vcp->vc_mutex);  	printk("Coda: Bye bye.\n");  } @@ -251,8 +259,6 @@ int coda_setattr(struct dentry *de, struct iattr *iattr)  	struct coda_vattr vattr;  	int error; -	lock_kernel(); -	  	memset(&vattr, 0, sizeof(vattr));   	inode->i_ctime = CURRENT_TIME_SEC; @@ -262,13 +268,10 @@ int coda_setattr(struct dentry *de, struct iattr *iattr)  	/* Venus is responsible for truncating the container-file!!! */  	error = venus_setattr(inode->i_sb, coda_i2f(inode), &vattr); -	if ( !error ) { +	if (!error) {  	        coda_vattr_to_iattr(inode, &vattr);   		coda_cache_clear_inode(inode);  	} - -	unlock_kernel(); -  	return error;  } @@ -282,12 +285,8 @@ static int coda_statfs(struct dentry *dentry, struct kstatfs *buf)  {  	int error; -	lock_kernel(); -  	error = venus_statfs(dentry, buf); -	unlock_kernel(); -  	if (error) {  		/* fake something like AFS does */  		buf->f_blocks = 9000000; |