From cb0e410f0df42e8e7af29dea7fd9e25292d14f9f Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Mon, 17 Jul 2023 21:18:09 +0000 Subject: [PATCH] WIP: acpiec(4): debug output --- sys/dev/acpi/acpi_ec.c | 168 +++++++++++++++++++++++++++++++++++++---- 1 file changed, 154 insertions(+), 14 deletions(-) diff --git a/sys/dev/acpi/acpi_ec.c b/sys/dev/acpi/acpi_ec.c index 16f852a53028..900b72fbdc26 100644 --- a/sys/dev/acpi/acpi_ec.c +++ b/sys/dev/acpi/acpi_ec.c @@ -58,6 +58,8 @@ * and the only option to recover on those models is to cut off all power. */ +#define ACPIEC_DEBUG -1 + #include __KERNEL_RCSID(0, "$NetBSD: acpi_ec.c,v 1.86 2021/12/31 17:22:25 riastradh Exp $"); @@ -101,24 +103,42 @@ ACPI_MODULE_NAME ("acpi_ec") #define EC_STATUS_SCI 0x20 #define EC_STATUS_SMI 0x40 +#define EC_STATUS_FMT \ + "\x10\7IGN7\6SMI\5SCI\4BURST\3CMD\2IGN2\1IBF\0OBF" + static const struct device_compatible_entry compat_data[] = { { .compat = "PNP0C09" }, DEVICE_COMPAT_EOL }; +#define EC_STATE_ENUM(F) \ + F(EC_STATE_QUERY, "query") \ + F(EC_STATE_QUERY_VAL, "query_val") \ + F(EC_STATE_READ, "read") \ + F(EC_STATE_READ_ADDR, "read_addr") \ + F(EC_STATE_READ_VAL, "read_val") \ + F(EC_STATE_WRITE, "write") \ + F(EC_STATE_WRITE_ADDR, "write_addr") \ + F(EC_STATE_WRITE_VAL, "write_val") \ + F(EC_STATE_FREE, "free") \ + enum ec_state_t { - EC_STATE_QUERY, - EC_STATE_QUERY_VAL, - EC_STATE_READ, - EC_STATE_READ_ADDR, - EC_STATE_READ_VAL, - EC_STATE_WRITE, - EC_STATE_WRITE_ADDR, - EC_STATE_WRITE_VAL, - EC_STATE_FREE +#define F(N, S) N, + EC_STATE_ENUM(F) +#undef F }; +#ifdef ACPIEC_DEBUG +static const char *const acpiec_state_names[] = { +#define F(N, S) [N] = S, + EC_STATE_ENUM(F) +#undef F +}; +#endif + struct acpiec_softc { + device_t sc_dev; + ACPI_HANDLE sc_ech; ACPI_HANDLE sc_gpeh; @@ -142,6 +162,32 @@ struct acpiec_softc { uint8_t sc_cur_addr, sc_cur_val; }; +#ifdef ACPIEC_DEBUG + +enum { + ACPIEC_DEBUG_REG, + ACPIEC_DEBUG_RW, + ACPIEC_DEBUG_QUERY, + ACPIEC_DEBUG_TRANSITION, + ACPIEC_DEBUG_INTR, +}; + +int acpiec_debug = ACPIEC_DEBUG; + +#define DPRINTF(n, sc, fmt, ...) do \ +{ \ + if (acpiec_debug & __BIT(n)) { \ + device_printf((sc)->sc_dev, "[%d %s] "fmt, \ + (int)(sc)->sc_state, \ + ((sc)->sc_state < __arraycount(acpiec_state_names) \ + ? acpiec_state_names[(sc)->sc_state] \ + : "(unknown)"), \ + ##__VA_ARGS__); \ + } \ +} while (0) + +#endif + static int acpiecdt_match(device_t, cfdata_t, void *); static void acpiecdt_attach(device_t, device_t, void *); @@ -313,6 +359,8 @@ acpiec_common_attach(device_t parent, device_t self, ACPI_STATUS rv; ACPI_INTEGER val; + sc->sc_dev = self; + sc->sc_csr_st = cmdt; sc->sc_data_st = datat; @@ -491,24 +539,46 @@ acpiec_parse_gpe_package(device_t self, ACPI_HANDLE ec_handle, static uint8_t acpiec_read_data(struct acpiec_softc *sc) { - return bus_space_read_1(sc->sc_data_st, sc->sc_data_sh, 0); + uint8_t x; + + KASSERT(mutex_owned(&sc->sc_mtx)); + + x = bus_space_read_1(sc->sc_data_st, sc->sc_data_sh, 0); + DPRINTF(ACPIEC_DEBUG_REG, sc, "read data=0x%"PRIx8"\n", x); + + return x; } static void acpiec_write_data(struct acpiec_softc *sc, uint8_t val) { + + KASSERT(mutex_owned(&sc->sc_mtx)); + + DPRINTF(ACPIEC_DEBUG_REG, sc, "write data=0x%"PRIx8"\n", val); bus_space_write_1(sc->sc_data_st, sc->sc_data_sh, 0, val); } static uint8_t acpiec_read_status(struct acpiec_softc *sc) { - return bus_space_read_1(sc->sc_csr_st, sc->sc_csr_sh, 0); + uint8_t x; + + KASSERT(mutex_owned(&sc->sc_mtx)); + + x = bus_space_read_1(sc->sc_csr_st, sc->sc_csr_sh, 0); + DPRINTF(ACPIEC_DEBUG_REG, sc, "read status=0x%"PRIx8"\n", x); + + return x; } static void acpiec_write_command(struct acpiec_softc *sc, uint8_t cmd) { + + KASSERT(mutex_owned(&sc->sc_mtx)); + + DPRINTF(ACPIEC_DEBUG_REG, sc, "write command=0x%"PRIx8"\n", cmd); bus_space_write_1(sc->sc_csr_st, sc->sc_csr_sh, 0, cmd); } @@ -571,6 +641,13 @@ acpiec_read(device_t dv, uint8_t addr, uint8_t *val) acpiec_lock(dv); mutex_enter(&sc->sc_mtx); + DPRINTF(ACPIEC_DEBUG_RW, sc, + "[pid %ld %s] [lid %ld%s%s] read addr 0x%"PRIx8"\n", + (long)curproc->p_pid, curproc->p_comm, + (long)curlwp->l_lid, curlwp->l_name ? " " : "", + curlwp->l_name ? curlwp->l_name : "", + addr); + sc->sc_cur_addr = addr; sc->sc_state = EC_STATE_READ; @@ -594,6 +671,11 @@ acpiec_read(device_t dv, uint8_t addr, uint8_t *val) return AE_ERROR; } } else if (cv_timedwait(&sc->sc_cv, &sc->sc_mtx, EC_CMD_TIMEOUT * hz)) { + /* + * XXX while (sc->sc_state != EC_STATE_FREE) + * cv_timedwait(...), + * plus deadline + */ mutex_exit(&sc->sc_mtx); acpiec_unlock(dv); aprint_error_dev(dv, @@ -602,6 +684,13 @@ acpiec_read(device_t dv, uint8_t addr, uint8_t *val) } done: + DPRINTF(ACPIEC_DEBUG_RW, sc, + "[pid %ld %s] [lid %ld%s%s] read addr 0x%"PRIx8": 0x%"PRIx8"\n", + (long)curproc->p_pid, curproc->p_comm, + (long)curlwp->l_lid, curlwp->l_name ? " " : "", + curlwp->l_name ? curlwp->l_name : "", + addr, sc->sc_cur_val); + *val = sc->sc_cur_val; mutex_exit(&sc->sc_mtx); @@ -618,6 +707,13 @@ acpiec_write(device_t dv, uint8_t addr, uint8_t val) acpiec_lock(dv); mutex_enter(&sc->sc_mtx); + DPRINTF(ACPIEC_DEBUG_RW, sc, + "[pid %ld %s] [lid %ld%s%s] write addr 0x%"PRIx8": 0x%"PRIx8"\n", + (long)curproc->p_pid, curproc->p_comm, + (long)curlwp->l_lid, curlwp->l_name ? " " : "", + curlwp->l_name ? curlwp->l_name : "", + addr, val); + sc->sc_cur_addr = addr; sc->sc_cur_val = val; sc->sc_state = EC_STATE_WRITE; @@ -642,6 +738,11 @@ acpiec_write(device_t dv, uint8_t addr, uint8_t val) return AE_ERROR; } } else if (cv_timedwait(&sc->sc_cv, &sc->sc_mtx, EC_CMD_TIMEOUT * hz)) { + /* + * XXX while (sc->sc_state != EC_STATE_FREE) + * cv_timedwait(...), + * plus deadline + */ mutex_exit(&sc->sc_mtx); acpiec_unlock(dv); aprint_error_dev(dv, @@ -650,6 +751,14 @@ acpiec_write(device_t dv, uint8_t addr, uint8_t val) } done: + DPRINTF(ACPIEC_DEBUG_RW, sc, + "[pid %ld %s] [lid %ld%s%s] write addr 0x%"PRIx8": 0x%"PRIx8 + " done\n", + (long)curproc->p_pid, curproc->p_comm, + (long)curlwp->l_lid, curlwp->l_name ? " " : "", + curlwp->l_name ? curlwp->l_name : "", + addr, val); + mutex_exit(&sc->sc_mtx); acpiec_unlock(dv); return AE_OK; @@ -747,14 +856,25 @@ acpiec_gpe_query(void *arg) int i; loop: + /* + * Wait until the EC sends an SCI requesting a query. + */ mutex_enter(&sc->sc_mtx); - - if (sc->sc_got_sci == false) + while (!sc->sc_got_sci) cv_wait(&sc->sc_cv_sci, &sc->sc_mtx); + DPRINTF(ACPIEC_DEBUG_QUERY, sc, "SCI query requested\n"); mutex_exit(&sc->sc_mtx); + /* + * EC is about to submit a query to us. Claim exclusive access + * to the interface. + */ acpiec_lock(dv); + mutex_enter(&sc->sc_mtx); + KASSERT(sc->sc_state == EC_STATE_FREE); + + DPRINTF(ACPIEC_DEBUG_QUERY, sc, "handling SCI query\n"); /* The Query command can always be issued, so be defensive here. */ sc->sc_got_sci = false; @@ -767,10 +887,13 @@ loop: delay(1); } + DPRINTF(ACPIEC_DEBUG_QUERY, sc, "SCI query timeout\n"); + /* XXX while (sc->sc_state != EC_STATE_FREE) cv_wait(...) */ cv_wait(&sc->sc_cv, &sc->sc_mtx); done: reg = sc->sc_cur_val; + DPRINTF(ACPIEC_DEBUG_QUERY, sc, "query returned 0x%"PRIx8"\n", reg); mutex_exit(&sc->sc_mtx); acpiec_unlock(dv); @@ -797,8 +920,19 @@ acpiec_gpe_state_machine(device_t dv) struct acpiec_softc *sc = device_private(dv); uint8_t reg; + KASSERT(mutex_owned(&sc->sc_mtx)); + reg = acpiec_read_status(sc); +#ifdef ACPIEC_DEBUG + if (acpiec_debug & __BIT(ACPIEC_DEBUG_TRANSITION)) { + char buf[128]; + + snprintb(buf, sizeof(buf), EC_STATUS_FMT, reg); + DPRINTF(ACPIEC_DEBUG_TRANSITION, sc, "status=%s\n", buf); + } +#endif + if (reg & EC_STATUS_SCI) sc->sc_got_sci = true; @@ -877,8 +1011,12 @@ acpiec_gpe_state_machine(device_t dv) panic("invalid state"); } - if (sc->sc_state != EC_STATE_FREE) + DPRINTF(ACPIEC_DEBUG_TRANSITION, sc, "transitioned\n"); + + if (sc->sc_state != EC_STATE_FREE) { + DPRINTF(ACPIEC_DEBUG_INTR, sc, "schedule callout\n"); callout_schedule(&sc->sc_pseudo_intr, 1); + } } static void @@ -888,6 +1026,7 @@ acpiec_callout(void *arg) struct acpiec_softc *sc = device_private(dv); mutex_enter(&sc->sc_mtx); + DPRINTF(ACPIEC_DEBUG_INTR, sc, "callout\n"); acpiec_gpe_state_machine(dv); mutex_exit(&sc->sc_mtx); } @@ -899,6 +1038,7 @@ acpiec_gpe_handler(ACPI_HANDLE hdl, uint32_t gpebit, void *arg) struct acpiec_softc *sc = device_private(dv); mutex_enter(&sc->sc_mtx); + DPRINTF(ACPIEC_DEBUG_INTR, sc, "GPE\n"); acpiec_gpe_state_machine(dv); mutex_exit(&sc->sc_mtx);