提交 410fdc72 编写于 作者: D Darrick J. Wong 提交者: Dave Chinner

xfs: zero posteof blocks when cloning above eof

When we're reflinking between two files and the destination file range
is well beyond the destination file's EOF marker, zero any posteof
speculative preallocations in the destination file so that we don't
expose stale disk contents.  The previous strategy of trying to clear
the preallocations does not work if the destination file has the
PREALLOC flag set.

Uncovered by shared/010.
Reported-by: NZorro Lang <zlang@redhat.com>
Bugzilla-id: https://bugzilla.kernel.org/show_bug.cgi?id=201259Signed-off-by: NDarrick J. Wong <darrick.wong@oracle.com>
Reviewed-by: NDave Chinner <dchinner@redhat.com>
Signed-off-by: NDave Chinner <david@fromorbit.com>
上级 0d41e1d2
...@@ -1240,6 +1240,26 @@ xfs_reflink_remap_unlock( ...@@ -1240,6 +1240,26 @@ xfs_reflink_remap_unlock(
inode_unlock_shared(inode_in); inode_unlock_shared(inode_in);
} }
/*
* If we're reflinking to a point past the destination file's EOF, we must
* zero any speculative post-EOF preallocations that sit between the old EOF
* and the destination file offset.
*/
static int
xfs_reflink_zero_posteof(
struct xfs_inode *ip,
loff_t pos)
{
loff_t isize = i_size_read(VFS_I(ip));
if (pos <= isize)
return 0;
trace_xfs_zero_eof(ip, isize, pos - isize);
return iomap_zero_range(VFS_I(ip), isize, pos - isize, NULL,
&xfs_iomap_ops);
}
/* /*
* Prepare two files for range cloning. Upon a successful return both inodes * Prepare two files for range cloning. Upon a successful return both inodes
* will have the iolock and mmaplock held, the page cache of the out file * will have the iolock and mmaplock held, the page cache of the out file
...@@ -1292,15 +1312,12 @@ xfs_reflink_remap_prep( ...@@ -1292,15 +1312,12 @@ xfs_reflink_remap_prep(
goto out_unlock; goto out_unlock;
/* /*
* Clear out post-eof preallocations because we don't have page cache * Zero existing post-eof speculative preallocations in the destination
* backing the delayed allocations and they'll never get freed on * file.
* their own.
*/ */
if (xfs_can_free_eofblocks(dest, true)) { ret = xfs_reflink_zero_posteof(dest, pos_out);
ret = xfs_free_eofblocks(dest); if (ret)
if (ret) goto out_unlock;
goto out_unlock;
}
/* Set flags and remap blocks. */ /* Set flags and remap blocks. */
ret = xfs_reflink_set_inode_flag(src, dest); ret = xfs_reflink_set_inode_flag(src, dest);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册