diff options
| author | Linus Torvalds <torvalds@linux-foundation.org> | 2008-03-19 00:26:24 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2008-03-19 00:26:24 -0400 |
| commit | 6c3c3158a81d6a92d335dd27ad9eb43f6b4c664b (patch) | |
| tree | fc740e2fff9c0f3d9e75848b6e106f4251dc7ec7 | |
| parent | d5eee405723eedbd621275e045ac9b36f668c39f (diff) | |
IDE: Make taskfile interface more robust wrt unexpected end-of-command
Now that we handle all the special commands using REQ_TYPE_ATA_TASKFILE
rather than using the old REQ_TYPE_ATA_CMD model, we need to also
emulate the lack of full taskfile data that comes with the old command
model (ie when commands are generated with the HDIO_DRIVE_CMD ioctl
rather than using the HDIO_DRIVE_TASK[FILE] ioctls).
In particular, this means that we should handle command completion the
more relaxed way that the old drive_cmd_intr() code did. It allows
commands to finish early even if they don't use up all the data that we
thought we had for them.
This fixes a regression seen by Anders Eriksson where some SMART
commands sent by smartd would cause a boot-time system hang on his
machine because the IDE command handling code didn't realize that the
command had completed.
Tested-by: Anders Eriksson <aeriksson@fastmail.fm>
Cc: Jens Axboe <jens.axboe@oracle.com>
Cc: Ingo Molnar <mingo@elte.hu>
Acked-by: Bartlomiej Zolnierkiewicz <bzolnier@gmail.com>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
| -rw-r--r-- | drivers/ide/ide-taskfile.c | 36 |
1 files changed, 27 insertions, 9 deletions
diff --git a/drivers/ide/ide-taskfile.c b/drivers/ide/ide-taskfile.c index 0518a2e948cf..4c86a8d84b4c 100644 --- a/drivers/ide/ide-taskfile.c +++ b/drivers/ide/ide-taskfile.c | |||
| @@ -423,6 +423,25 @@ void task_end_request(ide_drive_t *drive, struct request *rq, u8 stat) | |||
| 423 | } | 423 | } |
| 424 | 424 | ||
| 425 | /* | 425 | /* |
| 426 | * We got an interrupt on a task_in case, but no errors and no DRQ. | ||
| 427 | * | ||
| 428 | * It might be a spurious irq (shared irq), but it might be a | ||
| 429 | * command that had no output. | ||
| 430 | */ | ||
| 431 | static ide_startstop_t task_in_unexpected(ide_drive_t *drive, struct request *rq, u8 stat) | ||
| 432 | { | ||
| 433 | /* Command all done? */ | ||
| 434 | if (OK_STAT(stat, READY_STAT, BUSY_STAT)) { | ||
| 435 | task_end_request(drive, rq, stat); | ||
| 436 | return ide_stopped; | ||
| 437 | } | ||
| 438 | |||
| 439 | /* Assume it was a spurious irq */ | ||
| 440 | ide_set_handler(drive, &task_in_intr, WAIT_WORSTCASE, NULL); | ||
| 441 | return ide_started; | ||
| 442 | } | ||
| 443 | |||
| 444 | /* | ||
| 426 | * Handler for command with PIO data-in phase (Read/Read Multiple). | 445 | * Handler for command with PIO data-in phase (Read/Read Multiple). |
| 427 | */ | 446 | */ |
| 428 | static ide_startstop_t task_in_intr(ide_drive_t *drive) | 447 | static ide_startstop_t task_in_intr(ide_drive_t *drive) |
| @@ -431,18 +450,17 @@ static ide_startstop_t task_in_intr(ide_drive_t *drive) | |||
| 431 | struct request *rq = HWGROUP(drive)->rq; | 450 | struct request *rq = HWGROUP(drive)->rq; |
| 432 | u8 stat = ide_read_status(drive); | 451 | u8 stat = ide_read_status(drive); |
| 433 | 452 | ||
| 434 | /* new way for dealing with premature shared PCI interrupts */ | 453 | /* Error? */ |
| 435 | if (!OK_STAT(stat, DRQ_STAT, BAD_R_STAT)) { | 454 | if (stat & ERR_STAT) |
| 436 | if (stat & (ERR_STAT | DRQ_STAT)) | 455 | return task_error(drive, rq, __FUNCTION__, stat); |
| 437 | return task_error(drive, rq, __FUNCTION__, stat); | 456 | |
| 438 | /* No data yet, so wait for another IRQ. */ | 457 | /* Didn't want any data? Odd. */ |
| 439 | ide_set_handler(drive, &task_in_intr, WAIT_WORSTCASE, NULL); | 458 | if (!(stat & DRQ_STAT)) |
| 440 | return ide_started; | 459 | return task_in_unexpected(drive, rq, stat); |
| 441 | } | ||
| 442 | 460 | ||
| 443 | ide_pio_datablock(drive, rq, 0); | 461 | ide_pio_datablock(drive, rq, 0); |
| 444 | 462 | ||
| 445 | /* If it was the last datablock check status and finish transfer. */ | 463 | /* Are we done? Check status and finish transfer. */ |
| 446 | if (!hwif->nleft) { | 464 | if (!hwif->nleft) { |
| 447 | stat = wait_drive_not_busy(drive); | 465 | stat = wait_drive_not_busy(drive); |
| 448 | if (!OK_STAT(stat, 0, BAD_STAT)) | 466 | if (!OK_STAT(stat, 0, BAD_STAT)) |
