diff options
Diffstat (limited to 'fs/sysv')
| -rw-r--r-- | fs/sysv/balloc.c | 18 | ||||
| -rw-r--r-- | fs/sysv/dir.c | 37 | ||||
| -rw-r--r-- | fs/sysv/file.c | 13 | ||||
| -rw-r--r-- | fs/sysv/ialloc.c | 16 | ||||
| -rw-r--r-- | fs/sysv/inode.c | 50 | ||||
| -rw-r--r-- | fs/sysv/itree.c | 20 | ||||
| -rw-r--r-- | fs/sysv/namei.c | 31 | ||||
| -rw-r--r-- | fs/sysv/super.c | 41 | ||||
| -rw-r--r-- | fs/sysv/sysv.h | 5 | 
9 files changed, 108 insertions, 123 deletions
diff --git a/fs/sysv/balloc.c b/fs/sysv/balloc.c index 9a6ad96acf2..921c053fc05 100644 --- a/fs/sysv/balloc.c +++ b/fs/sysv/balloc.c @@ -60,12 +60,12 @@ void sysv_free_block(struct super_block * sb, sysv_zone_t nr)  		return;  	} -	lock_super(sb); +	mutex_lock(&sbi->s_lock);  	count = fs16_to_cpu(sbi, *sbi->s_bcache_count);  	if (count > sbi->s_flc_size) {  		printk("sysv_free_block: flc_count > flc_size\n"); -		unlock_super(sb); +		mutex_unlock(&sbi->s_lock);  		return;  	}  	/* If the free list head in super-block is full, it is copied @@ -77,7 +77,7 @@ void sysv_free_block(struct super_block * sb, sysv_zone_t nr)  		bh = sb_getblk(sb, block);  		if (!bh) {  			printk("sysv_free_block: getblk() failed\n"); -			unlock_super(sb); +			mutex_unlock(&sbi->s_lock);  			return;  		}  		memset(bh->b_data, 0, sb->s_blocksize); @@ -93,7 +93,7 @@ void sysv_free_block(struct super_block * sb, sysv_zone_t nr)  	*sbi->s_bcache_count = cpu_to_fs16(sbi, count);  	fs32_add(sbi, sbi->s_free_blocks, 1);  	dirty_sb(sb); -	unlock_super(sb); +	mutex_unlock(&sbi->s_lock);  }  sysv_zone_t sysv_new_block(struct super_block * sb) @@ -104,7 +104,7 @@ sysv_zone_t sysv_new_block(struct super_block * sb)  	struct buffer_head * bh;  	unsigned count; -	lock_super(sb); +	mutex_lock(&sbi->s_lock);  	count = fs16_to_cpu(sbi, *sbi->s_bcache_count);  	if (count == 0) /* Applies only to Coherent FS */ @@ -147,11 +147,11 @@ sysv_zone_t sysv_new_block(struct super_block * sb)  	/* Now the free list head in the superblock is valid again. */  	fs32_add(sbi, sbi->s_free_blocks, -1);  	dirty_sb(sb); -	unlock_super(sb); +	mutex_unlock(&sbi->s_lock);  	return nr;  Enospc: -	unlock_super(sb); +	mutex_unlock(&sbi->s_lock);  	return 0;  } @@ -173,7 +173,7 @@ unsigned long sysv_count_free_blocks(struct super_block * sb)  	if (sbi->s_type == FSTYPE_AFS)  		return 0; -	lock_super(sb); +	mutex_lock(&sbi->s_lock);  	sb_count = fs32_to_cpu(sbi, *sbi->s_free_blocks);  	if (0) @@ -211,7 +211,7 @@ unsigned long sysv_count_free_blocks(struct super_block * sb)  	if (count != sb_count)  		goto Ecount;  done: -	unlock_super(sb); +	mutex_unlock(&sbi->s_lock);  	return count;  Einval: diff --git a/fs/sysv/dir.c b/fs/sysv/dir.c index a77c4215762..d42291d0821 100644 --- a/fs/sysv/dir.c +++ b/fs/sysv/dir.c @@ -18,12 +18,12 @@  #include <linux/swap.h>  #include "sysv.h" -static int sysv_readdir(struct file *, void *, filldir_t); +static int sysv_readdir(struct file *, struct dir_context *);  const struct file_operations sysv_dir_operations = {  	.llseek		= generic_file_llseek,  	.read		= generic_read_dir, -	.readdir	= sysv_readdir, +	.iterate	= sysv_readdir,  	.fsync		= generic_file_fsync,  }; @@ -65,18 +65,21 @@ static struct page * dir_get_page(struct inode *dir, unsigned long n)  	return page;  } -static int sysv_readdir(struct file * filp, void * dirent, filldir_t filldir) +static int sysv_readdir(struct file *file, struct dir_context *ctx)  { -	unsigned long pos = filp->f_pos; -	struct inode *inode = filp->f_path.dentry->d_inode; +	unsigned long pos = ctx->pos; +	struct inode *inode = file_inode(file);  	struct super_block *sb = inode->i_sb; -	unsigned offset = pos & ~PAGE_CACHE_MASK; -	unsigned long n = pos >> PAGE_CACHE_SHIFT;  	unsigned long npages = dir_pages(inode); +	unsigned offset; +	unsigned long n; -	pos = (pos + SYSV_DIRSIZE-1) & ~(SYSV_DIRSIZE-1); +	ctx->pos = pos = (pos + SYSV_DIRSIZE-1) & ~(SYSV_DIRSIZE-1);  	if (pos >= inode->i_size) -		goto done; +		return 0; + +	offset = pos & ~PAGE_CACHE_MASK; +	n = pos >> PAGE_CACHE_SHIFT;  	for ( ; n < npages; n++, offset = 0) {  		char *kaddr, *limit; @@ -88,29 +91,21 @@ static int sysv_readdir(struct file * filp, void * dirent, filldir_t filldir)  		kaddr = (char *)page_address(page);  		de = (struct sysv_dir_entry *)(kaddr+offset);  		limit = kaddr + PAGE_CACHE_SIZE - SYSV_DIRSIZE; -		for ( ;(char*)de <= limit; de++) { +		for ( ;(char*)de <= limit; de++, ctx->pos += sizeof(*de)) {  			char *name = de->name; -			int over;  			if (!de->inode)  				continue; -			offset = (char *)de - kaddr; - -			over = filldir(dirent, name, strnlen(name,SYSV_NAMELEN), -					((loff_t)n<<PAGE_CACHE_SHIFT) | offset, +			if (!dir_emit(ctx, name, strnlen(name,SYSV_NAMELEN),  					fs16_to_cpu(SYSV_SB(sb), de->inode), -					DT_UNKNOWN); -			if (over) { +					DT_UNKNOWN)) {  				dir_put_page(page); -				goto done; +				return 0;  			}  		}  		dir_put_page(page);  	} - -done: -	filp->f_pos = ((loff_t)n << PAGE_CACHE_SHIFT) | offset;  	return 0;  } diff --git a/fs/sysv/file.c b/fs/sysv/file.c index 0a65939508e..b00811c75b2 100644 --- a/fs/sysv/file.c +++ b/fs/sysv/file.c @@ -21,10 +21,10 @@   */  const struct file_operations sysv_file_operations = {  	.llseek		= generic_file_llseek, -	.read		= do_sync_read, -	.aio_read	= generic_file_aio_read, -	.write		= do_sync_write, -	.aio_write	= generic_file_aio_write, +	.read		= new_sync_read, +	.read_iter	= generic_file_read_iter, +	.write		= new_sync_write, +	.write_iter	= generic_file_write_iter,  	.mmap		= generic_file_mmap,  	.fsync		= generic_file_fsync,  	.splice_read	= generic_file_splice_read, @@ -41,9 +41,11 @@ static int sysv_setattr(struct dentry *dentry, struct iattr *attr)  	if ((attr->ia_valid & ATTR_SIZE) &&  	    attr->ia_size != i_size_read(inode)) { -		error = vmtruncate(inode, attr->ia_size); +		error = inode_newsize_ok(inode, attr->ia_size);  		if (error)  			return error; +		truncate_setsize(inode, attr->ia_size); +		sysv_truncate(inode);  	}  	setattr_copy(inode, attr); @@ -52,7 +54,6 @@ static int sysv_setattr(struct dentry *dentry, struct iattr *attr)  }  const struct inode_operations sysv_file_inode_operations = { -	.truncate	= sysv_truncate,  	.setattr	= sysv_setattr,  	.getattr	= sysv_getattr,  }; diff --git a/fs/sysv/ialloc.c b/fs/sysv/ialloc.c index 0c96c98bd1d..f9db4eb31db 100644 --- a/fs/sysv/ialloc.c +++ b/fs/sysv/ialloc.c @@ -118,7 +118,7 @@ void sysv_free_inode(struct inode * inode)  		       "%s\n", inode->i_sb->s_id);  		return;  	} -	lock_super(sb); +	mutex_lock(&sbi->s_lock);  	count = fs16_to_cpu(sbi, *sbi->s_sb_fic_count);  	if (count < sbi->s_fic_size) {  		*sv_sb_fic_inode(sb,count++) = cpu_to_fs16(sbi, ino); @@ -128,11 +128,11 @@ void sysv_free_inode(struct inode * inode)  	dirty_sb(sb);  	memset(raw_inode, 0, sizeof(struct sysv_inode));  	mark_buffer_dirty(bh); -	unlock_super(sb); +	mutex_unlock(&sbi->s_lock);  	brelse(bh);  } -struct inode * sysv_new_inode(const struct inode * dir, mode_t mode) +struct inode * sysv_new_inode(const struct inode * dir, umode_t mode)  {  	struct super_block *sb = dir->i_sb;  	struct sysv_sb_info *sbi = SYSV_SB(sb); @@ -147,13 +147,13 @@ struct inode * sysv_new_inode(const struct inode * dir, mode_t mode)  	if (!inode)  		return ERR_PTR(-ENOMEM); -	lock_super(sb); +	mutex_lock(&sbi->s_lock);  	count = fs16_to_cpu(sbi, *sbi->s_sb_fic_count);  	if (count == 0 || (*sv_sb_fic_inode(sb,count-1) == 0)) {  		count = refill_free_cache(sb);  		if (count == 0) {  			iput(inode); -			unlock_super(sb); +			mutex_unlock(&sbi->s_lock);  			return ERR_PTR(-ENOSPC);  		}  	} @@ -174,7 +174,7 @@ struct inode * sysv_new_inode(const struct inode * dir, mode_t mode)  	sysv_write_inode(inode, &wbc);	/* ensure inode not allocated again */  	mark_inode_dirty(inode);	/* cleared by sysv_write_inode() */  	/* That's it. */ -	unlock_super(sb); +	mutex_unlock(&sbi->s_lock);  	return inode;  } @@ -185,7 +185,7 @@ unsigned long sysv_count_free_inodes(struct super_block * sb)  	struct sysv_inode * raw_inode;  	int ino, count, sb_count; -	lock_super(sb); +	mutex_lock(&sbi->s_lock);  	sb_count = fs16_to_cpu(sbi, *sbi->s_sb_total_free_inodes); @@ -213,7 +213,7 @@ unsigned long sysv_count_free_inodes(struct super_block * sb)  	if (count != sb_count)  		goto Einval;  out: -	unlock_super(sb); +	mutex_unlock(&sbi->s_lock);  	return count;  Einval: diff --git a/fs/sysv/inode.c b/fs/sysv/inode.c index de44d067b9e..88956309cc8 100644 --- a/fs/sysv/inode.c +++ b/fs/sysv/inode.c @@ -36,14 +36,13 @@ static int sysv_sync_fs(struct super_block *sb, int wait)  	struct sysv_sb_info *sbi = SYSV_SB(sb);  	unsigned long time = get_seconds(), old_time; -	lock_super(sb); +	mutex_lock(&sbi->s_lock);  	/*  	 * If we are going to write out the super block,  	 * then attach current time stamp.  	 * But if the filesystem was marked clean, keep it clean.  	 */ -	sb->s_dirt = 0;  	old_time = fs32_to_cpu(sbi, *sbi->s_sb_time);  	if (sbi->s_type == FSTYPE_SYSV4) {  		if (*sbi->s_sb_state == cpu_to_fs32(sbi, 0x7c269d38 - old_time)) @@ -52,28 +51,18 @@ static int sysv_sync_fs(struct super_block *sb, int wait)  		mark_buffer_dirty(sbi->s_bh2);  	} -	unlock_super(sb); +	mutex_unlock(&sbi->s_lock);  	return 0;  } -static void sysv_write_super(struct super_block *sb) -{ -	if (!(sb->s_flags & MS_RDONLY)) -		sysv_sync_fs(sb, 1); -	else -		sb->s_dirt = 0; -} -  static int sysv_remount(struct super_block *sb, int *flags, char *data)  {  	struct sysv_sb_info *sbi = SYSV_SB(sb); -	lock_super(sb); + +	sync_filesystem(sb);  	if (sbi->s_forced_ro)  		*flags |= MS_RDONLY; -	if (*flags & MS_RDONLY) -		sysv_write_super(sb); -	unlock_super(sb);  	return 0;  } @@ -81,9 +70,6 @@ static void sysv_put_super(struct super_block *sb)  {  	struct sysv_sb_info *sbi = SYSV_SB(sb); -	if (sb->s_dirt) -		sysv_write_super(sb); -  	if (!(sb->s_flags & MS_RDONLY)) {  		/* XXX ext2 also updates the state here */  		mark_buffer_dirty(sbi->s_bh1); @@ -217,9 +203,9 @@ struct inode *sysv_iget(struct super_block *sb, unsigned int ino)  	}  	/* SystemV FS: kludge permissions if ino==SYSV_ROOT_INO ?? */  	inode->i_mode = fs16_to_cpu(sbi, raw_inode->i_mode); -	inode->i_uid = (uid_t)fs16_to_cpu(sbi, raw_inode->i_uid); -	inode->i_gid = (gid_t)fs16_to_cpu(sbi, raw_inode->i_gid); -	inode->i_nlink = fs16_to_cpu(sbi, raw_inode->i_nlink); +	i_uid_write(inode, (uid_t)fs16_to_cpu(sbi, raw_inode->i_uid)); +	i_gid_write(inode, (gid_t)fs16_to_cpu(sbi, raw_inode->i_gid)); +	set_nlink(inode, fs16_to_cpu(sbi, raw_inode->i_nlink));  	inode->i_size = fs32_to_cpu(sbi, raw_inode->i_size);  	inode->i_atime.tv_sec = fs32_to_cpu(sbi, raw_inode->i_atime);  	inode->i_mtime.tv_sec = fs32_to_cpu(sbi, raw_inode->i_mtime); @@ -271,8 +257,8 @@ static int __sysv_write_inode(struct inode *inode, int wait)  	}  	raw_inode->i_mode = cpu_to_fs16(sbi, inode->i_mode); -	raw_inode->i_uid = cpu_to_fs16(sbi, fs_high2lowuid(inode->i_uid)); -	raw_inode->i_gid = cpu_to_fs16(sbi, fs_high2lowgid(inode->i_gid)); +	raw_inode->i_uid = cpu_to_fs16(sbi, fs_high2lowuid(i_uid_read(inode))); +	raw_inode->i_gid = cpu_to_fs16(sbi, fs_high2lowgid(i_gid_read(inode)));  	raw_inode->i_nlink = cpu_to_fs16(sbi, inode->i_nlink);  	raw_inode->i_size = cpu_to_fs32(sbi, inode->i_size);  	raw_inode->i_atime = cpu_to_fs32(sbi, inode->i_atime.tv_sec); @@ -310,13 +296,13 @@ int sysv_sync_inode(struct inode *inode)  static void sysv_evict_inode(struct inode *inode)  { -	truncate_inode_pages(&inode->i_data, 0); +	truncate_inode_pages_final(&inode->i_data);  	if (!inode->i_nlink) {  		inode->i_size = 0;  		sysv_truncate(inode);  	}  	invalidate_inode_buffers(inode); -	end_writeback(inode); +	clear_inode(inode);  	if (!inode->i_nlink)  		sysv_free_inode(inode);  } @@ -333,11 +319,17 @@ static struct inode *sysv_alloc_inode(struct super_block *sb)  	return &si->vfs_inode;  } -static void sysv_destroy_inode(struct inode *inode) +static void sysv_i_callback(struct rcu_head *head)  { +	struct inode *inode = container_of(head, struct inode, i_rcu);  	kmem_cache_free(sysv_inode_cachep, SYSV_I(inode));  } +static void sysv_destroy_inode(struct inode *inode) +{ +	call_rcu(&inode->i_rcu, sysv_i_callback); +} +  static void init_once(void *p)  {  	struct sysv_inode_info *si = (struct sysv_inode_info *)p; @@ -351,7 +343,6 @@ const struct super_operations sysv_sops = {  	.write_inode	= sysv_write_inode,  	.evict_inode	= sysv_evict_inode,  	.put_super	= sysv_put_super, -	.write_super	= sysv_write_super,  	.sync_fs	= sysv_sync_fs,  	.remount_fs	= sysv_remount,  	.statfs		= sysv_statfs, @@ -370,5 +361,10 @@ int __init sysv_init_icache(void)  void sysv_destroy_icache(void)  { +	/* +	 * Make sure all delayed rcu free inodes are flushed before we +	 * destroy cache. +	 */ +	rcu_barrier();  	kmem_cache_destroy(sysv_inode_cachep);  } diff --git a/fs/sysv/itree.c b/fs/sysv/itree.c index 9ca66276315..66bc316927e 100644 --- a/fs/sysv/itree.c +++ b/fs/sysv/itree.c @@ -442,7 +442,7 @@ static unsigned sysv_nblocks(struct super_block *s, loff_t size)  int sysv_getattr(struct vfsmount *mnt, struct dentry *dentry, struct kstat *stat)  { -	struct super_block *s = mnt->mnt_sb; +	struct super_block *s = dentry->d_sb;  	generic_fillattr(dentry->d_inode, stat);  	stat->blocks = (s->s_blocksize / 512) * sysv_nblocks(s, stat->size);  	stat->blksize = s->s_blocksize; @@ -464,6 +464,16 @@ int sysv_prepare_chunk(struct page *page, loff_t pos, unsigned len)  	return __block_write_begin(page, pos, len, get_block);  } +static void sysv_write_failed(struct address_space *mapping, loff_t to) +{ +	struct inode *inode = mapping->host; + +	if (to > inode->i_size) { +		truncate_pagecache(inode, inode->i_size); +		sysv_truncate(inode); +	} +} +  static int sysv_write_begin(struct file *file, struct address_space *mapping,  			loff_t pos, unsigned len, unsigned flags,  			struct page **pagep, void **fsdata) @@ -471,11 +481,8 @@ static int sysv_write_begin(struct file *file, struct address_space *mapping,  	int ret;  	ret = block_write_begin(mapping, pos, len, flags, pagep, get_block); -	if (unlikely(ret)) { -		loff_t isize = mapping->host->i_size; -		if (pos + len > isize) -			vmtruncate(mapping->host, isize); -	} +	if (unlikely(ret)) +		sysv_write_failed(mapping, pos + len);  	return ret;  } @@ -488,7 +495,6 @@ static sector_t sysv_bmap(struct address_space *mapping, sector_t block)  const struct address_space_operations sysv_aops = {  	.readpage = sysv_readpage,  	.writepage = sysv_writepage, -	.sync_page = block_sync_page,  	.write_begin = sysv_write_begin,  	.write_end = generic_write_end,  	.bmap = sysv_bmap diff --git a/fs/sysv/namei.c b/fs/sysv/namei.c index 11e7f7d11cd..731b2bbcaab 100644 --- a/fs/sysv/namei.c +++ b/fs/sysv/namei.c @@ -27,7 +27,7 @@ static int add_nondir(struct dentry *dentry, struct inode *inode)  	return err;  } -static int sysv_hash(struct dentry *dentry, struct qstr *qstr) +static int sysv_hash(const struct dentry *dentry, struct qstr *qstr)  {  	/* Truncate the name in place, avoids having to define a compare  	   function. */ @@ -42,12 +42,11 @@ const struct dentry_operations sysv_dentry_operations = {  	.d_hash		= sysv_hash,  }; -static struct dentry *sysv_lookup(struct inode * dir, struct dentry * dentry, struct nameidata *nd) +static struct dentry *sysv_lookup(struct inode * dir, struct dentry * dentry, unsigned int flags)  {  	struct inode * inode = NULL;  	ino_t ino; -	dentry->d_op = dir->i_sb->s_root->d_op;  	if (dentry->d_name.len > SYSV_NAMELEN)  		return ERR_PTR(-ENAMETOOLONG);  	ino = sysv_inode_by_name(dentry); @@ -61,7 +60,7 @@ static struct dentry *sysv_lookup(struct inode * dir, struct dentry * dentry, st  	return NULL;  } -static int sysv_mknod(struct inode * dir, struct dentry * dentry, int mode, dev_t rdev) +static int sysv_mknod(struct inode * dir, struct dentry * dentry, umode_t mode, dev_t rdev)  {  	struct inode * inode;  	int err; @@ -80,7 +79,7 @@ static int sysv_mknod(struct inode * dir, struct dentry * dentry, int mode, dev_  	return err;  } -static int sysv_create(struct inode * dir, struct dentry * dentry, int mode, struct nameidata *nd) +static int sysv_create(struct inode * dir, struct dentry * dentry, umode_t mode, bool excl)  {  	return sysv_mknod(dir, dentry, mode, 0);  } @@ -121,9 +120,6 @@ static int sysv_link(struct dentry * old_dentry, struct inode * dir,  {  	struct inode *inode = old_dentry->d_inode; -	if (inode->i_nlink >= SYSV_SB(inode->i_sb)->s_link_max) -		return -EMLINK; -  	inode->i_ctime = CURRENT_TIME_SEC;  	inode_inc_link_count(inode);  	ihold(inode); @@ -131,13 +127,11 @@ static int sysv_link(struct dentry * old_dentry, struct inode * dir,  	return add_nondir(dentry, inode);  } -static int sysv_mkdir(struct inode * dir, struct dentry *dentry, int mode) +static int sysv_mkdir(struct inode * dir, struct dentry *dentry, umode_t mode)  {  	struct inode * inode; -	int err = -EMLINK; +	int err; -	if (dir->i_nlink >= SYSV_SB(dir->i_sb)->s_link_max)  -		goto out;  	inode_inc_link_count(dir);  	inode = sysv_new_inode(dir, S_IFDIR|mode); @@ -245,30 +239,21 @@ static int sysv_rename(struct inode * old_dir, struct dentry * old_dentry,  		new_de = sysv_find_entry(new_dentry, &new_page);  		if (!new_de)  			goto out_dir; -		inode_inc_link_count(old_inode);  		sysv_set_link(new_de, new_page, old_inode);  		new_inode->i_ctime = CURRENT_TIME_SEC;  		if (dir_de)  			drop_nlink(new_inode);  		inode_dec_link_count(new_inode);  	} else { -		if (dir_de) { -			err = -EMLINK; -			if (new_dir->i_nlink >= SYSV_SB(new_dir->i_sb)->s_link_max) -				goto out_dir; -		} -		inode_inc_link_count(old_inode);  		err = sysv_add_link(new_dentry, old_inode); -		if (err) { -			inode_dec_link_count(old_inode); +		if (err)  			goto out_dir; -		}  		if (dir_de)  			inode_inc_link_count(new_dir);  	}  	sysv_delete_entry(old_de, old_page); -	inode_dec_link_count(old_inode); +	mark_inode_dirty(old_inode);  	if (dir_de) {  		sysv_set_link(dir_de, dir_page, new_dir); diff --git a/fs/sysv/super.c b/fs/sysv/super.c index 3d9c62be0c1..eda10959714 100644 --- a/fs/sysv/super.c +++ b/fs/sysv/super.c @@ -44,7 +44,7 @@ enum {  	JAN_1_1980 = (10*365 + 2) * 24 * 60 * 60  }; -static void detected_xenix(struct sysv_sb_info *sbi) +static void detected_xenix(struct sysv_sb_info *sbi, unsigned *max_links)  {  	struct buffer_head *bh1 = sbi->s_bh1;  	struct buffer_head *bh2 = sbi->s_bh2; @@ -59,7 +59,7 @@ static void detected_xenix(struct sysv_sb_info *sbi)  		sbd2 = (struct xenix_super_block *) (bh2->b_data - 512);  	} -	sbi->s_link_max = XENIX_LINK_MAX; +	*max_links = XENIX_LINK_MAX;  	sbi->s_fic_size = XENIX_NICINOD;  	sbi->s_flc_size = XENIX_NICFREE;  	sbi->s_sbd1 = (char *)sbd1; @@ -75,7 +75,7 @@ static void detected_xenix(struct sysv_sb_info *sbi)  	sbi->s_nzones = fs32_to_cpu(sbi, sbd1->s_fsize);  } -static void detected_sysv4(struct sysv_sb_info *sbi) +static void detected_sysv4(struct sysv_sb_info *sbi, unsigned *max_links)  {  	struct sysv4_super_block * sbd;  	struct buffer_head *bh1 = sbi->s_bh1; @@ -86,7 +86,7 @@ static void detected_sysv4(struct sysv_sb_info *sbi)  	else  		sbd = (struct sysv4_super_block *) bh2->b_data; -	sbi->s_link_max = SYSV_LINK_MAX; +	*max_links = SYSV_LINK_MAX;  	sbi->s_fic_size = SYSV_NICINOD;  	sbi->s_flc_size = SYSV_NICFREE;  	sbi->s_sbd1 = (char *)sbd; @@ -103,7 +103,7 @@ static void detected_sysv4(struct sysv_sb_info *sbi)  	sbi->s_nzones = fs32_to_cpu(sbi, sbd->s_fsize);  } -static void detected_sysv2(struct sysv_sb_info *sbi) +static void detected_sysv2(struct sysv_sb_info *sbi, unsigned *max_links)  {  	struct sysv2_super_block *sbd;  	struct buffer_head *bh1 = sbi->s_bh1; @@ -114,7 +114,7 @@ static void detected_sysv2(struct sysv_sb_info *sbi)  	else  		sbd = (struct sysv2_super_block *) bh2->b_data; -	sbi->s_link_max = SYSV_LINK_MAX; +	*max_links = SYSV_LINK_MAX;  	sbi->s_fic_size = SYSV_NICINOD;  	sbi->s_flc_size = SYSV_NICFREE;  	sbi->s_sbd1 = (char *)sbd; @@ -131,14 +131,14 @@ static void detected_sysv2(struct sysv_sb_info *sbi)  	sbi->s_nzones = fs32_to_cpu(sbi, sbd->s_fsize);  } -static void detected_coherent(struct sysv_sb_info *sbi) +static void detected_coherent(struct sysv_sb_info *sbi, unsigned *max_links)  {  	struct coh_super_block * sbd;  	struct buffer_head *bh1 = sbi->s_bh1;  	sbd = (struct coh_super_block *) bh1->b_data; -	sbi->s_link_max = COH_LINK_MAX; +	*max_links = COH_LINK_MAX;  	sbi->s_fic_size = COH_NICINOD;  	sbi->s_flc_size = COH_NICFREE;  	sbi->s_sbd1 = (char *)sbd; @@ -154,12 +154,12 @@ static void detected_coherent(struct sysv_sb_info *sbi)  	sbi->s_nzones = fs32_to_cpu(sbi, sbd->s_fsize);  } -static void detected_v7(struct sysv_sb_info *sbi) +static void detected_v7(struct sysv_sb_info *sbi, unsigned *max_links)  {  	struct buffer_head *bh2 = sbi->s_bh2;  	struct v7_super_block *sbd = (struct v7_super_block *)bh2->b_data; -	sbi->s_link_max = V7_LINK_MAX; +	*max_links = V7_LINK_MAX;  	sbi->s_fic_size = V7_NICINOD;  	sbi->s_flc_size = V7_NICFREE;  	sbi->s_sbd1 = (char *)sbd; @@ -290,7 +290,7 @@ static char *flavour_names[] = {  	[FSTYPE_AFS]	= "AFS",  }; -static void (*flavour_setup[])(struct sysv_sb_info *) = { +static void (*flavour_setup[])(struct sysv_sb_info *, unsigned *) = {  	[FSTYPE_XENIX]	= detected_xenix,  	[FSTYPE_SYSV4]	= detected_sysv4,  	[FSTYPE_SYSV2]	= detected_sysv2, @@ -310,7 +310,7 @@ static int complete_read_super(struct super_block *sb, int silent, int size)  	sbi->s_firstinodezone = 2; -	flavour_setup[sbi->s_type](sbi); +	flavour_setup[sbi->s_type](sbi, &sb->s_max_links);  	sbi->s_truncate = 1;  	sbi->s_ndatazones = sbi->s_nzones - sbi->s_firstdatazone; @@ -332,21 +332,20 @@ static int complete_read_super(struct super_block *sb, int silent, int size)  	sb->s_magic = SYSV_MAGIC_BASE + sbi->s_type;  	/* set up enough so that it can read an inode */  	sb->s_op = &sysv_sops; +	if (sbi->s_forced_ro) +		sb->s_flags |= MS_RDONLY; +	if (sbi->s_truncate) +		sb->s_d_op = &sysv_dentry_operations;  	root_inode = sysv_iget(sb, SYSV_ROOT_INO);  	if (IS_ERR(root_inode)) {  		printk("SysV FS: get root inode failed\n");  		return 0;  	} -	sb->s_root = d_alloc_root(root_inode); +	sb->s_root = d_make_root(root_inode);  	if (!sb->s_root) { -		iput(root_inode);  		printk("SysV FS: get root dentry failed\n");  		return 0;  	} -	if (sbi->s_forced_ro) -		sb->s_flags |= MS_RDONLY; -	if (sbi->s_truncate) -		sb->s_root->d_op = &sysv_dentry_operations;  	return 1;  } @@ -369,6 +368,7 @@ static int sysv_fill_super(struct super_block *sb, void *data, int silent)  	sbi->s_sb = sb;  	sbi->s_block_base = 0; +	mutex_init(&sbi->s_lock);  	sb->s_fs_info = sbi;  	sb_set_blocksize(sb, BLOCK_SIZE); @@ -487,6 +487,7 @@ static int v7_fill_super(struct super_block *sb, void *data, int silent)  	sbi->s_sb = sb;  	sbi->s_block_base = 0;  	sbi->s_type = FSTYPE_V7; +	mutex_init(&sbi->s_lock);  	sb->s_fs_info = sbi;  	sb_set_blocksize(sb, 512); @@ -545,6 +546,7 @@ static struct file_system_type sysv_fs_type = {  	.kill_sb	= kill_block_super,  	.fs_flags	= FS_REQUIRES_DEV,  }; +MODULE_ALIAS_FS("sysv");  static struct file_system_type v7_fs_type = {  	.owner		= THIS_MODULE, @@ -553,6 +555,8 @@ static struct file_system_type v7_fs_type = {  	.kill_sb	= kill_block_super,  	.fs_flags	= FS_REQUIRES_DEV,  }; +MODULE_ALIAS_FS("v7"); +MODULE_ALIAS("v7");  static int __init init_sysv_fs(void)  { @@ -586,5 +590,4 @@ static void __exit exit_sysv_fs(void)  module_init(init_sysv_fs)  module_exit(exit_sysv_fs) -MODULE_ALIAS("v7");  MODULE_LICENSE("GPL"); diff --git a/fs/sysv/sysv.h b/fs/sysv/sysv.h index bb55cdb394b..69d488986cc 100644 --- a/fs/sysv/sysv.h +++ b/fs/sysv/sysv.h @@ -24,7 +24,6 @@ struct sysv_sb_info {  	char	       s_bytesex;	/* bytesex (le/be/pdp) */  	char	       s_truncate;	/* if 1: names > SYSV_NAMELEN chars are truncated */  					/* if 0: they are disallowed (ENAMETOOLONG) */ -	nlink_t        s_link_max;	/* max number of hard links to a file */  	unsigned int   s_inodes_per_block;	/* number of inodes per block */  	unsigned int   s_inodes_per_block_1;	/* inodes_per_block - 1 */  	unsigned int   s_inodes_per_block_bits;	/* log2(inodes_per_block) */ @@ -59,6 +58,7 @@ struct sysv_sb_info {  	u32            s_nzones;	/* same as s_sbd->s_fsize */  	u16	       s_namelen;       /* max length of dir entry */  	int	       s_forced_ro; +	struct mutex s_lock;  };  /* @@ -118,14 +118,13 @@ static inline void dirty_sb(struct super_block *sb)  	mark_buffer_dirty(sbi->s_bh1);  	if (sbi->s_bh1 != sbi->s_bh2)  		mark_buffer_dirty(sbi->s_bh2); -	sb->s_dirt = 1;  }  /* ialloc.c */  extern struct sysv_inode *sysv_raw_inode(struct super_block *, unsigned,  			struct buffer_head **); -extern struct inode * sysv_new_inode(const struct inode *, mode_t); +extern struct inode * sysv_new_inode(const struct inode *, umode_t);  extern void sysv_free_inode(struct inode *);  extern unsigned long sysv_count_free_inodes(struct super_block *);  | 
