diff options
author | Uwe Stuehler <uwe@cvs.openbsd.org> | 2007-03-29 18:42:39 +0000 |
---|---|---|
committer | Uwe Stuehler <uwe@cvs.openbsd.org> | 2007-03-29 18:42:39 +0000 |
commit | baa31b7505103721e06149694e3a0a0a6b726a75 (patch) | |
tree | 1e8e36dc507c5aee0c34c04ed2a52c394b9bf9f0 /sys/arch/zaurus/dev | |
parent | 39db385c566253b8b3e3fc4bf00808441f47b61c (diff) |
Add power hook to scoop(4), replacing the Zaurus-specific scoop_* calls in
pxa2x0_apm.c.
Set a flag to indicate that scoop is currently suspended. Test this flag
in scoop_timeout() (and maybe later in other places) to avoid manipulating
scoop registers while the driver is suspended. The scoop_checkdisk timeout
is intentionally not stopped, as it should be safe to have timeouts running
accross suspend/resume.
Put splhigh() in scoop_gpio_pin_write() to make changing the pin level an
atomic operation.
Result of prodding and diffs from Robert and Theo.
Diffstat (limited to 'sys/arch/zaurus/dev')
-rw-r--r-- | sys/arch/zaurus/dev/zaurus_scoop.c | 53 |
1 files changed, 50 insertions, 3 deletions
diff --git a/sys/arch/zaurus/dev/zaurus_scoop.c b/sys/arch/zaurus/dev/zaurus_scoop.c index 795a36d53ce..71d50a29465 100644 --- a/sys/arch/zaurus/dev/zaurus_scoop.c +++ b/sys/arch/zaurus/dev/zaurus_scoop.c @@ -1,4 +1,4 @@ -/* $OpenBSD: zaurus_scoop.c,v 1.15 2007/03/27 23:23:22 deraadt Exp $ */ +/* $OpenBSD: zaurus_scoop.c,v 1.16 2007/03/29 18:42:38 uwe Exp $ */ /* * Copyright (c) 2005 Uwe Stuehler <uwe@bsdx.de> @@ -40,6 +40,8 @@ struct scoop_softc { bus_space_tag_t sc_iot; bus_space_handle_t sc_ioh; u_int16_t sc_gpwr; /* GPIO state before suspend */ + void *sc_powerhook; + int sc_suspended; }; int scoopmatch(struct device *, void *, void *); @@ -65,6 +67,7 @@ void scoop0_set_card_power(enum card, int); struct timeout scoop_checkdisk; void scoop_timeout(void *); +void scoop_power(int, void *); int scoopmatch(struct device *parent, void *match, void *aux) @@ -112,9 +115,15 @@ scoopattach(struct device *parent, struct device *self, void *aux) scoop_gpio_pin_write(sc, SCOOP0_AKIN_PULLUP, GPIO_PIN_LOW); } - timeout_set(&scoop_checkdisk, scoop_timeout, sc); + if (sc->sc_dev.dv_unit == 0) + timeout_set(&scoop_checkdisk, scoop_timeout, sc); printf(": PCMCIA/GPIO controller\n"); + + sc->sc_powerhook = powerhook_establish(scoop_power, sc); + if (sc->sc_powerhook == NULL) + panic("Unable to establish %s powerhook", + sc->sc_dev.dv_xname); } int @@ -132,10 +141,13 @@ scoop_gpio_pin_write(struct scoop_softc *sc, int pin, int level) { u_int16_t rv; u_int16_t bit = (1 << pin); + int s; + s = splhigh(); rv = bus_space_read_2(sc->sc_iot, sc->sc_ioh, SCOOP_GPWR); bus_space_write_2(sc->sc_iot, sc->sc_ioh, SCOOP_GPWR, level == GPIO_PIN_LOW ? (rv & ~bit) : (rv | bit)); + splx(s); } void @@ -358,6 +370,8 @@ scoop_suspend(void) struct scoop_softc *sc; u_int32_t rv; + scoop_check_mcr(); + if (scoop_cd.cd_ndevs > 0 && scoop_cd.cd_devs[0] != NULL) { sc = scoop_cd.cd_devs[0]; sc->sc_gpwr = bus_space_read_2(sc->sc_iot, sc->sc_ioh, @@ -390,6 +404,8 @@ scoop_resume(void) { struct scoop_softc *sc; + scoop_check_mcr(); + if (scoop_cd.cd_ndevs > 0 && scoop_cd.cd_devs[0] != NULL) { sc = scoop_cd.cd_devs[0]; bus_space_write_2(sc->sc_iot, sc->sc_ioh, SCOOP_GPWR, @@ -407,6 +423,7 @@ void scoop_timeout(void *v) { extern struct disklist_head disklist; + struct scoop_softc *sc = v; static struct disk *dk; static int state = 0; @@ -418,7 +435,9 @@ scoop_timeout(void *v) break; } - if (dk) { + if (sc->sc_suspended) + state = -1; + else if (dk) { int newstate = (dk->dk_busy ? 1 : 0); if (newstate != state) { @@ -428,3 +447,31 @@ scoop_timeout(void *v) } timeout_add(&scoop_checkdisk, hz/25); } + +void +scoop_power(int why, void *arg) +{ + struct scoop_softc *sc = arg; + + switch (why) { + case PWR_STANDBY: + case PWR_SUSPEND: + /* + * Nothing should use the scoop from this point on. + * No timeouts, no interrupts (even though interrupts + * are still enabled). scoop_timeout() respects the + * sc_suspended flag. + */ + if (sc->sc_dev.dv_unit == 0) { + sc->sc_suspended = 1; + scoop_suspend(); + } + break; + case PWR_RESUME: + if (sc->sc_dev.dv_unit == 0) { + scoop_resume(); + sc->sc_suspended = 0; + } + break; + } +} |