Changes since last update:
- Fix invalid algorithm dereference in encoded extents;
- Add missing dax_break_layout_final(), since recent FSDAX fixes
didn't cover EROFS;
- Arrange long xattr name prefixes more properly.
-----BEGIN PGP SIGNATURE-----
iQJFBAABCgAvFiEEQ0A6bDUS9Y+83NPFUXZn5Zlu5qoFAmjGAqURHHhpYW5nQGtl
cm5lbC5vcmcACgkQUXZn5Zlu5qq2BA/+M205rUNBMV4FemrcWyf1VOJ4D2uQBp5T
JKKw3JcVg8lhYTDDrZHGcsauqRpBJ15xE+gafOi1cLgUoqS5ZEX2+Mey9Lu6ERWX
JRorsFYAdALdikCxhL5dfIlc+jpBxop/AzBLUTOoJvV+xZbeZUZXR1wD3VCmC2yf
Xrzn6BQCwoOsg5/k+jxzjPUEsVjQZo9glZaBJYgBssWNpGEb98qk8tuHncvTilNm
6Gy7+yUZVfNK9+hP8lUYs8+/k4pCmiMfpLYAAwGnH+zAw/gw3O6wIaHt9vwmMSlc
gWsPnhLacDk9xo8xmsi2t0u8zmfBl1MNDY58VYyUQNAeUAnSUzdOv+PTWwzQM4MS
m3pXyMNJNFlTJMufnqsftW3l2CB64EFMiF/IH09//EL+jInzKzkahoDqRFAv7LSm
yIuvGW9c1ZDoTnF95JBzQEnLJVoOjiy7i3n/YeB7Z2DWqG1L/GYZxHRrWuM72Fzj
21Te8OoMk9LtaOFTmN2o5G6KQN+s4w+ywIBWNItvou2lY1cZkVAWRa0Jy9+PHYss
InPUKXJP8JvE9Jfa+xO+Wz9t5cr6xNzbfhCs/vGpLkKh2/zk34Rsbh+4ovHZqNJG
rMDAE/GJc/kN/W0d8sEN/GfIPmq03dHIpKKeTty2bWDrjfjf/qYJq6rW18drpt0O
CT/fUDSL2ks=
=iM5A
-----END PGP SIGNATURE-----
Merge tag 'erofs-for-6.17-rc6-fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/xiang/erofs
Pull erofs fixes from Gao Xiang:
- Fix invalid algorithm dereference in encoded extents
- Add missing dax_break_layout_final(), since recent FSDAX fixes
didn't cover EROFS
- Arrange long xattr name prefixes more properly
* tag 'erofs-for-6.17-rc6-fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/xiang/erofs:
erofs: fix long xattr name prefix placement
erofs: fix runtime warning on truncate_folio_batch_exceptionals()
erofs: fix invalid algorithm for encoded extents
pull/1352/head
commit
f83a4f2a4d
|
|
@ -12,10 +12,12 @@
|
|||
/* to allow for x86 boot sectors and other oddities. */
|
||||
#define EROFS_SUPER_OFFSET 1024
|
||||
|
||||
#define EROFS_FEATURE_COMPAT_SB_CHKSUM 0x00000001
|
||||
#define EROFS_FEATURE_COMPAT_MTIME 0x00000002
|
||||
#define EROFS_FEATURE_COMPAT_XATTR_FILTER 0x00000004
|
||||
#define EROFS_FEATURE_COMPAT_SB_CHKSUM 0x00000001
|
||||
#define EROFS_FEATURE_COMPAT_MTIME 0x00000002
|
||||
#define EROFS_FEATURE_COMPAT_XATTR_FILTER 0x00000004
|
||||
#define EROFS_FEATURE_COMPAT_SHARED_EA_IN_METABOX 0x00000008
|
||||
#define EROFS_FEATURE_COMPAT_PLAIN_XATTR_PFX 0x00000010
|
||||
|
||||
|
||||
/*
|
||||
* Any bits that aren't in EROFS_ALL_FEATURE_INCOMPAT should
|
||||
|
|
|
|||
|
|
@ -234,6 +234,7 @@ EROFS_FEATURE_FUNCS(metabox, incompat, INCOMPAT_METABOX)
|
|||
EROFS_FEATURE_FUNCS(sb_chksum, compat, COMPAT_SB_CHKSUM)
|
||||
EROFS_FEATURE_FUNCS(xattr_filter, compat, COMPAT_XATTR_FILTER)
|
||||
EROFS_FEATURE_FUNCS(shared_ea_in_metabox, compat, COMPAT_SHARED_EA_IN_METABOX)
|
||||
EROFS_FEATURE_FUNCS(plain_xattr_pfx, compat, COMPAT_PLAIN_XATTR_PFX)
|
||||
|
||||
static inline u64 erofs_nid_to_ino64(struct erofs_sb_info *sbi, erofs_nid_t nid)
|
||||
{
|
||||
|
|
|
|||
|
|
@ -1018,10 +1018,22 @@ static int erofs_show_options(struct seq_file *seq, struct dentry *root)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static void erofs_evict_inode(struct inode *inode)
|
||||
{
|
||||
#ifdef CONFIG_FS_DAX
|
||||
if (IS_DAX(inode))
|
||||
dax_break_layout_final(inode);
|
||||
#endif
|
||||
|
||||
truncate_inode_pages_final(&inode->i_data);
|
||||
clear_inode(inode);
|
||||
}
|
||||
|
||||
const struct super_operations erofs_sops = {
|
||||
.put_super = erofs_put_super,
|
||||
.alloc_inode = erofs_alloc_inode,
|
||||
.free_inode = erofs_free_inode,
|
||||
.evict_inode = erofs_evict_inode,
|
||||
.statfs = erofs_statfs,
|
||||
.show_options = erofs_show_options,
|
||||
};
|
||||
|
|
|
|||
|
|
@ -482,6 +482,7 @@ int erofs_xattr_prefixes_init(struct super_block *sb)
|
|||
erofs_off_t pos = (erofs_off_t)sbi->xattr_prefix_start << 2;
|
||||
struct erofs_xattr_prefix_item *pfs;
|
||||
int ret = 0, i, len;
|
||||
bool plain = erofs_sb_has_plain_xattr_pfx(sbi);
|
||||
|
||||
if (!sbi->xattr_prefix_count)
|
||||
return 0;
|
||||
|
|
@ -490,9 +491,15 @@ int erofs_xattr_prefixes_init(struct super_block *sb)
|
|||
if (!pfs)
|
||||
return -ENOMEM;
|
||||
|
||||
if (sbi->packed_inode)
|
||||
buf.mapping = sbi->packed_inode->i_mapping;
|
||||
else
|
||||
if (!plain) {
|
||||
if (erofs_sb_has_metabox(sbi))
|
||||
(void)erofs_init_metabuf(&buf, sb, true);
|
||||
else if (sbi->packed_inode)
|
||||
buf.mapping = sbi->packed_inode->i_mapping;
|
||||
else
|
||||
plain = true;
|
||||
}
|
||||
if (plain)
|
||||
(void)erofs_init_metabuf(&buf, sb, false);
|
||||
|
||||
for (i = 0; i < sbi->xattr_prefix_count; i++) {
|
||||
|
|
|
|||
|
|
@ -394,10 +394,10 @@ static int z_erofs_map_blocks_fo(struct inode *inode,
|
|||
.map = map,
|
||||
.in_mbox = erofs_inode_in_metabox(inode),
|
||||
};
|
||||
int err = 0;
|
||||
unsigned int endoff, afmt;
|
||||
unsigned int endoff;
|
||||
unsigned long initial_lcn;
|
||||
unsigned long long ofs, end;
|
||||
int err;
|
||||
|
||||
ofs = flags & EROFS_GET_BLOCKS_FINDTAIL ? inode->i_size - 1 : map->m_la;
|
||||
if (fragment && !(flags & EROFS_GET_BLOCKS_FINDTAIL) &&
|
||||
|
|
@ -482,20 +482,15 @@ static int z_erofs_map_blocks_fo(struct inode *inode,
|
|||
err = -EFSCORRUPTED;
|
||||
goto unmap_out;
|
||||
}
|
||||
afmt = vi->z_advise & Z_EROFS_ADVISE_INTERLACED_PCLUSTER ?
|
||||
Z_EROFS_COMPRESSION_INTERLACED :
|
||||
Z_EROFS_COMPRESSION_SHIFTED;
|
||||
if (vi->z_advise & Z_EROFS_ADVISE_INTERLACED_PCLUSTER)
|
||||
map->m_algorithmformat = Z_EROFS_COMPRESSION_INTERLACED;
|
||||
else
|
||||
map->m_algorithmformat = Z_EROFS_COMPRESSION_SHIFTED;
|
||||
} else if (m.headtype == Z_EROFS_LCLUSTER_TYPE_HEAD2) {
|
||||
map->m_algorithmformat = vi->z_algorithmtype[1];
|
||||
} else {
|
||||
afmt = m.headtype == Z_EROFS_LCLUSTER_TYPE_HEAD2 ?
|
||||
vi->z_algorithmtype[1] : vi->z_algorithmtype[0];
|
||||
if (!(EROFS_I_SB(inode)->available_compr_algs & (1 << afmt))) {
|
||||
erofs_err(sb, "inconsistent algorithmtype %u for nid %llu",
|
||||
afmt, vi->nid);
|
||||
err = -EFSCORRUPTED;
|
||||
goto unmap_out;
|
||||
}
|
||||
map->m_algorithmformat = vi->z_algorithmtype[0];
|
||||
}
|
||||
map->m_algorithmformat = afmt;
|
||||
|
||||
if ((flags & EROFS_GET_BLOCKS_FIEMAP) ||
|
||||
((flags & EROFS_GET_BLOCKS_READMORE) &&
|
||||
|
|
@ -626,9 +621,9 @@ static int z_erofs_fill_inode(struct inode *inode, struct erofs_map_blocks *map)
|
|||
{
|
||||
struct erofs_inode *const vi = EROFS_I(inode);
|
||||
struct super_block *const sb = inode->i_sb;
|
||||
int err, headnr;
|
||||
erofs_off_t pos;
|
||||
struct z_erofs_map_header *h;
|
||||
erofs_off_t pos;
|
||||
int err = 0;
|
||||
|
||||
if (test_bit(EROFS_I_Z_INITED_BIT, &vi->flags)) {
|
||||
/*
|
||||
|
|
@ -642,7 +637,6 @@ static int z_erofs_fill_inode(struct inode *inode, struct erofs_map_blocks *map)
|
|||
if (wait_on_bit_lock(&vi->flags, EROFS_I_BL_Z_BIT, TASK_KILLABLE))
|
||||
return -ERESTARTSYS;
|
||||
|
||||
err = 0;
|
||||
if (test_bit(EROFS_I_Z_INITED_BIT, &vi->flags))
|
||||
goto out_unlock;
|
||||
|
||||
|
|
@ -679,15 +673,6 @@ static int z_erofs_fill_inode(struct inode *inode, struct erofs_map_blocks *map)
|
|||
else if (vi->z_advise & Z_EROFS_ADVISE_INLINE_PCLUSTER)
|
||||
vi->z_idata_size = le16_to_cpu(h->h_idata_size);
|
||||
|
||||
headnr = 0;
|
||||
if (vi->z_algorithmtype[0] >= Z_EROFS_COMPRESSION_MAX ||
|
||||
vi->z_algorithmtype[++headnr] >= Z_EROFS_COMPRESSION_MAX) {
|
||||
erofs_err(sb, "unknown HEAD%u format %u for nid %llu, please upgrade kernel",
|
||||
headnr + 1, vi->z_algorithmtype[headnr], vi->nid);
|
||||
err = -EOPNOTSUPP;
|
||||
goto out_unlock;
|
||||
}
|
||||
|
||||
if (!erofs_sb_has_big_pcluster(EROFS_SB(sb)) &&
|
||||
vi->z_advise & (Z_EROFS_ADVISE_BIG_PCLUSTER_1 |
|
||||
Z_EROFS_ADVISE_BIG_PCLUSTER_2)) {
|
||||
|
|
@ -726,6 +711,30 @@ out_unlock:
|
|||
return err;
|
||||
}
|
||||
|
||||
static int z_erofs_map_sanity_check(struct inode *inode,
|
||||
struct erofs_map_blocks *map)
|
||||
{
|
||||
struct erofs_sb_info *sbi = EROFS_I_SB(inode);
|
||||
|
||||
if (!(map->m_flags & EROFS_MAP_ENCODED))
|
||||
return 0;
|
||||
if (unlikely(map->m_algorithmformat >= Z_EROFS_COMPRESSION_RUNTIME_MAX)) {
|
||||
erofs_err(inode->i_sb, "unknown algorithm %d @ pos %llu for nid %llu, please upgrade kernel",
|
||||
map->m_algorithmformat, map->m_la, EROFS_I(inode)->nid);
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
if (unlikely(map->m_algorithmformat < Z_EROFS_COMPRESSION_MAX &&
|
||||
!(sbi->available_compr_algs & (1 << map->m_algorithmformat)))) {
|
||||
erofs_err(inode->i_sb, "inconsistent algorithmtype %u for nid %llu",
|
||||
map->m_algorithmformat, EROFS_I(inode)->nid);
|
||||
return -EFSCORRUPTED;
|
||||
}
|
||||
if (unlikely(map->m_plen > Z_EROFS_PCLUSTER_MAX_SIZE ||
|
||||
map->m_llen > Z_EROFS_PCLUSTER_MAX_DSIZE))
|
||||
return -EOPNOTSUPP;
|
||||
return 0;
|
||||
}
|
||||
|
||||
int z_erofs_map_blocks_iter(struct inode *inode, struct erofs_map_blocks *map,
|
||||
int flags)
|
||||
{
|
||||
|
|
@ -746,10 +755,8 @@ int z_erofs_map_blocks_iter(struct inode *inode, struct erofs_map_blocks *map,
|
|||
else
|
||||
err = z_erofs_map_blocks_fo(inode, map, flags);
|
||||
}
|
||||
if (!err && (map->m_flags & EROFS_MAP_ENCODED) &&
|
||||
unlikely(map->m_plen > Z_EROFS_PCLUSTER_MAX_SIZE ||
|
||||
map->m_llen > Z_EROFS_PCLUSTER_MAX_DSIZE))
|
||||
err = -EOPNOTSUPP;
|
||||
if (!err)
|
||||
err = z_erofs_map_sanity_check(inode, map);
|
||||
if (err)
|
||||
map->m_llen = 0;
|
||||
}
|
||||
|
|
|
|||
Loading…
Reference in New Issue