diff options
Diffstat (limited to 'fs/bcachefs/super-io.c')
| -rw-r--r-- | fs/bcachefs/super-io.c | 52 | 
1 files changed, 30 insertions, 22 deletions
| diff --git a/fs/bcachefs/super-io.c b/fs/bcachefs/super-io.c index d60c7d27a047..ad28e370b640 100644 --- a/fs/bcachefs/super-io.c +++ b/fs/bcachefs/super-io.c @@ -142,8 +142,8 @@ void bch2_sb_field_delete(struct bch_sb_handle *sb,  void bch2_free_super(struct bch_sb_handle *sb)  {  	kfree(sb->bio); -	if (!IS_ERR_OR_NULL(sb->bdev_handle)) -		bdev_release(sb->bdev_handle); +	if (!IS_ERR_OR_NULL(sb->s_bdev_file)) +		fput(sb->s_bdev_file);  	kfree(sb->holder);  	kfree(sb->sb_name); @@ -470,6 +470,14 @@ static int bch2_sb_validate(struct bch_sb_handle *disk_sb, struct printbuf *out,  			return ret;  	} +	if (rw == WRITE && +	    bch2_sb_member_get(sb, sb->dev_idx).seq != sb->seq) { +		prt_printf(out, "Invalid superblock: member seq %llu != sb seq %llu", +			   le64_to_cpu(bch2_sb_member_get(sb, sb->dev_idx).seq), +			   le64_to_cpu(sb->seq)); +		return -BCH_ERR_invalid_sb_members_missing; +	} +  	return 0;  } @@ -704,22 +712,23 @@ retry:  	if (!opt_get(*opts, nochanges))  		sb->mode |= BLK_OPEN_WRITE; -	sb->bdev_handle = bdev_open_by_path(path, sb->mode, sb->holder, &bch2_sb_handle_bdev_ops); -	if (IS_ERR(sb->bdev_handle) && -	    PTR_ERR(sb->bdev_handle) == -EACCES && +	sb->s_bdev_file = bdev_file_open_by_path(path, sb->mode, sb->holder, &bch2_sb_handle_bdev_ops); +	if (IS_ERR(sb->s_bdev_file) && +	    PTR_ERR(sb->s_bdev_file) == -EACCES &&  	    opt_get(*opts, read_only)) {  		sb->mode &= ~BLK_OPEN_WRITE; -		sb->bdev_handle = bdev_open_by_path(path, sb->mode, sb->holder, &bch2_sb_handle_bdev_ops); -		if (!IS_ERR(sb->bdev_handle)) +		sb->s_bdev_file = bdev_file_open_by_path(path, sb->mode, sb->holder, &bch2_sb_handle_bdev_ops); +		if (!IS_ERR(sb->s_bdev_file))  			opt_set(*opts, nochanges, true);  	} -	if (IS_ERR(sb->bdev_handle)) { -		ret = PTR_ERR(sb->bdev_handle); -		goto out; +	if (IS_ERR(sb->s_bdev_file)) { +		ret = PTR_ERR(sb->s_bdev_file); +		prt_printf(&err, "error opening %s: %s", path, bch2_err_str(ret)); +		goto err;  	} -	sb->bdev = sb->bdev_handle->bdev; +	sb->bdev = file_bdev(sb->s_bdev_file);  	ret = bch2_sb_realloc(sb, 0);  	if (ret) { @@ -743,9 +752,9 @@ retry:  	prt_printf(&err2, "bcachefs (%s): error reading default superblock: %s\n",  	       path, err.buf);  	if (ret == -BCH_ERR_invalid_sb_magic && ignore_notbchfs_msg) -		printk(KERN_INFO "%s", err2.buf); +		bch2_print_opts(opts, KERN_INFO "%s", err2.buf);  	else -		printk(KERN_ERR "%s", err2.buf); +		bch2_print_opts(opts, KERN_ERR "%s", err2.buf);  	printbuf_exit(&err2);  	printbuf_reset(&err); @@ -803,21 +812,20 @@ got_super:  		goto err;  	} -	ret = 0;  	sb->have_layout = true;  	ret = bch2_sb_validate(sb, &err, READ);  	if (ret) { -		printk(KERN_ERR "bcachefs (%s): error validating superblock: %s\n", -		       path, err.buf); +		bch2_print_opts(opts, KERN_ERR "bcachefs (%s): error validating superblock: %s\n", +				path, err.buf);  		goto err_no_print;  	}  out:  	printbuf_exit(&err);  	return ret;  err: -	printk(KERN_ERR "bcachefs (%s): error reading superblock: %s\n", -	       path, err.buf); +	bch2_print_opts(opts, KERN_ERR "bcachefs (%s): error reading superblock: %s\n", +			path, err.buf);  err_no_print:  	bch2_free_super(sb);  	goto out; @@ -977,7 +985,7 @@ int bch2_write_super(struct bch_fs *c)  		prt_str(&buf, " > ");  		bch2_version_to_text(&buf, bcachefs_metadata_version_current);  		prt_str(&buf, ")"); -		bch2_fs_fatal_error(c, "%s", buf.buf); +		bch2_fs_fatal_error(c, ": %s", buf.buf);  		printbuf_exit(&buf);  		return -BCH_ERR_sb_not_downgraded;  	} @@ -997,7 +1005,7 @@ int bch2_write_super(struct bch_fs *c)  		if (le64_to_cpu(ca->sb_read_scratch->seq) < ca->disk_sb.seq) {  			bch2_fs_fatal_error(c, -				"Superblock write was silently dropped! (seq %llu expected %llu)", +				": Superblock write was silently dropped! (seq %llu expected %llu)",  				le64_to_cpu(ca->sb_read_scratch->seq),  				ca->disk_sb.seq);  			percpu_ref_put(&ca->io_ref); @@ -1007,7 +1015,7 @@ int bch2_write_super(struct bch_fs *c)  		if (le64_to_cpu(ca->sb_read_scratch->seq) > ca->disk_sb.seq) {  			bch2_fs_fatal_error(c, -				"Superblock modified by another process (seq %llu expected %llu)", +				": Superblock modified by another process (seq %llu expected %llu)",  				le64_to_cpu(ca->sb_read_scratch->seq),  				ca->disk_sb.seq);  			percpu_ref_put(&ca->io_ref); @@ -1058,7 +1066,7 @@ int bch2_write_super(struct bch_fs *c)  				 !can_mount_with_written ||  				 (can_mount_without_written &&  				  !can_mount_with_written), c, -		"Unable to write superblock to sufficient devices (from %ps)", +		": Unable to write superblock to sufficient devices (from %ps)",  		(void *) _RET_IP_))  		ret = -1;  out: |