diff options
author | Mark Lord <liml@rtr.ca> | 2009-04-06 15:24:14 -0400 |
---|---|---|
committer | Jeff Garzik <jgarzik@redhat.com> | 2009-04-06 20:13:48 -0400 |
commit | 65ad7fef10b50b6c06d6165fa847e2d3636b0a66 (patch) | |
tree | 500afce9205d59853e14901813ae4f6689210019 | |
parent | c361acbc59c434315f8649ab06e5b7d5b297d1b7 (diff) |
sata_mv: workaround errata PCI#7
Workaround for 60x1-B2 errata PCI#7.
Write-combining may be unreliable when chip operates in PCI-X mode,
so disable write-combining when in PCI-X mode.
Also, update the errata comments at the top of sata_mv,
and include a note about errata PCI#11.
Signed-off-by: Mark Lord <mlord@pobox.com>
Signed-off-by: Jeff Garzik <jgarzik@redhat.com>
-rw-r--r-- | drivers/ata/sata_mv.c | 27 |
1 files changed, 23 insertions, 4 deletions
diff --git a/drivers/ata/sata_mv.c b/drivers/ata/sata_mv.c index a49643d16f3..b0c929d3623 100644 --- a/drivers/ata/sata_mv.c +++ b/drivers/ata/sata_mv.c | |||
@@ -28,10 +28,6 @@ | |||
28 | /* | 28 | /* |
29 | * sata_mv TODO list: | 29 | * sata_mv TODO list: |
30 | * | 30 | * |
31 | * --> More errata workarounds for PCI-X. | ||
32 | * | ||
33 | * --> Complete a full errata audit for all chipsets to identify others. | ||
34 | * | ||
35 | * --> Develop a low-power-consumption strategy, and implement it. | 31 | * --> Develop a low-power-consumption strategy, and implement it. |
36 | * | 32 | * |
37 | * --> Add sysfs attributes for per-chip / per-HC IRQ coalescing thresholds. | 33 | * --> Add sysfs attributes for per-chip / per-HC IRQ coalescing thresholds. |
@@ -44,6 +40,15 @@ | |||
44 | * connect two SATA ports. | 40 | * connect two SATA ports. |
45 | */ | 41 | */ |
46 | 42 | ||
43 | /* | ||
44 | * 80x1-B2 errata PCI#11: | ||
45 | * | ||
46 | * Users of the 6041/6081 Rev.B2 chips (current is C0) | ||
47 | * should be careful to insert those cards only onto PCI-X bus #0, | ||
48 | * and only in device slots 0..7, not higher. The chips may not | ||
49 | * work correctly otherwise (note: this is a pretty rare condition). | ||
50 | */ | ||
51 | |||
47 | #include <linux/kernel.h> | 52 | #include <linux/kernel.h> |
48 | #include <linux/module.h> | 53 | #include <linux/module.h> |
49 | #include <linux/pci.h> | 54 | #include <linux/pci.h> |
@@ -181,6 +186,7 @@ enum { | |||
181 | /* PCI interface registers */ | 186 | /* PCI interface registers */ |
182 | 187 | ||
183 | PCI_COMMAND_OFS = 0xc00, | 188 | PCI_COMMAND_OFS = 0xc00, |
189 | PCI_COMMAND_MWRCOM = (1 << 4), /* PCI Master Write Combining */ | ||
184 | PCI_COMMAND_MRDTRIG = (1 << 7), /* PCI Master Read Trigger */ | 190 | PCI_COMMAND_MRDTRIG = (1 << 7), /* PCI Master Read Trigger */ |
185 | 191 | ||
186 | PCI_MAIN_CMD_STS_OFS = 0xd30, | 192 | PCI_MAIN_CMD_STS_OFS = 0xd30, |
@@ -3527,6 +3533,18 @@ static int mv_pci_cut_through_okay(struct ata_host *host) | |||
3527 | return 1; /* okay */ | 3533 | return 1; /* okay */ |
3528 | } | 3534 | } |
3529 | 3535 | ||
3536 | static void mv_60x1b2_errata_pci7(struct ata_host *host) | ||
3537 | { | ||
3538 | struct mv_host_priv *hpriv = host->private_data; | ||
3539 | void __iomem *mmio = hpriv->base; | ||
3540 | |||
3541 | /* workaround for 60x1-B2 errata PCI#7 */ | ||
3542 | if (mv_in_pcix_mode(host)) { | ||
3543 | u32 reg = readl(mmio + PCI_COMMAND_OFS); | ||
3544 | writelfl(reg & ~PCI_COMMAND_MWRCOM, mmio + PCI_COMMAND_OFS); | ||
3545 | } | ||
3546 | } | ||
3547 | |||
3530 | static int mv_chip_id(struct ata_host *host, unsigned int board_idx) | 3548 | static int mv_chip_id(struct ata_host *host, unsigned int board_idx) |
3531 | { | 3549 | { |
3532 | struct pci_dev *pdev = to_pci_dev(host->dev); | 3550 | struct pci_dev *pdev = to_pci_dev(host->dev); |
@@ -3580,6 +3598,7 @@ static int mv_chip_id(struct ata_host *host, unsigned int board_idx) | |||
3580 | 3598 | ||
3581 | switch (pdev->revision) { | 3599 | switch (pdev->revision) { |
3582 | case 0x7: | 3600 | case 0x7: |
3601 | mv_60x1b2_errata_pci7(host); | ||
3583 | hp_flags |= MV_HP_ERRATA_60X1B2; | 3602 | hp_flags |= MV_HP_ERRATA_60X1B2; |
3584 | break; | 3603 | break; |
3585 | case 0x9: | 3604 | case 0x9: |