diff options
Diffstat (limited to 'sys')
-rw-r--r-- | sys/dev/ipmi.c | 51 | ||||
-rw-r--r-- | sys/dev/ipmivar.h | 4 |
2 files changed, 50 insertions, 5 deletions
diff --git a/sys/dev/ipmi.c b/sys/dev/ipmi.c index 3fc7759410c..50833d1e872 100644 --- a/sys/dev/ipmi.c +++ b/sys/dev/ipmi.c @@ -1,4 +1,4 @@ -/* $OpenBSD: ipmi.c,v 1.89 2016/01/12 09:11:59 uebayasi Exp $ */ +/* $OpenBSD: ipmi.c,v 1.90 2016/01/12 10:44:32 uebayasi Exp $ */ /* * Copyright (c) 2005 Jordan Hargrave @@ -153,7 +153,10 @@ int get_sdr(struct ipmi_softc *, u_int16_t, u_int16_t *); int ipmi_sendcmd(struct ipmi_cmd *); int ipmi_recvcmd(struct ipmi_cmd *); void ipmi_delay(struct ipmi_softc *, int); -void ipmi_cmd(void *); +void ipmi_cmd(struct ipmi_cmd *); +void ipmi_cmd_poll(struct ipmi_cmd *); +void ipmi_cmd_wait(struct ipmi_cmd *); +void ipmi_cmd_wait_cb(void *); int ipmi_watchdog(void *, int); void ipmi_watchdog_tickle(void *); @@ -1069,9 +1072,18 @@ ipmi_delay(struct ipmi_softc *sc, int period) } void -ipmi_cmd(void *arg) +ipmi_cmd(struct ipmi_cmd *c) { - struct ipmi_cmd *c = arg; + if (cold || panicstr != NULL) + ipmi_cmd_poll(c); + else + ipmi_cmd_wait(c); +} + +void +ipmi_cmd_poll(struct ipmi_cmd *c) +{ + mtx_enter(&c->c_sc->sc_cmd_mtx); c->c_sc->sc_cmd = c; if (ipmi_sendcmd(c)) { @@ -1079,6 +1091,33 @@ ipmi_cmd(void *arg) } c->c_ccode = ipmi_recvcmd(c); c->c_sc->sc_cmd = NULL; + + mtx_leave(&c->c_sc->sc_cmd_mtx); +} + +void +ipmi_cmd_wait(struct ipmi_cmd *c) +{ + struct task t; + int res; + + task_set(&t, ipmi_cmd_wait_cb, c); + res = task_add(c->c_sc->sc_cmd_taskq, &t); + KASSERT(res == 1); + + tsleep(c, PWAIT, "ipmicmd", 0); + + res = task_del(c->c_sc->sc_cmd_taskq, &t); + KASSERT(res == 0); +} + +void +ipmi_cmd_wait_cb(void *arg) +{ + struct ipmi_cmd *c = arg; + + ipmi_cmd_poll(c); + wakeup(c); } /* Read a partial SDR entry */ @@ -1665,6 +1704,7 @@ ipmi_match(struct device *parent, void *match, void *aux) return (0); /* XXX local softc is wrong wrong wrong */ + mtx_init(&sc.sc_cmd_mtx, IPL_NONE); strlcpy(sc.sc_dev.dv_xname, "ipmi0", sizeof(sc.sc_dev.dv_xname)); /* Map registers */ @@ -1736,6 +1776,9 @@ ipmi_attach(struct device *parent, struct device *self, void *aux) sc->sc_wakeup = 0; sc->sc_max_retries = 50; /* 50 * 1/100 = 0.5 seconds max */ timeout_set(&sc->sc_timeout, _bmc_io_wait, sc); + + sc->sc_cmd_taskq = taskq_create("ipmicmd", 1, IPL_NONE, TASKQ_MPSAFE); + mtx_init(&sc->sc_cmd_mtx, IPL_NONE); } int diff --git a/sys/dev/ipmivar.h b/sys/dev/ipmivar.h index 04b7126fd8f..0e2bc938129 100644 --- a/sys/dev/ipmivar.h +++ b/sys/dev/ipmivar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: ipmivar.h,v 1.26 2016/01/12 09:11:59 uebayasi Exp $ */ +/* $OpenBSD: ipmivar.h,v 1.27 2016/01/12 10:44:32 uebayasi Exp $ */ /* * Copyright (c) 2005 Jordan Hargrave @@ -112,6 +112,8 @@ struct ipmi_softc { int sc_btseq; u_int8_t sc_buf[IPMI_MAX_RX + 16]; struct ipmi_cmd *sc_cmd; + struct taskq *sc_cmd_taskq; + struct mutex sc_cmd_mtx; int sc_wdog_period; struct task sc_wdog_tickle_task; |