提交 17030c18 编写于 作者: E Eric Biggers 提交者: Zheng Zengkai

f2fs: fix error handling in f2fs_end_enable_verity()

stable inclusion
from stable-5.10.36
commit 39624749c52dc015d358f8c894bd2236702d07bb
bugzilla: 51867
CVE: NA

--------------------------------

commit 3c031542 upstream.

f2fs didn't properly clean up if verity failed to be enabled on a file:

- It left verity metadata (pages past EOF) in the page cache, which
  would be exposed to userspace if the file was later extended.

- It didn't truncate the verity metadata at all (either from cache or
  from disk) if an error occurred while setting the verity bit.

Fix these bugs by adding a call to truncate_inode_pages() and ensuring
that we truncate the verity metadata (both from cache and from disk) in
all error paths.  Also rework the code to cleanly separate the success
path from the error paths, which makes it much easier to understand.

Finally, log a message if f2fs_truncate() fails, since it might
otherwise fail silently.
Reported-by: NYunlei He <heyunlei@hihonor.com>
Fixes: 95ae251f ("f2fs: add fs-verity support")
Cc: <stable@vger.kernel.org> # v5.4+
Signed-off-by: NEric Biggers <ebiggers@google.com>
Reviewed-by: NChao Yu <yuchao0@huawei.com>
Signed-off-by: NJaegeuk Kim <jaegeuk@kernel.org>
Signed-off-by: NGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: NChen Jun <chenjun102@huawei.com>
Acked-by: NWeilong Chen <chenweilong@huawei.com>
Signed-off-by: NZheng Zengkai <zhengzengkai@huawei.com>
上级 00a9aad1
...@@ -152,40 +152,73 @@ static int f2fs_end_enable_verity(struct file *filp, const void *desc, ...@@ -152,40 +152,73 @@ static int f2fs_end_enable_verity(struct file *filp, const void *desc,
size_t desc_size, u64 merkle_tree_size) size_t desc_size, u64 merkle_tree_size)
{ {
struct inode *inode = file_inode(filp); struct inode *inode = file_inode(filp);
struct f2fs_sb_info *sbi = F2FS_I_SB(inode);
u64 desc_pos = f2fs_verity_metadata_pos(inode) + merkle_tree_size; u64 desc_pos = f2fs_verity_metadata_pos(inode) + merkle_tree_size;
struct fsverity_descriptor_location dloc = { struct fsverity_descriptor_location dloc = {
.version = cpu_to_le32(F2FS_VERIFY_VER), .version = cpu_to_le32(F2FS_VERIFY_VER),
.size = cpu_to_le32(desc_size), .size = cpu_to_le32(desc_size),
.pos = cpu_to_le64(desc_pos), .pos = cpu_to_le64(desc_pos),
}; };
int err = 0; int err = 0, err2 = 0;
if (desc != NULL) { /*
/* Succeeded; write the verity descriptor. */ * If an error already occurred (which fs/verity/ signals by passing
err = pagecache_write(inode, desc, desc_size, desc_pos); * desc == NULL), then only clean-up is needed.
*/
if (desc == NULL)
goto cleanup;
/* Write all pages before clearing FI_VERITY_IN_PROGRESS. */ /* Append the verity descriptor. */
if (!err) err = pagecache_write(inode, desc, desc_size, desc_pos);
err = filemap_write_and_wait(inode->i_mapping); if (err)
} goto cleanup;
/*
* Write all pages (both data and verity metadata). Note that this must
* happen before clearing FI_VERITY_IN_PROGRESS; otherwise pages beyond
* i_size won't be written properly. For crash consistency, this also
* must happen before the verity inode flag gets persisted.
*/
err = filemap_write_and_wait(inode->i_mapping);
if (err)
goto cleanup;
/* Set the verity xattr. */
err = f2fs_setxattr(inode, F2FS_XATTR_INDEX_VERITY,
F2FS_XATTR_NAME_VERITY, &dloc, sizeof(dloc),
NULL, XATTR_CREATE);
if (err)
goto cleanup;
/* If we failed, truncate anything we wrote past i_size. */ /* Finally, set the verity inode flag. */
if (desc == NULL || err) file_set_verity(inode);
f2fs_truncate(inode); f2fs_set_inode_flags(inode);
f2fs_mark_inode_dirty_sync(inode, true);
clear_inode_flag(inode, FI_VERITY_IN_PROGRESS); clear_inode_flag(inode, FI_VERITY_IN_PROGRESS);
return 0;
if (desc != NULL && !err) { cleanup:
err = f2fs_setxattr(inode, F2FS_XATTR_INDEX_VERITY, /*
F2FS_XATTR_NAME_VERITY, &dloc, sizeof(dloc), * Verity failed to be enabled, so clean up by truncating any verity
NULL, XATTR_CREATE); * metadata that was written beyond i_size (both from cache and from
if (!err) { * disk) and clearing FI_VERITY_IN_PROGRESS.
file_set_verity(inode); *
f2fs_set_inode_flags(inode); * Taking i_gc_rwsem[WRITE] is needed to stop f2fs garbage collection
f2fs_mark_inode_dirty_sync(inode, true); * from re-instantiating cached pages we are truncating (since unlike
} * normal file accesses, garbage collection isn't limited by i_size).
*/
down_write(&F2FS_I(inode)->i_gc_rwsem[WRITE]);
truncate_inode_pages(inode->i_mapping, inode->i_size);
err2 = f2fs_truncate(inode);
if (err2) {
f2fs_err(sbi, "Truncating verity metadata failed (errno=%d)",
err2);
set_sbi_flag(sbi, SBI_NEED_FSCK);
} }
return err; up_write(&F2FS_I(inode)->i_gc_rwsem[WRITE]);
clear_inode_flag(inode, FI_VERITY_IN_PROGRESS);
return err ?: err2;
} }
static int f2fs_get_verity_descriptor(struct inode *inode, void *buf, static int f2fs_get_verity_descriptor(struct inode *inode, void *buf,
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册