Changes since last update:

- Better error message when prepare_ondemand_read failed;
 
  - Fix unmount of bdev-based mode if CONFIG_EROFS_FS_ONDEMAND is on.
 -----BEGIN PGP SIGNATURE-----
 
 iQJFBAABCgAvFiEEQ0A6bDUS9Y+83NPFUXZn5Zlu5qoFAmYvafURHHhpYW5nQGtl
 cm5lbC5vcmcACgkQUXZn5Zlu5qrgAg/+PK6WKc2t5SiLpG+dSLgQLhWfebJXhf3+
 IWr4hT04JcFJgmxBJAcgh7uoJFrxjTxGDX0vtiLhAAOqRCXb9rDlpqbu+PXfURa1
 YyiBAEuqnwqKCm3BixVixV8JCSLHu+UzJ72mZGw+wtSdhInXS/ddBAc8WNVbfjyU
 cUJb30+N5vjYVRYtDQG5qP0dbpstm1cTjNJIhjStcSpk9LQkQ9degYctjJj1mk8B
 ilplRIyy0bEBVbupUHPtmrHzpDZYe4QXFTtQaEGst1uDw/Cxw8I8zpwO4wavdxVz
 DZwDF6bcKGsaCFIV4UuJvowU7syEDkBHk5JvJoyNEhRuZQYOIvMdYRxl6AxAp1Jk
 vV5sn/LbWaqF7uR2Lw4NJBapulcDjEMfHuP2FyzzRqMmPO/4BiGrLnK+IRu3cSTH
 0/bh6oRdplx8Q7faylMnSTXxYigXfu6vjPXuIZ95Wisk9L+URzhmdwkhnnu8MRPn
 DSQfD2VvRrRr52uFiR+LppspPrb8ru6EqORW9ShpugM0V9XPiGyYnNFlCMtoCpgf
 fYx5p2F23fXNq461dB6UwZfIM9jvMvrNsb3h3+pvpN/y5UbsW6kMUD38MwZxlSPT
 z18P87HKHXGNc6+pQK7zI89I7g+2W5TCZVg0/w5DGTXkOlDQPi9X9woaOYcJIPYJ
 LcDG0uM18kI=
 =v0WH
 -----END PGP SIGNATURE-----

Merge tag 'erofs-for-6.9-rc7-fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/xiang/erofs

Pull erofs fixes from Gao Xiang:
 "Three fixes related to EROFS fscache mode. The most important two
  patches fix calling kill_block_super() in bdev-based mode instead of
  kill_anon_super(). The remaining patch is an informative one.

  Summary:

   - Better error message when prepare_ondemand_read failed

   - Fix unmount of bdev-based mode if CONFIG_EROFS_FS_ONDEMAND is on"

* tag 'erofs-for-6.9-rc7-fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/xiang/erofs:
  erofs: reliably distinguish block based and fscache mode
  erofs: get rid of erofs_fs_context
  erofs: modify the error message when prepare_ondemand_read failed
This commit is contained in:
Linus Torvalds 2024-04-29 08:52:08 -07:00
commit b947cc5bf6
3 changed files with 56 additions and 77 deletions

View File

@ -151,7 +151,7 @@ static int erofs_fscache_read_io_async(struct fscache_cookie *cookie,
if (WARN_ON(len == 0)) if (WARN_ON(len == 0))
source = NETFS_INVALID_READ; source = NETFS_INVALID_READ;
if (source != NETFS_READ_FROM_CACHE) { if (source != NETFS_READ_FROM_CACHE) {
erofs_err(NULL, "prepare_read failed (source %d)", source); erofs_err(NULL, "prepare_ondemand_read failed (source %d)", source);
return -EIO; return -EIO;
} }

View File

@ -84,13 +84,6 @@ struct erofs_dev_context {
bool flatdev; bool flatdev;
}; };
struct erofs_fs_context {
struct erofs_mount_opts opt;
struct erofs_dev_context *devs;
char *fsid;
char *domain_id;
};
/* all filesystem-wide lz4 configurations */ /* all filesystem-wide lz4 configurations */
struct erofs_sb_lz4_info { struct erofs_sb_lz4_info {
/* # of pages needed for EROFS lz4 rolling decompression */ /* # of pages needed for EROFS lz4 rolling decompression */

View File

@ -370,18 +370,18 @@ out:
return ret; return ret;
} }
static void erofs_default_options(struct erofs_fs_context *ctx) static void erofs_default_options(struct erofs_sb_info *sbi)
{ {
#ifdef CONFIG_EROFS_FS_ZIP #ifdef CONFIG_EROFS_FS_ZIP
ctx->opt.cache_strategy = EROFS_ZIP_CACHE_READAROUND; sbi->opt.cache_strategy = EROFS_ZIP_CACHE_READAROUND;
ctx->opt.max_sync_decompress_pages = 3; sbi->opt.max_sync_decompress_pages = 3;
ctx->opt.sync_decompress = EROFS_SYNC_DECOMPRESS_AUTO; sbi->opt.sync_decompress = EROFS_SYNC_DECOMPRESS_AUTO;
#endif #endif
#ifdef CONFIG_EROFS_FS_XATTR #ifdef CONFIG_EROFS_FS_XATTR
set_opt(&ctx->opt, XATTR_USER); set_opt(&sbi->opt, XATTR_USER);
#endif #endif
#ifdef CONFIG_EROFS_FS_POSIX_ACL #ifdef CONFIG_EROFS_FS_POSIX_ACL
set_opt(&ctx->opt, POSIX_ACL); set_opt(&sbi->opt, POSIX_ACL);
#endif #endif
} }
@ -426,16 +426,16 @@ static const struct fs_parameter_spec erofs_fs_parameters[] = {
static bool erofs_fc_set_dax_mode(struct fs_context *fc, unsigned int mode) static bool erofs_fc_set_dax_mode(struct fs_context *fc, unsigned int mode)
{ {
#ifdef CONFIG_FS_DAX #ifdef CONFIG_FS_DAX
struct erofs_fs_context *ctx = fc->fs_private; struct erofs_sb_info *sbi = fc->s_fs_info;
switch (mode) { switch (mode) {
case EROFS_MOUNT_DAX_ALWAYS: case EROFS_MOUNT_DAX_ALWAYS:
set_opt(&ctx->opt, DAX_ALWAYS); set_opt(&sbi->opt, DAX_ALWAYS);
clear_opt(&ctx->opt, DAX_NEVER); clear_opt(&sbi->opt, DAX_NEVER);
return true; return true;
case EROFS_MOUNT_DAX_NEVER: case EROFS_MOUNT_DAX_NEVER:
set_opt(&ctx->opt, DAX_NEVER); set_opt(&sbi->opt, DAX_NEVER);
clear_opt(&ctx->opt, DAX_ALWAYS); clear_opt(&sbi->opt, DAX_ALWAYS);
return true; return true;
default: default:
DBG_BUGON(1); DBG_BUGON(1);
@ -450,7 +450,7 @@ static bool erofs_fc_set_dax_mode(struct fs_context *fc, unsigned int mode)
static int erofs_fc_parse_param(struct fs_context *fc, static int erofs_fc_parse_param(struct fs_context *fc,
struct fs_parameter *param) struct fs_parameter *param)
{ {
struct erofs_fs_context *ctx = fc->fs_private; struct erofs_sb_info *sbi = fc->s_fs_info;
struct fs_parse_result result; struct fs_parse_result result;
struct erofs_device_info *dif; struct erofs_device_info *dif;
int opt, ret; int opt, ret;
@ -463,9 +463,9 @@ static int erofs_fc_parse_param(struct fs_context *fc,
case Opt_user_xattr: case Opt_user_xattr:
#ifdef CONFIG_EROFS_FS_XATTR #ifdef CONFIG_EROFS_FS_XATTR
if (result.boolean) if (result.boolean)
set_opt(&ctx->opt, XATTR_USER); set_opt(&sbi->opt, XATTR_USER);
else else
clear_opt(&ctx->opt, XATTR_USER); clear_opt(&sbi->opt, XATTR_USER);
#else #else
errorfc(fc, "{,no}user_xattr options not supported"); errorfc(fc, "{,no}user_xattr options not supported");
#endif #endif
@ -473,16 +473,16 @@ static int erofs_fc_parse_param(struct fs_context *fc,
case Opt_acl: case Opt_acl:
#ifdef CONFIG_EROFS_FS_POSIX_ACL #ifdef CONFIG_EROFS_FS_POSIX_ACL
if (result.boolean) if (result.boolean)
set_opt(&ctx->opt, POSIX_ACL); set_opt(&sbi->opt, POSIX_ACL);
else else
clear_opt(&ctx->opt, POSIX_ACL); clear_opt(&sbi->opt, POSIX_ACL);
#else #else
errorfc(fc, "{,no}acl options not supported"); errorfc(fc, "{,no}acl options not supported");
#endif #endif
break; break;
case Opt_cache_strategy: case Opt_cache_strategy:
#ifdef CONFIG_EROFS_FS_ZIP #ifdef CONFIG_EROFS_FS_ZIP
ctx->opt.cache_strategy = result.uint_32; sbi->opt.cache_strategy = result.uint_32;
#else #else
errorfc(fc, "compression not supported, cache_strategy ignored"); errorfc(fc, "compression not supported, cache_strategy ignored");
#endif #endif
@ -504,27 +504,27 @@ static int erofs_fc_parse_param(struct fs_context *fc,
kfree(dif); kfree(dif);
return -ENOMEM; return -ENOMEM;
} }
down_write(&ctx->devs->rwsem); down_write(&sbi->devs->rwsem);
ret = idr_alloc(&ctx->devs->tree, dif, 0, 0, GFP_KERNEL); ret = idr_alloc(&sbi->devs->tree, dif, 0, 0, GFP_KERNEL);
up_write(&ctx->devs->rwsem); up_write(&sbi->devs->rwsem);
if (ret < 0) { if (ret < 0) {
kfree(dif->path); kfree(dif->path);
kfree(dif); kfree(dif);
return ret; return ret;
} }
++ctx->devs->extra_devices; ++sbi->devs->extra_devices;
break; break;
#ifdef CONFIG_EROFS_FS_ONDEMAND #ifdef CONFIG_EROFS_FS_ONDEMAND
case Opt_fsid: case Opt_fsid:
kfree(ctx->fsid); kfree(sbi->fsid);
ctx->fsid = kstrdup(param->string, GFP_KERNEL); sbi->fsid = kstrdup(param->string, GFP_KERNEL);
if (!ctx->fsid) if (!sbi->fsid)
return -ENOMEM; return -ENOMEM;
break; break;
case Opt_domain_id: case Opt_domain_id:
kfree(ctx->domain_id); kfree(sbi->domain_id);
ctx->domain_id = kstrdup(param->string, GFP_KERNEL); sbi->domain_id = kstrdup(param->string, GFP_KERNEL);
if (!ctx->domain_id) if (!sbi->domain_id)
return -ENOMEM; return -ENOMEM;
break; break;
#else #else
@ -581,8 +581,7 @@ static const struct export_operations erofs_export_ops = {
static int erofs_fc_fill_super(struct super_block *sb, struct fs_context *fc) static int erofs_fc_fill_super(struct super_block *sb, struct fs_context *fc)
{ {
struct inode *inode; struct inode *inode;
struct erofs_sb_info *sbi; struct erofs_sb_info *sbi = EROFS_SB(sb);
struct erofs_fs_context *ctx = fc->fs_private;
int err; int err;
sb->s_magic = EROFS_SUPER_MAGIC; sb->s_magic = EROFS_SUPER_MAGIC;
@ -590,19 +589,6 @@ static int erofs_fc_fill_super(struct super_block *sb, struct fs_context *fc)
sb->s_maxbytes = MAX_LFS_FILESIZE; sb->s_maxbytes = MAX_LFS_FILESIZE;
sb->s_op = &erofs_sops; sb->s_op = &erofs_sops;
sbi = kzalloc(sizeof(*sbi), GFP_KERNEL);
if (!sbi)
return -ENOMEM;
sb->s_fs_info = sbi;
sbi->opt = ctx->opt;
sbi->devs = ctx->devs;
ctx->devs = NULL;
sbi->fsid = ctx->fsid;
ctx->fsid = NULL;
sbi->domain_id = ctx->domain_id;
ctx->domain_id = NULL;
sbi->blkszbits = PAGE_SHIFT; sbi->blkszbits = PAGE_SHIFT;
if (erofs_is_fscache_mode(sb)) { if (erofs_is_fscache_mode(sb)) {
sb->s_blocksize = PAGE_SIZE; sb->s_blocksize = PAGE_SIZE;
@ -706,9 +692,9 @@ static int erofs_fc_fill_super(struct super_block *sb, struct fs_context *fc)
static int erofs_fc_get_tree(struct fs_context *fc) static int erofs_fc_get_tree(struct fs_context *fc)
{ {
struct erofs_fs_context *ctx = fc->fs_private; struct erofs_sb_info *sbi = fc->s_fs_info;
if (IS_ENABLED(CONFIG_EROFS_FS_ONDEMAND) && ctx->fsid) if (IS_ENABLED(CONFIG_EROFS_FS_ONDEMAND) && sbi->fsid)
return get_tree_nodev(fc, erofs_fc_fill_super); return get_tree_nodev(fc, erofs_fc_fill_super);
return get_tree_bdev(fc, erofs_fc_fill_super); return get_tree_bdev(fc, erofs_fc_fill_super);
@ -718,19 +704,19 @@ static int erofs_fc_reconfigure(struct fs_context *fc)
{ {
struct super_block *sb = fc->root->d_sb; struct super_block *sb = fc->root->d_sb;
struct erofs_sb_info *sbi = EROFS_SB(sb); struct erofs_sb_info *sbi = EROFS_SB(sb);
struct erofs_fs_context *ctx = fc->fs_private; struct erofs_sb_info *new_sbi = fc->s_fs_info;
DBG_BUGON(!sb_rdonly(sb)); DBG_BUGON(!sb_rdonly(sb));
if (ctx->fsid || ctx->domain_id) if (new_sbi->fsid || new_sbi->domain_id)
erofs_info(sb, "ignoring reconfiguration for fsid|domain_id."); erofs_info(sb, "ignoring reconfiguration for fsid|domain_id.");
if (test_opt(&ctx->opt, POSIX_ACL)) if (test_opt(&new_sbi->opt, POSIX_ACL))
fc->sb_flags |= SB_POSIXACL; fc->sb_flags |= SB_POSIXACL;
else else
fc->sb_flags &= ~SB_POSIXACL; fc->sb_flags &= ~SB_POSIXACL;
sbi->opt = ctx->opt; sbi->opt = new_sbi->opt;
fc->sb_flags |= SB_RDONLY; fc->sb_flags |= SB_RDONLY;
return 0; return 0;
@ -761,12 +747,15 @@ static void erofs_free_dev_context(struct erofs_dev_context *devs)
static void erofs_fc_free(struct fs_context *fc) static void erofs_fc_free(struct fs_context *fc)
{ {
struct erofs_fs_context *ctx = fc->fs_private; struct erofs_sb_info *sbi = fc->s_fs_info;
erofs_free_dev_context(ctx->devs); if (!sbi)
kfree(ctx->fsid); return;
kfree(ctx->domain_id);
kfree(ctx); erofs_free_dev_context(sbi->devs);
kfree(sbi->fsid);
kfree(sbi->domain_id);
kfree(sbi);
} }
static const struct fs_context_operations erofs_context_ops = { static const struct fs_context_operations erofs_context_ops = {
@ -778,38 +767,35 @@ static const struct fs_context_operations erofs_context_ops = {
static int erofs_init_fs_context(struct fs_context *fc) static int erofs_init_fs_context(struct fs_context *fc)
{ {
struct erofs_fs_context *ctx; struct erofs_sb_info *sbi;
ctx = kzalloc(sizeof(*ctx), GFP_KERNEL); sbi = kzalloc(sizeof(*sbi), GFP_KERNEL);
if (!ctx) if (!sbi)
return -ENOMEM; return -ENOMEM;
ctx->devs = kzalloc(sizeof(struct erofs_dev_context), GFP_KERNEL);
if (!ctx->devs) { sbi->devs = kzalloc(sizeof(struct erofs_dev_context), GFP_KERNEL);
kfree(ctx); if (!sbi->devs) {
kfree(sbi);
return -ENOMEM; return -ENOMEM;
} }
fc->fs_private = ctx; fc->s_fs_info = sbi;
idr_init(&ctx->devs->tree); idr_init(&sbi->devs->tree);
init_rwsem(&ctx->devs->rwsem); init_rwsem(&sbi->devs->rwsem);
erofs_default_options(ctx); erofs_default_options(sbi);
fc->ops = &erofs_context_ops; fc->ops = &erofs_context_ops;
return 0; return 0;
} }
static void erofs_kill_sb(struct super_block *sb) static void erofs_kill_sb(struct super_block *sb)
{ {
struct erofs_sb_info *sbi; struct erofs_sb_info *sbi = EROFS_SB(sb);
if (erofs_is_fscache_mode(sb)) if (IS_ENABLED(CONFIG_EROFS_FS_ONDEMAND) && sbi->fsid)
kill_anon_super(sb); kill_anon_super(sb);
else else
kill_block_super(sb); kill_block_super(sb);
sbi = EROFS_SB(sb);
if (!sbi)
return;
erofs_free_dev_context(sbi->devs); erofs_free_dev_context(sbi->devs);
fs_put_dax(sbi->dax_dev, NULL); fs_put_dax(sbi->dax_dev, NULL);
erofs_fscache_unregister_fs(sb); erofs_fscache_unregister_fs(sb);