/* $OpenBSD: acpi_x86.c,v 1.31 2024/08/31 15:53:44 deraadt Exp $ */ /* * Copyright (c) 2005 Thorsten Lockert * Copyright (c) 2005 Jordan Hargrave * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #include int sleep_showstate(void *v, int sleepmode) { struct acpi_softc *sc = v; int fallback_state = -1; switch (sleepmode) { case SLEEP_SUSPEND: sc->sc_state = ACPI_STATE_S3; #ifdef __amd64__ if (sc->sc_fadt->flags & FADT_POWER_S0_IDLE_CAPABLE) sc->sc_state = ACPI_STATE_S0; fallback_state = ACPI_STATE_S0; /* No S3, use S0 */ #endif break; case SLEEP_HIBERNATE: sc->sc_state = ACPI_STATE_S4; fallback_state = ACPI_STATE_S5; /* No S4, use S5 */ break; default: return (EOPNOTSUPP); } if (sc->sc_sleeptype[sc->sc_state].slp_typa == -1 || sc->sc_sleeptype[sc->sc_state].slp_typb == -1) { if (fallback_state != -1) { printf("%s: S%d unavailable, using S%d\n", sc->sc_dev.dv_xname, sc->sc_state, fallback_state); sc->sc_state = fallback_state; } else { printf("%s: state S%d unavailable\n", sc->sc_dev.dv_xname, sc->sc_state); return (EOPNOTSUPP); } } /* 1st suspend AML step: _TTS(tostate) */ if (sc->sc_state != ACPI_STATE_S0) { if (aml_node_setval(sc, sc->sc_tts, sc->sc_state) != 0) return (EINVAL); } acpi_indicator(sc, ACPI_SST_WAKING); /* blink */ return 0; } int sleep_setstate(void *v) { struct acpi_softc *sc = v; /* 2nd suspend AML step: _PTS(tostate) */ if (sc->sc_state != ACPI_STATE_S0) { if (aml_node_setval(sc, sc->sc_pts, sc->sc_state) != 0) return (EINVAL); } return 0; } int gosleep(void *v) { extern int cpu_wakeups; struct acpi_softc *sc = v; int ret; acpibtn_enable_psw(); /* enable _LID for wakeup */ acpi_indicator(sc, ACPI_SST_SLEEPING); /* 3rd suspend AML step: _GTS(tostate) */ if (sc->sc_state != ACPI_STATE_S0) aml_node_setval(sc, sc->sc_gts, sc->sc_state); /* Clear fixed event status */ acpi_write_pmreg(sc, ACPIREG_PM1_STS, 0, ACPI_PM1_ALL_STS); /* Enable wake GPEs */ acpi_disable_allgpes(sc); acpi_enable_wakegpes(sc, sc->sc_state); if (sc->sc_pmc_suspend) sc->sc_pmc_suspend(sc->sc_pmc_cookie); cpu_wakeups = 0; sc->sc_wakeup = 0; sc->sc_wakeups = 0; while (!sc->sc_wakeup) { ret = acpi_sleep_cpu(sc, sc->sc_state); acpi_resume_cpu(sc, sc->sc_state); sc->sc_wakeups++; if (sc->sc_ec && sc->sc_wakegpe == sc->sc_ec->sc_gpe) { sc->sc_wakeup = 0; acpiec_gpehandler(sc, sc->sc_wakegpe, sc->sc_ec); } else sc->sc_wakeup = 1; } if (sc->sc_pmc_resume) sc->sc_pmc_resume(sc->sc_pmc_cookie); acpi_indicator(sc, ACPI_SST_WAKING); /* blink */ /* 1st resume AML step: _WAK(fromstate) */ if (sc->sc_state != ACPI_STATE_S0) aml_node_setval(sc, sc->sc_wak, sc->sc_state); return ret; } int sleep_resume(void *v) { struct acpi_softc *sc = v; sc->sc_resume_time = getuptime(); acpibtn_disable_psw(); /* disable _LID for wakeup */ /* 3rd resume AML step: _TTS(runstate) */ if (sc->sc_state != ACPI_STATE_S0) { if (aml_node_setval(sc, sc->sc_tts, ACPI_STATE_S0) != 0) return (EINVAL); } return 0; } static int checklids(struct acpi_softc *sc) { extern int lid_action; int lids; lids = acpibtn_numopenlids(); if (lids == 0 && lid_action != 0) return EAGAIN; return 0; } int suspend_finish(void *v) { extern int cpu_wakeups; struct acpi_softc *sc = v; printf("wakeups: %d %d\n", cpu_wakeups, sc->sc_wakeups); printf("wakeup event: "); switch (sc->sc_wakegpe) { case 0: printf("unknown\n"); break; case -1: printf("PWRBTN\n"); break; case -2: printf("SLPTN\n"); break; default: printf("GPE 0x%x\n", sc->sc_wakegpe); break; } sc->sc_wakegpe = 0; acpi_record_event(sc, APM_NORMAL_RESUME); acpi_indicator(sc, ACPI_SST_WORKING); sc->sc_state = ACPI_STATE_S0; /* If we woke up but all the lids are closed, go back to sleep */ return checklids(sc); }