When we're writing zeroes to a reflinked block (such as when we're punching a reflinked range), we need to fork the the block and write to that, otherwise we can corrupt the other reflinks. v2: Only call the end_cow functions if we had set should_fork, and release the buffer if xfs_map_cow_blocks fails. Signed-off-by: Darrick J. Wong <darrick.wong@xxxxxxxxxx> [hch@xxxxxx: Only call the end_cow functions if we had set should_fork] Signed-off-by: Christoph Hellwig <hch@xxxxxx> --- fs/xfs/xfs_aops.c | 12 ++++++++++++ fs/xfs/xfs_bmap_util.c | 38 ++++++++++++++++++++++++++++++++++++-- fs/xfs/xfs_reflink.h | 4 ++++ 3 files changed, 52 insertions(+), 2 deletions(-) diff --git a/fs/xfs/xfs_aops.c b/fs/xfs/xfs_aops.c index 31318b3..812bae5 100644 --- a/fs/xfs/xfs_aops.c +++ b/fs/xfs/xfs_aops.c @@ -394,6 +394,18 @@ xfs_map_blocks( return 0; } +/* + * Find a CoW mapping and ensure that blocks have been allocated to it. + */ +int +xfs_map_cow_blocks( + struct inode *inode, + xfs_off_t offset, + struct xfs_bmbt_irec *imap) +{ + return xfs_map_blocks(inode, offset, imap, XFS_IO_COW); +} + STATIC bool xfs_imap_valid( struct inode *inode, diff --git a/fs/xfs/xfs_bmap_util.c b/fs/xfs/xfs_bmap_util.c index 8666873..79225fb 100644 --- a/fs/xfs/xfs_bmap_util.c +++ b/fs/xfs/xfs_bmap_util.c @@ -42,6 +42,8 @@ #include "xfs_icache.h" #include "xfs_log.h" #include "xfs_rmap_btree.h" +#include "xfs_iomap.h" +#include "xfs_reflink.h" /* Kernel only BMAP related definitions and functions */ @@ -1042,7 +1044,8 @@ xfs_zero_remaining_bytes( xfs_buf_t *bp; xfs_mount_t *mp = ip->i_mount; int nimap; - int error = 0; + int error = 0, err2; + bool should_fork = false; /* * Avoid doing I/O beyond eof - it's not necessary @@ -1055,6 +1058,11 @@ xfs_zero_remaining_bytes( if (endoff > XFS_ISIZE(ip)) endoff = XFS_ISIZE(ip); + error = xfs_reflink_reserve_cow_range(ip, startoff, + endoff - startoff + 1); + if (error) + return error; + for (offset = startoff; offset <= endoff; offset = lastoffset + 1) { uint lock_mode; @@ -1063,6 +1071,10 @@ xfs_zero_remaining_bytes( lock_mode = xfs_ilock_data_map_shared(ip); error = xfs_bmapi_read(ip, offset_fsb, 1, &imap, &nimap, 0); + + /* Do we need to CoW this block? */ + if (error == 0 && nimap == 1) + should_fork = xfs_reflink_is_cow_pending(ip, offset); xfs_iunlock(ip, lock_mode); if (error || nimap < 1) @@ -1084,7 +1096,7 @@ xfs_zero_remaining_bytes( lastoffset = endoff; /* DAX can just zero the backing device directly */ - if (IS_DAX(VFS_I(ip))) { + if (IS_DAX(VFS_I(ip)) && !should_fork) { error = dax_zero_page_range(VFS_I(ip), offset, lastoffset - offset + 1, xfs_get_blocks_direct); @@ -1105,8 +1117,30 @@ xfs_zero_remaining_bytes( (offset - XFS_FSB_TO_B(mp, imap.br_startoff)), 0, lastoffset - offset + 1); + if (should_fork) { + xfs_fsblock_t new_fsbno; + + error = xfs_map_cow_blocks(VFS_I(ip), offset, &imap); + if (error) { + xfs_buf_relse(bp); + return error; + } + new_fsbno = imap.br_startblock + + (offset_fsb - imap.br_startoff); + XFS_BUF_SET_ADDR(bp, XFS_FSB_TO_DADDR(mp, new_fsbno)); + } + error = xfs_bwrite(bp); xfs_buf_relse(bp); + if (should_fork) { + if (error) { + err2 = xfs_reflink_cancel_cow_range(ip, offset, + lastoffset - offset + 1); + return error; + } + error = xfs_reflink_end_cow(ip, offset, + lastoffset - offset + 1); + } if (error) return error; } diff --git a/fs/xfs/xfs_reflink.h b/fs/xfs/xfs_reflink.h index fb128dd..2f3c829 100644 --- a/fs/xfs/xfs_reflink.h +++ b/fs/xfs/xfs_reflink.h @@ -41,4 +41,8 @@ extern int xfs_reflink_cancel_cow_range(struct xfs_inode *ip, xfs_off_t offset, extern int xfs_reflink_end_cow(struct xfs_inode *ip, xfs_off_t offset, xfs_off_t count); +/* xfs_aops.c */ +extern int xfs_map_cow_blocks(struct inode *inode, xfs_off_t offset, + struct xfs_bmbt_irec *imap); + #endif /* __XFS_REFLINK_H */ -- To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html