2005-04-16 15:20:36 -07:00
/*
2005-11-02 14:58:39 +11:00
* Copyright ( c ) 2004 - 2005 Silicon Graphics , Inc .
* 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 <linux/compat.h>
# include <linux/init.h>
# include <linux/ioctl.h>
# include <linux/syscalls.h>
# include <linux/types.h>
# include <linux/fs.h>
# include <asm/uaccess.h>
# include "xfs.h"
# include "xfs_types.h"
# include "xfs_fs.h"
# include "xfs_vfs.h"
# include "xfs_vnode.h"
# include "xfs_dfrag.h"
2005-09-05 08:25:06 +10:00
# define _NATIVE_IOC(cmd, type) \
_IOC ( _IOC_DIR ( cmd ) , _IOC_TYPE ( cmd ) , _IOC_NR ( cmd ) , sizeof ( type ) )
2005-04-16 15:20:36 -07:00
# if defined(CONFIG_IA64) || defined(CONFIG_X86_64)
# define BROKEN_X86_ALIGNMENT
2005-09-05 08:25:06 +10:00
/* on ia32 l_start is on a 32-bit boundary */
typedef struct xfs_flock64_32 {
__s16 l_type ;
__s16 l_whence ;
__s64 l_start __attribute__ ( ( packed ) ) ;
/* len == 0 means until end of file */
__s64 l_len __attribute__ ( ( packed ) ) ;
__s32 l_sysid ;
__u32 l_pid ;
__s32 l_pad [ 4 ] ; /* reserve area */
} xfs_flock64_32_t ;
# define XFS_IOC_ALLOCSP_32 _IOW ('X', 10, struct xfs_flock64_32)
# define XFS_IOC_FREESP_32 _IOW ('X', 11, struct xfs_flock64_32)
# define XFS_IOC_ALLOCSP64_32 _IOW ('X', 36, struct xfs_flock64_32)
# define XFS_IOC_FREESP64_32 _IOW ('X', 37, struct xfs_flock64_32)
# define XFS_IOC_RESVSP_32 _IOW ('X', 40, struct xfs_flock64_32)
# define XFS_IOC_UNRESVSP_32 _IOW ('X', 41, struct xfs_flock64_32)
# define XFS_IOC_RESVSP64_32 _IOW ('X', 42, struct xfs_flock64_32)
# define XFS_IOC_UNRESVSP64_32 _IOW ('X', 43, struct xfs_flock64_32)
/* just account for different alignment */
STATIC unsigned long
xfs_ioctl32_flock (
unsigned long arg )
{
xfs_flock64_32_t __user * p32 = ( void __user * ) arg ;
xfs_flock64_t __user * p = compat_alloc_user_space ( sizeof ( * p ) ) ;
if ( copy_in_user ( & p - > l_type , & p32 - > l_type , sizeof ( s16 ) ) | |
copy_in_user ( & p - > l_whence , & p32 - > l_whence , sizeof ( s16 ) ) | |
copy_in_user ( & p - > l_start , & p32 - > l_start , sizeof ( s64 ) ) | |
copy_in_user ( & p - > l_len , & p32 - > l_len , sizeof ( s64 ) ) | |
copy_in_user ( & p - > l_sysid , & p32 - > l_sysid , sizeof ( s32 ) ) | |
copy_in_user ( & p - > l_pid , & p32 - > l_pid , sizeof ( u32 ) ) | |
copy_in_user ( & p - > l_pad , & p32 - > l_pad , 4 * sizeof ( u32 ) ) )
return - EFAULT ;
2006-03-29 08:55:47 +10:00
2005-09-05 08:25:06 +10:00
return ( unsigned long ) p ;
}
2007-07-11 11:09:57 +10:00
typedef struct compat_xfs_fsop_geom_v1 {
__u32 blocksize ; /* filesystem (data) block size */
__u32 rtextsize ; /* realtime extent size */
__u32 agblocks ; /* fsblocks in an AG */
__u32 agcount ; /* number of allocation groups */
__u32 logblocks ; /* fsblocks in the log */
__u32 sectsize ; /* (data) sector size, bytes */
__u32 inodesize ; /* inode size in bytes */
__u32 imaxpct ; /* max allowed inode space(%) */
__u64 datablocks ; /* fsblocks in data subvolume */
__u64 rtblocks ; /* fsblocks in realtime subvol */
__u64 rtextents ; /* rt extents in realtime subvol*/
__u64 logstart ; /* starting fsblock of the log */
unsigned char uuid [ 16 ] ; /* unique id of the filesystem */
__u32 sunit ; /* stripe unit, fsblocks */
__u32 swidth ; /* stripe width, fsblocks */
__s32 version ; /* structure version */
__u32 flags ; /* superblock version flags */
__u32 logsectsize ; /* log sector size, bytes */
__u32 rtsectsize ; /* realtime sector size, bytes */
__u32 dirblocksize ; /* directory block size, bytes */
} __attribute__ ( ( packed ) ) compat_xfs_fsop_geom_v1_t ;
# define XFS_IOC_FSGEOMETRY_V1_32 \
_IOR ( ' X ' , 100 , struct compat_xfs_fsop_geom_v1 )
STATIC unsigned long xfs_ioctl32_geom_v1 ( unsigned long arg )
{
compat_xfs_fsop_geom_v1_t __user * p32 = ( void __user * ) arg ;
xfs_fsop_geom_v1_t __user * p = compat_alloc_user_space ( sizeof ( * p ) ) ;
if ( copy_in_user ( p , p32 , sizeof ( * p32 ) ) )
return - EFAULT ;
return ( unsigned long ) p ;
}
2005-04-16 15:20:36 -07:00
# else
typedef struct xfs_fsop_bulkreq32 {
compat_uptr_t lastip ; /* last inode # pointer */
__s32 icount ; /* count of entries in buffer */
compat_uptr_t ubuffer ; /* user buffer for inode desc. */
__s32 ocount ; /* output count pointer */
} xfs_fsop_bulkreq32_t ;
2005-05-06 06:44:46 -07:00
STATIC unsigned long
xfs_ioctl32_bulkstat (
unsigned long arg )
2005-04-16 15:20:36 -07:00
{
xfs_fsop_bulkreq32_t __user * p32 = ( void __user * ) arg ;
xfs_fsop_bulkreq_t __user * p = compat_alloc_user_space ( sizeof ( * p ) ) ;
u32 addr ;
if ( get_user ( addr , & p32 - > lastip ) | |
put_user ( compat_ptr ( addr ) , & p - > lastip ) | |
copy_in_user ( & p - > icount , & p32 - > icount , sizeof ( s32 ) ) | |
get_user ( addr , & p32 - > ubuffer ) | |
put_user ( compat_ptr ( addr ) , & p - > ubuffer ) | |
get_user ( addr , & p32 - > ocount ) | |
put_user ( compat_ptr ( addr ) , & p - > ocount ) )
return - EFAULT ;
return ( unsigned long ) p ;
}
# endif
2007-07-11 11:10:09 +10:00
typedef struct compat_xfs_fsop_handlereq {
__u32 fd ; /* fd for FD_TO_HANDLE */
compat_uptr_t path ; /* user pathname */
__u32 oflags ; /* open flags */
compat_uptr_t ihandle ; /* user supplied handle */
__u32 ihandlen ; /* user supplied length */
compat_uptr_t ohandle ; /* user buffer for handle */
compat_uptr_t ohandlen ; /* user buffer length */
} compat_xfs_fsop_handlereq_t ;
# define XFS_IOC_PATH_TO_FSHANDLE_32 \
_IOWR ( ' X ' , 104 , struct compat_xfs_fsop_handlereq )
# define XFS_IOC_PATH_TO_HANDLE_32 \
_IOWR ( ' X ' , 105 , struct compat_xfs_fsop_handlereq )
# define XFS_IOC_FD_TO_HANDLE_32 \
_IOWR ( ' X ' , 106 , struct compat_xfs_fsop_handlereq )
# define XFS_IOC_OPEN_BY_HANDLE_32 \
_IOWR ( ' X ' , 107 , struct compat_xfs_fsop_handlereq )
# define XFS_IOC_READLINK_BY_HANDLE_32 \
_IOWR ( ' X ' , 108 , struct compat_xfs_fsop_handlereq )
STATIC unsigned long xfs_ioctl32_fshandle ( unsigned long arg )
{
compat_xfs_fsop_handlereq_t __user * p32 = ( void __user * ) arg ;
xfs_fsop_handlereq_t __user * p = compat_alloc_user_space ( sizeof ( * p ) ) ;
u32 addr ;
if ( copy_in_user ( & p - > fd , & p32 - > fd , sizeof ( __u32 ) ) | |
get_user ( addr , & p32 - > path ) | |
put_user ( compat_ptr ( addr ) , & p - > path ) | |
copy_in_user ( & p - > oflags , & p32 - > oflags , sizeof ( __u32 ) ) | |
get_user ( addr , & p32 - > ihandle ) | |
put_user ( compat_ptr ( addr ) , & p - > ihandle ) | |
copy_in_user ( & p - > ihandlen , & p32 - > ihandlen , sizeof ( __u32 ) ) | |
get_user ( addr , & p32 - > ohandle ) | |
put_user ( compat_ptr ( addr ) , & p - > ohandle ) | |
get_user ( addr , & p32 - > ohandlen ) | |
put_user ( compat_ptr ( addr ) , & p - > ohandlen ) )
return - EFAULT ;
return ( unsigned long ) p ;
}
2005-05-06 06:44:46 -07:00
STATIC long
2006-03-29 08:55:47 +10:00
xfs_compat_ioctl (
int mode ,
struct file * file ,
unsigned cmd ,
unsigned long arg )
2005-04-16 15:20:36 -07:00
{
2006-12-08 02:36:49 -08:00
struct inode * inode = file - > f_path . dentry - > d_inode ;
2006-06-09 17:00:52 +10:00
bhv_vnode_t * vp = vn_from_inode ( inode ) ;
2005-04-16 15:20:36 -07:00
int error ;
switch ( cmd ) {
case XFS_IOC_DIOINFO :
case XFS_IOC_FSGEOMETRY :
case XFS_IOC_GETVERSION :
case XFS_IOC_GETXFLAGS :
case XFS_IOC_SETXFLAGS :
case XFS_IOC_FSGETXATTR :
case XFS_IOC_FSSETXATTR :
case XFS_IOC_FSGETXATTRA :
case XFS_IOC_FSSETDM :
case XFS_IOC_GETBMAP :
case XFS_IOC_GETBMAPA :
case XFS_IOC_GETBMAPX :
/* not handled
case XFS_IOC_FSSETDM_BY_HANDLE :
case XFS_IOC_ATTRLIST_BY_HANDLE :
case XFS_IOC_ATTRMULTI_BY_HANDLE :
*/
case XFS_IOC_FSCOUNTS :
case XFS_IOC_SET_RESBLKS :
case XFS_IOC_GET_RESBLKS :
case XFS_IOC_FSGROWFSDATA :
case XFS_IOC_FSGROWFSLOG :
case XFS_IOC_FSGROWFSRT :
case XFS_IOC_FREEZE :
case XFS_IOC_THAW :
case XFS_IOC_GOINGDOWN :
case XFS_IOC_ERROR_INJECTION :
case XFS_IOC_ERROR_CLEARALL :
break ;
2005-09-05 08:25:06 +10:00
# ifdef BROKEN_X86_ALIGNMENT
/* xfs_flock_t has wrong u32 vs u64 alignment */
case XFS_IOC_ALLOCSP_32 :
case XFS_IOC_FREESP_32 :
case XFS_IOC_ALLOCSP64_32 :
case XFS_IOC_FREESP64_32 :
case XFS_IOC_RESVSP_32 :
case XFS_IOC_UNRESVSP_32 :
case XFS_IOC_RESVSP64_32 :
case XFS_IOC_UNRESVSP64_32 :
arg = xfs_ioctl32_flock ( arg ) ;
cmd = _NATIVE_IOC ( cmd , struct xfs_flock64 ) ;
break ;
2007-07-11 11:09:57 +10:00
case XFS_IOC_FSGEOMETRY_V1_32 :
arg = xfs_ioctl32_geom_v1 ( arg ) ;
cmd = _NATIVE_IOC ( cmd , struct xfs_fsop_geom_v1 ) ;
break ;
2005-09-05 08:25:06 +10:00
# else /* These are handled fine if no alignment issues */
2005-04-16 15:20:36 -07:00
case XFS_IOC_ALLOCSP :
case XFS_IOC_FREESP :
case XFS_IOC_RESVSP :
case XFS_IOC_UNRESVSP :
case XFS_IOC_ALLOCSP64 :
case XFS_IOC_FREESP64 :
case XFS_IOC_RESVSP64 :
case XFS_IOC_UNRESVSP64 :
2007-07-11 11:09:57 +10:00
case XFS_IOC_FSGEOMETRY_V1 :
2005-09-05 08:25:06 +10:00
break ;
/* xfs_bstat_t still has wrong u32 vs u64 alignment */
2005-04-16 15:20:36 -07:00
case XFS_IOC_SWAPEXT :
break ;
case XFS_IOC_FSBULKSTAT_SINGLE :
case XFS_IOC_FSBULKSTAT :
case XFS_IOC_FSINUMBERS :
arg = xfs_ioctl32_bulkstat ( arg ) ;
break ;
# endif
2007-07-11 11:10:09 +10:00
case XFS_IOC_FD_TO_HANDLE_32 :
case XFS_IOC_PATH_TO_HANDLE_32 :
case XFS_IOC_PATH_TO_FSHANDLE_32 :
case XFS_IOC_OPEN_BY_HANDLE_32 :
case XFS_IOC_READLINK_BY_HANDLE_32 :
arg = xfs_ioctl32_fshandle ( arg ) ;
cmd = _NATIVE_IOC ( cmd , struct xfs_fsop_handlereq ) ;
break ;
2005-04-16 15:20:36 -07:00
default :
return - ENOIOCTLCMD ;
}
2006-06-09 17:00:52 +10:00
error = bhv_vop_ioctl ( vp , inode , file , mode , cmd , ( void __user * ) arg ) ;
2005-04-16 15:20:36 -07:00
VMODIFY ( vp ) ;
return error ;
}
2005-05-06 06:44:46 -07:00
long
2006-03-14 14:06:18 +11:00
xfs_file_compat_ioctl (
2006-03-29 08:55:47 +10:00
struct file * file ,
2005-05-06 06:44:46 -07:00
unsigned cmd ,
unsigned long arg )
2005-04-16 15:20:36 -07:00
{
2006-03-29 08:55:47 +10:00
return xfs_compat_ioctl ( 0 , file , cmd , arg ) ;
2005-04-16 15:20:36 -07:00
}
2005-05-06 06:44:46 -07:00
long
2006-03-14 14:06:18 +11:00
xfs_file_compat_invis_ioctl (
2006-03-29 08:55:47 +10:00
struct file * file ,
2005-05-06 06:44:46 -07:00
unsigned cmd ,
unsigned long arg )
2005-04-16 15:20:36 -07:00
{
2006-03-29 08:55:47 +10:00
return xfs_compat_ioctl ( IO_INVIS , file , cmd , arg ) ;
2005-04-16 15:20:36 -07:00
}