aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTejun Heo <tj@kernel.org>2011-07-22 05:41:26 -0400
committerJeff Garzik <jgarzik@redhat.com>2012-01-08 19:14:57 -0500
commit7faa33da9b7add01db9f1ad92c6a5d9145e940a7 (patch)
tree13b91e9292ff56f7dc67b14a9cf93f30889be13f
parent805a6af8dba5dfdd35ec35dc52ec0122400b2610 (diff)
ahci: start engine only during soft/hard resets
This is another attempt at fixing the same problem that 270dac35c2 (libata: ahci_start_engine compliant to AHCI spec) tried to solve. Unfortunately, 270dac35c2 created regressions for a lot more common controllers and got reverted. This specific AHCI IP block becomes a brick if the DMA engine is started while DRQ is set. It is not possible to avoid the condition completely but the most common occurrence is caused by spurious use of ahci_start_engine() from ahci_start_port() during init sequence. DMA engine is started after both soft and hard resets and ahci_start_port() is always followed by resets, so there is no reason to start DMA engine from ahci_start_port(). This patch removes ahci_start_engine() invocation from ahci_start_port(). This change makes failure path of ahci_port_suspend() leave engine stopped without following resets. This is resolved by replacing ahci_start_port() call with ata_port_freeze() which forces resets afterwards, which is the better behavior anyway. Signed-off-by: Tejun Heo <tj@kernel.org> Reported-by: Brian Norris <computersforpeace@gmail.com> Reported-by: Jian Peng <jipeng2005@gmail.com> Signed-off-by: Jeff Garzik <jgarzik@redhat.com>
-rw-r--r--drivers/ata/libahci.c5
1 files changed, 1 insertions, 4 deletions
diff --git a/drivers/ata/libahci.c b/drivers/ata/libahci.c
index 3c92dbd751e0..a72bfd0ecfee 100644
--- a/drivers/ata/libahci.c
+++ b/drivers/ata/libahci.c
@@ -746,9 +746,6 @@ static void ahci_start_port(struct ata_port *ap)
746 /* enable FIS reception */ 746 /* enable FIS reception */
747 ahci_start_fis_rx(ap); 747 ahci_start_fis_rx(ap);
748 748
749 /* enable DMA */
750 ahci_start_engine(ap);
751
752 /* turn on LEDs */ 749 /* turn on LEDs */
753 if (ap->flags & ATA_FLAG_EM) { 750 if (ap->flags & ATA_FLAG_EM) {
754 ata_for_each_link(link, ap, EDGE) { 751 ata_for_each_link(link, ap, EDGE) {
@@ -2022,7 +2019,7 @@ static int ahci_port_suspend(struct ata_port *ap, pm_message_t mesg)
2022 ahci_power_down(ap); 2019 ahci_power_down(ap);
2023 else { 2020 else {
2024 ata_port_err(ap, "%s (%d)\n", emsg, rc); 2021 ata_port_err(ap, "%s (%d)\n", emsg, rc);
2025 ahci_start_port(ap); 2022 ata_port_freeze(ap);
2026 } 2023 }
2027 2024
2028 return rc; 2025 return rc;