2019-05-19 13:08:55 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2005-04-16 15:20:36 -07:00
/*
* Changes :
* Arnaldo Carvalho de Melo < acme @ conectiva . com . br > 08 / 23 / 2000
* - get rid of some verify_areas and use __copy * user and __get / put_user
* for the ones that remain
*/
# include <linux/module.h>
# include <linux/blkdev.h>
# include <linux/interrupt.h>
# include <linux/errno.h>
# include <linux/kernel.h>
# include <linux/sched.h>
# include <linux/mm.h>
# include <linux/string.h>
2016-12-24 11:46:01 -08:00
# include <linux/uaccess.h>
2021-07-24 09:20:27 +02:00
# include <linux/cdrom.h>
2005-04-16 15:20:36 -07:00
# include <scsi/scsi.h>
2006-06-10 18:01:03 +02:00
# include <scsi/scsi_cmnd.h>
2005-04-16 15:20:36 -07:00
# include <scsi/scsi_device.h>
# include <scsi/scsi_eh.h>
# include <scsi/scsi_host.h>
# include <scsi/scsi_ioctl.h>
# include <scsi/sg.h>
# include <scsi/scsi_dbg.h>
# include "scsi_logging.h"
# define NORMAL_RETRIES 5
# define IOCTL_NORMAL_TIMEOUT (10 * HZ)
# define MAX_BUF PAGE_SIZE
2005-09-06 14:03:44 +02:00
/**
* ioctl_probe - - return host identification
* @ host : host to identify
* @ buffer : userspace buffer for identification
*
* Return an identifying string at @ buffer , if @ buffer is non - NULL , filling
* to the length stored at * ( int * ) @ buffer .
2005-04-16 15:20:36 -07:00
*/
static int ioctl_probe ( struct Scsi_Host * host , void __user * buffer )
{
unsigned int len , slen ;
const char * string ;
2005-09-06 14:03:44 +02:00
if ( buffer ) {
2005-04-16 15:20:36 -07:00
if ( get_user ( len , ( unsigned int __user * ) buffer ) )
return - EFAULT ;
if ( host - > hostt - > info )
string = host - > hostt - > info ( host ) ;
else
string = host - > hostt - > name ;
if ( string ) {
slen = strlen ( string ) ;
if ( len > slen )
len = slen + 1 ;
if ( copy_to_user ( buffer , string , len ) )
return - EFAULT ;
}
}
2005-09-06 14:03:44 +02:00
return 1 ;
2005-04-16 15:20:36 -07:00
}
static int ioctl_internal_command ( struct scsi_device * sdev , char * cmd ,
int timeout , int retries )
{
int result ;
struct scsi_sense_hdr sshdr ;
2014-06-25 16:39:58 +02:00
SCSI_LOG_IOCTL ( 1 , sdev_printk ( KERN_INFO , sdev ,
" Trying ioctl with scsi command %d \n " , * cmd ) ) ;
2005-04-16 15:20:36 -07:00
2005-08-28 11:27:01 -05:00
result = scsi_execute_req ( sdev , cmd , DMA_NONE , NULL , 0 ,
2008-12-04 14:24:39 +09:00
& sshdr , timeout , retries , NULL ) ;
2005-04-16 15:20:36 -07:00
2014-06-25 16:39:58 +02:00
SCSI_LOG_IOCTL ( 2 , sdev_printk ( KERN_INFO , sdev ,
" Ioctl returned 0x%x \n " , result ) ) ;
2005-04-16 15:20:36 -07:00
2021-04-27 10:30:12 +02:00
if ( result < 0 )
goto out ;
2021-04-27 10:30:15 +02:00
if ( scsi_sense_valid ( & sshdr ) ) {
2005-04-16 15:20:36 -07:00
switch ( sshdr . sense_key ) {
case ILLEGAL_REQUEST :
if ( cmd [ 0 ] = = ALLOW_MEDIUM_REMOVAL )
sdev - > lockable = 0 ;
else
2014-06-25 16:39:58 +02:00
sdev_printk ( KERN_INFO , sdev ,
" ioctl_internal_command: "
" ILLEGAL REQUEST "
" asc=0x%x ascq=0x%x \n " ,
sshdr . asc , sshdr . ascq ) ;
2005-04-16 15:20:36 -07:00
break ;
case NOT_READY : /* This happens if there is no disc in drive */
2006-07-28 09:04:09 +02:00
if ( sdev - > removable )
2005-04-16 15:20:36 -07:00
break ;
2020-08-23 17:36:59 -05:00
fallthrough ;
2005-04-16 15:20:36 -07:00
case UNIT_ATTENTION :
if ( sdev - > removable ) {
sdev - > changed = 1 ;
2005-08-28 11:27:01 -05:00
result = 0 ; /* This is no longer considered an error */
2005-04-16 15:20:36 -07:00
break ;
}
2020-08-23 17:36:59 -05:00
fallthrough ; /* for non-removable media */
2017-08-25 13:46:29 -07:00
default :
2005-10-02 11:45:08 -05:00
sdev_printk ( KERN_INFO , sdev ,
" ioctl_internal_command return code = %x \n " ,
result ) ;
2014-10-24 14:26:45 +02:00
scsi_print_sense_hdr ( sdev , NULL , & sshdr ) ;
2005-04-16 15:20:36 -07:00
break ;
}
}
2021-04-27 10:30:12 +02:00
out :
2014-06-25 16:39:58 +02:00
SCSI_LOG_IOCTL ( 2 , sdev_printk ( KERN_INFO , sdev ,
" IOCTL Releasing command \n " ) ) ;
2005-04-16 15:20:36 -07:00
return result ;
}
int scsi_set_medium_removal ( struct scsi_device * sdev , char state )
{
char scsi_cmd [ MAX_COMMAND_SIZE ] ;
int ret ;
if ( ! sdev - > removable | | ! sdev - > lockable )
return 0 ;
scsi_cmd [ 0 ] = ALLOW_MEDIUM_REMOVAL ;
scsi_cmd [ 1 ] = 0 ;
scsi_cmd [ 2 ] = 0 ;
scsi_cmd [ 3 ] = 0 ;
scsi_cmd [ 4 ] = state ;
scsi_cmd [ 5 ] = 0 ;
ret = ioctl_internal_command ( sdev , scsi_cmd ,
IOCTL_NORMAL_TIMEOUT , NORMAL_RETRIES ) ;
if ( ret = = 0 )
sdev - > locked = ( state = = SCSI_REMOVAL_PREVENT ) ;
return ret ;
}
EXPORT_SYMBOL ( scsi_set_medium_removal ) ;
/*
* The scsi_ioctl_get_pci ( ) function places into arg the value
* pci_dev : : slot_name ( 8 characters ) for the PCI device ( if any ) .
* Returns : 0 on success
* - ENXIO if there isn ' t a PCI device pointer
* ( could be because the SCSI driver hasn ' t been
* updated yet , or because it isn ' t a SCSI
* device )
* any copy_to_user ( ) error on failure there
*/
static int scsi_ioctl_get_pci ( struct scsi_device * sdev , void __user * arg )
{
struct device * dev = scsi_get_device ( sdev - > host ) ;
2008-12-03 22:41:36 +01:00
const char * name ;
2005-04-16 15:20:36 -07:00
if ( ! dev )
return - ENXIO ;
2008-12-03 22:41:36 +01:00
name = dev_name ( dev ) ;
/* compatibility with old ioctl which only returned
* 20 characters */
return copy_to_user ( arg , name , min ( strlen ( name ) , ( size_t ) 20 ) )
? - EFAULT : 0 ;
2005-04-16 15:20:36 -07:00
}
2021-07-24 09:20:27 +02:00
static int sg_get_version ( int __user * p )
{
static const int sg_version_num = 30527 ;
return put_user ( sg_version_num , p ) ;
}
static int sg_get_timeout ( struct request_queue * q )
{
return jiffies_to_clock_t ( q - > sg_timeout ) ;
}
static int sg_set_timeout ( struct request_queue * q , int __user * p )
{
int timeout , err = get_user ( timeout , p ) ;
if ( ! err )
q - > sg_timeout = clock_t_to_jiffies ( timeout ) ;
return err ;
}
static int sg_get_reserved_size ( struct request_queue * q , int __user * p )
{
int val = min ( q - > sg_reserved_size , queue_max_bytes ( q ) ) ;
return put_user ( val , p ) ;
}
static int sg_set_reserved_size ( struct request_queue * q , int __user * p )
{
int size , err = get_user ( size , p ) ;
if ( err )
return err ;
if ( size < 0 )
return - EINVAL ;
q - > sg_reserved_size = min_t ( unsigned int , size , queue_max_bytes ( q ) ) ;
return 0 ;
}
/*
* will always return that we are ATAPI even for a real SCSI drive , I ' m not
* so sure this is worth doing anything about ( why would you care ? ? )
*/
static int sg_emulated_host ( struct request_queue * q , int __user * p )
{
return put_user ( 1 , p ) ;
}
/* Send basic block requests */
static int __blk_send_generic ( struct request_queue * q , struct gendisk * bd_disk ,
int cmd , int data )
{
struct request * rq ;
int err ;
rq = blk_get_request ( q , REQ_OP_DRV_OUT , 0 ) ;
if ( IS_ERR ( rq ) )
return PTR_ERR ( rq ) ;
rq - > timeout = BLK_DEFAULT_SG_TIMEOUT ;
scsi_req ( rq ) - > cmd [ 0 ] = cmd ;
scsi_req ( rq ) - > cmd [ 4 ] = data ;
scsi_req ( rq ) - > cmd_len = 6 ;
blk_execute_rq ( bd_disk , rq , 0 ) ;
err = scsi_req ( rq ) - > result ? - EIO : 0 ;
blk_put_request ( rq ) ;
return err ;
}
static inline int blk_send_start_stop ( struct request_queue * q ,
struct gendisk * bd_disk , int data )
{
return __blk_send_generic ( q , bd_disk , GPCMD_START_STOP_UNIT , data ) ;
}
/*
* Check if the given command is allowed .
*
* Only a subset of commands are allowed for unprivileged users . Commands used
* to format the media , update the firmware , etc . are not permitted .
*/
bool scsi_cmd_allowed ( unsigned char * cmd , fmode_t mode )
{
/* root can do any command. */
if ( capable ( CAP_SYS_RAWIO ) )
return true ;
/* Anybody who can open the device can do a read-safe command */
switch ( cmd [ 0 ] ) {
/* Basic read-only commands */
case TEST_UNIT_READY :
case REQUEST_SENSE :
case READ_6 :
case READ_10 :
case READ_12 :
case READ_16 :
case READ_BUFFER :
case READ_DEFECT_DATA :
case READ_CAPACITY : /* also GPCMD_READ_CDVD_CAPACITY */
case READ_LONG :
case INQUIRY :
case MODE_SENSE :
case MODE_SENSE_10 :
case LOG_SENSE :
case START_STOP :
case GPCMD_VERIFY_10 :
case VERIFY_16 :
case REPORT_LUNS :
case SERVICE_ACTION_IN_16 :
case RECEIVE_DIAGNOSTIC :
case MAINTENANCE_IN : /* also GPCMD_SEND_KEY, which is a write command */
case GPCMD_READ_BUFFER_CAPACITY :
/* Audio CD commands */
case GPCMD_PLAY_CD :
case GPCMD_PLAY_AUDIO_10 :
case GPCMD_PLAY_AUDIO_MSF :
case GPCMD_PLAY_AUDIO_TI :
case GPCMD_PAUSE_RESUME :
/* CD/DVD data reading */
case GPCMD_READ_CD :
case GPCMD_READ_CD_MSF :
case GPCMD_READ_DISC_INFO :
case GPCMD_READ_DVD_STRUCTURE :
case GPCMD_READ_HEADER :
case GPCMD_READ_TRACK_RZONE_INFO :
case GPCMD_READ_SUBCHANNEL :
case GPCMD_READ_TOC_PMA_ATIP :
case GPCMD_REPORT_KEY :
case GPCMD_SCAN :
case GPCMD_GET_CONFIGURATION :
case GPCMD_READ_FORMAT_CAPACITIES :
case GPCMD_GET_EVENT_STATUS_NOTIFICATION :
case GPCMD_GET_PERFORMANCE :
case GPCMD_SEEK :
case GPCMD_STOP_PLAY_SCAN :
/* ZBC */
case ZBC_IN :
return true ;
/* Basic writing commands */
case WRITE_6 :
case WRITE_10 :
case WRITE_VERIFY :
case WRITE_12 :
case WRITE_VERIFY_12 :
case WRITE_16 :
case WRITE_LONG :
case WRITE_LONG_2 :
case WRITE_SAME :
case WRITE_SAME_16 :
case WRITE_SAME_32 :
case ERASE :
case GPCMD_MODE_SELECT_10 :
case MODE_SELECT :
case LOG_SELECT :
case GPCMD_BLANK :
case GPCMD_CLOSE_TRACK :
case GPCMD_FLUSH_CACHE :
case GPCMD_FORMAT_UNIT :
case GPCMD_REPAIR_RZONE_TRACK :
case GPCMD_RESERVE_RZONE_TRACK :
case GPCMD_SEND_DVD_STRUCTURE :
case GPCMD_SEND_EVENT :
case GPCMD_SEND_OPC :
case GPCMD_SEND_CUE_SHEET :
case GPCMD_SET_SPEED :
case GPCMD_PREVENT_ALLOW_MEDIUM_REMOVAL :
case GPCMD_LOAD_UNLOAD :
case GPCMD_SET_STREAMING :
case GPCMD_SET_READ_AHEAD :
/* ZBC */
case ZBC_OUT :
return ( mode & FMODE_WRITE ) ;
default :
return false ;
}
}
EXPORT_SYMBOL ( scsi_cmd_allowed ) ;
static int scsi_fill_sghdr_rq ( struct request_queue * q , struct request * rq ,
struct sg_io_hdr * hdr , fmode_t mode )
{
struct scsi_request * req = scsi_req ( rq ) ;
if ( copy_from_user ( req - > cmd , hdr - > cmdp , hdr - > cmd_len ) )
return - EFAULT ;
if ( ! scsi_cmd_allowed ( req - > cmd , mode ) )
return - EPERM ;
/*
* fill in request structure
*/
req - > cmd_len = hdr - > cmd_len ;
rq - > timeout = msecs_to_jiffies ( hdr - > timeout ) ;
if ( ! rq - > timeout )
rq - > timeout = q - > sg_timeout ;
if ( ! rq - > timeout )
rq - > timeout = BLK_DEFAULT_SG_TIMEOUT ;
if ( rq - > timeout < BLK_MIN_SG_TIMEOUT )
rq - > timeout = BLK_MIN_SG_TIMEOUT ;
return 0 ;
}
static int scsi_complete_sghdr_rq ( struct request * rq , struct sg_io_hdr * hdr ,
struct bio * bio )
{
struct scsi_request * req = scsi_req ( rq ) ;
int r , ret = 0 ;
/*
* fill in all the output members
*/
hdr - > status = req - > result & 0xff ;
hdr - > masked_status = status_byte ( req - > result ) ;
hdr - > msg_status = COMMAND_COMPLETE ;
hdr - > host_status = host_byte ( req - > result ) ;
hdr - > driver_status = 0 ;
if ( scsi_status_is_check_condition ( hdr - > status ) )
hdr - > driver_status = DRIVER_SENSE ;
hdr - > info = 0 ;
if ( hdr - > masked_status | | hdr - > host_status | | hdr - > driver_status )
hdr - > info | = SG_INFO_CHECK ;
hdr - > resid = req - > resid_len ;
hdr - > sb_len_wr = 0 ;
if ( req - > sense_len & & hdr - > sbp ) {
int len = min ( ( unsigned int ) hdr - > mx_sb_len , req - > sense_len ) ;
if ( ! copy_to_user ( hdr - > sbp , req - > sense , len ) )
hdr - > sb_len_wr = len ;
else
ret = - EFAULT ;
}
r = blk_rq_unmap_user ( bio ) ;
if ( ! ret )
ret = r ;
return ret ;
}
static int sg_io ( struct request_queue * q , struct gendisk * bd_disk ,
struct sg_io_hdr * hdr , fmode_t mode )
{
unsigned long start_time ;
ssize_t ret = 0 ;
int writing = 0 ;
int at_head = 0 ;
struct request * rq ;
struct scsi_request * req ;
struct bio * bio ;
if ( hdr - > interface_id ! = ' S ' )
return - EINVAL ;
if ( hdr - > dxfer_len > ( queue_max_hw_sectors ( q ) < < 9 ) )
return - EIO ;
if ( hdr - > dxfer_len )
switch ( hdr - > dxfer_direction ) {
default :
return - EINVAL ;
case SG_DXFER_TO_DEV :
writing = 1 ;
break ;
case SG_DXFER_TO_FROM_DEV :
case SG_DXFER_FROM_DEV :
break ;
}
if ( hdr - > flags & SG_FLAG_Q_AT_HEAD )
at_head = 1 ;
ret = - ENOMEM ;
rq = blk_get_request ( q , writing ? REQ_OP_DRV_OUT : REQ_OP_DRV_IN , 0 ) ;
if ( IS_ERR ( rq ) )
return PTR_ERR ( rq ) ;
req = scsi_req ( rq ) ;
if ( hdr - > cmd_len > BLK_MAX_CDB ) {
req - > cmd = kzalloc ( hdr - > cmd_len , GFP_KERNEL ) ;
if ( ! req - > cmd )
goto out_put_request ;
}
ret = scsi_fill_sghdr_rq ( q , rq , hdr , mode ) ;
if ( ret < 0 )
goto out_free_cdb ;
ret = 0 ;
if ( hdr - > iovec_count ) {
struct iov_iter i ;
struct iovec * iov = NULL ;
ret = import_iovec ( rq_data_dir ( rq ) , hdr - > dxferp ,
hdr - > iovec_count , 0 , & iov , & i ) ;
if ( ret < 0 )
goto out_free_cdb ;
/* SG_IO howto says that the shorter of the two wins */
iov_iter_truncate ( & i , hdr - > dxfer_len ) ;
ret = blk_rq_map_user_iov ( q , rq , NULL , & i , GFP_KERNEL ) ;
kfree ( iov ) ;
} else if ( hdr - > dxfer_len )
ret = blk_rq_map_user ( q , rq , NULL , hdr - > dxferp , hdr - > dxfer_len ,
GFP_KERNEL ) ;
if ( ret )
goto out_free_cdb ;
bio = rq - > bio ;
req - > retries = 0 ;
start_time = jiffies ;
blk_execute_rq ( bd_disk , rq , at_head ) ;
hdr - > duration = jiffies_to_msecs ( jiffies - start_time ) ;
ret = scsi_complete_sghdr_rq ( rq , hdr , bio ) ;
out_free_cdb :
scsi_req_free_cmd ( req ) ;
out_put_request :
blk_put_request ( rq ) ;
return ret ;
}
/**
* sg_scsi_ioctl - - handle deprecated SCSI_IOCTL_SEND_COMMAND ioctl
* @ q : request queue to send scsi commands down
* @ disk : gendisk to operate on ( option )
* @ mode : mode used to open the file through which the ioctl has been
* submitted
* @ sic : userspace structure describing the command to perform
*
* Send down the scsi command described by @ sic to the device below
* the request queue @ q . If @ file is non - NULL it ' s used to perform
* fine - grained permission checks that allow users to send down
* non - destructive SCSI commands . If the caller has a struct gendisk
* available it should be passed in as @ disk to allow the low level
* driver to use the information contained in it . A non - NULL @ disk
* is only allowed if the caller knows that the low level driver doesn ' t
* need it ( e . g . in the scsi subsystem ) .
*
* Notes :
* - This interface is deprecated - users should use the SG_IO
* interface instead , as this is a more flexible approach to
* performing SCSI commands on a device .
* - The SCSI command length is determined by examining the 1 st byte
* of the given command . There is no way to override this .
* - Data transfers are limited to PAGE_SIZE
* - The length ( x + y ) must be at least OMAX_SB_LEN bytes long to
* accommodate the sense buffer when an error occurs .
* The sense buffer is truncated to OMAX_SB_LEN ( 16 ) bytes so that
* old code will not be surprised .
* - If a Unix error occurs ( e . g . ENOMEM ) then the user will receive
* a negative return and the Unix error code in ' errno ' .
* If the SCSI command succeeds then 0 is returned .
* Positive numbers returned are the compacted SCSI error codes ( 4
* bytes in one int ) where the lowest byte is the SCSI status .
*/
int sg_scsi_ioctl ( struct request_queue * q , struct gendisk * disk , fmode_t mode ,
struct scsi_ioctl_command __user * sic )
{
enum { OMAX_SB_LEN = 16 } ; /* For backward compatibility */
struct request * rq ;
struct scsi_request * req ;
int err ;
unsigned int in_len , out_len , bytes , opcode , cmdlen ;
char * buffer = NULL ;
if ( ! sic )
return - EINVAL ;
/*
* get in an out lengths , verify they don ' t exceed a page worth of data
*/
if ( get_user ( in_len , & sic - > inlen ) )
return - EFAULT ;
if ( get_user ( out_len , & sic - > outlen ) )
return - EFAULT ;
if ( in_len > PAGE_SIZE | | out_len > PAGE_SIZE )
return - EINVAL ;
if ( get_user ( opcode , sic - > data ) )
return - EFAULT ;
bytes = max ( in_len , out_len ) ;
if ( bytes ) {
buffer = kzalloc ( bytes , GFP_NOIO | GFP_USER | __GFP_NOWARN ) ;
if ( ! buffer )
return - ENOMEM ;
}
rq = blk_get_request ( q , in_len ? REQ_OP_DRV_OUT : REQ_OP_DRV_IN , 0 ) ;
if ( IS_ERR ( rq ) ) {
err = PTR_ERR ( rq ) ;
goto error_free_buffer ;
}
req = scsi_req ( rq ) ;
cmdlen = COMMAND_SIZE ( opcode ) ;
/*
* get command and data to send to device , if any
*/
err = - EFAULT ;
req - > cmd_len = cmdlen ;
if ( copy_from_user ( req - > cmd , sic - > data , cmdlen ) )
goto error ;
if ( in_len & & copy_from_user ( buffer , sic - > data + cmdlen , in_len ) )
goto error ;
err = - EPERM ;
if ( ! scsi_cmd_allowed ( req - > cmd , mode ) )
goto error ;
/* default. possible overridden later */
req - > retries = 5 ;
switch ( opcode ) {
case SEND_DIAGNOSTIC :
case FORMAT_UNIT :
rq - > timeout = FORMAT_UNIT_TIMEOUT ;
req - > retries = 1 ;
break ;
case START_STOP :
rq - > timeout = START_STOP_TIMEOUT ;
break ;
case MOVE_MEDIUM :
rq - > timeout = MOVE_MEDIUM_TIMEOUT ;
break ;
case READ_ELEMENT_STATUS :
rq - > timeout = READ_ELEMENT_STATUS_TIMEOUT ;
break ;
case READ_DEFECT_DATA :
rq - > timeout = READ_DEFECT_DATA_TIMEOUT ;
req - > retries = 1 ;
break ;
default :
rq - > timeout = BLK_DEFAULT_SG_TIMEOUT ;
break ;
}
if ( bytes ) {
err = blk_rq_map_kern ( q , rq , buffer , bytes , GFP_NOIO ) ;
if ( err )
goto error ;
}
blk_execute_rq ( disk , rq , 0 ) ;
err = req - > result & 0xff ; /* only 8 bit SCSI status */
if ( err ) {
if ( req - > sense_len & & req - > sense ) {
bytes = ( OMAX_SB_LEN > req - > sense_len ) ?
req - > sense_len : OMAX_SB_LEN ;
if ( copy_to_user ( sic - > data , req - > sense , bytes ) )
err = - EFAULT ;
}
} else {
if ( copy_to_user ( sic - > data , buffer , out_len ) )
err = - EFAULT ;
}
error :
blk_put_request ( rq ) ;
error_free_buffer :
kfree ( buffer ) ;
return err ;
}
EXPORT_SYMBOL_GPL ( sg_scsi_ioctl ) ;
int put_sg_io_hdr ( const struct sg_io_hdr * hdr , void __user * argp )
{
# ifdef CONFIG_COMPAT
if ( in_compat_syscall ( ) ) {
struct compat_sg_io_hdr hdr32 = {
. interface_id = hdr - > interface_id ,
. dxfer_direction = hdr - > dxfer_direction ,
. cmd_len = hdr - > cmd_len ,
. mx_sb_len = hdr - > mx_sb_len ,
. iovec_count = hdr - > iovec_count ,
. dxfer_len = hdr - > dxfer_len ,
. dxferp = ( uintptr_t ) hdr - > dxferp ,
. cmdp = ( uintptr_t ) hdr - > cmdp ,
. sbp = ( uintptr_t ) hdr - > sbp ,
. timeout = hdr - > timeout ,
. flags = hdr - > flags ,
. pack_id = hdr - > pack_id ,
. usr_ptr = ( uintptr_t ) hdr - > usr_ptr ,
. status = hdr - > status ,
. masked_status = hdr - > masked_status ,
. msg_status = hdr - > msg_status ,
. sb_len_wr = hdr - > sb_len_wr ,
. host_status = hdr - > host_status ,
. driver_status = hdr - > driver_status ,
. resid = hdr - > resid ,
. duration = hdr - > duration ,
. info = hdr - > info ,
} ;
if ( copy_to_user ( argp , & hdr32 , sizeof ( hdr32 ) ) )
return - EFAULT ;
return 0 ;
}
# endif
if ( copy_to_user ( argp , hdr , sizeof ( * hdr ) ) )
return - EFAULT ;
return 0 ;
}
EXPORT_SYMBOL ( put_sg_io_hdr ) ;
int get_sg_io_hdr ( struct sg_io_hdr * hdr , const void __user * argp )
{
# ifdef CONFIG_COMPAT
struct compat_sg_io_hdr hdr32 ;
if ( in_compat_syscall ( ) ) {
if ( copy_from_user ( & hdr32 , argp , sizeof ( hdr32 ) ) )
return - EFAULT ;
* hdr = ( struct sg_io_hdr ) {
. interface_id = hdr32 . interface_id ,
. dxfer_direction = hdr32 . dxfer_direction ,
. cmd_len = hdr32 . cmd_len ,
. mx_sb_len = hdr32 . mx_sb_len ,
. iovec_count = hdr32 . iovec_count ,
. dxfer_len = hdr32 . dxfer_len ,
. dxferp = compat_ptr ( hdr32 . dxferp ) ,
. cmdp = compat_ptr ( hdr32 . cmdp ) ,
. sbp = compat_ptr ( hdr32 . sbp ) ,
. timeout = hdr32 . timeout ,
. flags = hdr32 . flags ,
. pack_id = hdr32 . pack_id ,
. usr_ptr = compat_ptr ( hdr32 . usr_ptr ) ,
. status = hdr32 . status ,
. masked_status = hdr32 . masked_status ,
. msg_status = hdr32 . msg_status ,
. sb_len_wr = hdr32 . sb_len_wr ,
. host_status = hdr32 . host_status ,
. driver_status = hdr32 . driver_status ,
. resid = hdr32 . resid ,
. duration = hdr32 . duration ,
. info = hdr32 . info ,
} ;
return 0 ;
}
# endif
if ( copy_from_user ( hdr , argp , sizeof ( * hdr ) ) )
return - EFAULT ;
return 0 ;
}
EXPORT_SYMBOL ( get_sg_io_hdr ) ;
# ifdef CONFIG_COMPAT
struct compat_cdrom_generic_command {
unsigned char cmd [ CDROM_PACKET_SIZE ] ;
compat_caddr_t buffer ;
compat_uint_t buflen ;
compat_int_t stat ;
compat_caddr_t sense ;
unsigned char data_direction ;
unsigned char pad [ 3 ] ;
compat_int_t quiet ;
compat_int_t timeout ;
compat_caddr_t unused ;
} ;
# endif
static int scsi_get_cdrom_generic_arg ( struct cdrom_generic_command * cgc ,
const void __user * arg )
{
# ifdef CONFIG_COMPAT
if ( in_compat_syscall ( ) ) {
struct compat_cdrom_generic_command cgc32 ;
if ( copy_from_user ( & cgc32 , arg , sizeof ( cgc32 ) ) )
return - EFAULT ;
* cgc = ( struct cdrom_generic_command ) {
. buffer = compat_ptr ( cgc32 . buffer ) ,
. buflen = cgc32 . buflen ,
. stat = cgc32 . stat ,
. sense = compat_ptr ( cgc32 . sense ) ,
. data_direction = cgc32 . data_direction ,
. quiet = cgc32 . quiet ,
. timeout = cgc32 . timeout ,
. unused = compat_ptr ( cgc32 . unused ) ,
} ;
memcpy ( & cgc - > cmd , & cgc32 . cmd , CDROM_PACKET_SIZE ) ;
return 0 ;
}
# endif
if ( copy_from_user ( cgc , arg , sizeof ( * cgc ) ) )
return - EFAULT ;
return 0 ;
}
static int scsi_put_cdrom_generic_arg ( const struct cdrom_generic_command * cgc ,
void __user * arg )
{
# ifdef CONFIG_COMPAT
if ( in_compat_syscall ( ) ) {
struct compat_cdrom_generic_command cgc32 = {
. buffer = ( uintptr_t ) ( cgc - > buffer ) ,
. buflen = cgc - > buflen ,
. stat = cgc - > stat ,
. sense = ( uintptr_t ) ( cgc - > sense ) ,
. data_direction = cgc - > data_direction ,
. quiet = cgc - > quiet ,
. timeout = cgc - > timeout ,
. unused = ( uintptr_t ) ( cgc - > unused ) ,
} ;
memcpy ( & cgc32 . cmd , & cgc - > cmd , CDROM_PACKET_SIZE ) ;
if ( copy_to_user ( arg , & cgc32 , sizeof ( cgc32 ) ) )
return - EFAULT ;
return 0 ;
}
# endif
if ( copy_to_user ( arg , cgc , sizeof ( * cgc ) ) )
return - EFAULT ;
return 0 ;
}
static int scsi_cdrom_send_packet ( struct request_queue * q ,
struct gendisk * bd_disk ,
fmode_t mode , void __user * arg )
{
struct cdrom_generic_command cgc ;
struct sg_io_hdr hdr ;
int err ;
err = scsi_get_cdrom_generic_arg ( & cgc , arg ) ;
if ( err )
return err ;
cgc . timeout = clock_t_to_jiffies ( cgc . timeout ) ;
memset ( & hdr , 0 , sizeof ( hdr ) ) ;
hdr . interface_id = ' S ' ;
hdr . cmd_len = sizeof ( cgc . cmd ) ;
hdr . dxfer_len = cgc . buflen ;
switch ( cgc . data_direction ) {
case CGC_DATA_UNKNOWN :
hdr . dxfer_direction = SG_DXFER_UNKNOWN ;
break ;
case CGC_DATA_WRITE :
hdr . dxfer_direction = SG_DXFER_TO_DEV ;
break ;
case CGC_DATA_READ :
hdr . dxfer_direction = SG_DXFER_FROM_DEV ;
break ;
case CGC_DATA_NONE :
hdr . dxfer_direction = SG_DXFER_NONE ;
break ;
default :
return - EINVAL ;
}
hdr . dxferp = cgc . buffer ;
hdr . sbp = cgc . sense ;
if ( hdr . sbp )
hdr . mx_sb_len = sizeof ( struct request_sense ) ;
hdr . timeout = jiffies_to_msecs ( cgc . timeout ) ;
hdr . cmdp = ( ( struct cdrom_generic_command __user * ) arg ) - > cmd ;
hdr . cmd_len = sizeof ( cgc . cmd ) ;
err = sg_io ( q , bd_disk , & hdr , mode ) ;
if ( err = = - EFAULT )
return - EFAULT ;
if ( hdr . status )
return - EIO ;
cgc . stat = err ;
cgc . buflen = hdr . resid ;
if ( scsi_put_cdrom_generic_arg ( & cgc , arg ) )
return - EFAULT ;
return err ;
}
2021-07-24 09:20:15 +02:00
/**
* scsi_ioctl - Dispatch ioctl to scsi device
* @ sdev : scsi device receiving ioctl
2021-07-24 09:20:20 +02:00
* @ disk : disk receiving the ioctl
* @ mode : mode the block / char device is opened with
2021-07-24 09:20:15 +02:00
* @ cmd : which ioctl is it
* @ arg : data associated with ioctl
*
* Description : The scsi_ioctl ( ) function differs from most ioctls in that it
* does not take a major / minor number as the dev field . Rather , it takes
* a pointer to a & struct scsi_device .
*/
2021-07-24 09:20:20 +02:00
int scsi_ioctl ( struct scsi_device * sdev , struct gendisk * disk , fmode_t mode ,
int cmd , void __user * arg )
2005-04-16 15:20:36 -07:00
{
2021-07-24 09:20:20 +02:00
struct request_queue * q = sdev - > request_queue ;
2005-04-16 15:20:36 -07:00
char scsi_cmd [ MAX_COMMAND_SIZE ] ;
2017-02-14 20:15:57 +01:00
struct scsi_sense_hdr sense_hdr ;
2021-07-24 09:20:20 +02:00
int error ;
2005-04-16 15:20:36 -07:00
/* Check for deprecated ioctls ... all the ioctls which don't
* follow the new unique numbering scheme are deprecated */
switch ( cmd ) {
case SCSI_IOCTL_SEND_COMMAND :
case SCSI_IOCTL_TEST_UNIT_READY :
case SCSI_IOCTL_BENCHMARK_COMMAND :
case SCSI_IOCTL_SYNC :
case SCSI_IOCTL_START_UNIT :
case SCSI_IOCTL_STOP_UNIT :
printk ( KERN_WARNING " program %s is using a deprecated SCSI "
" ioctl, please convert it to SG_IO \n " , current - > comm ) ;
break ;
default :
break ;
}
2021-07-24 09:20:27 +02:00
switch ( cmd ) {
case SG_GET_VERSION_NUM :
return sg_get_version ( arg ) ;
case SG_SET_TIMEOUT :
return sg_set_timeout ( q , arg ) ;
case SG_GET_TIMEOUT :
return sg_get_timeout ( q ) ;
case SG_GET_RESERVED_SIZE :
return sg_get_reserved_size ( q , arg ) ;
case SG_SET_RESERVED_SIZE :
return sg_set_reserved_size ( q , arg ) ;
case SG_EMULATED_HOST :
return sg_emulated_host ( q , arg ) ;
case SG_IO : {
struct sg_io_hdr hdr ;
error = get_sg_io_hdr ( & hdr , arg ) ;
if ( error )
2021-07-24 09:20:20 +02:00
return error ;
2021-07-24 09:20:27 +02:00
error = sg_io ( q , disk , & hdr , mode ) ;
if ( error = = - EFAULT )
return error ;
if ( put_sg_io_hdr ( & hdr , arg ) )
return - EFAULT ;
return 0 ;
}
case SCSI_IOCTL_SEND_COMMAND :
return sg_scsi_ioctl ( q , disk , mode , arg ) ;
case CDROM_SEND_PACKET :
return scsi_cdrom_send_packet ( q , disk , mode , arg ) ;
case CDROMCLOSETRAY :
return blk_send_start_stop ( q , disk , 0x03 ) ;
case CDROMEJECT :
return blk_send_start_stop ( q , disk , 0x02 ) ;
2020-02-12 11:36:05 -05:00
case SCSI_IOCTL_GET_IDLUN : {
struct scsi_idlun v = {
. dev_id = ( sdev - > id & 0xff )
+ ( ( sdev - > lun & 0xff ) < < 8 )
+ ( ( sdev - > channel & 0xff ) < < 16 )
+ ( ( sdev - > host - > host_no & 0xff ) < < 24 ) ,
. host_unique_id = sdev - > host - > unique_id
} ;
if ( copy_to_user ( arg , & v , sizeof ( struct scsi_idlun ) ) )
2005-04-16 15:20:36 -07:00
return - EFAULT ;
return 0 ;
2020-02-12 11:36:05 -05:00
}
2005-04-16 15:20:36 -07:00
case SCSI_IOCTL_GET_BUS_NUMBER :
return put_user ( sdev - > host - > host_no , ( int __user * ) arg ) ;
case SCSI_IOCTL_PROBE_HOST :
return ioctl_probe ( sdev - > host , arg ) ;
case SCSI_IOCTL_DOORLOCK :
return scsi_set_medium_removal ( sdev , SCSI_REMOVAL_PREVENT ) ;
case SCSI_IOCTL_DOORUNLOCK :
return scsi_set_medium_removal ( sdev , SCSI_REMOVAL_ALLOW ) ;
case SCSI_IOCTL_TEST_UNIT_READY :
return scsi_test_unit_ready ( sdev , IOCTL_NORMAL_TIMEOUT ,
2017-02-14 20:15:57 +01:00
NORMAL_RETRIES , & sense_hdr ) ;
2005-04-16 15:20:36 -07:00
case SCSI_IOCTL_START_UNIT :
scsi_cmd [ 0 ] = START_STOP ;
scsi_cmd [ 1 ] = 0 ;
scsi_cmd [ 2 ] = scsi_cmd [ 3 ] = scsi_cmd [ 5 ] = 0 ;
scsi_cmd [ 4 ] = 1 ;
return ioctl_internal_command ( sdev , scsi_cmd ,
START_STOP_TIMEOUT , NORMAL_RETRIES ) ;
case SCSI_IOCTL_STOP_UNIT :
scsi_cmd [ 0 ] = START_STOP ;
scsi_cmd [ 1 ] = 0 ;
scsi_cmd [ 2 ] = scsi_cmd [ 3 ] = scsi_cmd [ 5 ] = 0 ;
scsi_cmd [ 4 ] = 0 ;
return ioctl_internal_command ( sdev , scsi_cmd ,
START_STOP_TIMEOUT , NORMAL_RETRIES ) ;
case SCSI_IOCTL_GET_PCI :
return scsi_ioctl_get_pci ( sdev , arg ) ;
2014-10-11 16:25:31 +02:00
case SG_SCSI_RESET :
return scsi_ioctl_reset ( sdev , arg ) ;
2005-04-16 15:20:36 -07:00
}
2019-03-15 16:45:16 +01:00
# ifdef CONFIG_COMPAT
2021-07-24 09:20:15 +02:00
if ( in_compat_syscall ( ) ) {
if ( ! sdev - > host - > hostt - > compat_ioctl )
return - EINVAL ;
2019-03-15 16:45:16 +01:00
return sdev - > host - > hostt - > compat_ioctl ( sdev , cmd , arg ) ;
2021-07-24 09:20:15 +02:00
}
2019-03-15 16:45:16 +01:00
# endif
2021-07-24 09:20:15 +02:00
if ( ! sdev - > host - > hostt - > ioctl )
return - EINVAL ;
return sdev - > host - > hostt - > ioctl ( sdev , cmd , arg ) ;
}
EXPORT_SYMBOL ( scsi_ioctl ) ;
2019-03-15 16:45:16 +01:00
2014-10-11 16:25:31 +02:00
/*
* We can process a reset even when a device isn ' t fully operable .
2005-04-16 15:20:36 -07:00
*/
2014-10-11 16:25:31 +02:00
int scsi_ioctl_block_when_processing_errors ( struct scsi_device * sdev , int cmd ,
bool ndelay )
2005-04-16 15:20:36 -07:00
{
2014-10-11 16:25:31 +02:00
if ( cmd = = SG_SCSI_RESET & & ndelay ) {
2005-09-18 15:05:20 -05:00
if ( scsi_host_in_recovery ( sdev - > host ) )
2014-10-27 16:28:13 +01:00
return - EAGAIN ;
2014-10-11 16:25:31 +02:00
} else {
if ( ! scsi_block_when_processing_errors ( sdev ) )
return - ENODEV ;
2005-04-16 15:20:36 -07:00
}
2014-10-11 16:25:31 +02:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2014-10-11 16:25:31 +02:00
EXPORT_SYMBOL_GPL ( scsi_ioctl_block_when_processing_errors ) ;