aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/ide/ide-lib.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/ide/ide-lib.c')
-rw-r--r--drivers/ide/ide-lib.c69
1 files changed, 37 insertions, 32 deletions
diff --git a/drivers/ide/ide-lib.c b/drivers/ide/ide-lib.c
index 001085845a79..15736d4ce9b4 100644
--- a/drivers/ide/ide-lib.c
+++ b/drivers/ide/ide-lib.c
@@ -479,6 +479,42 @@ static void ide_dump_opcode(ide_drive_t *drive)
479 printk("0x%02x\n", opcode); 479 printk("0x%02x\n", opcode);
480} 480}
481 481
482static u64 ide_get_lba_addr(struct ide_taskfile *tf, int lba48)
483{
484 u32 high, low;
485
486 if (lba48)
487 high = (tf->hob_lbah << 16) | (tf->hob_lbam << 8) |
488 tf->hob_lbal;
489 else
490 high = tf->device & 0xf;
491 low = (tf->lbah << 16) | (tf->lbam << 8) | tf->lbal;
492
493 return ((u64)high << 24) | low;
494}
495
496static void ide_dump_sector(ide_drive_t *drive)
497{
498 ide_task_t task;
499 struct ide_taskfile *tf = &task.tf;
500 int lba48 = (drive->addressing == 1) ? 1 : 0;
501
502 memset(&task, 0, sizeof(task));
503 if (lba48)
504 task.tf_flags = IDE_TFLAG_IN_LBA | IDE_TFLAG_IN_HOB_LBA |
505 IDE_TFLAG_LBA48;
506 else
507 task.tf_flags = IDE_TFLAG_IN_LBA | IDE_TFLAG_IN_DEVICE;
508
509 ide_tf_read(drive, &task);
510
511 if (lba48 || (tf->device & ATA_LBA))
512 printk(", LBAsect=%llu", ide_get_lba_addr(tf, lba48));
513 else
514 printk(", CHS=%d/%d/%d", (tf->lbah << 8) + tf->lbam,
515 tf->device & 0xf, tf->lbal);
516}
517
482static u8 ide_dump_ata_status(ide_drive_t *drive, const char *msg, u8 stat) 518static u8 ide_dump_ata_status(ide_drive_t *drive, const char *msg, u8 stat)
483{ 519{
484 ide_hwif_t *hwif = HWIF(drive); 520 ide_hwif_t *hwif = HWIF(drive);
@@ -512,38 +548,7 @@ static u8 ide_dump_ata_status(ide_drive_t *drive, const char *msg, u8 stat)
512 printk("}"); 548 printk("}");
513 if ((err & (BBD_ERR | ABRT_ERR)) == BBD_ERR || 549 if ((err & (BBD_ERR | ABRT_ERR)) == BBD_ERR ||
514 (err & (ECC_ERR|ID_ERR|MARK_ERR))) { 550 (err & (ECC_ERR|ID_ERR|MARK_ERR))) {
515 if (drive->addressing == 1) { 551 ide_dump_sector(drive);
516 __u64 sectors = 0;
517 u32 low = 0, high = 0;
518 hwif->OUTB(drive->ctl&~0x80, IDE_CONTROL_REG);
519 low = ide_read_24(drive);
520 hwif->OUTB(drive->ctl|0x80, IDE_CONTROL_REG);
521 high = ide_read_24(drive);
522 sectors = ((__u64)high << 24) | low;
523 printk(", LBAsect=%llu, high=%d, low=%d",
524 (unsigned long long) sectors,
525 high, low);
526 } else {
527 u8 sector, lcyl, hcyl, cur;
528
529 sector = hwif->INB(IDE_SECTOR_REG);
530 lcyl = hwif->INB(IDE_LCYL_REG);
531 hcyl = hwif->INB(IDE_HCYL_REG);
532 cur = hwif->INB(IDE_SELECT_REG);
533
534 if (cur & 0x40) { /* using LBA? */
535 printk(", LBAsect=%ld", (unsigned long)
536 ((cur & 0xf) << 24) |
537 (hcyl << 16) |
538 (lcyl << 8) |
539 sector);
540 } else {
541 printk(", CHS=%d/%d/%d",
542 (hcyl << 8) + lcyl,
543 cur & 0xf,
544 sector);
545 }
546 }
547 if (HWGROUP(drive) && HWGROUP(drive)->rq) 552 if (HWGROUP(drive) && HWGROUP(drive)->rq)
548 printk(", sector=%llu", 553 printk(", sector=%llu",
549 (unsigned long long)HWGROUP(drive)->rq->sector); 554 (unsigned long long)HWGROUP(drive)->rq->sector);