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/ioctl.h>
2009-01-19 02:03:03 +01:00
# include <linux/mount.h>
2005-04-16 15:20:36 -07:00
# include <asm/uaccess.h>
# include "xfs.h"
# include "xfs_fs.h"
2007-07-11 11:10:19 +10:00
# include "xfs_bit.h"
# include "xfs_log.h"
# include "xfs_inum.h"
# include "xfs_trans.h"
# include "xfs_sb.h"
# include "xfs_ag.h"
# include "xfs_dir2.h"
# include "xfs_dmapi.h"
# include "xfs_mount.h"
# include "xfs_bmap_btree.h"
# include "xfs_attr_sf.h"
# include "xfs_dir2_sf.h"
2005-04-16 15:20:36 -07:00
# include "xfs_vnode.h"
2007-07-11 11:10:19 +10:00
# include "xfs_dinode.h"
# include "xfs_inode.h"
# include "xfs_itable.h"
# include "xfs_error.h"
2005-04-16 15:20:36 -07:00
# include "xfs_dfrag.h"
2007-08-29 10:58:01 +10:00
# include "xfs_vnodeops.h"
2008-11-25 21:20:08 -06:00
# include "xfs_fsops.h"
2008-11-25 21:20:10 -06:00
# include "xfs_alloc.h"
# include "xfs_rtalloc.h"
2008-11-25 21:20:14 -06:00
# include "xfs_attr.h"
2008-11-25 21:20:08 -06:00
# include "xfs_ioctl.h"
2007-11-23 16:28:09 +11:00
# include "xfs_ioctl32.h"
2005-04-16 15:20:36 -07:00
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 ) )
2008-11-25 21:20:07 -06:00
# ifdef BROKEN_X86_ALIGNMENT
2008-11-25 21:20:08 -06:00
STATIC int
xfs_compat_flock64_copyin (
xfs_flock64_t * bf ,
compat_xfs_flock64_t __user * arg32 )
2005-09-05 08:25:06 +10:00
{
2008-11-25 21:20:08 -06:00
if ( get_user ( bf - > l_type , & arg32 - > l_type ) | |
get_user ( bf - > l_whence , & arg32 - > l_whence ) | |
get_user ( bf - > l_start , & arg32 - > l_start ) | |
get_user ( bf - > l_len , & arg32 - > l_len ) | |
get_user ( bf - > l_sysid , & arg32 - > l_sysid ) | |
get_user ( bf - > l_pid , & arg32 - > l_pid ) | |
copy_from_user ( bf - > l_pad , & arg32 - > l_pad , 4 * sizeof ( u32 ) ) )
return - XFS_ERROR ( EFAULT ) ;
return 0 ;
2005-09-05 08:25:06 +10:00
}
2008-11-25 21:20:08 -06:00
STATIC int
xfs_compat_ioc_fsgeometry_v1 (
struct xfs_mount * mp ,
compat_xfs_fsop_geom_v1_t __user * arg32 )
2007-07-11 11:09:57 +10:00
{
2008-11-25 21:20:08 -06:00
xfs_fsop_geom_t fsgeo ;
int error ;
2007-07-11 11:09:57 +10:00
2008-11-25 21:20:08 -06:00
error = xfs_fs_geometry ( mp , & fsgeo , 3 ) ;
if ( error )
return - error ;
/* The 32-bit variant simply has some padding at the end */
if ( copy_to_user ( arg32 , & fsgeo , sizeof ( struct compat_xfs_fsop_geom_v1 ) ) )
return - XFS_ERROR ( EFAULT ) ;
return 0 ;
2007-07-11 11:09:57 +10:00
}
2008-11-25 21:20:10 -06:00
STATIC int
xfs_compat_growfs_data_copyin (
struct xfs_growfs_data * in ,
compat_xfs_growfs_data_t __user * arg32 )
{
if ( get_user ( in - > newblocks , & arg32 - > newblocks ) | |
get_user ( in - > imaxpct , & arg32 - > imaxpct ) )
return - XFS_ERROR ( EFAULT ) ;
return 0 ;
}
STATIC int
xfs_compat_growfs_rt_copyin (
struct xfs_growfs_rt * in ,
compat_xfs_growfs_rt_t __user * arg32 )
{
if ( get_user ( in - > newblocks , & arg32 - > newblocks ) | |
get_user ( in - > extsize , & arg32 - > extsize ) )
return - XFS_ERROR ( EFAULT ) ;
return 0 ;
}
2008-11-25 21:20:08 -06:00
STATIC int
xfs_inumbers_fmt_compat (
void __user * ubuffer ,
const xfs_inogrp_t * buffer ,
long count ,
long * written )
2007-07-11 11:10:19 +10:00
{
2008-11-25 21:20:08 -06:00
compat_xfs_inogrp_t __user * p32 = ubuffer ;
long i ;
2007-07-11 11:10:19 +10:00
for ( i = 0 ; i < count ; i + + ) {
if ( put_user ( buffer [ i ] . xi_startino , & p32 [ i ] . xi_startino ) | |
put_user ( buffer [ i ] . xi_alloccount , & p32 [ i ] . xi_alloccount ) | |
put_user ( buffer [ i ] . xi_allocmask , & p32 [ i ] . xi_allocmask ) )
2008-11-25 21:20:08 -06:00
return - XFS_ERROR ( EFAULT ) ;
2007-07-11 11:10:19 +10:00
}
* written = count * sizeof ( * p32 ) ;
return 0 ;
}
2005-04-16 15:20:36 -07:00
# else
2007-07-11 11:10:19 +10:00
# define xfs_inumbers_fmt_compat xfs_inumbers_fmt
2008-11-25 21:20:17 -06:00
# endif /* BROKEN_X86_ALIGNMENT */
2007-07-11 11:10:19 +10:00
2008-11-25 21:20:09 -06:00
STATIC int
xfs_ioctl32_bstime_copyin (
xfs_bstime_t * bstime ,
compat_xfs_bstime_t __user * bstime32 )
{
compat_time_t sec32 ; /* tv_sec differs on 64 vs. 32 */
if ( get_user ( sec32 , & bstime32 - > tv_sec ) | |
get_user ( bstime - > tv_nsec , & bstime32 - > tv_nsec ) )
return - XFS_ERROR ( EFAULT ) ;
bstime - > tv_sec = sec32 ;
return 0 ;
}
/* xfs_bstat_t has differing alignment on intel, & bstime_t sizes everywhere */
STATIC int
xfs_ioctl32_bstat_copyin (
xfs_bstat_t * bstat ,
compat_xfs_bstat_t __user * bstat32 )
{
if ( get_user ( bstat - > bs_ino , & bstat32 - > bs_ino ) | |
get_user ( bstat - > bs_mode , & bstat32 - > bs_mode ) | |
get_user ( bstat - > bs_nlink , & bstat32 - > bs_nlink ) | |
get_user ( bstat - > bs_uid , & bstat32 - > bs_uid ) | |
get_user ( bstat - > bs_gid , & bstat32 - > bs_gid ) | |
get_user ( bstat - > bs_rdev , & bstat32 - > bs_rdev ) | |
get_user ( bstat - > bs_blksize , & bstat32 - > bs_blksize ) | |
get_user ( bstat - > bs_size , & bstat32 - > bs_size ) | |
xfs_ioctl32_bstime_copyin ( & bstat - > bs_atime , & bstat32 - > bs_atime ) | |
xfs_ioctl32_bstime_copyin ( & bstat - > bs_mtime , & bstat32 - > bs_mtime ) | |
xfs_ioctl32_bstime_copyin ( & bstat - > bs_ctime , & bstat32 - > bs_ctime ) | |
get_user ( bstat - > bs_blocks , & bstat32 - > bs_size ) | |
get_user ( bstat - > bs_xflags , & bstat32 - > bs_size ) | |
get_user ( bstat - > bs_extsize , & bstat32 - > bs_extsize ) | |
get_user ( bstat - > bs_extents , & bstat32 - > bs_extents ) | |
get_user ( bstat - > bs_gen , & bstat32 - > bs_gen ) | |
get_user ( bstat - > bs_projid , & bstat32 - > bs_projid ) | |
get_user ( bstat - > bs_dmevmask , & bstat32 - > bs_dmevmask ) | |
get_user ( bstat - > bs_dmstate , & bstat32 - > bs_dmstate ) | |
get_user ( bstat - > bs_aextents , & bstat32 - > bs_aextents ) )
return - XFS_ERROR ( EFAULT ) ;
return 0 ;
}
2007-07-11 11:10:19 +10:00
/* XFS_IOC_FSBULKSTAT and friends */
2008-11-25 21:20:08 -06:00
STATIC int
xfs_bstime_store_compat (
compat_xfs_bstime_t __user * p32 ,
const xfs_bstime_t * p )
2007-07-11 11:10:19 +10:00
{
2008-11-25 21:20:08 -06:00
__s32 sec32 ;
2007-07-11 11:10:19 +10:00
sec32 = p - > tv_sec ;
if ( put_user ( sec32 , & p32 - > tv_sec ) | |
put_user ( p - > tv_nsec , & p32 - > tv_nsec ) )
2008-11-25 21:20:08 -06:00
return - XFS_ERROR ( EFAULT ) ;
2007-07-11 11:10:19 +10:00
return 0 ;
}
2008-11-25 21:20:12 -06:00
/* Return 0 on success or positive error (to xfs_bulkstat()) */
2008-11-25 21:20:08 -06:00
STATIC int
xfs_bulkstat_one_fmt_compat (
2007-07-11 11:10:19 +10:00
void __user * ubuffer ,
2008-11-25 21:20:12 -06:00
int ubsize ,
int * ubused ,
2007-07-11 11:10:19 +10:00
const xfs_bstat_t * buffer )
{
2008-11-25 21:20:08 -06:00
compat_xfs_bstat_t __user * p32 = ubuffer ;
2007-07-11 11:10:19 +10:00
2008-11-25 21:20:12 -06:00
if ( ubsize < sizeof ( * p32 ) )
return XFS_ERROR ( ENOMEM ) ;
if ( put_user ( buffer - > bs_ino , & p32 - > bs_ino ) | |
put_user ( buffer - > bs_mode , & p32 - > bs_mode ) | |
put_user ( buffer - > bs_nlink , & p32 - > bs_nlink ) | |
put_user ( buffer - > bs_uid , & p32 - > bs_uid ) | |
put_user ( buffer - > bs_gid , & p32 - > bs_gid ) | |
put_user ( buffer - > bs_rdev , & p32 - > bs_rdev ) | |
put_user ( buffer - > bs_blksize , & p32 - > bs_blksize ) | |
put_user ( buffer - > bs_size , & p32 - > bs_size ) | |
2007-07-11 11:10:19 +10:00
xfs_bstime_store_compat ( & p32 - > bs_atime , & buffer - > bs_atime ) | |
xfs_bstime_store_compat ( & p32 - > bs_mtime , & buffer - > bs_mtime ) | |
xfs_bstime_store_compat ( & p32 - > bs_ctime , & buffer - > bs_ctime ) | |
2008-11-25 21:20:12 -06:00
put_user ( buffer - > bs_blocks , & p32 - > bs_blocks ) | |
put_user ( buffer - > bs_xflags , & p32 - > bs_xflags ) | |
put_user ( buffer - > bs_extsize , & p32 - > bs_extsize ) | |
put_user ( buffer - > bs_extents , & p32 - > bs_extents ) | |
put_user ( buffer - > bs_gen , & p32 - > bs_gen ) | |
put_user ( buffer - > bs_projid , & p32 - > bs_projid ) | |
put_user ( buffer - > bs_dmevmask , & p32 - > bs_dmevmask ) | |
put_user ( buffer - > bs_dmstate , & p32 - > bs_dmstate ) | |
2007-07-11 11:10:19 +10:00
put_user ( buffer - > bs_aextents , & p32 - > bs_aextents ) )
2008-11-25 21:20:12 -06:00
return XFS_ERROR ( EFAULT ) ;
if ( ubused )
* ubused = sizeof ( * p32 ) ;
return 0 ;
2007-07-11 11:10:19 +10:00
}
2008-11-25 21:20:11 -06:00
STATIC int
xfs_bulkstat_one_compat (
xfs_mount_t * mp , /* mount point for filesystem */
xfs_ino_t ino , /* inode number to get data for */
void __user * buffer , /* buffer to place output in */
int ubsize , /* size of buffer */
void * private_data , /* my private data */
xfs_daddr_t bno , /* starting bno of inode cluster */
int * ubused , /* bytes used by me */
void * dibuff , /* on-disk inode buffer */
int * stat ) /* BULKSTAT_RV_... */
{
return xfs_bulkstat_one_int ( mp , ino , buffer , ubsize ,
xfs_bulkstat_one_fmt_compat , bno ,
ubused , dibuff , stat ) ;
}
2007-07-11 11:10:19 +10:00
/* copied from xfs_ioctl.c */
STATIC int
2008-11-25 21:20:11 -06:00
xfs_compat_ioc_bulkstat (
xfs_mount_t * mp ,
unsigned int cmd ,
compat_xfs_fsop_bulkreq_t __user * p32 )
2005-04-16 15:20:36 -07:00
{
u32 addr ;
2007-07-11 11:10:19 +10:00
xfs_fsop_bulkreq_t bulkreq ;
int count ; /* # of records returned */
xfs_ino_t inlast ; /* last inode number */
int done ;
int error ;
/* done = 1 if there are more stats to get and if bulkstat */
/* should be called again (unused here, but used in dmapi) */
2005-04-16 15:20:36 -07:00
2007-07-11 11:10:19 +10:00
if ( ! capable ( CAP_SYS_ADMIN ) )
2008-11-25 21:20:08 -06:00
return - XFS_ERROR ( EPERM ) ;
2007-07-11 11:10:19 +10:00
if ( XFS_FORCED_SHUTDOWN ( mp ) )
return - XFS_ERROR ( EIO ) ;
if ( get_user ( addr , & p32 - > lastip ) )
2008-11-25 21:20:08 -06:00
return - XFS_ERROR ( EFAULT ) ;
2007-07-11 11:10:19 +10:00
bulkreq . lastip = compat_ptr ( addr ) ;
if ( get_user ( bulkreq . icount , & p32 - > icount ) | |
get_user ( addr , & p32 - > ubuffer ) )
2008-11-25 21:20:08 -06:00
return - XFS_ERROR ( EFAULT ) ;
2007-07-11 11:10:19 +10:00
bulkreq . ubuffer = compat_ptr ( addr ) ;
if ( get_user ( addr , & p32 - > ocount ) )
2008-11-25 21:20:08 -06:00
return - XFS_ERROR ( EFAULT ) ;
2007-07-11 11:10:19 +10:00
bulkreq . ocount = compat_ptr ( addr ) ;
2005-04-16 15:20:36 -07:00
2007-07-11 11:10:19 +10:00
if ( copy_from_user ( & inlast , bulkreq . lastip , sizeof ( __s64 ) ) )
return - XFS_ERROR ( EFAULT ) ;
if ( ( count = bulkreq . icount ) < = 0 )
return - XFS_ERROR ( EINVAL ) ;
2007-11-23 16:30:32 +11:00
if ( bulkreq . ubuffer = = NULL )
return - XFS_ERROR ( EINVAL ) ;
2008-11-25 21:20:13 -06:00
if ( cmd = = XFS_IOC_FSINUMBERS_32 ) {
2007-07-11 11:10:19 +10:00
error = xfs_inumbers ( mp , & inlast , & count ,
bulkreq . ubuffer , xfs_inumbers_fmt_compat ) ;
2008-11-25 21:20:13 -06:00
} else if ( cmd = = XFS_IOC_FSBULKSTAT_SINGLE_32 ) {
int res ;
error = xfs_bulkstat_one_compat ( mp , inlast , bulkreq . ubuffer ,
sizeof ( compat_xfs_bstat_t ) ,
NULL , 0 , NULL , NULL , & res ) ;
} else if ( cmd = = XFS_IOC_FSBULKSTAT_32 ) {
2007-07-11 11:10:19 +10:00
error = xfs_bulkstat ( mp , & inlast , & count ,
2008-11-25 21:20:11 -06:00
xfs_bulkstat_one_compat , NULL ,
2007-07-11 11:10:19 +10:00
sizeof ( compat_xfs_bstat_t ) , bulkreq . ubuffer ,
BULKSTAT_FG_QUICK , & done ) ;
2008-11-25 21:20:13 -06:00
} else
error = XFS_ERROR ( EINVAL ) ;
2007-07-11 11:10:19 +10:00
if ( error )
return - error ;
if ( bulkreq . ocount ! = NULL ) {
if ( copy_to_user ( bulkreq . lastip , & inlast ,
sizeof ( xfs_ino_t ) ) )
return - XFS_ERROR ( EFAULT ) ;
if ( copy_to_user ( bulkreq . ocount , & count , sizeof ( count ) ) )
return - XFS_ERROR ( EFAULT ) ;
}
return 0 ;
2005-04-16 15:20:36 -07:00
}
2007-07-11 11:10:19 +10:00
2008-11-25 21:20:08 -06:00
STATIC int
xfs_compat_handlereq_copyin (
xfs_fsop_handlereq_t * hreq ,
compat_xfs_fsop_handlereq_t __user * arg32 )
2007-07-11 11:10:09 +10:00
{
2008-11-25 21:20:08 -06:00
compat_xfs_fsop_handlereq_t hreq32 ;
if ( copy_from_user ( & hreq32 , arg32 , sizeof ( compat_xfs_fsop_handlereq_t ) ) )
return - XFS_ERROR ( EFAULT ) ;
hreq - > fd = hreq32 . fd ;
hreq - > path = compat_ptr ( hreq32 . path ) ;
hreq - > oflags = hreq32 . oflags ;
hreq - > ihandle = compat_ptr ( hreq32 . ihandle ) ;
hreq - > ihandlen = hreq32 . ihandlen ;
hreq - > ohandle = compat_ptr ( hreq32 . ohandle ) ;
hreq - > ohandlen = compat_ptr ( hreq32 . ohandlen ) ;
return 0 ;
2007-07-11 11:10:09 +10:00
}
2009-01-19 02:02:57 +01:00
STATIC struct dentry *
xfs_compat_handlereq_to_dentry (
struct file * parfilp ,
compat_xfs_fsop_handlereq_t * hreq )
2008-11-25 21:20:14 -06:00
{
2009-01-19 02:02:57 +01:00
return xfs_handle_to_dentry ( parfilp ,
compat_ptr ( hreq - > ihandle ) , hreq - > ihandlen ) ;
2008-11-25 21:20:14 -06:00
}
STATIC int
xfs_compat_attrlist_by_handle (
2009-01-19 02:02:57 +01:00
struct file * parfilp ,
void __user * arg )
2008-11-25 21:20:14 -06:00
{
int error ;
attrlist_cursor_kern_t * cursor ;
compat_xfs_fsop_attrlist_handlereq_t al_hreq ;
2009-01-19 02:02:57 +01:00
struct dentry * dentry ;
2008-11-25 21:20:14 -06:00
char * kbuf ;
if ( ! capable ( CAP_SYS_ADMIN ) )
return - XFS_ERROR ( EPERM ) ;
if ( copy_from_user ( & al_hreq , arg ,
sizeof ( compat_xfs_fsop_attrlist_handlereq_t ) ) )
return - XFS_ERROR ( EFAULT ) ;
if ( al_hreq . buflen > XATTR_LIST_MAX )
return - XFS_ERROR ( EINVAL ) ;
/*
* Reject flags , only allow namespaces .
*/
if ( al_hreq . flags & ~ ( ATTR_ROOT | ATTR_SECURE ) )
return - XFS_ERROR ( EINVAL ) ;
2009-01-19 02:02:57 +01:00
dentry = xfs_compat_handlereq_to_dentry ( parfilp , & al_hreq . hreq ) ;
if ( IS_ERR ( dentry ) )
return PTR_ERR ( dentry ) ;
2008-11-25 21:20:14 -06:00
2009-01-19 02:02:57 +01:00
error = - ENOMEM ;
2008-11-25 21:20:14 -06:00
kbuf = kmalloc ( al_hreq . buflen , GFP_KERNEL ) ;
if ( ! kbuf )
2009-01-19 02:02:57 +01:00
goto out_dput ;
2008-11-25 21:20:14 -06:00
cursor = ( attrlist_cursor_kern_t * ) & al_hreq . pos ;
2009-01-19 02:02:57 +01:00
error = - xfs_attr_list ( XFS_I ( dentry - > d_inode ) , kbuf , al_hreq . buflen ,
2008-11-25 21:20:14 -06:00
al_hreq . flags , cursor ) ;
if ( error )
goto out_kfree ;
if ( copy_to_user ( compat_ptr ( al_hreq . buffer ) , kbuf , al_hreq . buflen ) )
error = - EFAULT ;
out_kfree :
kfree ( kbuf ) ;
2009-01-19 02:02:57 +01:00
out_dput :
dput ( dentry ) ;
return error ;
2008-11-25 21:20:14 -06:00
}
2008-11-25 21:20:15 -06:00
STATIC int
xfs_compat_attrmulti_by_handle (
2009-01-19 02:02:57 +01:00
struct file * parfilp ,
void __user * arg )
2008-11-25 21:20:15 -06:00
{
int error ;
compat_xfs_attr_multiop_t * ops ;
compat_xfs_fsop_attrmulti_handlereq_t am_hreq ;
2009-01-19 02:02:57 +01:00
struct dentry * dentry ;
2008-11-25 21:20:15 -06:00
unsigned int i , size ;
char * attr_name ;
if ( ! capable ( CAP_SYS_ADMIN ) )
return - XFS_ERROR ( EPERM ) ;
if ( copy_from_user ( & am_hreq , arg ,
sizeof ( compat_xfs_fsop_attrmulti_handlereq_t ) ) )
return - XFS_ERROR ( EFAULT ) ;
2009-01-19 02:02:57 +01:00
dentry = xfs_compat_handlereq_to_dentry ( parfilp , & am_hreq . hreq ) ;
if ( IS_ERR ( dentry ) )
return PTR_ERR ( dentry ) ;
2008-11-25 21:20:15 -06:00
error = E2BIG ;
size = am_hreq . opcount * sizeof ( compat_xfs_attr_multiop_t ) ;
if ( ! size | | size > 16 * PAGE_SIZE )
2009-01-19 02:02:57 +01:00
goto out_dput ;
2008-11-25 21:20:15 -06:00
2009-04-08 15:08:04 +08:00
ops = memdup_user ( compat_ptr ( am_hreq . ops ) , size ) ;
if ( IS_ERR ( ops ) ) {
error = PTR_ERR ( ops ) ;
2009-01-19 02:02:57 +01:00
goto out_dput ;
2009-04-08 15:08:04 +08:00
}
2008-11-25 21:20:15 -06:00
attr_name = kmalloc ( MAXNAMELEN , GFP_KERNEL ) ;
if ( ! attr_name )
goto out_kfree_ops ;
error = 0 ;
for ( i = 0 ; i < am_hreq . opcount ; i + + ) {
ops [ i ] . am_error = strncpy_from_user ( attr_name ,
compat_ptr ( ops [ i ] . am_attrname ) ,
MAXNAMELEN ) ;
if ( ops [ i ] . am_error = = 0 | | ops [ i ] . am_error = = MAXNAMELEN )
error = - ERANGE ;
if ( ops [ i ] . am_error < 0 )
break ;
switch ( ops [ i ] . am_opcode ) {
case ATTR_OP_GET :
2009-01-19 02:02:57 +01:00
ops [ i ] . am_error = xfs_attrmulti_attr_get (
dentry - > d_inode , attr_name ,
2008-11-25 21:20:15 -06:00
compat_ptr ( ops [ i ] . am_attrvalue ) ,
& ops [ i ] . am_length , ops [ i ] . am_flags ) ;
break ;
case ATTR_OP_SET :
2009-01-19 02:03:03 +01:00
ops [ i ] . am_error = mnt_want_write ( parfilp - > f_path . mnt ) ;
if ( ops [ i ] . am_error )
break ;
2009-01-19 02:02:57 +01:00
ops [ i ] . am_error = xfs_attrmulti_attr_set (
dentry - > d_inode , attr_name ,
2008-11-25 21:20:15 -06:00
compat_ptr ( ops [ i ] . am_attrvalue ) ,
ops [ i ] . am_length , ops [ i ] . am_flags ) ;
2009-01-19 02:03:03 +01:00
mnt_drop_write ( parfilp - > f_path . mnt ) ;
2008-11-25 21:20:15 -06:00
break ;
case ATTR_OP_REMOVE :
2009-01-19 02:03:03 +01:00
ops [ i ] . am_error = mnt_want_write ( parfilp - > f_path . mnt ) ;
if ( ops [ i ] . am_error )
break ;
2009-01-19 02:02:57 +01:00
ops [ i ] . am_error = xfs_attrmulti_attr_remove (
dentry - > d_inode , attr_name ,
ops [ i ] . am_flags ) ;
2009-01-19 02:03:03 +01:00
mnt_drop_write ( parfilp - > f_path . mnt ) ;
2008-11-25 21:20:15 -06:00
break ;
default :
ops [ i ] . am_error = EINVAL ;
}
}
if ( copy_to_user ( compat_ptr ( am_hreq . ops ) , ops , size ) )
error = XFS_ERROR ( EFAULT ) ;
kfree ( attr_name ) ;
out_kfree_ops :
kfree ( ops ) ;
2009-01-19 02:02:57 +01:00
out_dput :
dput ( dentry ) ;
2008-11-25 21:20:15 -06:00
return - error ;
}
2008-11-25 21:20:16 -06:00
STATIC int
xfs_compat_fssetdm_by_handle (
2009-01-19 02:02:57 +01:00
struct file * parfilp ,
void __user * arg )
2008-11-25 21:20:16 -06:00
{
int error ;
struct fsdmidata fsd ;
compat_xfs_fsop_setdm_handlereq_t dmhreq ;
2009-01-19 02:02:57 +01:00
struct dentry * dentry ;
2008-11-25 21:20:16 -06:00
if ( ! capable ( CAP_MKNOD ) )
return - XFS_ERROR ( EPERM ) ;
if ( copy_from_user ( & dmhreq , arg ,
sizeof ( compat_xfs_fsop_setdm_handlereq_t ) ) )
return - XFS_ERROR ( EFAULT ) ;
2009-01-19 02:02:57 +01:00
dentry = xfs_compat_handlereq_to_dentry ( parfilp , & dmhreq . hreq ) ;
if ( IS_ERR ( dentry ) )
return PTR_ERR ( dentry ) ;
2008-11-25 21:20:16 -06:00
2009-01-19 02:02:57 +01:00
if ( IS_IMMUTABLE ( dentry - > d_inode ) | | IS_APPEND ( dentry - > d_inode ) ) {
2008-11-25 21:20:16 -06:00
error = - XFS_ERROR ( EPERM ) ;
goto out ;
}
if ( copy_from_user ( & fsd , compat_ptr ( dmhreq . data ) , sizeof ( fsd ) ) ) {
error = - XFS_ERROR ( EFAULT ) ;
goto out ;
}
2009-01-19 02:02:57 +01:00
error = - xfs_set_dmattrs ( XFS_I ( dentry - > d_inode ) , fsd . fsd_dmevmask ,
2008-11-25 21:20:16 -06:00
fsd . fsd_dmstate ) ;
out :
2009-01-19 02:02:57 +01:00
dput ( dentry ) ;
2008-11-25 21:20:16 -06:00
return error ;
}
2008-12-09 04:47:33 -05:00
long
xfs_file_compat_ioctl (
struct file * filp ,
unsigned cmd ,
unsigned long p )
2005-04-16 15:20:36 -07:00
{
2008-12-09 04:47:33 -05:00
struct inode * inode = filp - > f_path . dentry - > d_inode ;
struct xfs_inode * ip = XFS_I ( inode ) ;
struct xfs_mount * mp = ip - > i_mount ;
void __user * arg = ( void __user * ) p ;
int ioflags = 0 ;
int error ;
if ( filp - > f_mode & FMODE_NOCMTIME )
ioflags | = IO_INVIS ;
xfs_itrace_entry ( ip ) ;
2005-04-16 15:20:36 -07:00
switch ( cmd ) {
2008-11-25 21:20:17 -06:00
/* No size or alignment issues on any arch */
2005-04-16 15:20:36 -07:00
case XFS_IOC_DIOINFO :
case XFS_IOC_FSGEOMETRY :
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 :
case XFS_IOC_FSCOUNTS :
case XFS_IOC_SET_RESBLKS :
case XFS_IOC_GET_RESBLKS :
case XFS_IOC_FSGROWFSLOG :
case XFS_IOC_GOINGDOWN :
case XFS_IOC_ERROR_INJECTION :
case XFS_IOC_ERROR_CLEARALL :
2008-12-09 04:47:33 -05:00
return xfs_file_ioctl ( filp , cmd , p ) ;
2008-11-25 21:20:17 -06:00
# ifndef BROKEN_X86_ALIGNMENT
/* These are handled fine if no alignment issues */
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 :
case XFS_IOC_FSGEOMETRY_V1 :
case XFS_IOC_FSGROWFSDATA :
case XFS_IOC_FSGROWFSRT :
2008-12-09 04:47:33 -05:00
return xfs_file_ioctl ( filp , cmd , p ) ;
2008-11-25 21:20:17 -06:00
# else
2005-09-05 08:25:06 +10:00
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 :
2008-11-25 21:20:08 -06:00
case XFS_IOC_UNRESVSP64_32 : {
struct xfs_flock64 bf ;
if ( xfs_compat_flock64_copyin ( & bf , arg ) )
return - XFS_ERROR ( EFAULT ) ;
2005-09-05 08:25:06 +10:00
cmd = _NATIVE_IOC ( cmd , struct xfs_flock64 ) ;
2008-11-25 21:20:08 -06:00
return xfs_ioc_space ( ip , inode , filp , ioflags , cmd , & bf ) ;
}
2007-07-11 11:09:57 +10:00
case XFS_IOC_FSGEOMETRY_V1_32 :
2008-11-25 21:20:08 -06:00
return xfs_compat_ioc_fsgeometry_v1 ( mp , arg ) ;
2008-11-25 21:20:10 -06:00
case XFS_IOC_FSGROWFSDATA_32 : {
struct xfs_growfs_data in ;
if ( xfs_compat_growfs_data_copyin ( & in , arg ) )
return - XFS_ERROR ( EFAULT ) ;
error = xfs_growfs_data ( mp , & in ) ;
return - error ;
}
case XFS_IOC_FSGROWFSRT_32 : {
struct xfs_growfs_rt in ;
if ( xfs_compat_growfs_rt_copyin ( & in , arg ) )
return - XFS_ERROR ( EFAULT ) ;
error = xfs_growfs_rt ( mp , & in ) ;
return - error ;
}
2005-04-16 15:20:36 -07:00
# endif
2008-11-25 21:20:17 -06:00
/* long changes size, but xfs only copiese out 32 bits */
case XFS_IOC_GETXFLAGS_32 :
case XFS_IOC_SETXFLAGS_32 :
case XFS_IOC_GETVERSION_32 :
cmd = _NATIVE_IOC ( cmd , long ) ;
2008-12-09 04:47:33 -05:00
return xfs_file_ioctl ( filp , cmd , p ) ;
2008-11-25 21:20:17 -06:00
case XFS_IOC_SWAPEXT : {
struct xfs_swapext sxp ;
struct compat_xfs_swapext __user * sxu = arg ;
/* Bulk copy in up to the sx_stat field, then copy bstat */
if ( copy_from_user ( & sxp , sxu ,
offsetof ( struct xfs_swapext , sx_stat ) ) | |
xfs_ioctl32_bstat_copyin ( & sxp . sx_stat , & sxu - > sx_stat ) )
return - XFS_ERROR ( EFAULT ) ;
error = xfs_swapext ( & sxp ) ;
return - error ;
}
2007-07-11 11:10:19 +10:00
case XFS_IOC_FSBULKSTAT_32 :
case XFS_IOC_FSBULKSTAT_SINGLE_32 :
case XFS_IOC_FSINUMBERS_32 :
2008-11-25 21:20:11 -06:00
return xfs_compat_ioc_bulkstat ( mp , cmd , arg ) ;
2007-07-11 11:10:09 +10:00
case XFS_IOC_FD_TO_HANDLE_32 :
case XFS_IOC_PATH_TO_HANDLE_32 :
2008-11-25 21:20:08 -06:00
case XFS_IOC_PATH_TO_FSHANDLE_32 : {
struct xfs_fsop_handlereq hreq ;
if ( xfs_compat_handlereq_copyin ( & hreq , arg ) )
return - XFS_ERROR ( EFAULT ) ;
2007-07-11 11:10:09 +10:00
cmd = _NATIVE_IOC ( cmd , struct xfs_fsop_handlereq ) ;
2008-11-25 21:20:08 -06:00
return xfs_find_handle ( cmd , & hreq ) ;
}
case XFS_IOC_OPEN_BY_HANDLE_32 : {
struct xfs_fsop_handlereq hreq ;
if ( xfs_compat_handlereq_copyin ( & hreq , arg ) )
return - XFS_ERROR ( EFAULT ) ;
2009-01-19 02:02:57 +01:00
return xfs_open_by_handle ( filp , & hreq ) ;
2008-11-25 21:20:08 -06:00
}
case XFS_IOC_READLINK_BY_HANDLE_32 : {
struct xfs_fsop_handlereq hreq ;
if ( xfs_compat_handlereq_copyin ( & hreq , arg ) )
return - XFS_ERROR ( EFAULT ) ;
2009-01-19 02:02:57 +01:00
return xfs_readlink_by_handle ( filp , & hreq ) ;
2008-11-25 21:20:08 -06:00
}
2008-11-25 21:20:14 -06:00
case XFS_IOC_ATTRLIST_BY_HANDLE_32 :
2009-01-19 02:02:57 +01:00
return xfs_compat_attrlist_by_handle ( filp , arg ) ;
2008-11-25 21:20:15 -06:00
case XFS_IOC_ATTRMULTI_BY_HANDLE_32 :
2009-01-19 02:02:57 +01:00
return xfs_compat_attrmulti_by_handle ( filp , arg ) ;
2008-11-25 21:20:16 -06:00
case XFS_IOC_FSSETDM_BY_HANDLE_32 :
2009-01-19 02:02:57 +01:00
return xfs_compat_fssetdm_by_handle ( filp , arg ) ;
2005-04-16 15:20:36 -07:00
default :
2008-11-25 21:20:08 -06:00
return - XFS_ERROR ( ENOIOCTLCMD ) ;
2005-04-16 15:20:36 -07:00
}
}