diff options
Diffstat (limited to 'fs/xfs/xfs_reflink.c')
| -rw-r--r-- | fs/xfs/xfs_reflink.c | 138 | 
1 files changed, 23 insertions, 115 deletions
| diff --git a/fs/xfs/xfs_reflink.c b/fs/xfs/xfs_reflink.c index 0f08153b4994..de451235c4ee 100644 --- a/fs/xfs/xfs_reflink.c +++ b/fs/xfs/xfs_reflink.c @@ -308,13 +308,13 @@ static int  xfs_find_trim_cow_extent(  	struct xfs_inode	*ip,  	struct xfs_bmbt_irec	*imap, +	struct xfs_bmbt_irec	*cmap,  	bool			*shared,  	bool			*found)  {  	xfs_fileoff_t		offset_fsb = imap->br_startoff;  	xfs_filblks_t		count_fsb = imap->br_blockcount;  	struct xfs_iext_cursor	icur; -	struct xfs_bmbt_irec	got;  	*found = false; @@ -322,23 +322,22 @@ xfs_find_trim_cow_extent(  	 * If we don't find an overlapping extent, trim the range we need to  	 * allocate to fit the hole we found.  	 */ -	if (!xfs_iext_lookup_extent(ip, ip->i_cowfp, offset_fsb, &icur, &got)) -		got.br_startoff = offset_fsb + count_fsb; -	if (got.br_startoff > offset_fsb) { +	if (!xfs_iext_lookup_extent(ip, ip->i_cowfp, offset_fsb, &icur, cmap)) +		cmap->br_startoff = offset_fsb + count_fsb; +	if (cmap->br_startoff > offset_fsb) {  		xfs_trim_extent(imap, imap->br_startoff, -				got.br_startoff - imap->br_startoff); +				cmap->br_startoff - imap->br_startoff);  		return xfs_inode_need_cow(ip, imap, shared);  	}  	*shared = true; -	if (isnullstartblock(got.br_startblock)) { -		xfs_trim_extent(imap, got.br_startoff, got.br_blockcount); +	if (isnullstartblock(cmap->br_startblock)) { +		xfs_trim_extent(imap, cmap->br_startoff, cmap->br_blockcount);  		return 0;  	}  	/* real extent found - no need to allocate */ -	xfs_trim_extent(&got, offset_fsb, count_fsb); -	*imap = got; +	xfs_trim_extent(cmap, offset_fsb, count_fsb);  	*found = true;  	return 0;  } @@ -348,6 +347,7 @@ int  xfs_reflink_allocate_cow(  	struct xfs_inode	*ip,  	struct xfs_bmbt_irec	*imap, +	struct xfs_bmbt_irec	*cmap,  	bool			*shared,  	uint			*lockmode,  	bool			convert_now) @@ -367,7 +367,7 @@ xfs_reflink_allocate_cow(  		xfs_ifork_init_cow(ip);  	} -	error = xfs_find_trim_cow_extent(ip, imap, shared, &found); +	error = xfs_find_trim_cow_extent(ip, imap, cmap, shared, &found);  	if (error || !*shared)  		return error;  	if (found) @@ -392,7 +392,7 @@ xfs_reflink_allocate_cow(  	/*  	 * Check for an overlapping extent again now that we dropped the ilock.  	 */ -	error = xfs_find_trim_cow_extent(ip, imap, shared, &found); +	error = xfs_find_trim_cow_extent(ip, imap, cmap, shared, &found);  	if (error || !*shared)  		goto out_trans_cancel;  	if (found) { @@ -410,8 +410,8 @@ xfs_reflink_allocate_cow(  	/* Allocate the entire reservation as unwritten blocks. */  	nimaps = 1;  	error = xfs_bmapi_write(tp, ip, imap->br_startoff, imap->br_blockcount, -			XFS_BMAPI_COWFORK | XFS_BMAPI_PREALLOC, -			resblks, imap, &nimaps); +			XFS_BMAPI_COWFORK | XFS_BMAPI_PREALLOC, 0, cmap, +			&nimaps);  	if (error)  		goto out_unreserve; @@ -427,15 +427,15 @@ xfs_reflink_allocate_cow(  	if (nimaps == 0)  		return -ENOSPC;  convert: -	xfs_trim_extent(imap, offset_fsb, count_fsb); +	xfs_trim_extent(cmap, offset_fsb, count_fsb);  	/*  	 * COW fork extents are supposed to remain unwritten until we're ready  	 * to initiate a disk write.  For direct I/O we are going to write the  	 * data and need the conversion, but for buffered writes we're done.  	 */ -	if (!convert_now || imap->br_state == XFS_EXT_NORM) +	if (!convert_now || cmap->br_state == XFS_EXT_NORM)  		return 0; -	trace_xfs_reflink_convert_cow(ip, imap); +	trace_xfs_reflink_convert_cow(ip, cmap);  	return xfs_reflink_convert_cow_locked(ip, offset_fsb, count_fsb);  out_unreserve: @@ -1270,7 +1270,7 @@ xfs_reflink_zero_posteof(  	trace_xfs_zero_eof(ip, isize, pos - isize);  	return iomap_zero_range(VFS_I(ip), isize, pos - isize, NULL, -			&xfs_iomap_ops); +			&xfs_buffered_write_iomap_ops);  }  /* @@ -1381,85 +1381,6 @@ out_unlock:  	return ret;  } -/* - * The user wants to preemptively CoW all shared blocks in this file, - * which enables us to turn off the reflink flag.  Iterate all - * extents which are not prealloc/delalloc to see which ranges are - * mentioned in the refcount tree, then read those blocks into the - * pagecache, dirty them, fsync them back out, and then we can update - * the inode flag.  What happens if we run out of memory? :) - */ -STATIC int -xfs_reflink_dirty_extents( -	struct xfs_inode	*ip, -	xfs_fileoff_t		fbno, -	xfs_filblks_t		end, -	xfs_off_t		isize) -{ -	struct xfs_mount	*mp = ip->i_mount; -	xfs_agnumber_t		agno; -	xfs_agblock_t		agbno; -	xfs_extlen_t		aglen; -	xfs_agblock_t		rbno; -	xfs_extlen_t		rlen; -	xfs_off_t		fpos; -	xfs_off_t		flen; -	struct xfs_bmbt_irec	map[2]; -	int			nmaps; -	int			error = 0; - -	while (end - fbno > 0) { -		nmaps = 1; -		/* -		 * Look for extents in the file.  Skip holes, delalloc, or -		 * unwritten extents; they can't be reflinked. -		 */ -		error = xfs_bmapi_read(ip, fbno, end - fbno, map, &nmaps, 0); -		if (error) -			goto out; -		if (nmaps == 0) -			break; -		if (!xfs_bmap_is_real_extent(&map[0])) -			goto next; - -		map[1] = map[0]; -		while (map[1].br_blockcount) { -			agno = XFS_FSB_TO_AGNO(mp, map[1].br_startblock); -			agbno = XFS_FSB_TO_AGBNO(mp, map[1].br_startblock); -			aglen = map[1].br_blockcount; - -			error = xfs_reflink_find_shared(mp, NULL, agno, agbno, -					aglen, &rbno, &rlen, true); -			if (error) -				goto out; -			if (rbno == NULLAGBLOCK) -				break; - -			/* Dirty the pages */ -			xfs_iunlock(ip, XFS_ILOCK_EXCL); -			fpos = XFS_FSB_TO_B(mp, map[1].br_startoff + -					(rbno - agbno)); -			flen = XFS_FSB_TO_B(mp, rlen); -			if (fpos + flen > isize) -				flen = isize - fpos; -			error = iomap_file_dirty(VFS_I(ip), fpos, flen, -					&xfs_iomap_ops); -			xfs_ilock(ip, XFS_ILOCK_EXCL); -			if (error) -				goto out; - -			map[1].br_blockcount -= (rbno - agbno + rlen); -			map[1].br_startoff += (rbno - agbno + rlen); -			map[1].br_startblock += (rbno - agbno + rlen); -		} - -next: -		fbno = map[0].br_startoff + map[0].br_blockcount; -	} -out: -	return error; -} -  /* Does this inode need the reflink flag? */  int  xfs_reflink_inode_has_shared_extents( @@ -1596,10 +1517,7 @@ xfs_reflink_unshare(  	xfs_off_t		offset,  	xfs_off_t		len)  { -	struct xfs_mount	*mp = ip->i_mount; -	xfs_fileoff_t		fbno; -	xfs_filblks_t		end; -	xfs_off_t		isize; +	struct inode		*inode = VFS_I(ip);  	int			error;  	if (!xfs_is_reflink_inode(ip)) @@ -1607,20 +1525,13 @@ xfs_reflink_unshare(  	trace_xfs_reflink_unshare(ip, offset, len); -	inode_dio_wait(VFS_I(ip)); +	inode_dio_wait(inode); -	/* Try to CoW the selected ranges */ -	xfs_ilock(ip, XFS_ILOCK_EXCL); -	fbno = XFS_B_TO_FSBT(mp, offset); -	isize = i_size_read(VFS_I(ip)); -	end = XFS_B_TO_FSB(mp, offset + len); -	error = xfs_reflink_dirty_extents(ip, fbno, end, isize); +	error = iomap_file_unshare(inode, offset, len, +			&xfs_buffered_write_iomap_ops);  	if (error) -		goto out_unlock; -	xfs_iunlock(ip, XFS_ILOCK_EXCL); - -	/* Wait for the IO to finish */ -	error = filemap_write_and_wait(VFS_I(ip)->i_mapping); +		goto out; +	error = filemap_write_and_wait(inode->i_mapping);  	if (error)  		goto out; @@ -1628,11 +1539,8 @@ xfs_reflink_unshare(  	error = xfs_reflink_try_clear_inode_flag(ip);  	if (error)  		goto out; -  	return 0; -out_unlock: -	xfs_iunlock(ip, XFS_ILOCK_EXCL);  out:  	trace_xfs_reflink_unshare_error(ip, error, _RET_IP_);  	return error; | 

