2005-04-17 02:20:36 +04:00
/*
2008-02-02 01:09:33 +03:00
* Copyright ( C ) 1996 - 1999 Gadi Oxman < gadio @ netvision . net . il >
* Copyright ( C ) 2004 - 2005 Bartlomiej Zolnierkiewicz
2005-04-17 02:20:36 +04:00
*/
/*
* Emulation of a SCSI host adapter for IDE ATAPI devices .
*
* With this driver , one can use the Linux SCSI drivers instead of the
* native IDE ATAPI drivers .
*
* Ver 0.1 Dec 3 96 Initial version .
* Ver 0.2 Jan 26 97 Fixed bug in cleanup_module ( ) and added emulation
* of MODE_SENSE_6 / MODE_SELECT_6 for cdroms . Thanks
* to Janos Farkas for pointing this out .
* Avoid using bitfields in structures for m68k .
* Added Scatter / Gather and DMA support .
* Ver 0.4 Dec 7 97 Add support for ATAPI PD / CD drives .
* Use variable timeout for each command .
* Ver 0.5 Jan 2 98 Fix previous PD / CD support .
* Allow disabling of SCSI - 6 to SCSI - 10 transformation .
* Ver 0.6 Jan 27 98 Allow disabling of SCSI command translation layer
* for access through / dev / sg .
* Fix MODE_SENSE_6 / MODE_SELECT_6 / INQUIRY translation .
* Ver 0.7 Dec 04 98 Ignore commands where lun ! = 0 to avoid multiple
* detection of devices with CONFIG_SCSI_MULTI_LUN
* Ver 0.8 Feb 05 99 Optical media need translation too . Reverse 0.7 .
* Ver 0.9 Jul 04 99 Fix a bug in SG_SET_TRANSFORM .
* Ver 0.91 Jun 10 02 Fix " off by one " error in transforms
* Ver 0.92 Dec 31 02 Implement new SCSI mid level API
*/
# define IDESCSI_VERSION "0.92"
# include <linux/module.h>
# include <linux/types.h>
# include <linux/string.h>
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/ioport.h>
# include <linux/blkdev.h>
# include <linux/errno.h>
# include <linux/hdreg.h>
# include <linux/slab.h>
# include <linux/ide.h>
# include <linux/scatterlist.h>
2005-05-28 15:57:14 +04:00
# include <linux/delay.h>
2006-03-23 14:00:22 +03:00
# include <linux/mutex.h>
2007-10-19 10:40:25 +04:00
# include <linux/bitops.h>
2005-04-17 02:20:36 +04:00
# include <asm/io.h>
# include <asm/uaccess.h>
# include <scsi/scsi.h>
# include <scsi/scsi_cmnd.h>
# include <scsi/scsi_device.h>
# include <scsi/scsi_host.h>
# include <scsi/scsi_tcq.h>
# include <scsi/sg.h>
# define IDESCSI_DEBUG_LOG 0
2008-07-15 23:21:52 +04:00
# if IDESCSI_DEBUG_LOG
# define debug_log(fmt, args...) \
printk ( KERN_INFO " ide-scsi: " fmt , # # args )
# else
# define debug_log(fmt, args...) do {} while (0)
# endif
2005-04-17 02:20:36 +04:00
/*
* SCSI command transformation layer
*/
# define IDESCSI_SG_TRANSFORM 1 /* /dev/sg transformation */
/*
* Log flags
*/
# define IDESCSI_LOG_CMD 0 /* Log SCSI commands */
typedef struct ide_scsi_obj {
ide_drive_t * drive ;
ide_driver_t * driver ;
struct gendisk * disk ;
struct Scsi_Host * host ;
2008-04-18 02:46:27 +04:00
struct ide_atapi_pc * pc ; /* Current packet command */
2005-04-17 02:20:36 +04:00
unsigned long flags ; /* Status/Action flags */
unsigned long transform ; /* SCSI cmd translation layer */
unsigned long log ; /* log flags */
} idescsi_scsi_t ;
2006-03-23 14:00:22 +03:00
static DEFINE_MUTEX ( idescsi_ref_mutex ) ;
2008-04-18 02:46:27 +04:00
/* Set by module param to skip cd */
static int idescsi_nocd ;
2005-04-17 02:20:36 +04:00
# define ide_scsi_g(disk) \
container_of ( ( disk ) - > private_data , struct ide_scsi_obj , driver )
static struct ide_scsi_obj * ide_scsi_get ( struct gendisk * disk )
{
struct ide_scsi_obj * scsi = NULL ;
2006-03-23 14:00:22 +03:00
mutex_lock ( & idescsi_ref_mutex ) ;
2005-04-17 02:20:36 +04:00
scsi = ide_scsi_g ( disk ) ;
if ( scsi )
scsi_host_get ( scsi - > host ) ;
2006-03-23 14:00:22 +03:00
mutex_unlock ( & idescsi_ref_mutex ) ;
2005-04-17 02:20:36 +04:00
return scsi ;
}
static void ide_scsi_put ( struct ide_scsi_obj * scsi )
{
2006-03-23 14:00:22 +03:00
mutex_lock ( & idescsi_ref_mutex ) ;
2005-04-17 02:20:36 +04:00
scsi_host_put ( scsi - > host ) ;
2006-03-23 14:00:22 +03:00
mutex_unlock ( & idescsi_ref_mutex ) ;
2005-04-17 02:20:36 +04:00
}
static inline idescsi_scsi_t * scsihost_to_idescsi ( struct Scsi_Host * host )
{
return ( idescsi_scsi_t * ) ( & host [ 1 ] ) ;
}
static inline idescsi_scsi_t * drive_to_idescsi ( ide_drive_t * ide_drive )
{
return scsihost_to_idescsi ( ide_drive - > driver_data ) ;
}
/*
* Per ATAPI device status bits .
*/
# define IDESCSI_DRQ_INTERRUPT 0 /* DRQ interrupt device */
/*
* ide - scsi requests .
*/
# define IDESCSI_PC_RQ 90
/*
2008-07-15 23:21:52 +04:00
* PIO data transfer routine using the scatter gather table .
2005-04-17 02:20:36 +04:00
*/
2008-07-15 23:21:52 +04:00
static void ide_scsi_io_buffers ( ide_drive_t * drive , struct ide_atapi_pc * pc ,
unsigned int bcount , int write )
2005-04-17 02:20:36 +04:00
{
2008-04-29 01:44:36 +04:00
ide_hwif_t * hwif = drive - > hwif ;
2008-07-23 21:55:56 +04:00
const struct ide_tp_ops * tp_ops = hwif - > tp_ops ;
xfer_func_t * xf = write ? tp_ops - > output_data : tp_ops - > input_data ;
2005-04-17 02:20:36 +04:00
char * buf ;
int count ;
while ( bcount ) {
count = min ( pc - > sg - > length - pc - > b_count , bcount ) ;
2007-10-22 23:19:53 +04:00
if ( PageHighMem ( sg_page ( pc - > sg ) ) ) {
2005-11-01 01:08:53 +03:00
unsigned long flags ;
local_irq_save ( flags ) ;
2007-10-22 23:19:53 +04:00
buf = kmap_atomic ( sg_page ( pc - > sg ) , KM_IRQ0 ) +
2008-07-15 23:21:52 +04:00
pc - > sg - > offset ;
xf ( drive , NULL , buf + pc - > b_count , count ) ;
2005-11-01 01:08:53 +03:00
kunmap_atomic ( buf - pc - > sg - > offset , KM_IRQ0 ) ;
local_irq_restore ( flags ) ;
} else {
2007-10-22 23:19:53 +04:00
buf = sg_virt ( pc - > sg ) ;
2008-07-15 23:21:52 +04:00
xf ( drive , NULL , buf + pc - > b_count , count ) ;
2005-11-01 01:08:53 +03:00
}
bcount - = count ; pc - > b_count + = count ;
2005-04-17 02:20:36 +04:00
if ( pc - > b_count = = pc - > sg - > length ) {
2007-10-17 15:16:35 +04:00
if ( ! - - pc - > sg_cnt )
2007-10-16 13:20:52 +04:00
break ;
pc - > sg = sg_next ( pc - > sg ) ;
2005-04-17 02:20:36 +04:00
pc - > b_count = 0 ;
}
}
2007-10-16 13:20:52 +04:00
if ( bcount ) {
2008-07-15 23:21:52 +04:00
printk ( KERN_ERR " %s: scatter gather table too small, %s \n " ,
drive - > name , write ? " padding with zeros "
: " discarding data " ) ;
ide_pad_transfer ( drive , write , bcount ) ;
2007-10-16 13:20:52 +04:00
}
2005-04-17 02:20:36 +04:00
}
2007-12-13 01:31:57 +03:00
static void ide_scsi_hex_dump ( u8 * data , int len )
{
print_hex_dump ( KERN_CONT , " " , DUMP_PREFIX_NONE , 16 , 1 , data , len , 0 ) ;
}
2008-07-15 23:22:02 +04:00
static int idescsi_end_request ( ide_drive_t * , int , int ) ;
static void ide_scsi_callback ( ide_drive_t * drive )
{
idescsi_scsi_t * scsi = drive_to_idescsi ( drive ) ;
struct ide_atapi_pc * pc = scsi - > pc ;
if ( pc - > flags & PC_FLAG_TIMEDOUT )
debug_log ( " %s: got timed out packet %lu at %lu \n " , __func__ ,
pc - > scsi_cmd - > serial_number , jiffies ) ;
/* end this request now - scsi should retry it*/
else if ( test_bit ( IDESCSI_LOG_CMD , & scsi - > log ) )
printk ( KERN_INFO " Packet command completed, %d bytes "
" transferred \n " , pc - > xferred ) ;
idescsi_end_request ( drive , 1 , 0 ) ;
}
2008-04-18 02:46:27 +04:00
static int idescsi_check_condition ( ide_drive_t * drive ,
struct request * failed_cmd )
2005-04-17 02:20:36 +04:00
{
idescsi_scsi_t * scsi = drive_to_idescsi ( drive ) ;
2008-04-18 02:46:27 +04:00
struct ide_atapi_pc * pc ;
2005-04-17 02:20:36 +04:00
struct request * rq ;
u8 * buf ;
/* stuff a sense request in front of our current request */
2008-04-18 02:46:27 +04:00
pc = kzalloc ( sizeof ( struct ide_atapi_pc ) , GFP_ATOMIC ) ;
2007-08-02 01:46:45 +04:00
rq = kmalloc ( sizeof ( struct request ) , GFP_ATOMIC ) ;
buf = kzalloc ( SCSI_SENSE_BUFFERSIZE , GFP_ATOMIC ) ;
if ( ! pc | | ! rq | | ! buf ) {
2005-11-07 12:01:26 +03:00
kfree ( buf ) ;
kfree ( rq ) ;
kfree ( pc ) ;
2005-04-17 02:20:36 +04:00
return - ENOMEM ;
}
2008-07-15 23:21:44 +04:00
blk_rq_init ( NULL , rq ) ;
2005-04-17 02:20:36 +04:00
rq - > special = ( char * ) pc ;
pc - > rq = rq ;
2008-04-18 02:46:27 +04:00
pc - > buf = buf ;
2005-04-17 02:20:36 +04:00
pc - > c [ 0 ] = REQUEST_SENSE ;
2008-04-18 02:46:27 +04:00
pc - > c [ 4 ] = pc - > req_xfer = pc - > buf_size = SCSI_SENSE_BUFFERSIZE ;
2006-08-10 10:44:47 +04:00
rq - > cmd_type = REQ_TYPE_SENSE ;
2008-07-15 23:21:41 +04:00
rq - > cmd_flags | = REQ_PREEMPT ;
2005-04-17 02:20:36 +04:00
pc - > timeout = jiffies + WAIT_READY ;
/* NOTE! Save the failed packet command in "rq->buffer" */
2008-04-18 02:46:27 +04:00
rq - > buffer = ( void * ) failed_cmd - > special ;
pc - > scsi_cmd = ( ( struct ide_atapi_pc * ) failed_cmd - > special ) - > scsi_cmd ;
2005-04-17 02:20:36 +04:00
if ( test_bit ( IDESCSI_LOG_CMD , & scsi - > log ) ) {
printk ( " ide-scsi: %s: queue cmd = " , drive - > name ) ;
2007-12-13 01:31:57 +03:00
ide_scsi_hex_dump ( pc - > c , 6 ) ;
2005-04-17 02:20:36 +04:00
}
rq - > rq_disk = scsi - > disk ;
2008-07-15 23:21:51 +04:00
ide_do_drive_cmd ( drive , rq ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
static ide_startstop_t
idescsi_atapi_error ( ide_drive_t * drive , struct request * rq , u8 stat , u8 err )
{
2008-04-18 02:46:26 +04:00
ide_hwif_t * hwif = drive - > hwif ;
2008-07-23 21:55:56 +04:00
if ( hwif - > tp_ops - > read_status ( hwif ) & ( BUSY_STAT | DRQ_STAT ) )
2005-04-17 02:20:36 +04:00
/* force an abort */
2008-07-23 21:55:56 +04:00
hwif - > tp_ops - > exec_command ( hwif , WIN_IDLEIMMEDIATE ) ;
2005-04-17 02:20:36 +04:00
rq - > errors + + ;
idescsi_end_request ( drive , 0 , 0 ) ;
return ide_stopped ;
}
static int idescsi_end_request ( ide_drive_t * drive , int uptodate , int nrsecs )
{
idescsi_scsi_t * scsi = drive_to_idescsi ( drive ) ;
struct request * rq = HWGROUP ( drive ) - > rq ;
2008-04-18 02:46:27 +04:00
struct ide_atapi_pc * pc = ( struct ide_atapi_pc * ) rq - > special ;
2005-04-17 02:20:36 +04:00
int log = test_bit ( IDESCSI_LOG_CMD , & scsi - > log ) ;
struct Scsi_Host * host ;
2005-10-31 03:03:49 +03:00
int errors = rq - > errors ;
2005-04-17 02:20:36 +04:00
unsigned long flags ;
2006-08-10 10:44:47 +04:00
if ( ! blk_special_request ( rq ) & & ! blk_sense_request ( rq ) ) {
2005-04-17 02:20:36 +04:00
ide_end_request ( drive , uptodate , nrsecs ) ;
return 0 ;
}
ide_end_drive_cmd ( drive , 0 , 0 ) ;
2006-08-10 10:44:47 +04:00
if ( blk_sense_request ( rq ) ) {
2008-04-18 02:46:27 +04:00
struct ide_atapi_pc * opc = ( struct ide_atapi_pc * ) rq - > buffer ;
2005-04-17 02:20:36 +04:00
if ( log ) {
printk ( " ide-scsi: %s: wrap up check %lu, rst = " , drive - > name , opc - > scsi_cmd - > serial_number ) ;
2008-04-18 02:46:27 +04:00
ide_scsi_hex_dump ( pc - > buf , 16 ) ;
2005-04-17 02:20:36 +04:00
}
2008-04-18 02:46:27 +04:00
memcpy ( ( void * ) opc - > scsi_cmd - > sense_buffer , pc - > buf ,
SCSI_SENSE_BUFFERSIZE ) ;
kfree ( pc - > buf ) ;
2005-04-17 02:20:36 +04:00
kfree ( pc ) ;
kfree ( rq ) ;
pc = opc ;
rq = pc - > rq ;
pc - > scsi_cmd - > result = ( CHECK_CONDITION < < 1 ) |
2008-04-18 02:46:27 +04:00
( ( ( pc - > flags & PC_FLAG_TIMEDOUT ) ?
DID_TIME_OUT :
DID_OK ) < < 16 ) ;
} else if ( pc - > flags & PC_FLAG_TIMEDOUT ) {
2005-04-17 02:20:36 +04:00
if ( log )
printk ( KERN_WARNING " ide-scsi: %s: timed out for %lu \n " ,
drive - > name , pc - > scsi_cmd - > serial_number ) ;
pc - > scsi_cmd - > result = DID_TIME_OUT < < 16 ;
2005-10-31 03:03:49 +03:00
} else if ( errors > = ERROR_MAX ) {
2005-04-17 02:20:36 +04:00
pc - > scsi_cmd - > result = DID_ERROR < < 16 ;
if ( log )
printk ( " ide-scsi: %s: I/O error for %lu \n " , drive - > name , pc - > scsi_cmd - > serial_number ) ;
2005-10-31 03:03:49 +03:00
} else if ( errors ) {
2005-04-17 02:20:36 +04:00
if ( log )
printk ( " ide-scsi: %s: check condition for %lu \n " , drive - > name , pc - > scsi_cmd - > serial_number ) ;
if ( ! idescsi_check_condition ( drive , rq ) )
/* we started a request sense, so we'll be back, exit for now */
return 0 ;
pc - > scsi_cmd - > result = ( CHECK_CONDITION < < 1 ) | ( DID_OK < < 16 ) ;
} else {
pc - > scsi_cmd - > result = DID_OK < < 16 ;
}
host = pc - > scsi_cmd - > device - > host ;
spin_lock_irqsave ( host - > host_lock , flags ) ;
pc - > done ( pc - > scsi_cmd ) ;
spin_unlock_irqrestore ( host - > host_lock , flags ) ;
kfree ( pc ) ;
kfree ( rq ) ;
scsi - > pc = NULL ;
return 0 ;
}
2008-04-18 02:46:27 +04:00
static inline unsigned long get_timeout ( struct ide_atapi_pc * pc )
2005-04-17 02:20:36 +04:00
{
return max_t ( unsigned long , WAIT_CMD , pc - > timeout - jiffies ) ;
}
static int idescsi_expiry ( ide_drive_t * drive )
{
2007-06-16 04:24:44 +04:00
idescsi_scsi_t * scsi = drive_to_idescsi ( drive ) ;
2008-04-18 02:46:27 +04:00
struct ide_atapi_pc * pc = scsi - > pc ;
2005-04-17 02:20:36 +04:00
2008-07-15 23:21:52 +04:00
debug_log ( " %s called for %lu at %lu \n " , __func__ ,
pc - > scsi_cmd - > serial_number , jiffies ) ;
2008-04-18 02:46:27 +04:00
pc - > flags | = PC_FLAG_TIMEDOUT ;
2005-04-17 02:20:36 +04:00
return 0 ; /* we do not want the ide subsystem to retry */
}
/*
* Our interrupt handler .
*/
static ide_startstop_t idescsi_pc_intr ( ide_drive_t * drive )
{
idescsi_scsi_t * scsi = drive_to_idescsi ( drive ) ;
2008-04-18 02:46:27 +04:00
struct ide_atapi_pc * pc = scsi - > pc ;
2005-04-17 02:20:36 +04:00
2008-07-15 23:22:03 +04:00
return ide_pc_intr ( drive , pc , idescsi_pc_intr , get_timeout ( pc ) ,
idescsi_expiry , NULL , NULL , NULL ,
ide_scsi_io_buffers ) ;
2005-04-17 02:20:36 +04:00
}
static ide_startstop_t idescsi_transfer_pc ( ide_drive_t * drive )
{
idescsi_scsi_t * scsi = drive_to_idescsi ( drive ) ;
2008-07-15 23:21:52 +04:00
2008-07-15 23:21:58 +04:00
return ide_transfer_pc ( drive , scsi - > pc , idescsi_pc_intr ,
get_timeout ( scsi - > pc ) , idescsi_expiry ) ;
2005-04-17 02:20:36 +04:00
}
2008-04-18 02:46:27 +04:00
static inline int idescsi_set_direction ( struct ide_atapi_pc * pc )
2005-04-17 02:20:36 +04:00
{
switch ( pc - > c [ 0 ] ) {
case READ_6 : case READ_10 : case READ_12 :
2008-04-18 02:46:27 +04:00
pc - > flags & = ~ PC_FLAG_WRITING ;
2005-04-17 02:20:36 +04:00
return 0 ;
case WRITE_6 : case WRITE_10 : case WRITE_12 :
2008-04-18 02:46:27 +04:00
pc - > flags | = PC_FLAG_WRITING ;
2005-04-17 02:20:36 +04:00
return 0 ;
default :
return 1 ;
}
}
2008-04-18 02:46:27 +04:00
static int idescsi_map_sg ( ide_drive_t * drive , struct ide_atapi_pc * pc )
2005-04-17 02:20:36 +04:00
{
ide_hwif_t * hwif = drive - > hwif ;
struct scatterlist * sg , * scsi_sg ;
int segments ;
2008-04-18 02:46:27 +04:00
if ( ! pc - > req_xfer | | pc - > req_xfer % 1024 )
2005-04-17 02:20:36 +04:00
return 1 ;
if ( idescsi_set_direction ( pc ) )
return 1 ;
sg = hwif - > sg_table ;
2007-09-18 14:27:43 +04:00
scsi_sg = scsi_sglist ( pc - > scsi_cmd ) ;
segments = scsi_sg_count ( pc - > scsi_cmd ) ;
2005-04-17 02:20:36 +04:00
if ( segments > hwif - > sg_max_nents )
return 1 ;
2007-09-18 14:27:43 +04:00
hwif - > sg_nents = segments ;
memcpy ( sg , scsi_sg , sizeof ( * sg ) * segments ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2008-04-18 02:46:27 +04:00
static ide_startstop_t idescsi_issue_pc ( ide_drive_t * drive ,
struct ide_atapi_pc * pc )
2005-04-17 02:20:36 +04:00
{
idescsi_scsi_t * scsi = drive_to_idescsi ( drive ) ;
2008-04-18 02:46:27 +04:00
/* Set the current packet command */
scsi - > pc = pc ;
2005-04-17 02:20:36 +04:00
2008-07-15 23:22:00 +04:00
return ide_issue_pc ( drive , pc , idescsi_transfer_pc ,
get_timeout ( pc ) , idescsi_expiry ) ;
2005-04-17 02:20:36 +04:00
}
/*
* idescsi_do_request is our request handling function .
*/
static ide_startstop_t idescsi_do_request ( ide_drive_t * drive , struct request * rq , sector_t block )
{
2008-07-15 23:21:52 +04:00
debug_log ( " dev: %s, cmd: %x, errors: %d \n " , rq - > rq_disk - > disk_name ,
rq - > cmd [ 0 ] , rq - > errors ) ;
debug_log ( " sector: %ld, nr_sectors: %ld, current_nr_sectors: %d \n " ,
rq - > sector , rq - > nr_sectors , rq - > current_nr_sectors ) ;
2005-04-17 02:20:36 +04:00
2006-08-10 10:44:47 +04:00
if ( blk_sense_request ( rq ) | | blk_special_request ( rq ) ) {
2008-07-15 23:21:58 +04:00
struct ide_atapi_pc * pc = ( struct ide_atapi_pc * ) rq - > special ;
2008-07-15 23:21:59 +04:00
idescsi_scsi_t * scsi = drive_to_idescsi ( drive ) ;
if ( test_bit ( IDESCSI_DRQ_INTERRUPT , & scsi - > flags ) )
pc - > flags | = PC_FLAG_DRQ_INTERRUPT ;
2008-07-15 23:21:58 +04:00
if ( drive - > using_dma & & ! idescsi_map_sg ( drive , pc ) )
pc - > flags | = PC_FLAG_DMA_OK ;
return idescsi_issue_pc ( drive , pc ) ;
2005-04-17 02:20:36 +04:00
}
blk_dump_rq_flags ( rq , " ide-scsi: unsup command " ) ;
idescsi_end_request ( drive , 0 , 0 ) ;
return ide_stopped ;
}
2007-05-10 02:01:10 +04:00
# ifdef CONFIG_IDE_PROC_FS
2005-04-17 02:20:36 +04:00
static void idescsi_add_settings ( ide_drive_t * drive )
{
idescsi_scsi_t * scsi = drive_to_idescsi ( drive ) ;
/*
2007-05-10 02:01:10 +04:00
* drive setting name read / write data type min max mul_factor div_factor data pointer set function
2005-04-17 02:20:36 +04:00
*/
2007-05-10 02:01:10 +04:00
ide_add_setting ( drive , " bios_cyl " , SETTING_RW , TYPE_INT , 0 , 1023 , 1 , 1 , & drive - > bios_cyl , NULL ) ;
ide_add_setting ( drive , " bios_head " , SETTING_RW , TYPE_BYTE , 0 , 255 , 1 , 1 , & drive - > bios_head , NULL ) ;
ide_add_setting ( drive , " bios_sect " , SETTING_RW , TYPE_BYTE , 0 , 63 , 1 , 1 , & drive - > bios_sect , NULL ) ;
ide_add_setting ( drive , " transform " , SETTING_RW , TYPE_INT , 0 , 3 , 1 , 1 , & scsi - > transform , NULL ) ;
ide_add_setting ( drive , " log " , SETTING_RW , TYPE_INT , 0 , 1 , 1 , 1 , & scsi - > log , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2007-05-10 02:01:10 +04:00
# else
static inline void idescsi_add_settings ( ide_drive_t * drive ) { ; }
# endif
2005-04-17 02:20:36 +04:00
/*
* Driver initialization .
*/
static void idescsi_setup ( ide_drive_t * drive , idescsi_scsi_t * scsi )
{
if ( drive - > id & & ( drive - > id - > config & 0x0060 ) = = 0x20 )
set_bit ( IDESCSI_DRQ_INTERRUPT , & scsi - > flags ) ;
clear_bit ( IDESCSI_SG_TRANSFORM , & scsi - > transform ) ;
# if IDESCSI_DEBUG_LOG
set_bit ( IDESCSI_LOG_CMD , & scsi - > log ) ;
# endif /* IDESCSI_DEBUG_LOG */
2008-07-23 21:55:59 +04:00
drive - > pc_callback = ide_scsi_callback ;
2005-04-17 02:20:36 +04:00
idescsi_add_settings ( drive ) ;
}
2006-02-01 14:04:45 +03:00
static void ide_scsi_remove ( ide_drive_t * drive )
2005-04-17 02:20:36 +04:00
{
struct Scsi_Host * scsihost = drive - > driver_data ;
struct ide_scsi_obj * scsi = scsihost_to_idescsi ( scsihost ) ;
struct gendisk * g = scsi - > disk ;
2007-08-15 22:57:01 +04:00
scsi_remove_host ( scsihost ) ;
2007-05-10 02:01:10 +04:00
ide_proc_unregister_driver ( drive , scsi - > driver ) ;
2005-04-17 02:20:36 +04:00
ide_unregister_region ( g ) ;
drive - > driver_data = NULL ;
g - > private_data = NULL ;
put_disk ( g ) ;
ide_scsi_put ( scsi ) ;
2008-07-15 23:21:58 +04:00
drive - > scsi = 0 ;
2005-04-17 02:20:36 +04:00
}
2006-02-01 14:04:45 +03:00
static int ide_scsi_probe ( ide_drive_t * ) ;
2005-04-17 02:20:36 +04:00
2007-05-10 02:01:09 +04:00
# ifdef CONFIG_IDE_PROC_FS
2005-04-17 02:20:36 +04:00
static ide_proc_entry_t idescsi_proc [ ] = {
{ " capacity " , S_IFREG | S_IRUGO , proc_ide_read_capacity , NULL } ,
{ NULL , 0 , NULL , NULL }
} ;
# endif
static ide_driver_t idescsi_driver = {
2005-05-26 16:55:34 +04:00
. gen_driver = {
2005-11-19 00:15:40 +03:00
. owner = THIS_MODULE ,
2005-05-26 16:55:34 +04:00
. name = " ide-scsi " ,
. bus = & ide_bus_type ,
} ,
2006-02-01 14:04:45 +03:00
. probe = ide_scsi_probe ,
. remove = ide_scsi_remove ,
2005-04-17 02:20:36 +04:00
. version = IDESCSI_VERSION ,
. media = ide_scsi ,
. supports_dsc_overlap = 0 ,
. do_request = idescsi_do_request ,
. end_request = idescsi_end_request ,
. error = idescsi_atapi_error ,
2007-05-10 02:01:10 +04:00
# ifdef CONFIG_IDE_PROC_FS
. proc = idescsi_proc ,
# endif
2005-04-17 02:20:36 +04:00
} ;
static int idescsi_ide_open ( struct inode * inode , struct file * filp )
{
struct gendisk * disk = inode - > i_bdev - > bd_disk ;
struct ide_scsi_obj * scsi ;
if ( ! ( scsi = ide_scsi_get ( disk ) ) )
return - ENXIO ;
return 0 ;
}
static int idescsi_ide_release ( struct inode * inode , struct file * filp )
{
struct gendisk * disk = inode - > i_bdev - > bd_disk ;
struct ide_scsi_obj * scsi = ide_scsi_g ( disk ) ;
ide_scsi_put ( scsi ) ;
return 0 ;
}
static int idescsi_ide_ioctl ( struct inode * inode , struct file * file ,
unsigned int cmd , unsigned long arg )
{
struct block_device * bdev = inode - > i_bdev ;
struct ide_scsi_obj * scsi = ide_scsi_g ( bdev - > bd_disk ) ;
return generic_ide_ioctl ( scsi - > drive , file , bdev , cmd , arg ) ;
}
static struct block_device_operations idescsi_ops = {
. owner = THIS_MODULE ,
. open = idescsi_ide_open ,
. release = idescsi_ide_release ,
. ioctl = idescsi_ide_ioctl ,
} ;
static int idescsi_slave_configure ( struct scsi_device * sdp )
{
/* Configure detected device */
2007-09-18 14:27:43 +04:00
sdp - > use_10_for_rw = 1 ;
sdp - > use_10_for_ms = 1 ;
2005-04-17 02:20:36 +04:00
scsi_adjust_queue_depth ( sdp , MSG_SIMPLE_TAG , sdp - > host - > cmd_per_lun ) ;
return 0 ;
}
static const char * idescsi_info ( struct Scsi_Host * host )
{
return " SCSI host adapter emulation for IDE ATAPI devices " ;
}
static int idescsi_ioctl ( struct scsi_device * dev , int cmd , void __user * arg )
{
idescsi_scsi_t * scsi = scsihost_to_idescsi ( dev - > host ) ;
if ( cmd = = SG_SET_TRANSFORM ) {
if ( arg )
set_bit ( IDESCSI_SG_TRANSFORM , & scsi - > transform ) ;
else
clear_bit ( IDESCSI_SG_TRANSFORM , & scsi - > transform ) ;
return 0 ;
} else if ( cmd = = SG_GET_TRANSFORM )
return put_user ( test_bit ( IDESCSI_SG_TRANSFORM , & scsi - > transform ) , ( int __user * ) arg ) ;
return - EINVAL ;
}
static int idescsi_queue ( struct scsi_cmnd * cmd ,
void ( * done ) ( struct scsi_cmnd * ) )
{
struct Scsi_Host * host = cmd - > device - > host ;
idescsi_scsi_t * scsi = scsihost_to_idescsi ( host ) ;
ide_drive_t * drive = scsi - > drive ;
struct request * rq = NULL ;
2008-04-18 02:46:27 +04:00
struct ide_atapi_pc * pc = NULL ;
2005-04-17 02:20:36 +04:00
if ( ! drive ) {
2005-10-25 02:04:36 +04:00
scmd_printk ( KERN_ERR , cmd , " drive not present \n " ) ;
2005-04-17 02:20:36 +04:00
goto abort ;
}
scsi = drive_to_idescsi ( drive ) ;
2008-04-18 02:46:27 +04:00
pc = kmalloc ( sizeof ( struct ide_atapi_pc ) , GFP_ATOMIC ) ;
rq = kmalloc ( sizeof ( struct request ) , GFP_ATOMIC ) ;
2005-04-17 02:20:36 +04:00
if ( rq = = NULL | | pc = = NULL ) {
printk ( KERN_ERR " ide-scsi: %s: out of memory \n " , drive - > name ) ;
goto abort ;
}
memset ( pc - > c , 0 , 12 ) ;
pc - > flags = 0 ;
2008-07-15 23:21:52 +04:00
if ( cmd - > sc_data_direction = = DMA_TO_DEVICE )
pc - > flags | = PC_FLAG_WRITING ;
2005-04-17 02:20:36 +04:00
pc - > rq = rq ;
memcpy ( pc - > c , cmd - > cmnd , cmd - > cmd_len ) ;
2008-04-18 02:46:27 +04:00
pc - > buf = NULL ;
2007-09-18 14:27:43 +04:00
pc - > sg = scsi_sglist ( cmd ) ;
2007-10-17 15:16:35 +04:00
pc - > sg_cnt = scsi_sg_count ( cmd ) ;
2005-04-17 02:20:36 +04:00
pc - > b_count = 0 ;
2008-04-18 02:46:27 +04:00
pc - > req_xfer = pc - > buf_size = scsi_bufflen ( cmd ) ;
2005-04-17 02:20:36 +04:00
pc - > scsi_cmd = cmd ;
pc - > done = done ;
pc - > timeout = jiffies + cmd - > timeout_per_command ;
if ( test_bit ( IDESCSI_LOG_CMD , & scsi - > log ) ) {
printk ( " ide-scsi: %s: que %lu, cmd = " , drive - > name , cmd - > serial_number ) ;
2007-12-13 01:31:57 +03:00
ide_scsi_hex_dump ( cmd - > cmnd , cmd - > cmd_len ) ;
2005-04-17 02:20:36 +04:00
if ( memcmp ( pc - > c , cmd - > cmnd , cmd - > cmd_len ) ) {
printk ( " ide-scsi: %s: que %lu, tsl = " , drive - > name , cmd - > serial_number ) ;
2007-12-13 01:31:57 +03:00
ide_scsi_hex_dump ( pc - > c , 12 ) ;
2005-04-17 02:20:36 +04:00
}
}
2008-07-15 23:21:44 +04:00
blk_rq_init ( NULL , rq ) ;
2005-04-17 02:20:36 +04:00
rq - > special = ( char * ) pc ;
2006-08-10 10:44:47 +04:00
rq - > cmd_type = REQ_TYPE_SPECIAL ;
2005-04-17 02:20:36 +04:00
spin_unlock_irq ( host - > host_lock ) ;
2008-07-15 23:21:50 +04:00
blk_execute_rq_nowait ( drive - > queue , scsi - > disk , rq , 0 , NULL ) ;
2005-04-17 02:20:36 +04:00
spin_lock_irq ( host - > host_lock ) ;
return 0 ;
abort :
2005-11-07 12:01:26 +03:00
kfree ( pc ) ;
kfree ( rq ) ;
2005-04-17 02:20:36 +04:00
cmd - > result = DID_ERROR < < 16 ;
done ( cmd ) ;
return 0 ;
}
static int idescsi_eh_abort ( struct scsi_cmnd * cmd )
{
idescsi_scsi_t * scsi = scsihost_to_idescsi ( cmd - > device - > host ) ;
ide_drive_t * drive = scsi - > drive ;
int busy ;
int ret = FAILED ;
/* In idescsi_eh_abort we try to gently pry our command from the ide subsystem */
if ( test_bit ( IDESCSI_LOG_CMD , & scsi - > log ) )
printk ( KERN_WARNING " ide-scsi: abort called for %lu \n " , cmd - > serial_number ) ;
if ( ! drive ) {
printk ( KERN_WARNING " ide-scsi: Drive not set in idescsi_eh_abort \n " ) ;
WARN_ON ( 1 ) ;
goto no_drive ;
}
/* First give it some more time, how much is "right" is hard to say :-( */
busy = ide_wait_not_busy ( HWIF ( drive ) , 100 ) ; /* FIXME - uses mdelay which causes latency? */
if ( test_bit ( IDESCSI_LOG_CMD , & scsi - > log ) )
printk ( KERN_WARNING " ide-scsi: drive did%s become ready \n " , busy ? " not " : " " ) ;
spin_lock_irq ( & ide_lock ) ;
/* If there is no pc running we're done (our interrupt took care of it) */
if ( ! scsi - > pc ) {
ret = SUCCESS ;
goto ide_unlock ;
}
/* It's somewhere in flight. Does ide subsystem agree? */
if ( scsi - > pc - > scsi_cmd - > serial_number = = cmd - > serial_number & & ! busy & &
elv_queue_empty ( drive - > queue ) & & HWGROUP ( drive ) - > rq ! = scsi - > pc - > rq ) {
/*
* FIXME - not sure this condition can ever occur
*/
printk ( KERN_ERR " ide-scsi: cmd aborted! \n " ) ;
2006-08-10 10:44:47 +04:00
if ( blk_sense_request ( scsi - > pc - > rq ) )
2008-04-18 02:46:27 +04:00
kfree ( scsi - > pc - > buf ) ;
2005-04-17 02:20:36 +04:00
kfree ( scsi - > pc - > rq ) ;
kfree ( scsi - > pc ) ;
scsi - > pc = NULL ;
ret = SUCCESS ;
}
ide_unlock :
spin_unlock_irq ( & ide_lock ) ;
no_drive :
if ( test_bit ( IDESCSI_LOG_CMD , & scsi - > log ) )
printk ( KERN_WARNING " ide-scsi: abort returns %s \n " , ret = = SUCCESS ? " success " : " failed " ) ;
return ret ;
}
static int idescsi_eh_reset ( struct scsi_cmnd * cmd )
{
struct request * req ;
idescsi_scsi_t * scsi = scsihost_to_idescsi ( cmd - > device - > host ) ;
ide_drive_t * drive = scsi - > drive ;
int ready = 0 ;
int ret = SUCCESS ;
/* In idescsi_eh_reset we forcefully remove the command from the ide subsystem and reset the device. */
if ( test_bit ( IDESCSI_LOG_CMD , & scsi - > log ) )
printk ( KERN_WARNING " ide-scsi: reset called for %lu \n " , cmd - > serial_number ) ;
if ( ! drive ) {
printk ( KERN_WARNING " ide-scsi: Drive not set in idescsi_eh_reset \n " ) ;
WARN_ON ( 1 ) ;
return FAILED ;
}
2005-05-28 15:57:14 +04:00
spin_lock_irq ( cmd - > device - > host - > host_lock ) ;
spin_lock ( & ide_lock ) ;
2005-04-17 02:20:36 +04:00
if ( ! scsi - > pc | | ( req = scsi - > pc - > rq ) ! = HWGROUP ( drive ) - > rq | | ! HWGROUP ( drive ) - > handler ) {
printk ( KERN_WARNING " ide-scsi: No active request in idescsi_eh_reset \n " ) ;
spin_unlock ( & ide_lock ) ;
2005-05-28 15:57:14 +04:00
spin_unlock_irq ( cmd - > device - > host - > host_lock ) ;
2005-04-17 02:20:36 +04:00
return FAILED ;
}
/* kill current request */
2007-12-12 01:49:29 +03:00
if ( __blk_end_request ( req , - EIO , 0 ) )
BUG ( ) ;
2006-08-10 10:44:47 +04:00
if ( blk_sense_request ( req ) )
2008-04-18 02:46:27 +04:00
kfree ( scsi - > pc - > buf ) ;
2005-04-17 02:20:36 +04:00
kfree ( scsi - > pc ) ;
scsi - > pc = NULL ;
kfree ( req ) ;
/* now nuke the drive queue */
while ( ( req = elv_next_request ( drive - > queue ) ) ) {
2007-12-12 01:49:29 +03:00
if ( __blk_end_request ( req , - EIO , 0 ) )
BUG ( ) ;
2005-04-17 02:20:36 +04:00
}
HWGROUP ( drive ) - > rq = NULL ;
HWGROUP ( drive ) - > handler = NULL ;
HWGROUP ( drive ) - > busy = 1 ; /* will set this to zero when ide reset finished */
2005-05-28 15:57:14 +04:00
spin_unlock ( & ide_lock ) ;
2005-04-17 02:20:36 +04:00
ide_do_reset ( drive ) ;
/* ide_do_reset starts a polling handler which restarts itself every 50ms until the reset finishes */
do {
spin_unlock_irq ( cmd - > device - > host - > host_lock ) ;
2005-05-28 15:57:14 +04:00
msleep ( 50 ) ;
2005-04-17 02:20:36 +04:00
spin_lock_irq ( cmd - > device - > host - > host_lock ) ;
} while ( HWGROUP ( drive ) - > handler ) ;
ready = drive_is_ready ( drive ) ;
HWGROUP ( drive ) - > busy - - ;
if ( ! ready ) {
printk ( KERN_ERR " ide-scsi: reset failed! \n " ) ;
ret = FAILED ;
}
2005-05-28 15:57:14 +04:00
spin_unlock_irq ( cmd - > device - > host - > host_lock ) ;
2005-04-17 02:20:36 +04:00
return ret ;
}
static int idescsi_bios ( struct scsi_device * sdev , struct block_device * bdev ,
sector_t capacity , int * parm )
{
idescsi_scsi_t * idescsi = scsihost_to_idescsi ( sdev - > host ) ;
ide_drive_t * drive = idescsi - > drive ;
if ( drive - > bios_cyl & & drive - > bios_head & & drive - > bios_sect ) {
parm [ 0 ] = drive - > bios_head ;
parm [ 1 ] = drive - > bios_sect ;
parm [ 2 ] = drive - > bios_cyl ;
}
return 0 ;
}
static struct scsi_host_template idescsi_template = {
. module = THIS_MODULE ,
. name = " idescsi " ,
. info = idescsi_info ,
. slave_configure = idescsi_slave_configure ,
. ioctl = idescsi_ioctl ,
. queuecommand = idescsi_queue ,
. eh_abort_handler = idescsi_eh_abort ,
. eh_host_reset_handler = idescsi_eh_reset ,
. bios_param = idescsi_bios ,
. can_queue = 40 ,
. this_id = - 1 ,
. sg_tablesize = 256 ,
. cmd_per_lun = 5 ,
. max_sectors = 128 ,
. use_clustering = DISABLE_CLUSTERING ,
. emulated = 1 ,
. proc_name = " ide-scsi " ,
} ;
2006-02-01 14:04:45 +03:00
static int ide_scsi_probe ( ide_drive_t * drive )
2005-04-17 02:20:36 +04:00
{
idescsi_scsi_t * idescsi ;
struct Scsi_Host * host ;
struct gendisk * g ;
static int warned ;
int err = - ENOMEM ;
if ( ! warned & & drive - > media = = ide_cdrom ) {
printk ( KERN_WARNING " ide-scsi is deprecated for cd burning! Use ide-cd and give dev=/dev/hdX as device \n " ) ;
warned = 1 ;
}
2006-12-07 07:39:33 +03:00
if ( idescsi_nocd & & drive - > media = = ide_cdrom )
return - ENODEV ;
2005-04-17 02:20:36 +04:00
if ( ! strstr ( " ide-scsi " , drive - > driver_req ) | |
! drive - > present | |
drive - > media = = ide_disk | |
! ( host = scsi_host_alloc ( & idescsi_template , sizeof ( idescsi_scsi_t ) ) ) )
2005-05-26 16:55:34 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2008-07-15 23:21:58 +04:00
drive - > scsi = 1 ;
2005-04-17 02:20:36 +04:00
g = alloc_disk ( 1 < < PARTN_BITS ) ;
if ( ! g )
goto out_host_put ;
ide_init_disk ( g , drive ) ;
host - > max_id = 1 ;
if ( drive - > id - > last_lun )
2008-07-15 23:21:52 +04:00
debug_log ( " %s: id->last_lun=%u \n " , drive - > name ,
drive - > id - > last_lun ) ;
2005-04-17 02:20:36 +04:00
if ( ( drive - > id - > last_lun & 0x7 ) ! = 7 )
host - > max_lun = ( drive - > id - > last_lun & 0x7 ) + 1 ;
else
host - > max_lun = 1 ;
drive - > driver_data = host ;
idescsi = scsihost_to_idescsi ( host ) ;
idescsi - > drive = drive ;
idescsi - > driver = & idescsi_driver ;
idescsi - > host = host ;
idescsi - > disk = g ;
g - > private_data = & idescsi - > driver ;
2007-05-10 02:01:10 +04:00
ide_proc_register_driver ( drive , & idescsi_driver ) ;
2005-05-26 16:55:34 +04:00
err = 0 ;
idescsi_setup ( drive , idescsi ) ;
g - > fops = & idescsi_ops ;
ide_register_region ( g ) ;
err = scsi_add_host ( host , & drive - > gendev ) ;
2005-04-17 02:20:36 +04:00
if ( ! err ) {
2005-05-26 16:55:34 +04:00
scsi_scan_host ( host ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
2005-05-26 16:55:34 +04:00
/* fall through on error */
ide_unregister_region ( g ) ;
2007-05-10 02:01:10 +04:00
ide_proc_unregister_driver ( drive , & idescsi_driver ) ;
2005-04-17 02:20:36 +04:00
put_disk ( g ) ;
out_host_put :
2008-07-15 23:21:58 +04:00
drive - > scsi = 0 ;
2005-04-17 02:20:36 +04:00
scsi_host_put ( host ) ;
return err ;
}
static int __init init_idescsi_module ( void )
{
2005-05-26 16:55:34 +04:00
return driver_register ( & idescsi_driver . gen_driver ) ;
2005-04-17 02:20:36 +04:00
}
static void __exit exit_idescsi_module ( void )
{
2005-05-26 16:55:34 +04:00
driver_unregister ( & idescsi_driver . gen_driver ) ;
2005-04-17 02:20:36 +04:00
}
2006-12-07 07:39:33 +03:00
module_param ( idescsi_nocd , int , 0600 ) ;
MODULE_PARM_DESC ( idescsi_nocd , " Disable handling of CD-ROMs so they may be driven by ide-cd " ) ;
2005-04-17 02:20:36 +04:00
module_init ( init_idescsi_module ) ;
module_exit ( exit_idescsi_module ) ;
MODULE_LICENSE ( " GPL " ) ;