summaryrefslogtreecommitdiff
path: root/sys/dev/acpi/acpi.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/acpi/acpi.c')
-rw-r--r--sys/dev/acpi/acpi.c214
1 files changed, 158 insertions, 56 deletions
diff --git a/sys/dev/acpi/acpi.c b/sys/dev/acpi/acpi.c
index 42aab210cab..b34d35e1a18 100644
--- a/sys/dev/acpi/acpi.c
+++ b/sys/dev/acpi/acpi.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: acpi.c,v 1.129 2009/02/10 02:13:19 jordan Exp $ */
+/* $OpenBSD: acpi.c,v 1.130 2009/02/19 21:02:05 marco Exp $ */
/*
* Copyright (c) 2005 Thorsten Lockert <tholo@sigmasoft.com>
* Copyright (c) 2005 Jordan Hargrave <jordan@openbsd.org>
@@ -26,6 +26,7 @@
#include <sys/signalvar.h>
#include <sys/proc.h>
#include <sys/kthread.h>
+#include <sys/workq.h>
#include <machine/conf.h>
#include <machine/cpufunc.h>
@@ -81,6 +82,10 @@ void acpi_init_states(struct acpi_softc *);
void acpi_init_gpes(struct acpi_softc *);
void acpi_init_pm(struct acpi_softc *);
+#ifdef ACPI_SLEEP_ENABLED
+void acpi_sleep_walk(struct acpi_softc *, int);
+#endif /* ACPI_SLEEP_ENABLED */
+
#ifndef SMALL_KERNEL
int acpi_add_device(struct aml_node *node, void *arg);
#endif /* SMALL_KERNEL */
@@ -841,6 +846,13 @@ acpiioctl(dev_t dev, u_long cmd, caddr_t data, int flag, struct proc *p)
ACPI_LOCK(sc);
/* fake APM */
switch (cmd) {
+#ifdef ACPI_SLEEP_ENABLED
+ case APM_IOC_SUSPEND:
+ case APM_IOC_STANDBY:
+ workq_add_task(NULL, 0, (workq_fn)acpi_sleep_state,
+ acpi_softc, (void *)ACPI_STATE_S3);
+ break;
+#endif /* ACPI_SLEEP_ENABLED */
case APM_IOC_GETPOWER:
/* A/C */
pi->ac_state = APM_AC_UNKNOWN;
@@ -1651,65 +1663,78 @@ acpi_init_pm(struct acpi_softc *sc)
sc->sc_gts = aml_searchname(&aml_root, "_GTS");
}
+#ifndef SMALL_KERNEL
void
-acpi_enter_sleep_state(struct acpi_softc *sc, int state)
+acpi_sleep_walk(struct acpi_softc *sc, int state)
{
- struct aml_value env;
- u_int16_t rega, regb;
- int retries;
+ struct acpi_wakeq *wentry;
+ int idx;
- if (sc == NULL || state == ACPI_STATE_S0)
- return;
- if (sc->sc_sleeptype[state].slp_typa == -1 ||
- sc->sc_sleeptype[state].slp_typb == -1) {
- printf("%s: state S%d unavailable\n",
- sc->sc_dev.dv_xname, state);
- return;
+ /* Clear GPE status */
+ for (idx = 0; idx < sc->sc_lastgpe; idx += 8) {
+ acpi_write_pmreg(sc, ACPIREG_GPE_EN, idx>>3, 0);
+ acpi_write_pmreg(sc, ACPIREG_GPE_STS, idx>>3, -1);
}
- memset(&env, 0, sizeof(env));
- env.type = AML_OBJTYPE_INTEGER;
- env.v_integer = state;
- /* _TTS(state) */
- if (sc->sc_tts) {
- if (aml_evalnode(sc, sc->sc_tts, 1, &env, NULL) != 0) {
- dnprintf(10, "%s evaluating method _TTS failed.\n",
- DEVNAME(sc));
- return;
- }
+ SIMPLEQ_FOREACH(wentry, &sc->sc_wakedevs, q_next) {
+ dnprintf(10, "%.4s(S%d) gpe %.2x\n", wentry->q_node->name,
+ wentry->q_state,
+ wentry->q_gpe);
+
+ if (state <= wentry->q_state)
+ acpi_enable_onegpe(sc, wentry->q_gpe, 1);
}
+}
+#endif /* ! SMALL_KERNEL */
+
+int
+acpi_sleep_state(struct acpi_softc *sc, int state)
+{
+ int ret;
+
switch (state) {
+ case ACPI_STATE_S0:
+ return (0);
+ case ACPI_STATE_S4:
+ return (EOPNOTSUPP);
+ case ACPI_STATE_S5:
+ break;
case ACPI_STATE_S1:
case ACPI_STATE_S2:
- resettodr();
- dopowerhooks(PWR_SUSPEND);
- break;
case ACPI_STATE_S3:
- resettodr();
- dopowerhooks(PWR_STANDBY);
- break;
- }
- /* _PTS(state) */
- if (sc->sc_pts) {
- if (aml_evalnode(sc, sc->sc_pts, 1, &env, NULL) != 0) {
- dnprintf(10, "%s evaluating method _PTS failed.\n",
- DEVNAME(sc));
- return;
- }
- }
- sc->sc_state = state;
- /* _GTS(state) */
- if (sc->sc_gts) {
- if (aml_evalnode(sc, sc->sc_gts, 1, &env, NULL) != 0) {
- dnprintf(10, "%s evaluating method _GTS failed.\n",
- DEVNAME(sc));
- return;
- }
+ if (sc->sc_sleeptype[state].slp_typa == -1 ||
+ sc->sc_sleeptype[state].slp_typb == -1)
+ return (EOPNOTSUPP);
}
- disable_intr();
+
+ acpi_sleep_walk(sc, state);
+
+ if ((ret = acpi_prepare_sleep_state(sc, state)) != 0)
+ return (ret);
+
+ if (state != ACPI_STATE_S1)
+ ret = acpi_sleep_machdep(sc, state);
+ else
+ ret = acpi_enter_sleep_state(sc, state);
+
+#ifndef SMALL_KERNEL
+ acpi_resume(sc);
+ Debugger();
+#endif /* ! SMALL_KERNEL */
+ return (ret);
+}
+
+int
+acpi_enter_sleep_state(struct acpi_softc *sc, int state)
+{
+ uint16_t rega, regb;
+ int retries;
/* Clear WAK_STS bit */
- acpi_write_pmreg(sc, ACPIREG_PM1_STS, 0, ACPI_PM1_WAK_STS);
+ acpi_write_pmreg(sc, ACPIREG_PM1_STS, 1, ACPI_PM1_WAK_STS);
+
+ /* Disable BM arbitration */
+ acpi_write_pmreg(sc, ACPIREG_PM2_CNT, 1, ACPI_PM2_ARB_DIS);
/* Write SLP_TYPx values */
rega = acpi_read_pmreg(sc, ACPIREG_PM1A_CNT, 0);
@@ -1724,9 +1749,15 @@ acpi_enter_sleep_state(struct acpi_softc *sc, int state)
/* Set SLP_EN bit */
rega |= ACPI_PM1_SLP_EN;
regb |= ACPI_PM1_SLP_EN;
+
+ /*
+ * Let the machdep code flush caches and do any other necessary
+ * tasks before going away.
+ */
+ acpi_cpu_flush(sc, state);
+
acpi_write_pmreg(sc, ACPIREG_PM1A_CNT, 0, rega);
acpi_write_pmreg(sc, ACPIREG_PM1B_CNT, 0, regb);
-
/* Loop on WAK_STS */
for (retries = 1000; retries > 0; retries--) {
rega = acpi_read_pmreg(sc, ACPIREG_PM1A_STS, 0);
@@ -1737,10 +1768,10 @@ acpi_enter_sleep_state(struct acpi_softc *sc, int state)
DELAY(10);
}
- enable_intr();
+ return (-1);
}
-#if 0
+#ifndef SMALL_KERNEL
void
acpi_resume(struct acpi_softc *sc)
{
@@ -1750,20 +1781,21 @@ acpi_resume(struct acpi_softc *sc)
env.type = AML_OBJTYPE_INTEGER;
env.v_integer = sc->sc_state;
- if (sc->sc_bfs) {
+ if (sc->sc_bfs)
if (aml_evalnode(sc, sc->sc_pts, 1, &env, NULL) != 0) {
dnprintf(10, "%s evaluating method _BFS failed.\n",
DEVNAME(sc));
}
- }
+
dopowerhooks(PWR_RESUME);
inittodr(0);
- if (sc->sc_wak) {
+
+ if (sc->sc_wak)
if (aml_evalnode(sc, sc->sc_wak, 1, &env, NULL) != 0) {
dnprintf(10, "%s evaluating method _WAK failed.\n",
DEVNAME(sc));
}
- }
+
sc->sc_state = ACPI_STATE_S0;
if (sc->sc_tts) {
env.v_integer = sc->sc_state;
@@ -1773,14 +1805,84 @@ acpi_resume(struct acpi_softc *sc)
}
}
}
-#endif
+#endif /* ! SMALL_KERNEL */
+
+int
+acpi_prepare_sleep_state(struct acpi_softc *sc, int state)
+{
+ struct aml_value env;
+
+ if (sc == NULL || state == ACPI_STATE_S0)
+ return(0);
+
+ if (sc->sc_sleeptype[state].slp_typa == -1 ||
+ sc->sc_sleeptype[state].slp_typb == -1) {
+ printf("%s: state S%d unavailable\n",
+ sc->sc_dev.dv_xname, state);
+ return (ENXIO);
+ }
+
+ memset(&env, 0, sizeof(env));
+ env.type = AML_OBJTYPE_INTEGER;
+ env.v_integer = state;
+ /* _TTS(state) */
+ if (sc->sc_tts)
+ if (aml_evalnode(sc, sc->sc_tts, 1, &env, NULL) != 0) {
+ dnprintf(10, "%s evaluating method _TTS failed.\n",
+ DEVNAME(sc));
+ return (ENXIO);
+ }
+
+ switch (state) {
+ case ACPI_STATE_S1:
+ case ACPI_STATE_S2:
+ resettodr();
+ dopowerhooks(PWR_SUSPEND);
+ break;
+ case ACPI_STATE_S3:
+ resettodr();
+ dopowerhooks(PWR_STANDBY);
+ break;
+ }
+
+ /* _PTS(state) */
+ if (sc->sc_pts)
+ if (aml_evalnode(sc, sc->sc_pts, 1, &env, NULL) != 0) {
+ dnprintf(10, "%s evaluating method _PTS failed.\n",
+ DEVNAME(sc));
+ return (ENXIO);
+ }
+
+ sc->sc_state = state;
+ /* _GTS(state) */
+ if (sc->sc_gts)
+ if (aml_evalnode(sc, sc->sc_gts, 1, &env, NULL) != 0) {
+ dnprintf(10, "%s evaluating method _GTS failed.\n",
+ DEVNAME(sc));
+ return (ENXIO);
+ }
+
+ disable_intr();
+ aml_evalname(sc, &aml_root, "\\_SST", 1, &env, NULL);
+ sc->sc_state = state;
+
+ return (0);
+}
+
+
void
acpi_powerdown(void)
{
- acpi_enter_sleep_state(acpi_softc, ACPI_STATE_S5);
+ /*
+ * In case acpi_prepare_sleep fails, we shouldn't try to enter
+ * the sleep state. It might cost us the battery.
+ */
+ if (acpi_prepare_sleep_state(acpi_softc, ACPI_STATE_S5) == 0)
+ acpi_enter_sleep_state(acpi_softc, ACPI_STATE_S5);
}
+
extern int aml_busy;
void