2005-04-16 15:20:36 -07:00
/*
2006-06-09 14:48:12 +10:00
* Copyright ( c ) 2000 - 2006 Silicon Graphics , Inc .
2005-11-02 14:58:39 +11:00
* All Rights Reserved .
2005-04-16 15:20:36 -07:00
*
2005-11-02 14:58:39 +11:00
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License as
2005-04-16 15:20:36 -07:00
* published by the Free Software Foundation .
*
2005-11-02 14:58:39 +11:00
* This program is distributed in the hope that it would be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2005-04-16 15:20:36 -07:00
*
2005-11-02 14:58:39 +11:00
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write the Free Software Foundation ,
* Inc . , 51 Franklin St , Fifth Floor , Boston , MA 02110 - 1301 USA
2005-04-16 15:20:36 -07:00
*/
# include "xfs.h"
2005-11-02 14:38:42 +11:00
# include "xfs_fs.h"
2005-04-16 15:20:36 -07:00
# include "xfs_types.h"
2005-11-02 14:38:42 +11:00
# include "xfs_bit.h"
2005-04-16 15:20:36 -07:00
# include "xfs_log.h"
2005-11-02 14:38:42 +11:00
# include "xfs_inum.h"
2005-04-16 15:20:36 -07:00
# include "xfs_trans.h"
# include "xfs_sb.h"
2005-11-02 14:38:42 +11:00
# include "xfs_ag.h"
2005-04-16 15:20:36 -07:00
# include "xfs_dir2.h"
# include "xfs_dmapi.h"
# include "xfs_mount.h"
# include "xfs_bmap_btree.h"
2005-11-02 14:38:42 +11:00
# include "xfs_alloc_btree.h"
2005-04-16 15:20:36 -07:00
# include "xfs_ialloc_btree.h"
# include "xfs_dir2_sf.h"
2005-11-02 14:38:42 +11:00
# include "xfs_attr_sf.h"
2005-04-16 15:20:36 -07:00
# include "xfs_dinode.h"
# include "xfs_inode.h"
2005-11-02 14:38:42 +11:00
# include "xfs_inode_item.h"
2005-04-16 15:20:36 -07:00
# include "xfs_bmap.h"
2005-11-02 14:38:42 +11:00
# include "xfs_btree.h"
2005-04-16 15:20:36 -07:00
# include "xfs_ialloc.h"
# include "xfs_itable.h"
# include "xfs_dfrag.h"
# include "xfs_error.h"
# include "xfs_rw.h"
2007-08-29 10:58:01 +10:00
# include "xfs_vnodeops.h"
2005-04-16 15:20:36 -07:00
/*
* Syssgi interface for swapext
*/
int
xfs_swapext (
2008-11-25 21:20:06 -06:00
xfs_swapext_t * sxp )
2005-04-16 15:20:36 -07:00
{
2008-02-05 12:13:07 +11:00
xfs_inode_t * ip , * tip ;
struct file * file , * target_file ;
2005-04-16 15:20:36 -07:00
int error = 0 ;
/* Pull information for the target fd */
2008-02-05 12:13:07 +11:00
file = fget ( ( int ) sxp - > sx_fdtarget ) ;
if ( ! file ) {
2005-04-16 15:20:36 -07:00
error = XFS_ERROR ( EINVAL ) ;
2009-01-25 20:53:00 -06:00
goto out ;
2005-04-16 15:20:36 -07:00
}
2008-02-05 12:13:15 +11:00
if ( ! ( file - > f_mode & FMODE_WRITE ) | | ( file - > f_flags & O_APPEND ) ) {
error = XFS_ERROR ( EBADF ) ;
goto out_put_file ;
}
2008-02-05 12:13:07 +11:00
target_file = fget ( ( int ) sxp - > sx_fdtmp ) ;
if ( ! target_file ) {
2005-04-16 15:20:36 -07:00
error = XFS_ERROR ( EINVAL ) ;
2008-02-05 12:13:07 +11:00
goto out_put_file ;
2005-04-16 15:20:36 -07:00
}
2008-02-05 12:13:15 +11:00
if ( ! ( target_file - > f_mode & FMODE_WRITE ) | |
( target_file - > f_flags & O_APPEND ) ) {
error = XFS_ERROR ( EBADF ) ;
goto out_put_target_file ;
}
2009-02-12 19:56:00 +01:00
if ( IS_SWAPFILE ( file - > f_path . dentry - > d_inode ) | |
IS_SWAPFILE ( target_file - > f_path . dentry - > d_inode ) ) {
error = XFS_ERROR ( EINVAL ) ;
goto out_put_target_file ;
}
2008-02-05 12:13:07 +11:00
ip = XFS_I ( file - > f_path . dentry - > d_inode ) ;
tip = XFS_I ( target_file - > f_path . dentry - > d_inode ) ;
2005-04-16 15:20:36 -07:00
if ( ip - > i_mount ! = tip - > i_mount ) {
2008-02-05 12:13:07 +11:00
error = XFS_ERROR ( EINVAL ) ;
goto out_put_target_file ;
2005-04-16 15:20:36 -07:00
}
if ( ip - > i_ino = = tip - > i_ino ) {
2008-02-05 12:13:07 +11:00
error = XFS_ERROR ( EINVAL ) ;
goto out_put_target_file ;
2005-04-16 15:20:36 -07:00
}
2008-02-05 12:13:07 +11:00
if ( XFS_FORCED_SHUTDOWN ( ip - > i_mount ) ) {
error = XFS_ERROR ( EIO ) ;
goto out_put_target_file ;
2005-04-16 15:20:36 -07:00
}
2007-10-11 17:34:33 +10:00
error = xfs_swap_extents ( ip , tip , sxp ) ;
2006-06-09 14:48:12 +10:00
2008-02-05 12:13:07 +11:00
out_put_target_file :
fput ( target_file ) ;
out_put_file :
fput ( file ) ;
out :
2006-06-09 14:48:12 +10:00
return error ;
}
int
xfs_swap_extents (
xfs_inode_t * ip ,
xfs_inode_t * tip ,
xfs_swapext_t * sxp )
{
xfs_mount_t * mp ;
xfs_trans_t * tp ;
xfs_bstat_t * sbp = & sxp - > sx_stat ;
xfs_ifork_t * tempifp , * ifp , * tifp ;
int ilf_fields , tilf_fields ;
int error = 0 ;
int aforkblks = 0 ;
int taforkblks = 0 ;
__uint64_t tmp ;
mp = ip - > i_mount ;
tempifp = kmem_alloc ( sizeof ( xfs_ifork_t ) , KM_MAYFAIL ) ;
if ( ! tempifp ) {
error = XFS_ERROR ( ENOMEM ) ;
2009-02-04 09:37:43 +01:00
goto out ;
2006-06-09 14:48:12 +10:00
}
sbp = & sxp - > sx_stat ;
2005-04-16 15:20:36 -07:00
2008-09-17 16:51:21 +10:00
/*
* we have to do two separate lock calls here to keep lockdep
* happy . If we try to get all the locks in one call , lock will
* report false positives when we drop the ILOCK and regain them
* below .
*/
xfs_lock_two_inodes ( ip , tip , XFS_IOLOCK_EXCL ) ;
xfs_lock_two_inodes ( ip , tip , XFS_ILOCK_EXCL ) ;
2005-04-16 15:20:36 -07:00
/* Verify that both files have the same format */
if ( ( ip - > i_d . di_mode & S_IFMT ) ! = ( tip - > i_d . di_mode & S_IFMT ) ) {
error = XFS_ERROR ( EINVAL ) ;
2009-02-04 09:37:43 +01:00
goto out_unlock ;
2005-04-16 15:20:36 -07:00
}
/* Verify both files are either real-time or non-realtime */
2007-11-23 16:29:42 +11:00
if ( XFS_IS_REALTIME_INODE ( ip ) ! = XFS_IS_REALTIME_INODE ( tip ) ) {
2005-04-16 15:20:36 -07:00
error = XFS_ERROR ( EINVAL ) ;
2009-02-04 09:37:43 +01:00
goto out_unlock ;
2005-04-16 15:20:36 -07:00
}
/* Should never get a local format */
if ( ip - > i_d . di_format = = XFS_DINODE_FMT_LOCAL | |
tip - > i_d . di_format = = XFS_DINODE_FMT_LOCAL ) {
error = XFS_ERROR ( EINVAL ) ;
2009-02-04 09:37:43 +01:00
goto out_unlock ;
2005-04-16 15:20:36 -07:00
}
2008-08-13 16:22:09 +10:00
if ( VN_CACHED ( VFS_I ( tip ) ) ! = 0 ) {
2007-10-11 17:44:08 +10:00
xfs_inval_cached_trace ( tip , 0 , - 1 , 0 , - 1 ) ;
2007-08-29 10:58:01 +10:00
error = xfs_flushinval_pages ( tip , 0 , - 1 ,
FI_REMAPF_LOCKED ) ;
2007-05-08 13:49:27 +10:00
if ( error )
2009-02-04 09:37:43 +01:00
goto out_unlock ;
2005-06-21 15:47:39 +10:00
}
2005-04-16 15:20:36 -07:00
/* Verify O_DIRECT for ftmp */
2008-08-13 16:22:09 +10:00
if ( VN_CACHED ( VFS_I ( tip ) ) ! = 0 ) {
2005-04-16 15:20:36 -07:00
error = XFS_ERROR ( EINVAL ) ;
2009-02-04 09:37:43 +01:00
goto out_unlock ;
2005-04-16 15:20:36 -07:00
}
/* Verify all data are being swapped */
2005-11-02 10:29:04 +11:00
if ( sxp - > sx_offset ! = 0 | |
sxp - > sx_length ! = ip - > i_d . di_size | |
sxp - > sx_length ! = tip - > i_d . di_size ) {
2005-04-16 15:20:36 -07:00
error = XFS_ERROR ( EFAULT ) ;
2009-02-04 09:37:43 +01:00
goto out_unlock ;
2005-04-16 15:20:36 -07:00
}
/*
* If the target has extended attributes , the tmp file
* must also in order to ensure the correct data fork
* format .
*/
if ( XFS_IFORK_Q ( ip ) ! = XFS_IFORK_Q ( tip ) ) {
error = XFS_ERROR ( EINVAL ) ;
2009-02-04 09:37:43 +01:00
goto out_unlock ;
2005-04-16 15:20:36 -07:00
}
/*
* Compare the current change & modify times with that
* passed in . If they differ , we abort this swap .
* This is the mechanism used to ensure the calling
* process that the file was not changed out from
* under it .
*/
if ( ( sbp - > bs_ctime . tv_sec ! = ip - > i_d . di_ctime . t_sec ) | |
( sbp - > bs_ctime . tv_nsec ! = ip - > i_d . di_ctime . t_nsec ) | |
( sbp - > bs_mtime . tv_sec ! = ip - > i_d . di_mtime . t_sec ) | |
( sbp - > bs_mtime . tv_nsec ! = ip - > i_d . di_mtime . t_nsec ) ) {
error = XFS_ERROR ( EBUSY ) ;
2009-02-04 09:37:43 +01:00
goto out_unlock ;
2005-04-16 15:20:36 -07:00
}
/* We need to fail if the file is memory mapped. Once we have tossed
* all existing pages , the page fault will have no option
* but to go to the filesystem for pages . By making the page fault call
2006-06-09 17:00:52 +10:00
* vop_read ( or write in the case of autogrow ) they block on the iolock
2005-04-16 15:20:36 -07:00
* until we have switched the extents .
*/
2008-08-13 16:22:09 +10:00
if ( VN_MAPPED ( VFS_I ( ip ) ) ) {
2005-04-16 15:20:36 -07:00
error = XFS_ERROR ( EBUSY ) ;
2009-02-04 09:37:43 +01:00
goto out_unlock ;
2005-04-16 15:20:36 -07:00
}
xfs_iunlock ( ip , XFS_ILOCK_EXCL ) ;
xfs_iunlock ( tip , XFS_ILOCK_EXCL ) ;
/*
* There is a race condition here since we gave up the
* ilock . However , the data fork will not change since
* we have the iolock ( locked for truncation too ) so we
* are safe . We don ' t really care if non - io related
* fields change .
*/
2007-08-29 10:58:01 +10:00
xfs_tosspages ( ip , 0 , - 1 , FI_REMAPF ) ;
2005-04-16 15:20:36 -07:00
tp = xfs_trans_alloc ( mp , XFS_TRANS_SWAPEXT ) ;
if ( ( error = xfs_trans_reserve ( tp , 0 ,
XFS_ICHANGE_LOG_RES ( mp ) , 0 ,
0 , 0 ) ) ) {
xfs_iunlock ( ip , XFS_IOLOCK_EXCL ) ;
xfs_iunlock ( tip , XFS_IOLOCK_EXCL ) ;
xfs_trans_cancel ( tp , 0 ) ;
2009-02-04 09:37:43 +01:00
goto out ;
2005-04-16 15:20:36 -07:00
}
2008-08-13 16:18:07 +10:00
xfs_lock_two_inodes ( ip , tip , XFS_ILOCK_EXCL ) ;
2005-04-16 15:20:36 -07:00
/*
* Count the number of extended attribute blocks
*/
if ( ( ( XFS_IFORK_Q ( ip ) ! = 0 ) & & ( ip - > i_d . di_anextents > 0 ) ) & &
( ip - > i_d . di_aformat ! = XFS_DINODE_FMT_LOCAL ) ) {
error = xfs_bmap_count_blocks ( tp , ip , XFS_ATTR_FORK , & aforkblks ) ;
2009-02-04 09:37:43 +01:00
if ( error )
goto out_trans_cancel ;
2005-04-16 15:20:36 -07:00
}
if ( ( ( XFS_IFORK_Q ( tip ) ! = 0 ) & & ( tip - > i_d . di_anextents > 0 ) ) & &
( tip - > i_d . di_aformat ! = XFS_DINODE_FMT_LOCAL ) ) {
error = xfs_bmap_count_blocks ( tp , tip , XFS_ATTR_FORK ,
& taforkblks ) ;
2009-02-04 09:37:43 +01:00
if ( error )
goto out_trans_cancel ;
2005-04-16 15:20:36 -07:00
}
/*
* Swap the data forks of the inodes
*/
ifp = & ip - > i_df ;
tifp = & tip - > i_df ;
2005-11-02 10:29:04 +11:00
* tempifp = * ifp ; /* struct copy */
* ifp = * tifp ; /* struct copy */
* tifp = * tempifp ; /* struct copy */
2005-04-16 15:20:36 -07:00
/*
* Fix the on - disk inode values
*/
tmp = ( __uint64_t ) ip - > i_d . di_nblocks ;
ip - > i_d . di_nblocks = tip - > i_d . di_nblocks - taforkblks + aforkblks ;
tip - > i_d . di_nblocks = tmp + taforkblks - aforkblks ;
tmp = ( __uint64_t ) ip - > i_d . di_nextents ;
ip - > i_d . di_nextents = tip - > i_d . di_nextents ;
tip - > i_d . di_nextents = tmp ;
tmp = ( __uint64_t ) ip - > i_d . di_format ;
ip - > i_d . di_format = tip - > i_d . di_format ;
tip - > i_d . di_format = tmp ;
ilf_fields = XFS_ILOG_CORE ;
switch ( ip - > i_d . di_format ) {
case XFS_DINODE_FMT_EXTENTS :
/* If the extents fit in the inode, fix the
* pointer . Otherwise it ' s already NULL or
* pointing to the extent .
*/
if ( ip - > i_d . di_nextents < = XFS_INLINE_EXTS ) {
ifp - > if_u1 . if_extents =
ifp - > if_u2 . if_inline_ext ;
}
ilf_fields | = XFS_ILOG_DEXT ;
break ;
case XFS_DINODE_FMT_BTREE :
ilf_fields | = XFS_ILOG_DBROOT ;
break ;
}
tilf_fields = XFS_ILOG_CORE ;
switch ( tip - > i_d . di_format ) {
case XFS_DINODE_FMT_EXTENTS :
/* If the extents fit in the inode, fix the
* pointer . Otherwise it ' s already NULL or
* pointing to the extent .
*/
if ( tip - > i_d . di_nextents < = XFS_INLINE_EXTS ) {
tifp - > if_u1 . if_extents =
tifp - > if_u2 . if_inline_ext ;
}
tilf_fields | = XFS_ILOG_DEXT ;
break ;
case XFS_DINODE_FMT_BTREE :
tilf_fields | = XFS_ILOG_DBROOT ;
break ;
}
2008-08-13 16:13:09 +10:00
IHOLD ( ip ) ;
2009-02-04 09:37:43 +01:00
xfs_trans_ijoin ( tp , ip , XFS_ILOCK_EXCL | XFS_IOLOCK_EXCL ) ;
2008-08-13 16:13:09 +10:00
IHOLD ( tip ) ;
2009-02-04 09:37:43 +01:00
xfs_trans_ijoin ( tp , tip , XFS_ILOCK_EXCL | XFS_IOLOCK_EXCL ) ;
2005-04-16 15:20:36 -07:00
xfs_trans_log_inode ( tp , ip , ilf_fields ) ;
xfs_trans_log_inode ( tp , tip , tilf_fields ) ;
/*
* If this is a synchronous mount , make sure that the
* transaction goes to disk before returning to the user .
*/
2009-02-04 09:37:43 +01:00
if ( mp - > m_flags & XFS_MOUNT_WSYNC )
2005-04-16 15:20:36 -07:00
xfs_trans_set_sync ( tp ) ;
2007-05-08 13:48:42 +10:00
error = xfs_trans_commit ( tp , XFS_TRANS_SWAPEXT ) ;
2005-04-16 15:20:36 -07:00
2009-02-04 09:37:43 +01:00
out :
kmem_free ( tempifp ) ;
2005-04-16 15:20:36 -07:00
return error ;
2009-02-04 09:37:43 +01:00
2009-05-07 19:49:45 -05:00
out_unlock :
xfs_iunlock ( ip , XFS_ILOCK_EXCL | XFS_IOLOCK_EXCL ) ;
xfs_iunlock ( tip , XFS_ILOCK_EXCL | XFS_IOLOCK_EXCL ) ;
goto out ;
2009-02-04 09:37:43 +01:00
out_trans_cancel :
xfs_trans_cancel ( tp , 0 ) ;
goto out_unlock ;
2005-04-16 15:20:36 -07:00
}