diff options
Diffstat (limited to 'fs/ext4/super.c')
| -rw-r--r-- | fs/ext4/super.c | 98 | 
1 files changed, 65 insertions, 33 deletions
| diff --git a/fs/ext4/super.c b/fs/ext4/super.c index 1145109968ef..53ff6c2a26ed 100644 --- a/fs/ext4/super.c +++ b/fs/ext4/super.c @@ -914,6 +914,18 @@ static inline void ext4_quota_off_umount(struct super_block *sb)  	for (type = 0; type < EXT4_MAXQUOTAS; type++)  		ext4_quota_off(sb, type);  } + +/* + * This is a helper function which is used in the mount/remount + * codepaths (which holds s_umount) to fetch the quota file name. + */ +static inline char *get_qf_name(struct super_block *sb, +				struct ext4_sb_info *sbi, +				int type) +{ +	return rcu_dereference_protected(sbi->s_qf_names[type], +					 lockdep_is_held(&sb->s_umount)); +}  #else  static inline void ext4_quota_off_umount(struct super_block *sb)  { @@ -965,7 +977,7 @@ static void ext4_put_super(struct super_block *sb)  	percpu_free_rwsem(&sbi->s_journal_flag_rwsem);  #ifdef CONFIG_QUOTA  	for (i = 0; i < EXT4_MAXQUOTAS; i++) -		kfree(sbi->s_qf_names[i]); +		kfree(get_qf_name(sb, sbi, i));  #endif  	/* Debugging code just in case the in-memory inode orphan list @@ -1040,6 +1052,7 @@ static struct inode *ext4_alloc_inode(struct super_block *sb)  	ei->i_da_metadata_calc_len = 0;  	ei->i_da_metadata_calc_last_lblock = 0;  	spin_lock_init(&(ei->i_block_reservation_lock)); +	ext4_init_pending_tree(&ei->i_pending_tree);  #ifdef CONFIG_QUOTA  	ei->i_reserved_quota = 0;  	memset(&ei->i_dquot, 0, sizeof(ei->i_dquot)); @@ -1530,11 +1543,10 @@ static const char deprecated_msg[] =  static int set_qf_name(struct super_block *sb, int qtype, substring_t *args)  {  	struct ext4_sb_info *sbi = EXT4_SB(sb); -	char *qname; +	char *qname, *old_qname = get_qf_name(sb, sbi, qtype);  	int ret = -1; -	if (sb_any_quota_loaded(sb) && -		!sbi->s_qf_names[qtype]) { +	if (sb_any_quota_loaded(sb) && !old_qname) {  		ext4_msg(sb, KERN_ERR,  			"Cannot change journaled "  			"quota options when quota turned on"); @@ -1551,8 +1563,8 @@ static int set_qf_name(struct super_block *sb, int qtype, substring_t *args)  			"Not enough memory for storing quotafile name");  		return -1;  	} -	if (sbi->s_qf_names[qtype]) { -		if (strcmp(sbi->s_qf_names[qtype], qname) == 0) +	if (old_qname) { +		if (strcmp(old_qname, qname) == 0)  			ret = 1;  		else  			ext4_msg(sb, KERN_ERR, @@ -1565,7 +1577,7 @@ static int set_qf_name(struct super_block *sb, int qtype, substring_t *args)  			"quotafile must be on filesystem root");  		goto errout;  	} -	sbi->s_qf_names[qtype] = qname; +	rcu_assign_pointer(sbi->s_qf_names[qtype], qname);  	set_opt(sb, QUOTA);  	return 1;  errout: @@ -1577,15 +1589,16 @@ static int clear_qf_name(struct super_block *sb, int qtype)  {  	struct ext4_sb_info *sbi = EXT4_SB(sb); +	char *old_qname = get_qf_name(sb, sbi, qtype); -	if (sb_any_quota_loaded(sb) && -		sbi->s_qf_names[qtype]) { +	if (sb_any_quota_loaded(sb) && old_qname) {  		ext4_msg(sb, KERN_ERR, "Cannot change journaled quota options"  			" when quota turned on");  		return -1;  	} -	kfree(sbi->s_qf_names[qtype]); -	sbi->s_qf_names[qtype] = NULL; +	rcu_assign_pointer(sbi->s_qf_names[qtype], NULL); +	synchronize_rcu(); +	kfree(old_qname);  	return 1;  }  #endif @@ -1960,7 +1973,7 @@ static int parse_options(char *options, struct super_block *sb,  			 int is_remount)  {  	struct ext4_sb_info *sbi = EXT4_SB(sb); -	char *p; +	char *p, __maybe_unused *usr_qf_name, __maybe_unused *grp_qf_name;  	substring_t args[MAX_OPT_ARGS];  	int token; @@ -1991,11 +2004,13 @@ static int parse_options(char *options, struct super_block *sb,  			 "Cannot enable project quota enforcement.");  		return 0;  	} -	if (sbi->s_qf_names[USRQUOTA] || sbi->s_qf_names[GRPQUOTA]) { -		if (test_opt(sb, USRQUOTA) && sbi->s_qf_names[USRQUOTA]) +	usr_qf_name = get_qf_name(sb, sbi, USRQUOTA); +	grp_qf_name = get_qf_name(sb, sbi, GRPQUOTA); +	if (usr_qf_name || grp_qf_name) { +		if (test_opt(sb, USRQUOTA) && usr_qf_name)  			clear_opt(sb, USRQUOTA); -		if (test_opt(sb, GRPQUOTA) && sbi->s_qf_names[GRPQUOTA]) +		if (test_opt(sb, GRPQUOTA) && grp_qf_name)  			clear_opt(sb, GRPQUOTA);  		if (test_opt(sb, GRPQUOTA) || test_opt(sb, USRQUOTA)) { @@ -2029,6 +2044,7 @@ static inline void ext4_show_quota_options(struct seq_file *seq,  {  #if defined(CONFIG_QUOTA)  	struct ext4_sb_info *sbi = EXT4_SB(sb); +	char *usr_qf_name, *grp_qf_name;  	if (sbi->s_jquota_fmt) {  		char *fmtname = ""; @@ -2047,11 +2063,14 @@ static inline void ext4_show_quota_options(struct seq_file *seq,  		seq_printf(seq, ",jqfmt=%s", fmtname);  	} -	if (sbi->s_qf_names[USRQUOTA]) -		seq_show_option(seq, "usrjquota", sbi->s_qf_names[USRQUOTA]); - -	if (sbi->s_qf_names[GRPQUOTA]) -		seq_show_option(seq, "grpjquota", sbi->s_qf_names[GRPQUOTA]); +	rcu_read_lock(); +	usr_qf_name = rcu_dereference(sbi->s_qf_names[USRQUOTA]); +	grp_qf_name = rcu_dereference(sbi->s_qf_names[GRPQUOTA]); +	if (usr_qf_name) +		seq_show_option(seq, "usrjquota", usr_qf_name); +	if (grp_qf_name) +		seq_show_option(seq, "grpjquota", grp_qf_name); +	rcu_read_unlock();  #endif  } @@ -4056,6 +4075,14 @@ static int ext4_fill_super(struct super_block *sb, void *data, int silent)  	sbi->s_groups_count = blocks_count;  	sbi->s_blockfile_groups = min_t(ext4_group_t, sbi->s_groups_count,  			(EXT4_MAX_BLOCK_FILE_PHYS / EXT4_BLOCKS_PER_GROUP(sb))); +	if (((u64)sbi->s_groups_count * sbi->s_inodes_per_group) != +	    le32_to_cpu(es->s_inodes_count)) { +		ext4_msg(sb, KERN_ERR, "inodes count not valid: %u vs %llu", +			 le32_to_cpu(es->s_inodes_count), +			 ((u64)sbi->s_groups_count * sbi->s_inodes_per_group)); +		ret = -EINVAL; +		goto failed_mount; +	}  	db_count = (sbi->s_groups_count + EXT4_DESC_PER_BLOCK(sb) - 1) /  		   EXT4_DESC_PER_BLOCK(sb);  	if (ext4_has_feature_meta_bg(sb)) { @@ -4075,14 +4102,6 @@ static int ext4_fill_super(struct super_block *sb, void *data, int silent)  		ret = -ENOMEM;  		goto failed_mount;  	} -	if (((u64)sbi->s_groups_count * sbi->s_inodes_per_group) != -	    le32_to_cpu(es->s_inodes_count)) { -		ext4_msg(sb, KERN_ERR, "inodes count not valid: %u vs %llu", -			 le32_to_cpu(es->s_inodes_count), -			 ((u64)sbi->s_groups_count * sbi->s_inodes_per_group)); -		ret = -EINVAL; -		goto failed_mount; -	}  	bgl_lock_init(sbi->s_blockgroup_lock); @@ -4491,6 +4510,7 @@ failed_mount6:  	percpu_counter_destroy(&sbi->s_freeinodes_counter);  	percpu_counter_destroy(&sbi->s_dirs_counter);  	percpu_counter_destroy(&sbi->s_dirtyclusters_counter); +	percpu_free_rwsem(&sbi->s_journal_flag_rwsem);  failed_mount5:  	ext4_ext_release(sb);  	ext4_release_system_zone(sb); @@ -5103,6 +5123,7 @@ static int ext4_remount(struct super_block *sb, int *flags, char *data)  	int err = 0;  #ifdef CONFIG_QUOTA  	int i, j; +	char *to_free[EXT4_MAXQUOTAS];  #endif  	char *orig_data = kstrdup(data, GFP_KERNEL); @@ -5122,8 +5143,9 @@ static int ext4_remount(struct super_block *sb, int *flags, char *data)  	old_opts.s_jquota_fmt = sbi->s_jquota_fmt;  	for (i = 0; i < EXT4_MAXQUOTAS; i++)  		if (sbi->s_qf_names[i]) { -			old_opts.s_qf_names[i] = kstrdup(sbi->s_qf_names[i], -							 GFP_KERNEL); +			char *qf_name = get_qf_name(sb, sbi, i); + +			old_opts.s_qf_names[i] = kstrdup(qf_name, GFP_KERNEL);  			if (!old_opts.s_qf_names[i]) {  				for (j = 0; j < i; j++)  					kfree(old_opts.s_qf_names[j]); @@ -5352,9 +5374,12 @@ restore_opts:  #ifdef CONFIG_QUOTA  	sbi->s_jquota_fmt = old_opts.s_jquota_fmt;  	for (i = 0; i < EXT4_MAXQUOTAS; i++) { -		kfree(sbi->s_qf_names[i]); -		sbi->s_qf_names[i] = old_opts.s_qf_names[i]; +		to_free[i] = get_qf_name(sb, sbi, i); +		rcu_assign_pointer(sbi->s_qf_names[i], old_opts.s_qf_names[i]);  	} +	synchronize_rcu(); +	for (i = 0; i < EXT4_MAXQUOTAS; i++) +		kfree(to_free[i]);  #endif  	kfree(orig_data);  	return err; @@ -5545,7 +5570,7 @@ static int ext4_write_info(struct super_block *sb, int type)   */  static int ext4_quota_on_mount(struct super_block *sb, int type)  { -	return dquot_quota_on_mount(sb, EXT4_SB(sb)->s_qf_names[type], +	return dquot_quota_on_mount(sb, get_qf_name(sb, EXT4_SB(sb), type),  					EXT4_SB(sb)->s_jquota_fmt, type);  } @@ -5954,6 +5979,10 @@ static int __init ext4_init_fs(void)  	if (err)  		return err; +	err = ext4_init_pending(); +	if (err) +		goto out6; +  	err = ext4_init_pageio();  	if (err)  		goto out5; @@ -5992,6 +6021,8 @@ out3:  out4:  	ext4_exit_pageio();  out5: +	ext4_exit_pending(); +out6:  	ext4_exit_es();  	return err; @@ -6009,6 +6040,7 @@ static void __exit ext4_exit_fs(void)  	ext4_exit_system_zone();  	ext4_exit_pageio();  	ext4_exit_es(); +	ext4_exit_pending();  }  MODULE_AUTHOR("Remy Card, Stephen Tweedie, Andrew Morton, Andreas Dilger, Theodore Ts'o and others"); | 
