Subject: Re: viaide(4) patch for suspend/resume
To: None <current-users@netbsd.org>
From: Joerg Sonnenberger <joerg@britannica.bec.de>
List: current-users
Date: 12/22/2007 00:45:27
--r5Pyd7+fXNt84Ff3
Content-Type: text/plain; charset=us-ascii
Content-Disposition: inline

On Fri, Dec 21, 2007 at 06:41:02PM -0500, Jared D. McNeill wrote:
> On Sat, 22 Dec 2007, Joerg Sonnenberger wrote:
>>> This is clearly wrong:
>>
>> You can just read the registers on other chipsets as well, I don't think
>> there's any hard in writing them back as well.
>
> You misunderstand. You are reading four PCI conf registers into a single 
> pcireg_t, and restoring garbage in the resume handler. This will never 
> work.

Ooops. Too much code in too few days :-) Better patch attached, thanks.

Joerg

--r5Pyd7+fXNt84Ff3
Content-Type: text/plain; charset=us-ascii
Content-Disposition: attachment; filename="viaide.diff"

Index: pciidevar.h
===================================================================
RCS file: /data/repo/netbsd/src/sys/dev/pci/pciidevar.h,v
retrieving revision 1.35
diff -u -p -r1.35 pciidevar.h
--- pciidevar.h	17 Oct 2006 13:45:05 -0000	1.35
+++ pciidevar.h	19 Dec 2007 14:25:34 -0000
@@ -150,12 +150,7 @@ struct pciide_softc {
 #endif	/* NATA_DMA */
 	} pciide_channels[PCIIDE_MAX_CHANNELS];
 
-	/* Power management */
-	void			*sc_powerhook;
-	struct pci_conf_state	sc_pciconf; /* Restore buffer */
-	/* Intel power management */
-	pcireg_t		sc_idetim;
-	pcireg_t		sc_udmatim;
+	pcireg_t		sc_pm_reg[4];
 };
 
 /* Given an ata_channel, get the pciide_softc. */
Index: piixide.c
===================================================================
RCS file: /data/repo/netbsd/src/sys/dev/pci/piixide.c,v
retrieving revision 1.42
diff -u -p -r1.42 piixide.c
--- piixide.c	9 Dec 2007 20:28:12 -0000	1.42
+++ piixide.c	19 Dec 2007 14:26:05 -0000
@@ -288,9 +288,9 @@ piixide_resume(device_t dv)
 	struct pciide_softc *sc = device_private(dv);
 
 	pci_conf_write(sc->sc_pc, sc->sc_tag, PIIX_IDETIM,
-	    sc->sc_idetim);
+	    sc->sc_pm_reg[0]);
 	pci_conf_write(sc->sc_pc, sc->sc_tag, PIIX_UDMATIM,
-	    sc->sc_udmatim);
+	    sc->sc_pm_reg[1]);
 
 	return true;
 }
@@ -300,9 +300,9 @@ piixide_suspend(device_t dv)
 {
 	struct pciide_softc *sc = device_private(dv);
 
-	sc->sc_idetim = pci_conf_read(sc->sc_pc, sc->sc_tag,
+	sc->sc_pm_reg[0] = pci_conf_read(sc->sc_pc, sc->sc_tag,
 	    PIIX_IDETIM);
-	sc->sc_udmatim = pci_conf_read(sc->sc_pc, sc->sc_tag,
+	sc->sc_pm_reg[1] = pci_conf_read(sc->sc_pc, sc->sc_tag,
 	    PIIX_UDMATIM);
 
 	return true;
Index: viaide.c
===================================================================
RCS file: /data/repo/netbsd/src/sys/dev/pci/viaide.c,v
retrieving revision 1.50
diff -u -p -r1.50 viaide.c
--- viaide.c	20 Dec 2007 22:24:40 -0000	1.50
+++ viaide.c	21 Dec 2007 23:44:44 -0000
@@ -67,6 +67,8 @@ static int	viaide_match(struct device *,
 static void	viaide_attach(struct device *, struct device *, void *);
 static const struct pciide_product_desc *
 		viaide_lookup(pcireg_t);
+static bool	viaide_suspend(device_t);
+static bool	viaide_resume(device_t);
 
 CFATTACH_DECL(viaide, sizeof(struct pciide_softc),
     viaide_match, viaide_attach, NULL, NULL);
@@ -374,6 +376,9 @@ viaide_attach(struct device *parent, str
 	if (pp == NULL)
 		panic("viaide_attach");
 	pciide_common_attach(sc, pa, pp);
+
+	if (!pmf_device_register(self, viaide_suspend, viaide_resume))
+		aprint_error_dev(self, "couldn't establish power handler\n");
 }
 
 static int
@@ -386,6 +391,39 @@ via_pcib_match(struct pci_attach_args *p
 	return 0;
 }
 
+static bool
+viaide_suspend(device_t dv)
+{
+	struct pciide_softc *sc = device_private(dv);
+
+	sc->sc_pm_reg[0] = pci_conf_read(sc->sc_pc, sc->sc_tag, APO_IDECONF(sc));
+	/* APO_DATATIM(sc) includes APO_UDMA(sc) */
+	sc->sc_pm_reg[1] = pci_conf_read(sc->sc_pc, sc->sc_tag, APO_DATATIM(sc));
+	/* This two are VIA-only, but should be ignored by other devices. */
+	sc->sc_pm_reg[2] = pci_conf_read(sc->sc_pc, sc->sc_tag, APO_CTLMISC(sc));
+	sc->sc_pm_reg[3] = pci_conf_read(sc->sc_pc, sc->sc_tag, APO_MISCTIM(sc));
+
+	return true;
+}
+
+static bool
+viaide_resume(device_t dv)
+{
+	struct pciide_softc *sc = device_private(dv);
+
+	pci_conf_write(sc->sc_pc, sc->sc_tag, APO_IDECONF(sc),
+	    sc->sc_pm_reg[0]);
+	pci_conf_write(sc->sc_pc, sc->sc_tag, APO_DATATIM(sc),
+	    sc->sc_pm_reg[1]);
+	/* This two are VIA-only, but should be ignored by other devices. */
+	pci_conf_write(sc->sc_pc, sc->sc_tag, APO_CTLMISC(sc),
+	    sc->sc_pm_reg[2]);
+	pci_conf_write(sc->sc_pc, sc->sc_tag, APO_MISCTIM(sc),
+	    sc->sc_pm_reg[3]);
+
+	return true;
+}
+
 static void
 via_chip_map(struct pciide_softc *sc, struct pci_attach_args *pa)
 {

--r5Pyd7+fXNt84Ff3--