2005-04-16 15:20:36 -07:00
/*
2008-02-01 23:09:33 +01:00
* Copyright ( C ) 2000 - 2002 Michael Cornwell < cornwell @ acm . org >
* Copyright ( C ) 2000 - 2002 Andre Hedrick < andre @ linux - ide . org >
* Copyright ( C ) 2001 - 2002 Klaus Smolin
2005-04-16 15:20:36 -07:00
* IBM Storage Technology Division
2008-02-01 23:09:33 +01:00
* Copyright ( C ) 2003 - 2004 , 2007 Bartlomiej Zolnierkiewicz
2005-04-16 15:20:36 -07:00
*
* The big the bad and the ugly .
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/string.h>
# include <linux/kernel.h>
# include <linux/timer.h>
# include <linux/mm.h>
2006-02-15 15:17:37 -08:00
# include <linux/sched.h>
2005-04-16 15:20:36 -07:00
# include <linux/interrupt.h>
# include <linux/major.h>
# include <linux/errno.h>
# include <linux/genhd.h>
# include <linux/blkpg.h>
# include <linux/slab.h>
# include <linux/pci.h>
# include <linux/delay.h>
# include <linux/hdreg.h>
# include <linux/ide.h>
# include <linux/bitops.h>
2007-07-25 08:13:56 +02:00
# include <linux/scatterlist.h>
2005-04-16 15:20:36 -07:00
# include <asm/byteorder.h>
# include <asm/irq.h>
# include <asm/uaccess.h>
# include <asm/io.h>
2008-04-28 23:44:39 +02:00
void ide_tf_dump ( const char * s , struct ide_taskfile * tf )
2008-01-25 22:17:07 +01:00
{
2008-01-25 22:17:10 +01:00
# ifdef DEBUG
printk ( " %s: tf: feat 0x%02x nsect 0x%02x lbal 0x%02x "
" lbam 0x%02x lbah 0x%02x dev 0x%02x cmd 0x%02x \n " ,
2008-04-28 23:44:39 +02:00
s , tf - > feature , tf - > nsect , tf - > lbal ,
2008-01-25 22:17:10 +01:00
tf - > lbam , tf - > lbah , tf - > device , tf - > command ) ;
2008-01-26 20:13:03 +01:00
printk ( " %s: hob: nsect 0x%02x lbal 0x%02x "
" lbam 0x%02x lbah 0x%02x \n " ,
2008-04-28 23:44:39 +02:00
s , tf - > hob_nsect , tf - > hob_lbal ,
2008-01-26 20:13:03 +01:00
tf - > hob_lbam , tf - > hob_lbah ) ;
2008-01-25 22:17:10 +01:00
# endif
2008-04-28 23:44:39 +02:00
}
2005-04-16 15:20:36 -07:00
int taskfile_lib_get_identify ( ide_drive_t * drive , u8 * buf )
{
ide_task_t args ;
2008-01-25 22:17:06 +01:00
2005-04-16 15:20:36 -07:00
memset ( & args , 0 , sizeof ( ide_task_t ) ) ;
2008-01-25 22:17:06 +01:00
args . tf . nsect = 0x01 ;
2005-04-16 15:20:36 -07:00
if ( drive - > media = = ide_disk )
2008-01-25 22:17:06 +01:00
args . tf . command = WIN_IDENTIFY ;
2005-04-16 15:20:36 -07:00
else
2008-01-25 22:17:06 +01:00
args . tf . command = WIN_PIDENTIFY ;
2008-01-26 20:13:10 +01:00
args . tf_flags = IDE_TFLAG_TF | IDE_TFLAG_DEVICE ;
2008-01-25 22:17:14 +01:00
args . data_phase = TASKFILE_IN ;
return ide_raw_taskfile ( drive , & args , buf , 1 ) ;
2005-04-16 15:20:36 -07:00
}
2008-01-25 22:17:07 +01:00
static int inline task_dma_ok ( ide_task_t * task )
{
2008-01-25 22:17:16 +01:00
if ( blk_fs_request ( task - > rq ) | | ( task - > tf_flags & IDE_TFLAG_FLAGGED ) )
2008-01-25 22:17:07 +01:00
return 1 ;
switch ( task - > tf . command ) {
case WIN_WRITEDMA_ONCE :
case WIN_WRITEDMA :
case WIN_WRITEDMA_EXT :
case WIN_READDMA_ONCE :
case WIN_READDMA :
case WIN_READDMA_EXT :
case WIN_IDENTIFY_DMA :
return 1 ;
}
return 0 ;
}
2008-01-25 22:17:16 +01:00
static ide_startstop_t task_no_data_intr ( ide_drive_t * ) ;
2008-01-25 22:17:16 +01:00
static ide_startstop_t set_geometry_intr ( ide_drive_t * ) ;
static ide_startstop_t recal_intr ( ide_drive_t * ) ;
static ide_startstop_t set_multmode_intr ( ide_drive_t * ) ;
2008-01-25 22:17:16 +01:00
static ide_startstop_t pre_task_out_intr ( ide_drive_t * , struct request * ) ;
static ide_startstop_t task_in_intr ( ide_drive_t * ) ;
2008-01-25 22:17:16 +01:00
2005-04-16 15:20:36 -07:00
ide_startstop_t do_rw_taskfile ( ide_drive_t * drive , ide_task_t * task )
{
ide_hwif_t * hwif = HWIF ( drive ) ;
2008-01-25 22:17:06 +01:00
struct ide_taskfile * tf = & task - > tf ;
2008-01-25 22:17:16 +01:00
ide_handler_t * handler = NULL ;
2008-04-26 22:25:24 +02:00
const struct ide_dma_ops * dma_ops = hwif - > dma_ops ;
2005-04-16 15:20:36 -07:00
2008-01-25 22:17:15 +01:00
if ( task - > data_phase = = TASKFILE_MULTI_IN | |
task - > data_phase = = TASKFILE_MULTI_OUT ) {
if ( ! drive - > mult_count ) {
printk ( KERN_ERR " %s: multimode not set! \n " ,
drive - > name ) ;
return ide_stopped ;
}
}
if ( task - > tf_flags & IDE_TFLAG_FLAGGED )
task - > tf_flags | = IDE_TFLAG_FLAGGED_SET_IN_FLAGS ;
2008-04-28 23:44:39 +02:00
if ( ( task - > tf_flags & IDE_TFLAG_DMA_PIO_FALLBACK ) = = 0 ) {
ide_tf_dump ( drive - > name , tf ) ;
2008-04-28 23:44:40 +02:00
hwif - > tf_load ( drive , task ) ;
2008-04-28 23:44:39 +02:00
}
2005-04-16 15:20:36 -07:00
2008-01-25 22:17:16 +01:00
switch ( task - > data_phase ) {
case TASKFILE_MULTI_OUT :
case TASKFILE_OUT :
2008-04-27 15:38:32 +02:00
hwif - > OUTBSYNC ( drive , tf - > command , hwif - > io_ports . command_addr ) ;
2008-01-25 22:17:16 +01:00
ndelay ( 400 ) ; /* FIXME */
return pre_task_out_intr ( drive , task - > rq ) ;
case TASKFILE_MULTI_IN :
case TASKFILE_IN :
2008-01-25 22:17:16 +01:00
handler = task_in_intr ;
2008-01-25 22:17:16 +01:00
/* fall-through */
2008-01-25 22:17:16 +01:00
case TASKFILE_NO_DATA :
2008-01-25 22:17:16 +01:00
if ( handler = = NULL )
handler = task_no_data_intr ;
2008-01-25 22:17:16 +01:00
/* WIN_{SPECIFY,RESTORE,SETMULT} use custom handlers */
2008-01-25 22:17:16 +01:00
if ( task - > tf_flags & IDE_TFLAG_CUSTOM_HANDLER ) {
switch ( tf - > command ) {
case WIN_SPECIFY : handler = set_geometry_intr ; break ;
case WIN_RESTORE : handler = recal_intr ; break ;
case WIN_SETMULT : handler = set_multmode_intr ; break ;
}
}
ide_execute_command ( drive , tf - > command , handler ,
WAIT_WORSTCASE , NULL ) ;
2005-04-16 15:20:36 -07:00
return ide_started ;
2008-01-25 22:17:16 +01:00
default :
if ( task_dma_ok ( task ) = = 0 | | drive - > using_dma = = 0 | |
2008-04-26 22:25:24 +02:00
dma_ops - > dma_setup ( drive ) )
2008-01-25 22:17:16 +01:00
return ide_stopped ;
2008-04-26 22:25:24 +02:00
dma_ops - > dma_exec_cmd ( drive , tf - > command ) ;
dma_ops - > dma_start ( drive ) ;
2008-01-25 22:17:07 +01:00
return ide_started ;
2005-04-16 15:20:36 -07:00
}
}
2008-01-25 22:17:16 +01:00
EXPORT_SYMBOL_GPL ( do_rw_taskfile ) ;
2005-04-16 15:20:36 -07:00
/*
* set_multmode_intr ( ) is invoked on completion of a WIN_SETMULT cmd .
*/
2008-01-25 22:17:16 +01:00
static ide_startstop_t set_multmode_intr ( ide_drive_t * drive )
2005-04-16 15:20:36 -07:00
{
2008-02-06 02:57:51 +01:00
u8 stat = ide_read_status ( drive ) ;
2005-04-16 15:20:36 -07:00
2008-02-06 02:57:51 +01:00
if ( OK_STAT ( stat , READY_STAT , BAD_STAT ) )
2005-04-16 15:20:36 -07:00
drive - > mult_count = drive - > mult_req ;
2008-02-06 02:57:51 +01:00
else {
2005-04-16 15:20:36 -07:00
drive - > mult_req = drive - > mult_count = 0 ;
drive - > special . b . recalibrate = 1 ;
( void ) ide_dump_status ( drive , " set_multmode " , stat ) ;
}
return ide_stopped ;
}
/*
* set_geometry_intr ( ) is invoked on completion of a WIN_SPECIFY cmd .
*/
2008-01-25 22:17:16 +01:00
static ide_startstop_t set_geometry_intr ( ide_drive_t * drive )
2005-04-16 15:20:36 -07:00
{
int retries = 5 ;
u8 stat ;
2008-02-06 02:57:51 +01:00
while ( ( ( stat = ide_read_status ( drive ) ) & BUSY_STAT ) & & retries - - )
2005-04-16 15:20:36 -07:00
udelay ( 10 ) ;
if ( OK_STAT ( stat , READY_STAT , BAD_STAT ) )
return ide_stopped ;
if ( stat & ( ERR_STAT | DRQ_STAT ) )
return ide_error ( drive , " set_geometry_intr " , stat ) ;
2006-06-23 02:06:06 -07:00
BUG_ON ( HWGROUP ( drive ) - > handler ! = NULL ) ;
2005-04-16 15:20:36 -07:00
ide_set_handler ( drive , & set_geometry_intr , WAIT_WORSTCASE , NULL ) ;
return ide_started ;
}
/*
* recal_intr ( ) is invoked on completion of a WIN_RESTORE ( recalibrate ) cmd .
*/
2008-01-25 22:17:16 +01:00
static ide_startstop_t recal_intr ( ide_drive_t * drive )
2005-04-16 15:20:36 -07:00
{
2008-02-06 02:57:51 +01:00
u8 stat = ide_read_status ( drive ) ;
2005-04-16 15:20:36 -07:00
2008-02-06 02:57:51 +01:00
if ( ! OK_STAT ( stat , READY_STAT , BAD_STAT ) )
2005-04-16 15:20:36 -07:00
return ide_error ( drive , " recal_intr " , stat ) ;
return ide_stopped ;
}
/*
* Handler for commands without a data phase
*/
2008-01-25 22:17:16 +01:00
static ide_startstop_t task_no_data_intr ( ide_drive_t * drive )
2005-04-16 15:20:36 -07:00
{
ide_task_t * args = HWGROUP ( drive ) - > rq - > special ;
u8 stat ;
2006-07-03 00:25:25 -07:00
local_irq_enable_in_hardirq ( ) ;
2008-02-06 02:57:51 +01:00
stat = ide_read_status ( drive ) ;
if ( ! OK_STAT ( stat , READY_STAT , BAD_STAT ) )
2005-04-16 15:20:36 -07:00
return ide_error ( drive , " task_no_data_intr " , stat ) ;
/* calls ide_end_drive_cmd */
2008-02-06 02:57:51 +01:00
2005-04-16 15:20:36 -07:00
if ( args )
2008-02-06 02:57:51 +01:00
ide_end_drive_cmd ( drive , stat , ide_read_error ( drive ) ) ;
2005-04-16 15:20:36 -07:00
return ide_stopped ;
}
2008-02-01 23:09:16 +01:00
static u8 wait_drive_not_busy ( ide_drive_t * drive )
2005-04-16 15:20:36 -07:00
{
2007-07-03 22:28:34 +02:00
int retries ;
2005-04-16 15:20:36 -07:00
u8 stat ;
/*
* Last sector was transfered , wait until drive is ready .
2008-01-26 20:13:11 +01:00
* This can take up to 10 usec , but we will wait max 1 ms .
2005-04-16 15:20:36 -07:00
*/
2007-07-03 22:28:34 +02:00
for ( retries = 0 ; retries < 100 ; retries + + ) {
2008-02-06 02:57:51 +01:00
stat = ide_read_status ( drive ) ;
if ( stat & BUSY_STAT )
2007-07-03 22:28:34 +02:00
udelay ( 10 ) ;
else
break ;
}
2005-04-16 15:20:36 -07:00
2007-07-03 22:28:34 +02:00
if ( stat & BUSY_STAT )
2005-04-16 15:20:36 -07:00
printk ( KERN_ERR " %s: drive still BUSY! \n " , drive - > name ) ;
return stat ;
}
2008-04-28 23:44:36 +02:00
static void ide_pio_sector ( ide_drive_t * drive , struct request * rq ,
unsigned int write )
2005-04-16 15:20:36 -07:00
{
ide_hwif_t * hwif = drive - > hwif ;
struct scatterlist * sg = hwif - > sg_table ;
2007-07-25 08:13:56 +02:00
struct scatterlist * cursg = hwif - > cursg ;
2005-04-16 15:20:36 -07:00
struct page * page ;
# ifdef CONFIG_HIGHMEM
unsigned long flags ;
# endif
unsigned int offset ;
u8 * buf ;
2007-07-25 08:13:56 +02:00
cursg = hwif - > cursg ;
if ( ! cursg ) {
cursg = sg ;
hwif - > cursg = sg ;
}
2007-10-22 21:19:53 +02:00
page = sg_page ( cursg ) ;
2007-07-25 08:13:56 +02:00
offset = cursg - > offset + hwif - > cursg_ofs * SECTOR_SIZE ;
2005-04-16 15:20:36 -07:00
/* get the current page and offset */
page = nth_page ( page , ( offset > > PAGE_SHIFT ) ) ;
offset % = PAGE_SIZE ;
# ifdef CONFIG_HIGHMEM
local_irq_save ( flags ) ;
# endif
buf = kmap_atomic ( page , KM_BIO_SRC_IRQ ) + offset ;
hwif - > nleft - - ;
hwif - > cursg_ofs + + ;
2007-07-25 08:13:56 +02:00
if ( ( hwif - > cursg_ofs * SECTOR_SIZE ) = = cursg - > length ) {
hwif - > cursg = sg_next ( hwif - > cursg ) ;
2005-04-16 15:20:36 -07:00
hwif - > cursg_ofs = 0 ;
}
/* do the actual data transfer */
if ( write )
2008-04-28 23:44:36 +02:00
hwif - > output_data ( drive , rq , buf , SECTOR_SIZE ) ;
2005-04-16 15:20:36 -07:00
else
2008-04-28 23:44:36 +02:00
hwif - > input_data ( drive , rq , buf , SECTOR_SIZE ) ;
2005-04-16 15:20:36 -07:00
kunmap_atomic ( buf , KM_BIO_SRC_IRQ ) ;
# ifdef CONFIG_HIGHMEM
local_irq_restore ( flags ) ;
# endif
}
2008-04-28 23:44:36 +02:00
static void ide_pio_multi ( ide_drive_t * drive , struct request * rq ,
unsigned int write )
2005-04-16 15:20:36 -07:00
{
unsigned int nsect ;
nsect = min_t ( unsigned int , drive - > hwif - > nleft , drive - > mult_count ) ;
while ( nsect - - )
2008-04-28 23:44:36 +02:00
ide_pio_sector ( drive , rq , write ) ;
2005-04-16 15:20:36 -07:00
}
2006-01-14 13:20:43 -08:00
static void ide_pio_datablock ( ide_drive_t * drive , struct request * rq ,
2005-04-16 15:20:36 -07:00
unsigned int write )
{
2008-01-26 20:13:10 +01:00
u8 saved_io_32bit = drive - > io_32bit ;
2005-04-16 15:20:36 -07:00
if ( rq - > bio ) /* fs request */
rq - > errors = 0 ;
2008-01-26 20:13:10 +01:00
if ( rq - > cmd_type = = REQ_TYPE_ATA_TASKFILE ) {
ide_task_t * task = rq - > special ;
if ( task - > tf_flags & IDE_TFLAG_IO_16BIT )
drive - > io_32bit = 0 ;
}
2006-02-15 15:17:37 -08:00
touch_softlockup_watchdog ( ) ;
2005-04-16 15:20:36 -07:00
switch ( drive - > hwif - > data_phase ) {
case TASKFILE_MULTI_IN :
case TASKFILE_MULTI_OUT :
2008-04-28 23:44:36 +02:00
ide_pio_multi ( drive , rq , write ) ;
2005-04-16 15:20:36 -07:00
break ;
default :
2008-04-28 23:44:36 +02:00
ide_pio_sector ( drive , rq , write ) ;
2005-04-16 15:20:36 -07:00
break ;
}
2008-01-26 20:13:10 +01:00
drive - > io_32bit = saved_io_32bit ;
2005-04-16 15:20:36 -07:00
}
static ide_startstop_t task_error ( ide_drive_t * drive , struct request * rq ,
const char * s , u8 stat )
{
if ( rq - > bio ) {
ide_hwif_t * hwif = drive - > hwif ;
int sectors = hwif - > nsect - hwif - > nleft ;
switch ( hwif - > data_phase ) {
case TASKFILE_IN :
if ( hwif - > nleft )
break ;
/* fall through */
case TASKFILE_OUT :
sectors - - ;
break ;
case TASKFILE_MULTI_IN :
if ( hwif - > nleft )
break ;
/* fall through */
case TASKFILE_MULTI_OUT :
sectors - = drive - > mult_count ;
default :
break ;
}
if ( sectors > 0 ) {
ide_driver_t * drv ;
drv = * ( ide_driver_t * * ) rq - > rq_disk - > private_data ;
drv - > end_request ( drive , 1 , sectors ) ;
}
}
return ide_error ( drive , s , stat ) ;
}
2008-01-26 20:13:11 +01:00
void task_end_request ( ide_drive_t * drive , struct request * rq , u8 stat )
2005-04-16 15:20:36 -07:00
{
2006-08-10 08:44:47 +02:00
if ( rq - > cmd_type = = REQ_TYPE_ATA_TASKFILE ) {
2008-02-06 02:57:51 +01:00
u8 err = ide_read_error ( drive ) ;
2005-04-16 15:20:36 -07:00
2008-01-26 20:13:11 +01:00
ide_end_drive_cmd ( drive , stat , err ) ;
return ;
2005-04-16 15:20:36 -07:00
}
2006-03-31 02:31:15 -08:00
if ( rq - > rq_disk ) {
ide_driver_t * drv ;
drv = * ( ide_driver_t * * ) rq - > rq_disk - > private_data ; ;
2008-01-26 20:13:11 +01:00
drv - > end_request ( drive , 1 , rq - > nr_sectors ) ;
2006-03-31 02:31:15 -08:00
} else
2008-01-26 20:13:11 +01:00
ide_end_request ( drive , 1 , rq - > nr_sectors ) ;
2005-04-16 15:20:36 -07:00
}
2008-03-18 21:26:24 -07:00
/*
* We got an interrupt on a task_in case , but no errors and no DRQ .
*
* It might be a spurious irq ( shared irq ) , but it might be a
* command that had no output .
*/
static ide_startstop_t task_in_unexpected ( ide_drive_t * drive , struct request * rq , u8 stat )
{
/* Command all done? */
if ( OK_STAT ( stat , READY_STAT , BUSY_STAT ) ) {
task_end_request ( drive , rq , stat ) ;
return ide_stopped ;
}
/* Assume it was a spurious irq */
ide_set_handler ( drive , & task_in_intr , WAIT_WORSTCASE , NULL ) ;
return ide_started ;
}
2005-04-16 15:20:36 -07:00
/*
* Handler for command with PIO data - in phase ( Read / Read Multiple ) .
*/
2008-01-25 22:17:16 +01:00
static ide_startstop_t task_in_intr ( ide_drive_t * drive )
2005-04-16 15:20:36 -07:00
{
ide_hwif_t * hwif = drive - > hwif ;
struct request * rq = HWGROUP ( drive ) - > rq ;
2008-02-06 02:57:51 +01:00
u8 stat = ide_read_status ( drive ) ;
2005-04-16 15:20:36 -07:00
2008-03-18 21:26:24 -07:00
/* Error? */
if ( stat & ERR_STAT )
2008-04-26 22:25:20 +02:00
return task_error ( drive , rq , __func__ , stat ) ;
2008-03-18 21:26:24 -07:00
/* Didn't want any data? Odd. */
if ( ! ( stat & DRQ_STAT ) )
return task_in_unexpected ( drive , rq , stat ) ;
2005-04-16 15:20:36 -07:00
ide_pio_datablock ( drive , rq , 0 ) ;
2008-03-18 21:26:24 -07:00
/* Are we done? Check status and finish transfer. */
2005-04-16 15:20:36 -07:00
if ( ! hwif - > nleft ) {
stat = wait_drive_not_busy ( drive ) ;
2008-01-26 20:13:10 +01:00
if ( ! OK_STAT ( stat , 0 , BAD_STAT ) )
2008-04-26 22:25:20 +02:00
return task_error ( drive , rq , __func__ , stat ) ;
2005-04-16 15:20:36 -07:00
task_end_request ( drive , rq , stat ) ;
return ide_stopped ;
}
/* Still data left to transfer. */
ide_set_handler ( drive , & task_in_intr , WAIT_WORSTCASE , NULL ) ;
return ide_started ;
}
/*
* Handler for command with PIO data - out phase ( Write / Write Multiple ) .
*/
static ide_startstop_t task_out_intr ( ide_drive_t * drive )
{
ide_hwif_t * hwif = drive - > hwif ;
struct request * rq = HWGROUP ( drive ) - > rq ;
2008-02-06 02:57:51 +01:00
u8 stat = ide_read_status ( drive ) ;
2005-04-16 15:20:36 -07:00
if ( ! OK_STAT ( stat , DRIVE_READY , drive - > bad_wstat ) )
2008-04-26 22:25:20 +02:00
return task_error ( drive , rq , __func__ , stat ) ;
2005-04-16 15:20:36 -07:00
/* Deal with unexpected ATA data phase. */
if ( ( ( stat & DRQ_STAT ) = = 0 ) ^ ! hwif - > nleft )
2008-04-26 22:25:20 +02:00
return task_error ( drive , rq , __func__ , stat ) ;
2005-04-16 15:20:36 -07:00
if ( ! hwif - > nleft ) {
task_end_request ( drive , rq , stat ) ;
return ide_stopped ;
}
/* Still data left to transfer. */
ide_pio_datablock ( drive , rq , 1 ) ;
ide_set_handler ( drive , & task_out_intr , WAIT_WORSTCASE , NULL ) ;
return ide_started ;
}
2008-01-25 22:17:16 +01:00
static ide_startstop_t pre_task_out_intr ( ide_drive_t * drive , struct request * rq )
2005-04-16 15:20:36 -07:00
{
ide_startstop_t startstop ;
2008-01-26 20:13:11 +01:00
if ( ide_wait_stat ( & startstop , drive , DRQ_STAT ,
2005-04-16 15:20:36 -07:00
drive - > bad_wstat , WAIT_DRQ ) ) {
printk ( KERN_ERR " %s: no DRQ after issuing %sWRITE%s \n " ,
drive - > name ,
drive - > hwif - > data_phase ? " MULT " : " " ,
drive - > addressing ? " _EXT " : " " ) ;
return startstop ;
}
if ( ! drive - > unmask )
local_irq_disable ( ) ;
ide_set_handler ( drive , & task_out_intr , WAIT_WORSTCASE , NULL ) ;
ide_pio_datablock ( drive , rq , 1 ) ;
return ide_started ;
}
2008-01-25 22:17:14 +01:00
int ide_raw_taskfile ( ide_drive_t * drive , ide_task_t * task , u8 * buf , u16 nsect )
2005-04-16 15:20:36 -07:00
{
struct request rq ;
2008-04-29 09:54:38 +02:00
blk_rq_init ( NULL , & rq ) ;
2006-08-10 08:44:47 +02:00
rq . cmd_type = REQ_TYPE_ATA_TASKFILE ;
2005-04-16 15:20:36 -07:00
rq . buffer = buf ;
/*
* ( ks ) We transfer currently only whole sectors .
* This is suffient for now . But , it would be great ,
* if we would find a solution to transfer any size .
* To support special commands like READ LONG .
*/
2008-01-25 22:17:14 +01:00
rq . hard_nr_sectors = rq . nr_sectors = nsect ;
rq . hard_cur_sectors = rq . current_nr_sectors = nsect ;
2005-04-16 15:20:36 -07:00
2008-01-25 22:17:14 +01:00
if ( task - > tf_flags & IDE_TFLAG_WRITE )
rq . cmd_flags | = REQ_RW ;
2005-04-16 15:20:36 -07:00
2008-01-25 22:17:14 +01:00
rq . special = task ;
task - > rq = & rq ;
2005-04-16 15:20:36 -07:00
return ide_do_drive_cmd ( drive , & rq , ide_wait ) ;
}
EXPORT_SYMBOL ( ide_raw_taskfile ) ;
2008-01-25 22:17:07 +01:00
int ide_no_data_taskfile ( ide_drive_t * drive , ide_task_t * task )
{
2008-01-25 22:17:14 +01:00
task - > data_phase = TASKFILE_NO_DATA ;
2008-01-25 22:17:07 +01:00
2008-01-25 22:17:14 +01:00
return ide_raw_taskfile ( drive , task , NULL , 0 ) ;
2008-01-25 22:17:07 +01:00
}
EXPORT_SYMBOL_GPL ( ide_no_data_taskfile ) ;
2007-11-05 21:42:27 +01:00
# ifdef CONFIG_IDE_TASK_IOCTL
2005-04-16 15:20:36 -07:00
int ide_taskfile_ioctl ( ide_drive_t * drive , unsigned int cmd , unsigned long arg )
{
ide_task_request_t * req_task ;
ide_task_t args ;
u8 * outbuf = NULL ;
u8 * inbuf = NULL ;
2008-01-25 22:17:14 +01:00
u8 * data_buf = NULL ;
2005-04-16 15:20:36 -07:00
int err = 0 ;
int tasksize = sizeof ( struct ide_task_request_s ) ;
2006-10-16 16:31:02 +01:00
unsigned int taskin = 0 ;
unsigned int taskout = 0 ;
2008-01-25 22:17:14 +01:00
u16 nsect = 0 ;
2005-04-16 15:20:36 -07:00
char __user * buf = ( char __user * ) arg ;
// printk("IDE Taskfile ...\n");
2005-11-07 01:01:25 -08:00
req_task = kzalloc ( tasksize , GFP_KERNEL ) ;
2005-04-16 15:20:36 -07:00
if ( req_task = = NULL ) return - ENOMEM ;
if ( copy_from_user ( req_task , buf , tasksize ) ) {
kfree ( req_task ) ;
return - EFAULT ;
}
2006-10-16 16:31:02 +01:00
taskout = req_task - > out_size ;
taskin = req_task - > in_size ;
if ( taskin > 65536 | | taskout > 65536 ) {
err = - EINVAL ;
goto abort ;
}
2005-04-16 15:20:36 -07:00
if ( taskout ) {
int outtotal = tasksize ;
2005-11-07 01:01:25 -08:00
outbuf = kzalloc ( taskout , GFP_KERNEL ) ;
2005-04-16 15:20:36 -07:00
if ( outbuf = = NULL ) {
err = - ENOMEM ;
goto abort ;
}
if ( copy_from_user ( outbuf , buf + outtotal , taskout ) ) {
err = - EFAULT ;
goto abort ;
}
}
if ( taskin ) {
int intotal = tasksize + taskout ;
2005-11-07 01:01:25 -08:00
inbuf = kzalloc ( taskin , GFP_KERNEL ) ;
2005-04-16 15:20:36 -07:00
if ( inbuf = = NULL ) {
err = - ENOMEM ;
goto abort ;
}
if ( copy_from_user ( inbuf , buf + intotal , taskin ) ) {
err = - EFAULT ;
goto abort ;
}
}
memset ( & args , 0 , sizeof ( ide_task_t ) ) ;
2008-01-25 22:17:06 +01:00
memcpy ( & args . tf_array [ 0 ] , req_task - > hob_ports , HDIO_DRIVE_HOB_HDR_SIZE - 2 ) ;
memcpy ( & args . tf_array [ 6 ] , req_task - > io_ports , HDIO_DRIVE_TASK_HDR_SIZE ) ;
2008-01-25 22:17:14 +01:00
args . data_phase = req_task - > data_phase ;
2005-04-16 15:20:36 -07:00
2008-01-26 20:13:10 +01:00
args . tf_flags = IDE_TFLAG_IO_16BIT | IDE_TFLAG_DEVICE |
IDE_TFLAG_IN_TF ;
2008-01-25 22:17:10 +01:00
if ( drive - > addressing = = 1 )
2008-01-26 20:13:10 +01:00
args . tf_flags | = ( IDE_TFLAG_LBA48 | IDE_TFLAG_IN_HOB ) ;
2008-01-25 22:17:10 +01:00
2008-01-25 22:17:07 +01:00
if ( req_task - > out_flags . all ) {
args . tf_flags | = IDE_TFLAG_FLAGGED ;
if ( req_task - > out_flags . b . data )
args . tf_flags | = IDE_TFLAG_OUT_DATA ;
if ( req_task - > out_flags . b . nsector_hob )
args . tf_flags | = IDE_TFLAG_OUT_HOB_NSECT ;
if ( req_task - > out_flags . b . sector_hob )
args . tf_flags | = IDE_TFLAG_OUT_HOB_LBAL ;
if ( req_task - > out_flags . b . lcyl_hob )
args . tf_flags | = IDE_TFLAG_OUT_HOB_LBAM ;
if ( req_task - > out_flags . b . hcyl_hob )
args . tf_flags | = IDE_TFLAG_OUT_HOB_LBAH ;
if ( req_task - > out_flags . b . error_feature )
args . tf_flags | = IDE_TFLAG_OUT_FEATURE ;
if ( req_task - > out_flags . b . nsector )
args . tf_flags | = IDE_TFLAG_OUT_NSECT ;
if ( req_task - > out_flags . b . sector )
args . tf_flags | = IDE_TFLAG_OUT_LBAL ;
if ( req_task - > out_flags . b . lcyl )
args . tf_flags | = IDE_TFLAG_OUT_LBAM ;
if ( req_task - > out_flags . b . hcyl )
args . tf_flags | = IDE_TFLAG_OUT_LBAH ;
2008-01-25 22:17:10 +01:00
} else {
args . tf_flags | = IDE_TFLAG_OUT_TF ;
if ( args . tf_flags & IDE_TFLAG_LBA48 )
args . tf_flags | = IDE_TFLAG_OUT_HOB ;
2008-01-25 22:17:07 +01:00
}
2008-01-25 22:17:14 +01:00
if ( req_task - > in_flags . b . data )
args . tf_flags | = IDE_TFLAG_IN_DATA ;
2005-04-16 15:20:36 -07:00
switch ( req_task - > data_phase ) {
case TASKFILE_MULTI_OUT :
if ( ! drive - > mult_count ) {
/* (hs): give up if multcount is not set */
printk ( KERN_ERR " %s: %s Multimode Write " \
" multcount is not set \n " ,
2008-04-26 22:25:20 +02:00
drive - > name , __func__ ) ;
2005-04-16 15:20:36 -07:00
err = - EPERM ;
goto abort ;
}
/* fall through */
case TASKFILE_OUT :
2008-01-25 22:17:14 +01:00
/* fall through */
case TASKFILE_OUT_DMAQ :
case TASKFILE_OUT_DMA :
nsect = taskout / SECTOR_SIZE ;
data_buf = outbuf ;
2005-04-16 15:20:36 -07:00
break ;
case TASKFILE_MULTI_IN :
if ( ! drive - > mult_count ) {
/* (hs): give up if multcount is not set */
printk ( KERN_ERR " %s: %s Multimode Read failure " \
" multcount is not set \n " ,
2008-04-26 22:25:20 +02:00
drive - > name , __func__ ) ;
2005-04-16 15:20:36 -07:00
err = - EPERM ;
goto abort ;
}
/* fall through */
case TASKFILE_IN :
2008-01-25 22:17:14 +01:00
/* fall through */
case TASKFILE_IN_DMAQ :
case TASKFILE_IN_DMA :
nsect = taskin / SECTOR_SIZE ;
data_buf = inbuf ;
2005-04-16 15:20:36 -07:00
break ;
case TASKFILE_NO_DATA :
break ;
default :
err = - EFAULT ;
goto abort ;
}
2008-01-25 22:17:14 +01:00
if ( req_task - > req_cmd = = IDE_DRIVE_TASK_NO_DATA )
nsect = 0 ;
else if ( ! nsect ) {
nsect = ( args . tf . hob_nsect < < 8 ) | args . tf . nsect ;
if ( ! nsect ) {
printk ( KERN_ERR " %s: in/out command without data \n " ,
drive - > name ) ;
err = - EFAULT ;
goto abort ;
}
}
if ( req_task - > req_cmd = = IDE_DRIVE_TASK_RAW_WRITE )
args . tf_flags | = IDE_TFLAG_WRITE ;
err = ide_raw_taskfile ( drive , & args , data_buf , nsect ) ;
2008-01-25 22:17:06 +01:00
memcpy ( req_task - > hob_ports , & args . tf_array [ 0 ] , HDIO_DRIVE_HOB_HDR_SIZE - 2 ) ;
memcpy ( req_task - > io_ports , & args . tf_array [ 6 ] , HDIO_DRIVE_TASK_HDR_SIZE ) ;
2008-01-25 22:17:14 +01:00
if ( ( args . tf_flags & IDE_TFLAG_FLAGGED_SET_IN_FLAGS ) & &
req_task - > in_flags . all = = 0 ) {
req_task - > in_flags . all = IDE_TASKFILE_STD_IN_FLAGS ;
if ( drive - > addressing = = 1 )
req_task - > in_flags . all | = ( IDE_HOB_STD_IN_FLAGS < < 8 ) ;
}
2005-04-16 15:20:36 -07:00
if ( copy_to_user ( buf , req_task , tasksize ) ) {
err = - EFAULT ;
goto abort ;
}
if ( taskout ) {
int outtotal = tasksize ;
if ( copy_to_user ( buf + outtotal , outbuf , taskout ) ) {
err = - EFAULT ;
goto abort ;
}
}
if ( taskin ) {
int intotal = tasksize + taskout ;
if ( copy_to_user ( buf + intotal , inbuf , taskin ) ) {
err = - EFAULT ;
goto abort ;
}
}
abort :
kfree ( req_task ) ;
2005-11-07 01:01:32 -08:00
kfree ( outbuf ) ;
kfree ( inbuf ) ;
2005-04-16 15:20:36 -07:00
// printk("IDE Taskfile ioctl ended. rc = %i\n", err);
return err ;
}
2007-11-05 21:42:27 +01:00
# endif
2005-04-16 15:20:36 -07:00
int ide_cmd_ioctl ( ide_drive_t * drive , unsigned int cmd , unsigned long arg )
{
2008-01-26 20:13:13 +01:00
u8 * buf = NULL ;
int bufsize = 0 , err = 0 ;
u8 args [ 4 ] , xfer_rate = 0 ;
2005-04-16 15:20:36 -07:00
ide_task_t tfargs ;
2008-01-25 22:17:06 +01:00
struct ide_taskfile * tf = & tfargs . tf ;
2008-02-02 19:56:46 +01:00
struct hd_driveid * id = drive - > id ;
2005-04-16 15:20:36 -07:00
if ( NULL = = ( void * ) arg ) {
struct request rq ;
2008-01-26 20:13:11 +01:00
2005-04-16 15:20:36 -07:00
ide_init_drive_cmd ( & rq ) ;
2008-01-26 20:13:12 +01:00
rq . cmd_type = REQ_TYPE_ATA_TASKFILE ;
2008-01-26 20:13:11 +01:00
2005-04-16 15:20:36 -07:00
return ide_do_drive_cmd ( drive , & rq , ide_wait ) ;
}
if ( copy_from_user ( args , ( void __user * ) arg , 4 ) )
return - EFAULT ;
memset ( & tfargs , 0 , sizeof ( ide_task_t ) ) ;
2008-01-25 22:17:06 +01:00
tf - > feature = args [ 2 ] ;
2008-01-26 20:13:13 +01:00
if ( args [ 0 ] = = WIN_SMART ) {
tf - > nsect = args [ 3 ] ;
tf - > lbal = args [ 1 ] ;
tf - > lbam = 0x4f ;
tf - > lbah = 0xc2 ;
tfargs . tf_flags = IDE_TFLAG_OUT_TF | IDE_TFLAG_IN_NSECT ;
} else {
tf - > nsect = args [ 1 ] ;
tfargs . tf_flags = IDE_TFLAG_OUT_FEATURE |
IDE_TFLAG_OUT_NSECT | IDE_TFLAG_IN_NSECT ;
}
2008-01-25 22:17:06 +01:00
tf - > command = args [ 0 ] ;
2008-01-26 20:13:13 +01:00
tfargs . data_phase = args [ 3 ] ? TASKFILE_IN : TASKFILE_NO_DATA ;
2005-04-16 15:20:36 -07:00
if ( args [ 3 ] ) {
2008-01-26 20:13:13 +01:00
tfargs . tf_flags | = IDE_TFLAG_IO_16BIT ;
bufsize = SECTOR_WORDS * 4 * args [ 3 ] ;
buf = kzalloc ( bufsize , GFP_KERNEL ) ;
if ( buf = = NULL )
2005-04-16 15:20:36 -07:00
return - ENOMEM ;
}
2008-01-26 20:13:13 +01:00
2008-02-02 19:56:46 +01:00
if ( tf - > command = = WIN_SETFEATURES & &
tf - > feature = = SETFEATURES_XFER & &
tf - > nsect > = XFER_SW_DMA_0 & &
( id - > dma_ultra | | id - > dma_mword | | id - > dma_1word ) ) {
2005-04-16 15:20:36 -07:00
xfer_rate = args [ 1 ] ;
2008-02-02 19:56:46 +01:00
if ( tf - > nsect > XFER_UDMA_2 & & ! eighty_ninty_three ( drive ) ) {
printk ( KERN_WARNING " %s: UDMA speeds >UDMA33 cannot "
" be set \n " , drive - > name ) ;
2005-04-16 15:20:36 -07:00
goto abort ;
2008-02-02 19:56:46 +01:00
}
2005-04-16 15:20:36 -07:00
}
2008-01-26 20:13:13 +01:00
err = ide_raw_taskfile ( drive , & tfargs , buf , args [ 3 ] ) ;
args [ 0 ] = tf - > status ;
args [ 1 ] = tf - > error ;
args [ 2 ] = tf - > nsect ;
2005-04-16 15:20:36 -07:00
if ( ! err & & xfer_rate ) {
/* active-retuning-calls future */
ide_set_xfer_rate ( drive , xfer_rate ) ;
ide_driveid_update ( drive ) ;
}
abort :
2008-01-26 20:13:13 +01:00
if ( copy_to_user ( ( void __user * ) arg , & args , 4 ) )
2005-04-16 15:20:36 -07:00
err = - EFAULT ;
2008-01-26 20:13:13 +01:00
if ( buf ) {
if ( copy_to_user ( ( void __user * ) ( arg + 4 ) , buf , bufsize ) )
err = - EFAULT ;
kfree ( buf ) ;
}
2005-04-16 15:20:36 -07:00
return err ;
}
int ide_task_ioctl ( ide_drive_t * drive , unsigned int cmd , unsigned long arg )
{
void __user * p = ( void __user * ) arg ;
int err = 0 ;
2008-01-25 22:17:11 +01:00
u8 args [ 7 ] ;
ide_task_t task ;
2005-04-16 15:20:36 -07:00
if ( copy_from_user ( args , p , 7 ) )
return - EFAULT ;
2008-01-25 22:17:11 +01:00
memset ( & task , 0 , sizeof ( task ) ) ;
memcpy ( & task . tf_array [ 7 ] , & args [ 1 ] , 6 ) ;
task . tf . command = args [ 0 ] ;
2008-01-26 20:13:10 +01:00
task . tf_flags = IDE_TFLAG_TF | IDE_TFLAG_DEVICE ;
2008-01-25 22:17:11 +01:00
err = ide_no_data_taskfile ( drive , & task ) ;
args [ 0 ] = task . tf . command ;
memcpy ( & args [ 1 ] , & task . tf_array [ 7 ] , 6 ) ;
if ( copy_to_user ( p , args , 7 ) )
2005-04-16 15:20:36 -07:00
err = - EFAULT ;
2008-01-25 22:17:11 +01:00
2005-04-16 15:20:36 -07:00
return err ;
}