Skip site navigation (1)Skip section navigation (2)
Date:      Wed, 28 Mar 2007 13:55:42 +0100
From:      Andrea Bittau <a.bittau@cs.ucl.ac.uk>
To:        freebsd-current@freebsd.org
Cc:        sos@freebsd.org
Subject:   attempt to fix suspend/resume of AHCI SATA
Message-ID:  <20070328125542.GA5457@shorty.sorbonet.org>

next in thread | raw e-mail | index | archive | help
I have a rough patch for fixing suspend/resume of AHCI SATA.  Perhaps someone
with more knowledge could write a proper patch.  In short, this needs fixing:

1) Restore controller registers
2) Restore channel registers
3) Do-not reattach stuff [e.g. HDD] when receiving a phy-change/attach interrupt
   on resume.

---

Index: ata-all.c
===================================================================
RCS file: /home/ncvs/src/sys/dev/ata/ata-all.c,v
retrieving revision 1.279
diff -u -p -r1.279 ata-all.c
--- ata-all.c	23 Feb 2007 16:25:08 -0000	1.279
+++ ata-all.c	28 Mar 2007 12:55:03 -0000
@@ -49,6 +49,7 @@ __FBSDID("$FreeBSD: src/sys/dev/ata/ata-
 #include <sys/rman.h>
 #include <dev/ata/ata-all.h>
 #include <ata_if.h>
+#include <dev/ata/ata-pci.h>
 
 /* device structure */
 static  d_ioctl_t       ata_ioctl;
@@ -293,11 +294,16 @@ int
 ata_resume(device_t dev)
 {
     struct ata_channel *ch;
+    struct ata_pci_controller *ctlr;
     int error;
 
     /* check for valid device */
     if (!dev || !(ch = device_get_softc(dev)))
 	return ENXIO;
+
+    ctlr = device_get_softc(device_get_parent(dev));
+    if ((error = ctlr->allocate(dev)))
+    	return error;
 
     /* reinit the devices, we dont know what mode/state they are in */
     error = ata_reinit(dev);
Index: ata-chipset.c
===================================================================
RCS file: /home/ncvs/src/sys/dev/ata/ata-chipset.c,v
retrieving revision 1.192
diff -u -p -r1.192 ata-chipset.c
--- ata-chipset.c	12 Mar 2007 15:34:08 -0000	1.192
+++ ata-chipset.c	28 Mar 2007 12:55:05 -0000
@@ -286,6 +286,11 @@ ata_sata_phy_event(void *context, int du
 
     mtx_lock(&Giant);   /* newbus suckage it needs Giant */
     if (tp->action == ATA_C_ATTACH) {
+    	/* XXX */
+        if (!device_get_children(tp->dev, &children, &nchildren)) {
+		device_printf(tp->dev, "Assuming resume attach skipped\n");
+		goto out;
+	}
 	if (bootverbose)
 	    device_printf(tp->dev, "CONNECTED\n");
 	ATA_RESET(tp->dev);
@@ -304,6 +309,7 @@ ata_sata_phy_event(void *context, int du
 	if (bootverbose)
 	    device_printf(tp->dev, "DISCONNECTED\n");
     }
+out:
     mtx_unlock(&Giant); /* suckage code dealt with, release Giant */
     free(tp, M_ATA);
 }
Index: ata-pci.c
===================================================================
RCS file: /home/ncvs/src/sys/dev/ata/ata-pci.c,v
retrieving revision 1.121
diff -u -p -r1.121 ata-pci.c
--- ata-pci.c	23 Feb 2007 12:18:33 -0000	1.121
+++ ata-pci.c	28 Mar 2007 12:55:05 -0000
@@ -251,6 +251,19 @@ ata_pci_detach(device_t dev)
     return 0;
 }
 
+static int
+ata_pci_resume(device_t dev)
+{
+    struct ata_pci_controller *ctlr = device_get_softc(dev);
+    int rc;
+
+    rc = ctlr->chipinit(dev);
+    if (rc)
+    	return rc;
+
+    return bus_generic_resume(dev);
+}
+
 struct resource *
 ata_pci_alloc_resource(device_t dev, device_t child, int type, int *rid,
 		       u_long start, u_long end, u_long count, u_int flags)
@@ -510,7 +523,7 @@ static device_method_t ata_pci_methods[]
     DEVMETHOD(device_detach,            ata_pci_detach),
     DEVMETHOD(device_shutdown,          bus_generic_shutdown),
     DEVMETHOD(device_suspend,           bus_generic_suspend),
-    DEVMETHOD(device_resume,            bus_generic_resume),
+    DEVMETHOD(device_resume,            ata_pci_resume),
 
     /* bus methods */
     DEVMETHOD(bus_alloc_resource,       ata_pci_alloc_resource),



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?20070328125542.GA5457>