Commit 7e33820d authored by Bartlomiej Zolnierkiewicz's avatar Bartlomiej Zolnierkiewicz Committed by Linus Torvalds

[PATCH] ide-disk.c: more write cache fixes

- many Maxtor disks incorrectly claim CACHE FLUSH EXT command support,
  fix it by checking both CACHE FLUSH EXT command and LBA48 support
  (thanks to Eric D. Mudama for help in fixing this)

- write_cache() was called with 'drive->id->cfs_enable_2 & 0x3000' as 'int arg'
  argument which was always truncated to zero due to 'u8 drive->wcache = arg'
  assignment so write cache was indeed enabled but drive->wcache was zero
  (thanks to Rene Herman for help in debugging this)

- flush cache in idedisk_start_power_step() only if ATA-6 CACHE FLUSH (EXT)
  bits are present in disk's identify data (prevents sending unknown commands)

- set drive->wcache in idedisk_setup() not idedisk_attach() (no need to check
  id->command_set_2 - we check id->cfs_enable_2 instead in write_cache() call)

- use ide_cacheflush_p() in idedisk_setup()

- minor cleanups
parent ab9e69a3
...@@ -740,8 +740,6 @@ static ide_startstop_t ide_do_rw_disk (ide_drive_t *drive, struct request *rq, s ...@@ -740,8 +740,6 @@ static ide_startstop_t ide_do_rw_disk (ide_drive_t *drive, struct request *rq, s
return __ide_do_rw_disk(drive, rq, block); return __ide_do_rw_disk(drive, rq, block);
} }
static int do_idedisk_flushcache(ide_drive_t *drive);
static u8 idedisk_dump_status (ide_drive_t *drive, const char *msg, u8 stat) static u8 idedisk_dump_status (ide_drive_t *drive, const char *msg, u8 stat)
{ {
ide_hwif_t *hwif = HWIF(drive); ide_hwif_t *hwif = HWIF(drive);
...@@ -1359,11 +1357,18 @@ static int set_nowerr(ide_drive_t *drive, int arg) ...@@ -1359,11 +1357,18 @@ static int set_nowerr(ide_drive_t *drive, int arg)
return 0; return 0;
} }
/* check if CACHE FLUSH (EXT) command is supported (bits defined in ATA-6) */
#define ide_id_has_flush_cache(id) ((id)->cfs_enable_2 & 0x3000)
/* some Maxtor disks have bit 13 defined incorrectly so check bit 10 too */
#define ide_id_has_flush_cache_ext(id) \
(((id)->cfs_enable_2 & 0x2400) == 0x2400)
static int write_cache (ide_drive_t *drive, int arg) static int write_cache (ide_drive_t *drive, int arg)
{ {
ide_task_t args; ide_task_t args;
if (!(drive->id->cfs_enable_2 & 0x3000)) if (!ide_id_has_flush_cache(drive->id))
return 1; return 1;
memset(&args, 0, sizeof(ide_task_t)); memset(&args, 0, sizeof(ide_task_t));
...@@ -1383,7 +1388,7 @@ static int do_idedisk_flushcache (ide_drive_t *drive) ...@@ -1383,7 +1388,7 @@ static int do_idedisk_flushcache (ide_drive_t *drive)
ide_task_t args; ide_task_t args;
memset(&args, 0, sizeof(ide_task_t)); memset(&args, 0, sizeof(ide_task_t));
if (drive->id->cfs_enable_2 & 0x2400) if (ide_id_has_flush_cache_ext(drive->id))
args.tfRegister[IDE_COMMAND_OFFSET] = WIN_FLUSH_CACHE_EXT; args.tfRegister[IDE_COMMAND_OFFSET] = WIN_FLUSH_CACHE_EXT;
else else
args.tfRegister[IDE_COMMAND_OFFSET] = WIN_FLUSH_CACHE; args.tfRegister[IDE_COMMAND_OFFSET] = WIN_FLUSH_CACHE;
...@@ -1513,11 +1518,11 @@ static ide_startstop_t idedisk_start_power_step (ide_drive_t *drive, struct requ ...@@ -1513,11 +1518,11 @@ static ide_startstop_t idedisk_start_power_step (ide_drive_t *drive, struct requ
switch (rq->pm->pm_step) { switch (rq->pm->pm_step) {
case idedisk_pm_flush_cache: /* Suspend step 1 (flush cache) */ case idedisk_pm_flush_cache: /* Suspend step 1 (flush cache) */
/* Not supported? Switch to next step now. */ /* Not supported? Switch to next step now. */
if (!drive->wcache) { if (!drive->wcache || !ide_id_has_flush_cache(drive->id)) {
idedisk_complete_power_step(drive, rq, 0, 0); idedisk_complete_power_step(drive, rq, 0, 0);
return ide_stopped; return ide_stopped;
} }
if (drive->id->cfs_enable_2 & 0x2400) if (ide_id_has_flush_cache_ext(drive->id))
args->tfRegister[IDE_COMMAND_OFFSET] = WIN_FLUSH_CACHE_EXT; args->tfRegister[IDE_COMMAND_OFFSET] = WIN_FLUSH_CACHE_EXT;
else else
args->tfRegister[IDE_COMMAND_OFFSET] = WIN_FLUSH_CACHE; args->tfRegister[IDE_COMMAND_OFFSET] = WIN_FLUSH_CACHE;
...@@ -1678,8 +1683,12 @@ static void idedisk_setup (ide_drive_t *drive) ...@@ -1678,8 +1683,12 @@ static void idedisk_setup (ide_drive_t *drive)
#endif /* CONFIG_IDEDISK_MULTI_MODE */ #endif /* CONFIG_IDEDISK_MULTI_MODE */
} }
drive->no_io_32bit = id->dword_io ? 1 : 0; drive->no_io_32bit = id->dword_io ? 1 : 0;
if (drive->id->cfs_enable_2 & 0x3000)
write_cache(drive, (id->cfs_enable_2 & 0x3000)); /* write cache enabled? */
if ((id->csfo & 1) || (id->cfs_enable_1 & (1 << 5)))
drive->wcache = 1;
write_cache(drive, 1);
#ifdef CONFIG_BLK_DEV_IDE_TCQ_DEFAULT #ifdef CONFIG_BLK_DEV_IDE_TCQ_DEFAULT
if (drive->using_dma) if (drive->using_dma)
...@@ -1687,9 +1696,17 @@ static void idedisk_setup (ide_drive_t *drive) ...@@ -1687,9 +1696,17 @@ static void idedisk_setup (ide_drive_t *drive)
#endif #endif
} }
static void ide_cacheflush_p(ide_drive_t *drive)
{
if (!drive->wcache || !ide_id_has_flush_cache(drive->id))
return;
if (do_idedisk_flushcache(drive))
printk(KERN_INFO "%s: wcache flush failed!\n", drive->name);
}
static int idedisk_cleanup (ide_drive_t *drive) static int idedisk_cleanup (ide_drive_t *drive)
{ {
static int ide_cacheflush_p(ide_drive_t *drive);
struct gendisk *g = drive->disk; struct gendisk *g = drive->disk;
ide_cacheflush_p(drive); ide_cacheflush_p(drive);
if (ide_unregister_subdriver(drive)) if (ide_unregister_subdriver(drive))
...@@ -1740,7 +1757,6 @@ static ide_driver_t idedisk_driver = { ...@@ -1740,7 +1757,6 @@ static ide_driver_t idedisk_driver = {
static int idedisk_open(struct inode *inode, struct file *filp) static int idedisk_open(struct inode *inode, struct file *filp)
{ {
u8 cf;
ide_drive_t *drive = inode->i_bdev->bd_disk->private_data; ide_drive_t *drive = inode->i_bdev->bd_disk->private_data;
drive->usage++; drive->usage++;
if (drive->removable && drive->usage == 1) { if (drive->removable && drive->usage == 1) {
...@@ -1758,35 +1774,6 @@ static int idedisk_open(struct inode *inode, struct file *filp) ...@@ -1758,35 +1774,6 @@ static int idedisk_open(struct inode *inode, struct file *filp)
if (drive->doorlocking && ide_raw_taskfile(drive, &args, NULL)) if (drive->doorlocking && ide_raw_taskfile(drive, &args, NULL))
drive->doorlocking = 0; drive->doorlocking = 0;
} }
drive->wcache = 0;
/* Cache enabled? */
if (drive->id->csfo & 1)
drive->wcache = 1;
/* Cache command set available? */
if (drive->id->cfs_enable_1 & (1 << 5))
drive->wcache = 1;
/* ATA6 cache extended commands */
cf = drive->id->command_set_2 >> 24;
if ((cf & 0xC0) == 0x40 && (cf & 0x30) != 0)
drive->wcache = 1;
return 0;
}
static int ide_cacheflush_p(ide_drive_t *drive)
{
if (!(drive->id->cfs_enable_2 & 0x3000))
return 0;
if(drive->wcache)
{
if (do_idedisk_flushcache(drive))
{
printk (KERN_INFO "%s: Write Cache FAILED Flushing!\n",
drive->name);
return -EIO;
}
return 1;
}
return 0; return 0;
} }
...@@ -1867,10 +1854,7 @@ static int idedisk_attach(ide_drive_t *drive) ...@@ -1867,10 +1854,7 @@ static int idedisk_attach(ide_drive_t *drive)
if ((!drive->head || drive->head > 16) && !drive->select.b.lba) { if ((!drive->head || drive->head > 16) && !drive->select.b.lba) {
printk(KERN_ERR "%s: INVALID GEOMETRY: %d PHYSICAL HEADS?\n", printk(KERN_ERR "%s: INVALID GEOMETRY: %d PHYSICAL HEADS?\n",
drive->name, drive->head); drive->name, drive->head);
if ((drive->id->cfs_enable_2 & 0x3000) && drive->wcache) ide_cacheflush_p(drive);
if (do_idedisk_flushcache(drive))
printk (KERN_INFO "%s: Write Cache FAILED Flushing!\n",
drive->name);
ide_unregister_subdriver(drive); ide_unregister_subdriver(drive);
DRIVER(drive)->busy--; DRIVER(drive)->busy--;
goto failed; goto failed;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment