summaryrefslogtreecommitdiff
path: root/sys/arch/i386
diff options
context:
space:
mode:
authorJordan Hargrave <jordan@cvs.openbsd.org>2005-11-21 17:23:20 +0000
committerJordan Hargrave <jordan@cvs.openbsd.org>2005-11-21 17:23:20 +0000
commit6716439710b044e2cec4df518884e986deb002e2 (patch)
treef4a3c4267c6ecb167e83c00f525af97fe59ccd92 /sys/arch/i386
parent24ae1eb837a8bb677dd453e04d69f366f2a990f3 (diff)
Add support for ESM (Embedded Server Management) for Dell PowerEdge servers.
Lot's of work by dlg@ ok marco@
Diffstat (limited to 'sys/arch/i386')
-rw-r--r--sys/arch/i386/i386/esm.c563
-rw-r--r--sys/arch/i386/i386/esmreg.h210
-rw-r--r--sys/arch/i386/i386/esmvar.h24
3 files changed, 797 insertions, 0 deletions
diff --git a/sys/arch/i386/i386/esm.c b/sys/arch/i386/i386/esm.c
new file mode 100644
index 00000000000..4b46260f3ad
--- /dev/null
+++ b/sys/arch/i386/i386/esm.c
@@ -0,0 +1,563 @@
+/* $OpenBSD: esm.c,v 1.1 2005/11/21 17:23:19 jordan Exp $ */
+
+/*
+ * Copyright (c) 2005 David Gwynne <dlg@openbsd.org>
+ *
+ * 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 <sys/param.h>
+#include <sys/systm.h>
+#include <sys/device.h>
+#include <sys/kernel.h>
+#include <sys/malloc.h>
+#include <sys/timeout.h>
+#include <sys/queue.h>
+#include <sys/sensors.h>
+
+#include <machine/bus.h>
+
+#include <arch/i386/i386/esmreg.h>
+#include <arch/i386/i386/esmvar.h>
+
+#define ESM_DEBUG
+
+#ifdef ESM_DEBUG
+#define DPRINTF(x...) do { if (esmdebug) printf(x); } while (0)
+#define DPRINTFN(n,x...) do { if (esmdebug > (n)) printf(x); } while (0)
+int esmdebug = 2;
+#else
+#define DPRINTF(x...) /* x */
+#define DPRINTFN(n,x...) /* n: x */
+#endif
+
+int esm_match(struct device *, void *, void *);
+void esm_attach(struct device *, struct device *, void *);
+
+struct esm_sensor_type {
+ enum sensor_type type;
+ long arg;
+#define ESM_A_PWRSUP_1 0x10
+#define ESM_A_PWRSUP_2 0x20
+#define ESM_A_PWRSUP_3 0x40
+#define ESM_A_PWRSUP_4 0x80
+ const char *name;
+};
+
+struct esm_sensor {
+ u_int8_t es_dev;
+ u_int8_t es_id;
+
+ long es_arg;
+
+ struct sensor es_sensor;
+ TAILQ_ENTRY(esm_sensor) es_entry;
+};
+
+struct esm_softc {
+ struct device sc_dev;
+ bus_space_tag_t sc_iot;
+ bus_space_handle_t sc_ioh;
+
+ TAILQ_HEAD(, esm_sensor) sc_sensors;
+ struct timeout sc_timeout;
+};
+
+struct cfattach esm_ca = {
+ sizeof(struct esm_softc), esm_match, esm_attach
+};
+
+struct cfdriver esm_cd = {
+ NULL, "esm", DV_DULL
+};
+
+#define DEVNAME(s) ((s)->sc_dev.dv_xname)
+
+#define EREAD(s, r) bus_space_read_1((s)->sc_iot, (s)->sc_ioh, (r))
+#define EWRITE(s, r, v) bus_space_write_1((s)->sc_iot, (s)->sc_ioh, (r), (v))
+
+#define ECTRLWR(s, v) EWRITE((s), ESM2_TC_REG, (v))
+#define EDATARD(s) EREAD((s), ESM2_TBUF_REG)
+#define EDATAWR(s, v) EWRITE((s), ESM2_TBUF_REG, (v))
+
+void esm_refresh(void *);
+
+int esm_get_devmap(struct esm_softc *, int, struct esm_devmap *);
+void esm_devmap(struct esm_softc *, struct esm_devmap *);
+void esm_make_sensors(struct esm_softc *, struct esm_devmap *,
+ struct esm_sensor_type *, int);
+
+int esm_bmc_ready(struct esm_softc *, int, u_int8_t, u_int8_t);
+int esm_bmc_wait(struct esm_softc *, int, u_int8_t, u_int8_t);
+int esm_cmd(struct esm_softc *, void *, size_t, void *, size_t);
+int esm_smb_cmd(struct esm_softc *, struct esm_smb_req *,
+ struct esm_smb_resp *);
+
+int64_t esm_val2temp(u_int16_t);
+
+int
+esm_match(struct device *parent, void *match, void *aux)
+{
+ struct esm_attach_args *eaa = aux;
+
+ if (strncmp(eaa->eaa_name, esm_cd.cd_name,
+ sizeof(esm_cd.cd_name)) == 0)
+ return (1);
+
+ return (0);
+}
+
+void
+esm_attach(struct device *parent, struct device *self, void *aux)
+{
+ struct esm_softc *sc = (struct esm_softc *)self;
+ struct esm_attach_args *eaa = aux;
+ u_int8_t x;
+
+ struct esm_devmap devmap;
+ int i;
+
+ sc->sc_iot = eaa->eaa_iot;
+ TAILQ_INIT(&sc->sc_sensors);
+
+ if (bus_space_map(sc->sc_iot, ESM2_BASE_PORT, 8, 0, &sc->sc_ioh) != 0) {
+ printf(": unable to map memory\n");
+ return;
+ }
+
+ /* turn off interrupts here */
+ x = EREAD(sc, ESM2_TIM_REG);
+ x &= ~(ESM2_TIM_SCI_EN|ESM2_TIM_SMI_EN|ESM2_TIM_NMI2SMI);
+ x |= ESM2_TIM_POWER_UP_BITS;
+ EWRITE(sc, ESM2_TIM_REG, x);
+
+ /* clear event doorbells */
+ x = EREAD(sc, ESM2_TC_REG);
+ x &= ~ESM2_TC_HOSTBUSY;
+ x |= ESM2_TC_POWER_UP_BITS;
+ EWRITE(sc, ESM2_TC_REG, x);
+
+ /* see if card is alive */
+ if (esm_bmc_wait(sc, ESM2_TC_REG, ESM2_TC_ECBUSY, 0) != 0) {
+ printf("%s: card is not alive\n");
+ bus_space_unmap(sc->sc_iot, sc->sc_ioh, 8);
+ return;
+ }
+
+ printf(": Watchdog shizz goes here\n");
+
+ for (i = 0; i <= 0xff; i++) {
+ if (esm_get_devmap(sc, i, &devmap) != 0)
+ break; /* XXX not continue? */
+ esm_devmap(sc, &devmap);
+ }
+
+ if (!TAILQ_EMPTY(&sc->sc_sensors)) {
+ DPRINTF("%s: starting refresh\n", DEVNAME(sc));
+ timeout_set(&sc->sc_timeout, esm_refresh, sc);
+ timeout_add(&sc->sc_timeout, hz * 10);
+ }
+}
+
+void
+esm_refresh(void *arg)
+{
+ struct esm_softc *sc = arg;
+ struct esm_sensor *sensor;
+ struct esm_smb_req req;
+ struct esm_smb_resp resp;
+ struct esm_smb_resp_val *val = &resp.resp_val;
+
+ memset(&req, 0, sizeof(req));
+ req.h_cmd = ESM2_CMD_SMB_XMIT_RECV;
+ req.h_txlen = sizeof(req.req_val);
+ req.h_rxlen = sizeof(resp.resp_val);
+ req.req_val.v_cmd = ESM2_SMB_SENSOR_VALUE;
+
+ TAILQ_FOREACH(sensor, &sc->sc_sensors, es_entry) {
+ req.h_dev = sensor->es_dev;
+ req.req_val.v_sensor = sensor->es_id;
+
+ if (esm_smb_cmd(sc, &req, &resp) != 0) {
+ sensor->es_sensor.flags |= SENSOR_FINVALID;
+ continue;
+ }
+
+ switch (sensor->es_sensor.type) {
+ case SENSOR_TEMP:
+ sensor->es_sensor.value = esm_val2temp(val->v_reading);
+ break;
+ default:
+ sensor->es_sensor.value = val->v_reading;
+ break;
+ }
+ }
+
+ timeout_add(&sc->sc_timeout, hz * 10);
+}
+
+int
+esm_get_devmap(struct esm_softc *sc, int dev, struct esm_devmap *devmap)
+{
+ struct esm_devmap_req req;
+ struct esm_devmap_resp resp;
+#ifdef ESM_DEBUG
+ int i;
+#endif
+
+ memset(&req, 0, sizeof(req));
+ memset(&resp, 0, sizeof(resp));
+
+ req.cmd = ESM2_CMD_DEVICEMAP;
+ req.action = ESM2_DEVICEMAP_READ;
+ req.index = dev;
+ req.ndev = 1;
+
+ if (esm_cmd(sc, &req, sizeof(req), &resp, sizeof(resp)) != 0)
+ return (1);
+
+ if (resp.status != 0)
+ return (1);
+
+ memcpy(devmap, &resp.devmap[0], sizeof(struct esm_devmap));
+
+#ifdef ESM_DEBUG
+ if (esmdebug > 5) {
+ printf("\n");
+ printf("Device Map(%d) returns:\n", dev);
+ printf(" status: %.2x\n", resp.status);
+ printf(" #devs : %.2x\n", resp.ndev);
+ printf(" index: %.2x\n", resp.devmap[0].index);
+ printf(" Type : %.2x.%.2x\n", resp.devmap[0].dev_major,
+ resp.devmap[0].dev_minor);
+ printf(" Rev : %.2x.%.2x\n", resp.devmap[0].rev_major,
+ resp.devmap[0].rev_minor);
+ printf(" ROM : %.2x\n", resp.devmap[0].rev_rom);
+ printf(" SMB : %.2x\n", resp.devmap[0].smb_addr);
+ printf(" Stat : %.2x\n", resp.devmap[0].status);
+ printf(" MonTy: %.2x\n", resp.devmap[0].monitor_type);
+ printf(" Poll : %.2x\n", resp.devmap[0].pollcycle);
+ printf(" UUID : ");
+ for (i = 0; i < ESM2_UUID_LEN; i++) {
+ printf("%02x", resp.devmap[0].uniqueid[i]);
+ }
+ printf("\n");
+ }
+#endif /* ESM_DEBUG */
+
+ return (0);
+}
+
+struct esm_sensor_type esm_sensors_esm2[] = {
+ { SENSOR_INTEGER, 0, "Motherboard" },
+ { SENSOR_TEMP, 0, "CPU 1" },
+ { SENSOR_TEMP, 0, "CPU 2" },
+ { SENSOR_TEMP, 0, "CPU 3" },
+ { SENSOR_TEMP, 0, "CPU 4" },
+
+ { SENSOR_TEMP, 0, "Mainboard" },
+ { SENSOR_TEMP, 0, "Ambient" },
+ { SENSOR_VOLTS_DC, 0, "CPU 1 Core" },
+ { SENSOR_VOLTS_DC, 0, "CPU 2 Core" },
+ { SENSOR_VOLTS_DC, 0, "CPU 3 Core" },
+
+ { SENSOR_VOLTS_DC, 0, "CPU 4 Core" },
+ { SENSOR_VOLTS_DC, 0, "Motherboard +5V" },
+ { SENSOR_VOLTS_DC, 0, "Motherboard +12V" },
+ { SENSOR_VOLTS_DC, 0, "Motherboard +3.3V" },
+ { SENSOR_VOLTS_DC, 0, "Motherboard +2.5V" },
+
+ { SENSOR_VOLTS_DC, 0, "Motherboard GTL Term" },
+ { SENSOR_VOLTS_DC, 0, "Motherboard Battery" },
+ { SENSOR_INDICATOR, 0, "Chassis Intrusion", },
+ { SENSOR_INTEGER, 0, "Chassis Fan Ctrl", },
+ { SENSOR_FANRPM, 0, "Fan 1" },
+
+ { SENSOR_FANRPM, 0, "Fan 2" }, /* 20 */
+ { SENSOR_FANRPM, 0, "Fan 3" },
+ { SENSOR_FANRPM, 0, "Power Supply Fan" },
+ { SENSOR_VOLTS_DC, 0, "CPU 1 cache" },
+ { SENSOR_VOLTS_DC, 0, "CPU 2 cache" },
+
+ { SENSOR_VOLTS_DC, 0, "CPU 3 cache" },
+ { SENSOR_VOLTS_DC, 0, "CPU 4 cache" },
+ { SENSOR_INTEGER, 0, "Power Ctrl" },
+ { SENSOR_INTEGER, ESM_A_PWRSUP_1, "Power Supply 1" },
+ { SENSOR_INTEGER, ESM_A_PWRSUP_2, "Power Supply 2" },
+
+ { SENSOR_VOLTS_DC, 0, "Mainboard +1.5V" }, /* 30 */
+ { SENSOR_VOLTS_DC, 0, "Motherboard +2.8V" },
+ { SENSOR_INTEGER, 0, "HotPlug Status" },
+ { SENSOR_INTEGER, 1, "PCI Slot 1" },
+ { SENSOR_INTEGER, 2, "PCI Slot 2" },
+
+ { SENSOR_INTEGER, 3, "PCI Slot 3" },
+ { SENSOR_INTEGER, 4, "PCI Slot 4" },
+ { SENSOR_INTEGER, 5, "PCI Slot 5" },
+ { SENSOR_INTEGER, 6, "PCI Slot 6" },
+ { SENSOR_INTEGER, 7, "PCI Slot 7" },
+
+ { SENSOR_VOLTS_DC, 0, "CPU 1 Cartridge" }, /* 40 */
+ { SENSOR_VOLTS_DC, 0, "CPU 2 Cartridge" },
+ { SENSOR_VOLTS_DC, 0, "CPU 3 Cartridge" },
+ { SENSOR_VOLTS_DC, 0, "CPU 4 Cartridge" },
+ { SENSOR_VOLTS_DC, 0, "Gigabit NIC +1.8V" },
+
+ { SENSOR_VOLTS_DC, 0, "Gigabit NIC +2.5V" },
+ { SENSOR_VOLTS_DC, 0, "Memory +3.3V" },
+ { SENSOR_VOLTS_DC, 0, "Video +2.5V" },
+ { SENSOR_INTEGER, ESM_A_PWRSUP_3, "Power Supply 3" },
+ { SENSOR_FANRPM, 0, "Fan 4" },
+
+ { SENSOR_FANRPM, 0, "Power Supply Fan" }, /* 50 */
+ { SENSOR_FANRPM, 0, "Power Supply Fan" },
+ { SENSOR_FANRPM, 0, "Power Supply Fan" },
+ { SENSOR_INTEGER, 0, "A/C Power Switch" },
+ { SENSOR_INTEGER, 0, "PS Over Temp" }
+};
+
+void
+esm_devmap(struct esm_softc *sc, struct esm_devmap *devmap)
+{
+ struct esm_sensor_type *sensor_types;
+ const char *maj_name, *min_name;
+ int nsensors;
+
+ switch (devmap->dev_major) {
+ case ESM2_DEV_ESM2:
+ maj_name = "Embedded Server Management";
+ sensor_types = esm_sensors_esm2;
+
+ switch (devmap->dev_minor) {
+ case ESM2_DEV_ESM2_2300:
+ min_name = "PowerEdge 2300";
+ nsensors = 23;
+ break;
+ case ESM2_DEV_ESM2_4300:
+ min_name = "PowerEdge 4300";
+ nsensors = 27;
+ break;
+ case ESM2_DEV_ESM2_6300:
+ min_name = "PowerEdge 6300";
+ nsensors = 27;
+ break;
+ case ESM2_DEV_ESM2_6400:
+ min_name = "PowerEdge 6400";
+ nsensors = 44;
+ break;
+ case ESM2_DEV_ESM2_2550:
+ min_name = "PowerEdge 2550";
+ nsensors = 48;
+ break;
+ case ESM2_DEV_ESM2_4350:
+ min_name = "PowerEdge 4350";
+ nsensors = 27;
+ break;
+ case ESM2_DEV_ESM2_6350:
+ min_name = "PowerEdge 6350";
+ nsensors = 27;
+ break;
+ case ESM2_DEV_ESM2_6450:
+ min_name = "PowerEdge 6450";
+ nsensors = 44;
+ break;
+ case ESM2_DEV_ESM2_2400:
+ min_name = "PowerEdge 2400";
+ nsensors = 30;
+ break;
+ case ESM2_DEV_ESM2_4400:
+ min_name = "PowerEdge 4400";
+ nsensors = 44;
+ break;
+ case ESM2_DEV_ESM2_2500:
+ min_name = "PowerEdge 2500";
+ nsensors = 55;
+ break;
+ case ESM2_DEV_ESM2_2450:
+ min_name = "PowerEdge 2450";
+ nsensors = 27;
+ break;
+ case ESM2_DEV_ESM2_2400EX:
+ min_name = "PowerEdge 2400";
+ nsensors = 27;
+ break;
+ case ESM2_DEV_ESM2_2450EX:
+ min_name = "PowerEdge 2450";
+ nsensors = 44;
+ break;
+
+ default:
+ return;
+ }
+
+ printf("%s: %s %s %d.%d\n", DEVNAME(sc), min_name, maj_name,
+ devmap->rev_major, devmap->rev_minor);
+ break;
+
+ default:
+ return;
+ }
+
+ esm_make_sensors(sc, devmap, sensor_types, nsensors);
+}
+
+void
+esm_make_sensors(struct esm_softc *sc, struct esm_devmap *devmap,
+ struct esm_sensor_type *sensor_types, int nsensors)
+{
+ struct esm_smb_req req;
+ struct esm_smb_resp resp;
+ struct esm_smb_resp_val *val = &resp.resp_val;
+ struct esm_sensor *sensor;
+ int i;
+
+ memset(&req, 0, sizeof(req));
+ req.h_cmd = ESM2_CMD_SMB_XMIT_RECV;
+ req.h_dev = devmap->index;
+ req.h_txlen = sizeof(req.req_val);
+ req.h_rxlen = sizeof(resp.resp_val);
+
+ req.req_val.v_cmd = ESM2_SMB_SENSOR_VALUE;
+
+ for (i = 0; i < nsensors; i++) {
+ req.req_val.v_sensor = i;
+ if (esm_smb_cmd(sc, &req, &resp) != 0)
+ continue;
+
+ DPRINTFN(1, "%s: dev: 0x%02x sensor: %d (%s) "
+ "reading: 0x%04x status: 0x%02x cksum: 0x%02x\n",
+ DEVNAME(sc), devmap->index, i, sensor_types[i].name,
+ val->v_reading, val->v_status, val->v_checksum);
+
+ if ((val->v_status & ESM2_VS_VALID) != ESM2_VS_VALID)
+ continue;
+
+ sensor = malloc(sizeof(struct esm_sensor), M_DEVBUF, M_NOWAIT);
+ if (sensor == NULL)
+ goto error;
+
+ memset(sensor, 0, sizeof(struct esm_sensor));
+ sensor->es_dev = devmap->index;
+ sensor->es_id = i;
+ sensor->es_arg = sensor_types[i].arg;
+ sensor->es_sensor.type = sensor_types[i].type;
+ strlcpy(sensor->es_sensor.device, DEVNAME(sc),
+ sizeof(sensor->es_sensor.device));
+ strlcpy(sensor->es_sensor.desc, sensor_types[i].name,
+ sizeof(sensor->es_sensor.desc));
+ TAILQ_INSERT_TAIL(&sc->sc_sensors, sensor, es_entry);
+ SENSOR_ADD(&sensor->es_sensor);
+ }
+
+ return;
+
+error:
+ while (!TAILQ_EMPTY(&sc->sc_sensors)) {
+ sensor = TAILQ_FIRST(&sc->sc_sensors);
+ TAILQ_REMOVE(&sc->sc_sensors, sensor, es_entry);
+ free(sensor, M_DEVBUF);
+ }
+}
+
+
+int
+esm_bmc_ready(struct esm_softc *sc, int port, u_int8_t mask, u_int8_t val)
+{
+ if ((EREAD(sc, port) & mask) == val)
+ return (1);
+
+ return (0);
+}
+
+int
+esm_bmc_wait(struct esm_softc *sc, int port, u_int8_t mask, u_int8_t val)
+{
+ unsigned int count;
+
+ for (count = 0; count < 0xffffL; count++) {
+ if (esm_bmc_ready(sc, port, mask, val))
+ return (0);
+ }
+
+ return (1);
+}
+
+int
+esm_cmd(struct esm_softc *sc, void *cmd, size_t cmdlen, void *resp,
+ size_t resplen)
+{
+ u_int8_t *tx = (u_int8_t *)cmd;
+ u_int8_t *rx = (u_int8_t *)resp;
+ int i;
+
+ /* Wait for card ready */
+ if (esm_bmc_wait(sc, ESM2_TC_REG, ESM2_TC_READY, 0) != 0)
+ return (1); /* busy */
+
+ /* Write command data to port */
+ ECTRLWR(sc, ESM2_TC_CLR_WPTR);
+ for (i = 0; i < cmdlen; i++) {
+ DPRINTFN(2, "write: %.2x\n", *tx);
+ EDATAWR(sc, *tx);
+ tx++;
+ }
+
+ /* Ring doorbell and wait */
+ ECTRLWR(sc, ESM2_TC_H2ECDB);
+ if (esm_bmc_wait(sc, ESM2_TC_REG, ESM2_TC_EC2HDB, ESM2_TC_EC2HDB) != 0)
+ printf("post\n");
+
+ /* Set host busy semaphore and clear doorbell */
+ ECTRLWR(sc, ESM2_TC_HOSTBUSY);
+ ECTRLWR(sc, ESM2_TC_EC2HDB);
+
+ /* Read response data from port */
+ ECTRLWR(sc, ESM2_TC_CLR_RPTR);
+ for (i = 0; i < resplen; i++) {
+ *rx = EDATARD(sc);
+ DPRINTFN(2, "read = %.2x\n", *rx);
+ rx++;
+ }
+
+ /* release semaphore */
+ ECTRLWR(sc, ESM2_TC_HOSTBUSY);
+
+ return (0);
+}
+
+int
+esm_smb_cmd(struct esm_softc *sc, struct esm_smb_req *req,
+ struct esm_smb_resp *resp)
+{
+ memset(resp, 0, sizeof(struct esm_smb_resp));
+
+ if (esm_cmd(sc, req, sizeof(req->hdr) + req->h_txlen, resp,
+ sizeof(resp->hdr) + req->h_rxlen) != 0)
+ return (1);
+
+ if (resp->h_status != 0 || resp->h_i2csts != 0) {
+ DPRINTFN(3, "%s: dev: 0x%02x error status: 0x%02x "
+ "i2csts: 0x%02x procsts: 0x%02x tx: 0x%02x rx: 0x%02x\n",
+ __func__, req->h_dev, resp->h_status, resp->h_i2csts,
+ resp->h_procsts, resp->h_rx, resp->h_tx);
+ return (1);
+ }
+
+ return (0);
+}
+
+int64_t
+esm_val2temp(u_int16_t value)
+{
+ return (((int64_t)value/10 * 1000000) + 273150000);
+}
diff --git a/sys/arch/i386/i386/esmreg.h b/sys/arch/i386/i386/esmreg.h
new file mode 100644
index 00000000000..5017d8a5c69
--- /dev/null
+++ b/sys/arch/i386/i386/esmreg.h
@@ -0,0 +1,210 @@
+/* $OpenBSD: esmreg.h,v 1.1 2005/11/21 17:23:19 jordan Exp $ */
+
+/*
+ * Copyright (c) 2005 David Gwynne <dlg@openbsd.org>
+ *
+ * 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.
+ */
+
+#define ESM2_BASE_PORT 0xe0
+
+#if 0
+#define ESM2_TC_REG (ESM2_BASE_PORT + 4)
+#define ESM2_TBUF_REG (ESM2_BASE_PORT + 5)
+#define ESM2_TIM_REG (ESM2_BASE_PORT + 6)
+#else
+#define ESM2_TC_REG 4
+#define ESM2_TBUF_REG 5
+#define ESM2_TIM_REG 6
+#endif
+
+#define ESM2_TC_CLR_WPTR (1L << 0)
+#define ESM2_TC_CLR_RPTR (1L << 1)
+#define ESM2_TC_H2ECDB (1L << 2)
+#define ESM2_TC_EC2HDB (1L << 3)
+#define ESM2_TC_EVENTDB (1L << 4)
+#define ESM2_TC_HBDB (1L << 5)
+#define ESM2_TC_HOSTBUSY (1L << 6)
+#define ESM2_TC_ECBUSY (1L << 7)
+#define ESM2_TC_READY (ESM2_TC_EC2HDB | ESM2_TC_H2ECDB | \
+ ESM2_TC_ECBUSY)
+#define ESM2_TC_POWER_UP_BITS (ESM2_TC_CLR_WPTR | ESM2_TC_CLR_RPTR | \
+ ESM2_TC_EC2HDB | ESM2_TC_EVENTDB)
+
+#define ESM2_TIM_HIRQ_PEND (1L << 1)
+#define ESM2_TIM_SCI_EN (1L << 2)
+#define ESM2_TIM_SMI_EN (1L << 3)
+#define ESM2_TIM_NMI2SMI (1L << 4)
+#define ESM2_TIM_POWER_UP_BITS (ESM2_TIM_HIRQ_PEND)
+
+#define ESM2_CMD_NOOP 0x00
+#define ESM2_CMD_ECHO 0x01
+#define ESM2_CMD_DEVICEMAP 0x03
+#define ESM2_DEVICEMAP_READ 0
+
+#define ESM2_CMD_SMB_BUF 0x20
+#define ESM2_CMD_SMB_BUF_XMIT_RECV 0x21
+#define ESM2_CMD_SMB_XMIT_RECV 0x22
+#define ESM2_SMB_SENSOR_VALUE 0x04
+#define ESM2_SMB_SENSOR_THRESHOLDS 0x19
+
+#define ESM2_CMD_HWDC 0x05 /* Host Watch Dog Control */
+#define ESM2_HWDC_WRITE_STATE 0x01
+#define ESM2_HWDC_READ_PROPERTY 0x02
+#define ESM2_HWDC_WRITE_PROPERTY 0x03
+
+#define ESM2_MAX_CMD_LEN 0x20
+#define ESM2_UUID_LEN 0x08
+
+struct esm_watchdog_prop {
+ u_int8_t cmd;
+ u_int8_t reserved;
+ u_int8_t subcmd;
+ u_int8_t action;
+ u_int32_t time;
+} __packed;
+
+struct esm_watchdog_state {
+ u_int8_t cmd;
+ u_int8_t reserved;
+ u_int8_t subcmd;
+ u_int8_t state;
+} __packed;
+
+struct esm_devmap {
+ u_int8_t index;
+ u_int8_t dev_major;
+ u_int8_t dev_minor;
+ u_int8_t rev_major;
+ u_int8_t rev_minor;
+ u_int8_t rev_rom;
+ u_int8_t smb_addr;
+ u_int8_t status;
+ u_int8_t monitor_type;
+ u_int8_t pollcycle;
+ u_int8_t uniqueid[ESM2_UUID_LEN];
+} __packed;
+
+struct esm_devmap_req {
+ u_int8_t cmd;
+ u_int8_t initiator;
+ u_int8_t action;
+ u_int8_t index;
+ u_int8_t ndev;
+} __packed;
+
+struct esm_devmap_resp {
+ u_int8_t status;
+ u_int8_t ndev;
+ struct esm_devmap devmap[1]; /* request one map at a time */
+} __packed;
+
+
+/* ESM SMB requests */
+
+struct esm_smb_req_hdr {
+} __packed;
+
+struct esm_smb_req_val {
+ u_int8_t v_cmd;
+ u_int8_t v_initiator;
+ u_int8_t v_sensor;
+} __packed;
+
+struct esm_smb_req {
+ struct {
+ u_int8_t _cmd;
+ u_int8_t _dev;
+ u_int8_t _txlen;
+ u_int8_t _rxlen;
+ } __packed hdr;
+#define h_cmd hdr._cmd
+#define h_dev hdr._dev
+#define h_txlen hdr._txlen
+#define h_rxlen hdr._rxlen
+
+ union {
+ struct esm_smb_req_val _val;
+ } __packed _;
+#define req_val _._val
+
+} __packed;
+
+/* ESM SMB responses */
+
+struct esm_smb_resp_val {
+ u_int16_t v_reading;
+ u_int8_t v_status;
+ u_int8_t v_checksum;
+} __packed;
+
+struct esm_smb_resp {
+ struct {
+ u_int8_t _status;
+ u_int8_t _i2csts;
+ u_int8_t _procsts;
+ u_int8_t _tx;
+ u_int8_t _rx;
+ } __packed hdr;
+#define h_status hdr._status
+#define h_i2csts hdr._i2csts
+#define h_procsts hdr._procsts
+#define h_tx hdr._tx
+#define h_rx hdr._rx
+
+ union {
+ struct esm_smb_resp_val _val;
+ } __packed _;
+#define resp_val _._val
+} __packed;
+
+#define ESM2_VS_VALID 0x07
+#define ESM2_VS_SLOT 0x04
+
+#define isValidSensor(state) (((state) & ESM_STATE_MASK) == ESM_VALID_SENSOR)
+#define isValidSlot(state) (((state) & ESM_VALID_SLOT) == ESM_VALID_SLOT)
+
+enum esm_dev_type {
+ ESM2_DEV_ESM2 = 1,
+ ESM2_DEV_DRACII,
+ ESM2_DEV_FRONT_PANEL,
+ ESM2_DEV_BACKPLANE2,
+ ESM2_DEV_POWERUNIT2,
+ ESM2_DEV_ENCL2_BACKPLANE,
+ ESM2_DEV_ENCL2_POWERUNIT,
+ ESM2_DEV_ENCL1_BACKPLANE,
+ ESM2_DEV_ENCL1_POWERUNIT,
+ ESM2_DEV_HPPCI,
+ ESM2_DEV_BACKPLANE3
+};
+
+enum esm_dev_esm2_type {
+ ESM2_DEV_ESM2_2300 = 0,
+ ESM2_DEV_ESM2_4300,
+ ESM2_DEV_ESM2_6300,
+ ESM2_DEV_ESM2_6400,
+ ESM2_DEV_ESM2_2550,
+ ESM2_DEV_ESM2_4350,
+ ESM2_DEV_ESM2_6350,
+ ESM2_DEV_ESM2_6450,
+ ESM2_DEV_ESM2_2400,
+ ESM2_DEV_ESM2_4400,
+ ESM2_DEV_ESM2_R0, /* reserved */
+ ESM2_DEV_ESM2_2500,
+ ESM2_DEV_ESM2_2450,
+ ESM2_DEV_ESM2_R1, /* reserved */
+ ESM2_DEV_ESM2_R2, /* reserved */
+ ESM2_DEV_ESM2_2400EX,
+ ESM2_DEV_ESM2_2450EX
+};
+
diff --git a/sys/arch/i386/i386/esmvar.h b/sys/arch/i386/i386/esmvar.h
new file mode 100644
index 00000000000..09d5a899de6
--- /dev/null
+++ b/sys/arch/i386/i386/esmvar.h
@@ -0,0 +1,24 @@
+/* $OpenBSD: esmvar.h,v 1.1 2005/11/21 17:23:19 jordan Exp $ */
+
+/*
+ * Copyright (c) David Gwynne <dlg@openbsd.org>
+ *
+ * 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.
+ */
+
+struct esm_attach_args {
+ char *eaa_name;
+ bus_space_tag_t eaa_iot;
+ bus_space_tag_t eaa_memt;
+};
+