2005-04-16 15:20:36 -07:00
/*
* fs / cifs / ioctl . c
*
* vfs operations that deal with io control
*
2013-10-14 15:31:32 -05:00
* Copyright ( C ) International Business Machines Corp . , 2005 , 2013
2005-04-16 15:20:36 -07:00
* Author ( s ) : Steve French ( sfrench @ us . ibm . com )
*
* This library is free software ; you can redistribute it and / or modify
* it under the terms of the GNU Lesser General Public License as published
* by the Free Software Foundation ; either version 2.1 of the License , or
* ( at your option ) any later version .
*
* This library is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See
* the GNU Lesser General Public License for more details .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this library ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
2005-04-28 22:41:04 -07:00
2005-04-16 15:20:36 -07:00
# include <linux/fs.h>
2013-11-14 00:05:36 -06:00
# include <linux/file.h>
# include <linux/mount.h>
# include <linux/mm.h>
# include <linux/pagemap.h>
2005-04-16 15:20:36 -07:00
# include "cifspdu.h"
# include "cifsglob.h"
# include "cifsproto.h"
# include "cifs_debug.h"
2005-04-28 22:41:04 -07:00
# include "cifsfs.h"
2015-07-04 18:40:10 -05:00
# include "cifs_ioctl.h"
2018-10-17 05:47:58 +10:00
# include "smb2proto.h"
2015-06-27 21:18:36 -07:00
# include <linux/btrfs.h>
2005-04-16 15:20:36 -07:00
2018-10-07 19:19:58 -05:00
static long cifs_ioctl_query_info ( unsigned int xid , struct file * filep ,
unsigned long p )
{
2018-10-17 05:47:58 +10:00
struct inode * inode = file_inode ( filep ) ;
struct cifs_sb_info * cifs_sb = CIFS_SB ( inode - > i_sb ) ;
struct cifs_tcon * tcon = cifs_sb_master_tcon ( cifs_sb ) ;
struct dentry * dentry = filep - > f_path . dentry ;
2021-03-18 15:47:35 -04:00
const unsigned char * path ;
2021-03-05 17:36:04 -05:00
void * page = alloc_dentry_path ( ) ;
2018-10-17 05:47:58 +10:00
__le16 * utf16_path = NULL , root_path ;
int rc = 0 ;
2018-10-07 19:19:58 -05:00
2021-03-05 17:36:04 -05:00
path = build_path_from_dentry ( dentry , page ) ;
if ( IS_ERR ( path ) ) {
free_dentry_path ( page ) ;
return PTR_ERR ( path ) ;
}
2018-10-17 05:47:58 +10:00
cifs_dbg ( FYI , " %s %s \n " , __func__ , path ) ;
if ( ! path [ 0 ] ) {
root_path = 0 ;
utf16_path = & root_path ;
} else {
utf16_path = cifs_convert_path_to_utf16 ( path + 1 , cifs_sb ) ;
if ( ! utf16_path ) {
rc = - ENOMEM ;
goto ici_exit ;
}
}
2018-10-07 19:19:58 -05:00
if ( tcon - > ses - > server - > ops - > ioctl_query_info )
2018-10-17 05:47:58 +10:00
rc = tcon - > ses - > server - > ops - > ioctl_query_info (
2020-02-03 21:46:43 +02:00
xid , tcon , cifs_sb , utf16_path ,
2018-10-17 05:47:58 +10:00
filep - > private_data ? 0 : 1 , p ) ;
2018-10-07 19:19:58 -05:00
else
2018-10-17 05:47:58 +10:00
rc = - EOPNOTSUPP ;
ici_exit :
if ( utf16_path ! = & root_path )
kfree ( utf16_path ) ;
2021-03-05 17:36:04 -05:00
free_dentry_path ( page ) ;
2018-10-17 05:47:58 +10:00
return rc ;
2018-10-07 19:19:58 -05:00
}
2017-04-04 02:12:04 -05:00
static long cifs_ioctl_copychunk ( unsigned int xid , struct file * dst_file ,
2015-12-03 12:59:50 +01:00
unsigned long srcfd )
{
int rc ;
struct fd src_file ;
struct inode * src_inode ;
2017-04-04 02:12:04 -05:00
cifs_dbg ( FYI , " ioctl copychunk range \n " ) ;
2015-12-03 12:59:50 +01:00
/* the destination must be opened for writing */
if ( ! ( dst_file - > f_mode & FMODE_WRITE ) ) {
cifs_dbg ( FYI , " file target not open for write \n " ) ;
return - EINVAL ;
}
/* check if target volume is readonly and take reference */
rc = mnt_want_write_file ( dst_file ) ;
if ( rc ) {
cifs_dbg ( FYI , " mnt_want_write failed with rc %d \n " , rc ) ;
return rc ;
}
src_file = fdget ( srcfd ) ;
if ( ! src_file . file ) {
rc = - EBADF ;
goto out_drop_write ;
}
if ( src_file . file - > f_op - > unlocked_ioctl ! = cifs_ioctl ) {
rc = - EBADF ;
cifs_dbg ( VFS , " src file seems to be from a different filesystem type \n " ) ;
goto out_fput ;
}
src_inode = file_inode ( src_file . file ) ;
rc = - EINVAL ;
if ( S_ISDIR ( src_inode - > i_mode ) )
goto out_fput ;
2017-02-10 16:03:51 +05:30
rc = cifs_file_copychunk_range ( xid , src_file . file , 0 , dst_file , 0 ,
src_inode - > i_size , 0 ) ;
2017-04-26 17:10:17 +01:00
if ( rc > 0 )
rc = 0 ;
2013-11-14 00:05:36 -06:00
out_fput :
fdput ( src_file ) ;
out_drop_write :
mnt_drop_write_file ( dst_file ) ;
return rc ;
}
2015-07-04 18:40:10 -05:00
static long smb_mnt_get_fsinfo ( unsigned int xid , struct cifs_tcon * tcon ,
void __user * arg )
{
int rc = 0 ;
struct smb_mnt_fs_info * fsinf ;
fsinf = kzalloc ( sizeof ( struct smb_mnt_fs_info ) , GFP_KERNEL ) ;
if ( fsinf = = NULL )
return - ENOMEM ;
fsinf - > version = 1 ;
fsinf - > protocol_id = tcon - > ses - > server - > vals - > protocol_id ;
fsinf - > device_characteristics =
le32_to_cpu ( tcon - > fsDevInfo . DeviceCharacteristics ) ;
fsinf - > device_type = le32_to_cpu ( tcon - > fsDevInfo . DeviceType ) ;
fsinf - > fs_attributes = le32_to_cpu ( tcon - > fsAttrInfo . Attributes ) ;
fsinf - > max_path_component =
le32_to_cpu ( tcon - > fsAttrInfo . MaxPathNameComponentLength ) ;
fsinf - > vol_serial_number = tcon - > vol_serial_number ;
fsinf - > vol_create_time = le64_to_cpu ( tcon - > vol_create_time ) ;
fsinf - > share_flags = tcon - > share_flags ;
fsinf - > share_caps = le32_to_cpu ( tcon - > capabilities ) ;
fsinf - > sector_flags = tcon - > ss_flags ;
fsinf - > optimal_sector_size = tcon - > perf_sector_size ;
fsinf - > max_bytes_chunk = tcon - > max_bytes_chunk ;
fsinf - > maximal_access = tcon - > maximal_access ;
fsinf - > cifs_posix_caps = le64_to_cpu ( tcon - > fsUnixInfo . Capability ) ;
if ( copy_to_user ( arg , fsinf , sizeof ( struct smb_mnt_fs_info ) ) )
rc = - EFAULT ;
kfree ( fsinf ) ;
return rc ;
}
2021-04-29 00:18:43 -05:00
static int cifs_shutdown ( struct super_block * sb , unsigned long arg )
{
struct cifs_sb_info * sbi = CIFS_SB ( sb ) ;
__u32 flags ;
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EPERM ;
if ( get_user ( flags , ( __u32 __user * ) arg ) )
return - EFAULT ;
if ( flags > CIFS_GOING_FLAGS_NOLOGFLUSH )
return - EINVAL ;
if ( cifs_forced_shutdown ( sbi ) )
return 0 ;
cifs_dbg ( VFS , " shut down requested (%d) " , flags ) ;
/* trace_cifs_shutdown(sb, flags);*/
/*
* see :
* https : //man7.org/linux/man-pages/man2/ioctl_xfs_goingdown.2.html
* for more information and description of original intent of the flags
*/
switch ( flags ) {
/*
* We could add support later for default flag which requires :
* " Flush all dirty data and metadata to disk "
* would need to call syncfs or equivalent to flush page cache for
* the mount and then issue fsync to server ( if nostrictsync not set )
*/
case CIFS_GOING_FLAGS_DEFAULT :
cifs_dbg ( FYI , " shutdown with default flag not supported \n " ) ;
return - EINVAL ;
/*
* FLAGS_LOGFLUSH is easy since it asks to write out metadata ( not
* data ) but metadata writes are not cached on the client , so can treat
* it similarly to NOLOGFLUSH
*/
case CIFS_GOING_FLAGS_LOGFLUSH :
case CIFS_GOING_FLAGS_NOLOGFLUSH :
sbi - > mnt_cifs_flags | = CIFS_MOUNT_SHUTDOWN ;
return 0 ;
default :
return - EINVAL ;
}
return 0 ;
}
2008-05-15 05:51:55 +00:00
long cifs_ioctl ( struct file * filep , unsigned int command , unsigned long arg )
2005-04-16 15:20:36 -07:00
{
2013-01-23 17:07:38 -05:00
struct inode * inode = file_inode ( filep ) ;
2019-09-19 04:00:55 -05:00
struct smb3_key_debug_info pkey_inf ;
2005-04-16 15:20:36 -07:00
int rc = - ENOTTY ; /* strange error - but the precedent */
2012-06-20 11:21:16 +04:00
unsigned int xid ;
2010-09-20 16:01:31 -07:00
struct cifsFileInfo * pSMBFile = filep - > private_data ;
2011-05-27 04:34:02 +00:00
struct cifs_tcon * tcon ;
2020-07-09 20:39:49 +10:00
struct tcon_link * tlink ;
2020-02-06 06:00:14 -06:00
struct cifs_sb_info * cifs_sb ;
2005-04-28 22:41:04 -07:00
__u64 ExtAttrBits = 0 ;
2010-11-08 07:28:32 -05:00
__u64 caps ;
2005-04-28 22:41:04 -07:00
2012-06-20 11:21:16 +04:00
xid = get_xid ( ) ;
2005-04-28 22:41:04 -07:00
2016-11-16 15:17:15 -08:00
cifs_dbg ( FYI , " cifs ioctl 0x%x \n " , command ) ;
2007-06-05 18:30:44 +00:00
switch ( command ) {
2006-08-29 19:06:16 +01:00
case FS_IOC_GETFLAGS :
2010-11-08 07:28:32 -05:00
if ( pSMBFile = = NULL )
break ;
tcon = tlink_tcon ( pSMBFile - > tlink ) ;
caps = le64_to_cpu ( tcon - > fsUnixInfo . Capability ) ;
2013-10-14 15:31:32 -05:00
# ifdef CONFIG_CIFS_POSIX
2007-06-05 18:30:44 +00:00
if ( CIFS_UNIX_EXTATTR_CAP & caps ) {
2013-10-13 22:32:30 -05:00
__u64 ExtAttrMask = 0 ;
2012-09-18 16:20:26 -07:00
rc = CIFSGetExtAttr ( xid , tcon ,
pSMBFile - > fid . netfid ,
& ExtAttrBits , & ExtAttrMask ) ;
2007-06-05 18:30:44 +00:00
if ( rc = = 0 )
2005-04-28 22:41:04 -07:00
rc = put_user ( ExtAttrBits &
2006-08-29 19:06:16 +01:00
FS_FL_USER_VISIBLE ,
2005-04-28 22:41:04 -07:00
( int __user * ) arg ) ;
2013-10-14 15:31:32 -05:00
if ( rc ! = EOPNOTSUPP )
break ;
}
# endif /* CONFIG_CIFS_POSIX */
rc = 0 ;
if ( CIFS_I ( inode ) - > cifsAttrs & ATTR_COMPRESSED ) {
/* add in the compressed bit */
ExtAttrBits = FS_COMPR_FL ;
rc = put_user ( ExtAttrBits & FS_FL_USER_VISIBLE ,
( int __user * ) arg ) ;
2005-04-28 22:41:04 -07:00
}
break ;
2006-08-29 19:06:16 +01:00
case FS_IOC_SETFLAGS :
2010-11-08 07:28:32 -05:00
if ( pSMBFile = = NULL )
break ;
tcon = tlink_tcon ( pSMBFile - > tlink ) ;
caps = le64_to_cpu ( tcon - > fsUnixInfo . Capability ) ;
2013-10-14 15:31:32 -05:00
if ( get_user ( ExtAttrBits , ( int __user * ) arg ) ) {
rc = - EFAULT ;
break ;
}
/*
* if ( CIFS_UNIX_EXTATTR_CAP & caps )
* rc = CIFSSetExtAttr ( xid , tcon ,
* pSMBFile - > fid . netfid ,
* extAttrBits ,
* & ExtAttrMask ) ;
* if ( rc ! = EOPNOTSUPP )
* break ;
*/
/* Currently only flag we can set is compressed flag */
if ( ( ExtAttrBits & FS_COMPR_FL ) = = 0 )
break ;
/* Try to set compress flag */
if ( tcon - > ses - > server - > ops - > set_compression ) {
rc = tcon - > ses - > server - > ops - > set_compression (
xid , tcon , pSMBFile ) ;
cifs_dbg ( FYI , " set compress flag rc %d \n " , rc ) ;
2005-04-28 22:41:04 -07:00
}
break ;
2013-11-24 21:53:17 -06:00
case CIFS_IOC_COPYCHUNK_FILE :
2017-04-04 02:12:04 -05:00
rc = cifs_ioctl_copychunk ( xid , filep , arg ) ;
2013-11-14 00:05:36 -06:00
break ;
2018-10-07 19:19:58 -05:00
case CIFS_QUERY_INFO :
rc = cifs_ioctl_query_info ( xid , filep , arg ) ;
break ;
2015-06-24 03:17:02 -05:00
case CIFS_IOC_SET_INTEGRITY :
if ( pSMBFile = = NULL )
break ;
tcon = tlink_tcon ( pSMBFile - > tlink ) ;
if ( tcon - > ses - > server - > ops - > set_integrity )
rc = tcon - > ses - > server - > ops - > set_integrity ( xid ,
tcon , pSMBFile ) ;
else
rc = - EOPNOTSUPP ;
break ;
2015-07-04 18:40:10 -05:00
case CIFS_IOC_GET_MNT_INFO :
2017-05-04 00:41:13 +02:00
if ( pSMBFile = = NULL )
break ;
2015-07-04 18:40:10 -05:00
tcon = tlink_tcon ( pSMBFile - > tlink ) ;
rc = smb_mnt_get_fsinfo ( xid , tcon , ( void __user * ) arg ) ;
break ;
2016-09-30 21:14:26 -05:00
case CIFS_ENUMERATE_SNAPSHOTS :
2017-05-03 17:39:08 +02:00
if ( pSMBFile = = NULL )
break ;
2016-09-30 21:14:26 -05:00
if ( arg = = 0 ) {
rc = - EINVAL ;
goto cifs_ioc_exit ;
}
tcon = tlink_tcon ( pSMBFile - > tlink ) ;
if ( tcon - > ses - > server - > ops - > enum_snapshots )
rc = tcon - > ses - > server - > ops - > enum_snapshots ( xid , tcon ,
pSMBFile , ( void __user * ) arg ) ;
else
rc = - EOPNOTSUPP ;
break ;
2019-09-19 04:00:55 -05:00
case CIFS_DUMP_KEY :
if ( pSMBFile = = NULL )
break ;
if ( ! capable ( CAP_SYS_ADMIN ) ) {
rc = - EACCES ;
break ;
}
tcon = tlink_tcon ( pSMBFile - > tlink ) ;
if ( ! smb3_encryption_required ( tcon ) ) {
rc = - EOPNOTSUPP ;
break ;
}
pkey_inf . cipher_type =
le16_to_cpu ( tcon - > ses - > server - > cipher_type ) ;
pkey_inf . Suid = tcon - > ses - > Suid ;
memcpy ( pkey_inf . auth_key , tcon - > ses - > auth_key . response ,
16 /* SMB2_NTLMV2_SESSKEY_SIZE */ ) ;
memcpy ( pkey_inf . smb3decryptionkey ,
tcon - > ses - > smb3decryptionkey , SMB3_SIGN_KEY_SIZE ) ;
memcpy ( pkey_inf . smb3encryptionkey ,
tcon - > ses - > smb3encryptionkey , SMB3_SIGN_KEY_SIZE ) ;
if ( copy_to_user ( ( void __user * ) arg , & pkey_inf ,
sizeof ( struct smb3_key_debug_info ) ) )
rc = - EFAULT ;
else
rc = 0 ;
break ;
2020-02-06 06:00:14 -06:00
case CIFS_IOC_NOTIFY :
if ( ! S_ISDIR ( inode - > i_mode ) ) {
/* Notify can only be done on directories */
rc = - EOPNOTSUPP ;
break ;
}
cifs_sb = CIFS_SB ( inode - > i_sb ) ;
2020-07-09 20:39:49 +10:00
tlink = cifs_sb_tlink ( cifs_sb ) ;
if ( IS_ERR ( tlink ) ) {
rc = PTR_ERR ( tlink ) ;
break ;
}
tcon = tlink_tcon ( tlink ) ;
2020-02-06 06:00:14 -06:00
if ( tcon & & tcon - > ses - > server - > ops - > notify ) {
rc = tcon - > ses - > server - > ops - > notify ( xid ,
filep , ( void __user * ) arg ) ;
cifs_dbg ( FYI , " ioctl notify rc %d \n " , rc ) ;
} else
rc = - EOPNOTSUPP ;
2020-07-09 20:39:49 +10:00
cifs_put_tlink ( tlink ) ;
2020-02-06 06:00:14 -06:00
break ;
2021-04-29 00:18:43 -05:00
case CIFS_IOC_SHUTDOWN :
rc = cifs_shutdown ( inode - > i_sb , arg ) ;
break ;
2005-04-16 15:20:36 -07:00
default :
2013-05-04 22:12:25 -05:00
cifs_dbg ( FYI , " unsupported ioctl \n " ) ;
2005-04-28 22:41:07 -07:00
break ;
2005-04-16 15:20:36 -07:00
}
2016-09-30 21:14:26 -05:00
cifs_ioc_exit :
2012-06-20 11:21:16 +04:00
free_xid ( xid ) ;
2005-04-16 15:20:36 -07:00
return rc ;
2007-06-05 18:30:44 +00:00
}