2008-10-13 23:39:50 +04:00
# include <linux/kernel.h>
# include <linux/ide.h>
# include <linux/jiffies.h>
# include <linux/blkdev.h>
DECLARE_WAIT_QUEUE_HEAD ( ide_park_wq ) ;
static void issue_park_cmd ( ide_drive_t * drive , unsigned long timeout )
{
2009-01-06 19:20:50 +03:00
ide_hwif_t * hwif = drive - > hwif ;
2008-10-13 23:39:50 +04:00
struct request_queue * q = drive - > queue ;
struct request * rq ;
int rc ;
timeout + = jiffies ;
2009-01-06 19:20:50 +03:00
spin_lock_irq ( & hwif - > lock ) ;
2008-10-13 23:39:50 +04:00
if ( drive - > dev_flags & IDE_DFLAG_PARKED ) {
2008-12-29 22:27:31 +03:00
int reset_timer = time_before ( timeout , drive - > sleep ) ;
2009-01-02 18:12:50 +03:00
int start_queue = 0 ;
2008-10-13 23:39:50 +04:00
drive - > sleep = timeout ;
wake_up_all ( & ide_park_wq ) ;
2009-01-06 19:20:50 +03:00
if ( reset_timer & & del_timer ( & hwif - > timer ) )
2009-01-02 18:12:50 +03:00
start_queue = 1 ;
2009-01-06 19:20:50 +03:00
spin_unlock_irq ( & hwif - > lock ) ;
2009-01-02 18:12:50 +03:00
2009-04-19 02:00:41 +04:00
if ( start_queue )
blk_run_queue ( q ) ;
2008-10-13 23:39:50 +04:00
return ;
}
2009-01-06 19:20:50 +03:00
spin_unlock_irq ( & hwif - > lock ) ;
2008-10-13 23:39:50 +04:00
rq = blk_get_request ( q , READ , __GFP_WAIT ) ;
rq - > cmd [ 0 ] = REQ_PARK_HEADS ;
rq - > cmd_len = 1 ;
rq - > cmd_type = REQ_TYPE_SPECIAL ;
rq - > special = & timeout ;
rc = blk_execute_rq ( q , NULL , rq , 1 ) ;
blk_put_request ( rq ) ;
if ( rc )
goto out ;
/*
* Make sure that * some * command is sent to the drive after the
* timeout has expired , so power management will be reenabled .
*/
rq = blk_get_request ( q , READ , GFP_NOWAIT ) ;
if ( unlikely ( ! rq ) )
goto out ;
rq - > cmd [ 0 ] = REQ_UNPARK_HEADS ;
rq - > cmd_len = 1 ;
rq - > cmd_type = REQ_TYPE_SPECIAL ;
elv_add_request ( q , rq , ELEVATOR_INSERT_FRONT , 1 ) ;
out :
return ;
}
2009-03-25 01:22:44 +03:00
ide_startstop_t ide_do_park_unpark ( ide_drive_t * drive , struct request * rq )
{
2009-03-27 14:46:37 +03:00
struct ide_cmd cmd ;
struct ide_taskfile * tf = & cmd . tf ;
2009-03-25 01:22:44 +03:00
2009-03-27 14:46:37 +03:00
memset ( & cmd , 0 , sizeof ( cmd ) ) ;
2009-03-25 01:22:44 +03:00
if ( rq - > cmd [ 0 ] = = REQ_PARK_HEADS ) {
drive - > sleep = * ( unsigned long * ) rq - > special ;
drive - > dev_flags | = IDE_DFLAG_SLEEPING ;
tf - > command = ATA_CMD_IDLEIMMEDIATE ;
tf - > feature = 0x44 ;
tf - > lbal = 0x4c ;
tf - > lbam = 0x4e ;
tf - > lbah = 0x55 ;
2009-04-08 16:13:01 +04:00
cmd . valid . out . tf = IDE_VALID_OUT_TF | IDE_VALID_DEVICE ;
cmd . valid . in . tf = IDE_VALID_IN_TF | IDE_VALID_DEVICE ;
2009-03-25 01:22:44 +03:00
} else /* cmd == REQ_UNPARK_HEADS */
tf - > command = ATA_CMD_CHK_POWER ;
2009-03-27 14:46:42 +03:00
cmd . tf_flags | = IDE_TFLAG_CUSTOM_HANDLER ;
2009-03-27 14:46:39 +03:00
cmd . protocol = ATA_PROT_NODATA ;
2009-03-27 14:46:37 +03:00
cmd . rq = rq ;
return do_rw_taskfile ( drive , & cmd ) ;
2009-03-25 01:22:44 +03:00
}
2008-10-13 23:39:50 +04:00
ssize_t ide_park_show ( struct device * dev , struct device_attribute * attr ,
char * buf )
{
ide_drive_t * drive = to_ide_device ( dev ) ;
2009-01-06 19:20:50 +03:00
ide_hwif_t * hwif = drive - > hwif ;
2008-10-13 23:39:50 +04:00
unsigned long now ;
unsigned int msecs ;
if ( drive - > dev_flags & IDE_DFLAG_NO_UNLOAD )
return - EOPNOTSUPP ;
2009-01-06 19:20:50 +03:00
spin_lock_irq ( & hwif - > lock ) ;
2008-10-13 23:39:50 +04:00
now = jiffies ;
if ( drive - > dev_flags & IDE_DFLAG_PARKED & &
time_after ( drive - > sleep , now ) )
msecs = jiffies_to_msecs ( drive - > sleep - now ) ;
else
msecs = 0 ;
2009-01-06 19:20:50 +03:00
spin_unlock_irq ( & hwif - > lock ) ;
2008-10-13 23:39:50 +04:00
return snprintf ( buf , 20 , " %u \n " , msecs ) ;
}
ssize_t ide_park_store ( struct device * dev , struct device_attribute * attr ,
const char * buf , size_t len )
{
# define MAX_PARK_TIMEOUT 30000
ide_drive_t * drive = to_ide_device ( dev ) ;
long int input ;
int rc ;
rc = strict_strtol ( buf , 10 , & input ) ;
if ( rc | | input < - 2 )
return - EINVAL ;
if ( input > MAX_PARK_TIMEOUT ) {
input = MAX_PARK_TIMEOUT ;
rc = - EOVERFLOW ;
}
mutex_lock ( & ide_setting_mtx ) ;
if ( input > = 0 ) {
if ( drive - > dev_flags & IDE_DFLAG_NO_UNLOAD )
rc = - EOPNOTSUPP ;
else if ( input | | drive - > dev_flags & IDE_DFLAG_PARKED )
issue_park_cmd ( drive , msecs_to_jiffies ( input ) ) ;
} else {
if ( drive - > media = = ide_disk )
switch ( input ) {
case - 1 :
drive - > dev_flags & = ~ IDE_DFLAG_NO_UNLOAD ;
break ;
case - 2 :
drive - > dev_flags | = IDE_DFLAG_NO_UNLOAD ;
break ;
}
else
rc = - EOPNOTSUPP ;
}
mutex_unlock ( & ide_setting_mtx ) ;
return rc ? rc : len ;
}