Skip to content

Commit 57e5568

Browse files
Ondrej Zaryhtejun
authored andcommitted
sata_via: Implement hotplug for VT6421
Enable IRQ on hotplug and add an interrupt handler to handle it. This allows hotplug to work: ata5: exception Emask 0x10 SAct 0x0 SErr 0x70000 action 0xe frozen ata5: SError: { PHYRdyChg PHYInt CommWake } ata5: hard resetting link ata5: SATA link up 1.5 Gbps (SStatus 113 SControl 310) ata5.00: LPM support broken, forcing max_power ata5.00: ATA-7: WDC WD800JD-75MSA3, 10.01E04, max UDMA/133 ata5.00: 156250000 sectors, multi 0: LBA48 NCQ (depth 0/32) ata5.00: LPM support broken, forcing max_power ata5.00: configured for UDMA/133 ata5: EH complete scsi 4:0:0:0: Direct-Access ATA WDC WD800JD-75MS 1E04 PQ: 0 ANSI: 5 sd 4:0:0:0: [sdb] 156250000 512-byte logical blocks: (80.0 GB/74.5 GiB) sd 4:0:0:0: [sdb] Write Protect is off sd 4:0:0:0: [sdb] Mode Sense: 00 3a 00 00 sd 4:0:0:0: [sdb] Write cache: enabled, read cache: enabled, doesn't support DPO or FUA sd 4:0:0:0: Attached scsi generic sg1 type 0 sd 4:0:0:0: [sdb] Attached SCSI disk And also hot unplug: ata5: exception Emask 0x10 SAct 0x0 SErr 0x1b0000 action 0xe frozen ata5: SError: { PHYRdyChg PHYInt 10B8B Dispar } ata5: hard resetting link ata5: SATA link down (SStatus 0 SControl 310) ata5: hard resetting link ata5: SATA link down (SStatus 0 SControl 310) ata5: hard resetting link ata5: SATA link down (SStatus 0 SControl 310) ata5.00: disabled ata5: EH complete ata5.00: detaching (SCSI 4:0:0:0) sd 4:0:0:0: [sdb] Synchronizing SCSI cache sd 4:0:0:0: [sdb] Synchronize Cache(10) failed: Result: hostbyte=DID_BAD_TARGET driverbyte=DRIVER_OK sd 4:0:0:0: [sdb] Stopping disk sd 4:0:0:0: [sdb] Start/Stop Unit failed: Result: hostbyte=DID_BAD_TARGET driverbyte=DRIVER_OK Signed-off-by: Ondrej Zary <linux@rainbow-software.org> Signed-off-by: Tejun Heo <tj@kernel.org>
1 parent 44a9b49 commit 57e5568

File tree

1 file changed

+50
-2
lines changed

1 file changed

+50
-2
lines changed

drivers/ata/sata_via.c

Lines changed: 50 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,7 @@ enum {
6161
SATA_CHAN_ENAB = 0x40, /* SATA channel enable */
6262
SATA_INT_GATE = 0x41, /* SATA interrupt gating */
6363
SATA_NATIVE_MODE = 0x42, /* Native mode enable */
64+
SVIA_MISC_3 = 0x46, /* Miscellaneous Control III */
6465
PATA_UDMA_TIMING = 0xB3, /* PATA timing for DMA/ cable detect */
6566
PATA_PIO_TIMING = 0xAB, /* PATA timing register */
6667

@@ -71,6 +72,8 @@ enum {
7172
NATIVE_MODE_ALL = (1 << 7) | (1 << 6) | (1 << 5) | (1 << 4),
7273

7374
SATA_EXT_PHY = (1 << 6), /* 0==use PATA, 1==ext phy */
75+
76+
SATA_HOTPLUG = (1 << 5), /* enable IRQ on hotplug */
7477
};
7578

7679
struct svia_priv {
@@ -553,6 +556,37 @@ static void svia_wd_fix(struct pci_dev *pdev)
553556
pci_write_config_byte(pdev, 0x52, tmp8 | BIT(2));
554557
}
555558

559+
static irqreturn_t vt6421_interrupt(int irq, void *dev_instance)
560+
{
561+
struct ata_host *host = dev_instance;
562+
irqreturn_t rc = ata_bmdma_interrupt(irq, dev_instance);
563+
564+
/* if the IRQ was not handled, it might be a hotplug IRQ */
565+
if (rc != IRQ_HANDLED) {
566+
u32 serror;
567+
unsigned long flags;
568+
569+
spin_lock_irqsave(&host->lock, flags);
570+
/* check for hotplug on port 0 */
571+
svia_scr_read(&host->ports[0]->link, SCR_ERROR, &serror);
572+
if (serror & SERR_PHYRDY_CHG) {
573+
ata_ehi_hotplugged(&host->ports[0]->link.eh_info);
574+
ata_port_freeze(host->ports[0]);
575+
rc = IRQ_HANDLED;
576+
}
577+
/* check for hotplug on port 1 */
578+
svia_scr_read(&host->ports[1]->link, SCR_ERROR, &serror);
579+
if (serror & SERR_PHYRDY_CHG) {
580+
ata_ehi_hotplugged(&host->ports[1]->link.eh_info);
581+
ata_port_freeze(host->ports[1]);
582+
rc = IRQ_HANDLED;
583+
}
584+
spin_unlock_irqrestore(&host->lock, flags);
585+
}
586+
587+
return rc;
588+
}
589+
556590
static void vt6421_error_handler(struct ata_port *ap)
557591
{
558592
struct svia_priv *hpriv = ap->host->private_data;
@@ -610,6 +644,16 @@ static void svia_configure(struct pci_dev *pdev, int board_id,
610644
pci_write_config_byte(pdev, SATA_NATIVE_MODE, tmp8);
611645
}
612646

647+
/* enable IRQ on hotplug */
648+
pci_read_config_byte(pdev, SVIA_MISC_3, &tmp8);
649+
if ((tmp8 & SATA_HOTPLUG) != SATA_HOTPLUG) {
650+
dev_dbg(&pdev->dev,
651+
"enabling SATA hotplug (0x%x)\n",
652+
(int) tmp8);
653+
tmp8 |= SATA_HOTPLUG;
654+
pci_write_config_byte(pdev, SVIA_MISC_3, tmp8);
655+
}
656+
613657
/*
614658
* vt6420/1 has problems talking to some drives. The following
615659
* is the fix from Joseph Chan <JosephChan@via.com.tw>.
@@ -698,8 +742,12 @@ static int svia_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
698742
svia_configure(pdev, board_id, hpriv);
699743

700744
pci_set_master(pdev);
701-
return ata_host_activate(host, pdev->irq, ata_bmdma_interrupt,
702-
IRQF_SHARED, &svia_sht);
745+
if (board_id == vt6421)
746+
return ata_host_activate(host, pdev->irq, vt6421_interrupt,
747+
IRQF_SHARED, &svia_sht);
748+
else
749+
return ata_host_activate(host, pdev->irq, ata_bmdma_interrupt,
750+
IRQF_SHARED, &svia_sht);
703751
}
704752

705753
#ifdef CONFIG_PM_SLEEP

0 commit comments

Comments
 (0)