2037 lines
47 KiB
C
2037 lines
47 KiB
C
/* $OpenBSD: ipmi.c,v 1.119 2024/04/03 18:32:47 gkoehler Exp $ */
|
|
|
|
/*
|
|
* Copyright (c) 2015 Masao Uebayashi
|
|
* Copyright (c) 2005 Jordan Hargrave
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE FOR
|
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/device.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/extent.h>
|
|
#include <sys/sensors.h>
|
|
#include <sys/malloc.h>
|
|
#include <sys/kthread.h>
|
|
#include <sys/task.h>
|
|
|
|
#include <machine/bus.h>
|
|
#include <machine/smbiosvar.h>
|
|
|
|
#include <dev/ipmivar.h>
|
|
#include <dev/ipmi.h>
|
|
|
|
struct ipmi_sensor {
|
|
u_int8_t *i_sdr;
|
|
int i_num;
|
|
int stype;
|
|
int etype;
|
|
struct ksensor i_sensor;
|
|
SLIST_ENTRY(ipmi_sensor) list;
|
|
};
|
|
|
|
int ipmi_enabled = 0;
|
|
|
|
#define SENSOR_REFRESH_RATE 5 /* seconds */
|
|
|
|
#define DEVNAME(s) ((s)->sc_dev.dv_xname)
|
|
|
|
#define IPMI_BTMSG_LEN 0
|
|
#define IPMI_BTMSG_NFLN 1
|
|
#define IPMI_BTMSG_SEQ 2
|
|
#define IPMI_BTMSG_CMD 3
|
|
#define IPMI_BTMSG_CCODE 4
|
|
#define IPMI_BTMSG_DATASND 4
|
|
#define IPMI_BTMSG_DATARCV 5
|
|
|
|
/* IPMI 2.0, Table 42-3: Sensor Type Codes */
|
|
#define IPMI_SENSOR_TYPE_TEMP 0x0101
|
|
#define IPMI_SENSOR_TYPE_VOLT 0x0102
|
|
#define IPMI_SENSOR_TYPE_CURRENT 0x0103
|
|
#define IPMI_SENSOR_TYPE_FAN 0x0104
|
|
#define IPMI_SENSOR_TYPE_INTRUSION 0x6F05
|
|
#define IPMI_SENSOR_TYPE_PWRSUPPLY 0x6F08
|
|
|
|
/* IPMI 2.0, Table 43-15: Sensor Unit Type Codes */
|
|
#define IPMI_UNIT_TYPE_DEGREE_C 1
|
|
#define IPMI_UNIT_TYPE_DEGREE_F 2
|
|
#define IPMI_UNIT_TYPE_DEGREE_K 3
|
|
#define IPMI_UNIT_TYPE_VOLTS 4
|
|
#define IPMI_UNIT_TYPE_AMPS 5
|
|
#define IPMI_UNIT_TYPE_WATTS 6
|
|
#define IPMI_UNIT_TYPE_RPM 18
|
|
|
|
#define IPMI_NAME_UNICODE 0x00
|
|
#define IPMI_NAME_BCDPLUS 0x01
|
|
#define IPMI_NAME_ASCII6BIT 0x02
|
|
#define IPMI_NAME_ASCII8BIT 0x03
|
|
|
|
#define IPMI_ENTITY_PWRSUPPLY 0x0A
|
|
|
|
#define IPMI_INVALID_SENSOR (1L << 5)
|
|
#define IPMI_DISABLED_SENSOR (1L << 6)
|
|
|
|
#define IPMI_SDR_TYPEFULL 1
|
|
#define IPMI_SDR_TYPECOMPACT 2
|
|
|
|
#define byteof(x) ((x) >> 3)
|
|
#define bitof(x) (1L << ((x) & 0x7))
|
|
#define TB(b,m) (data[2+byteof(b)] & bitof(b))
|
|
|
|
#ifdef IPMI_DEBUG
|
|
int ipmi_dbg = 0;
|
|
#define dbg_printf(lvl, fmt...) \
|
|
if (ipmi_dbg >= lvl) \
|
|
printf(fmt);
|
|
#define dbg_dump(lvl, msg, len, buf) \
|
|
if (len && ipmi_dbg >= lvl) \
|
|
dumpb(msg, len, (const u_int8_t *)(buf));
|
|
#else
|
|
#define dbg_printf(lvl, fmt...)
|
|
#define dbg_dump(lvl, msg, len, buf)
|
|
#endif
|
|
|
|
long signextend(unsigned long, int);
|
|
|
|
SLIST_HEAD(ipmi_sensors_head, ipmi_sensor);
|
|
struct ipmi_sensors_head ipmi_sensor_list =
|
|
SLIST_HEAD_INITIALIZER(ipmi_sensor_list);
|
|
|
|
void dumpb(const char *, int, const u_int8_t *);
|
|
|
|
int read_sensor(struct ipmi_softc *, struct ipmi_sensor *);
|
|
int add_sdr_sensor(struct ipmi_softc *, u_int8_t *, int);
|
|
int get_sdr_partial(struct ipmi_softc *, u_int16_t, u_int16_t,
|
|
u_int8_t, u_int8_t, void *, u_int16_t *);
|
|
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_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 *);
|
|
void ipmi_watchdog_set(void *);
|
|
|
|
struct ipmi_softc *ipmilookup(dev_t dev);
|
|
|
|
int ipmiopen(dev_t, int, int, struct proc *);
|
|
int ipmiclose(dev_t, int, int, struct proc *);
|
|
int ipmiioctl(dev_t, u_long, caddr_t, int, struct proc *);
|
|
|
|
long ipow(long, int);
|
|
long ipmi_convert(u_int8_t, struct sdrtype1 *, long);
|
|
int ipmi_sensor_name(char *, int, u_int8_t, u_int8_t *, int);
|
|
|
|
/* BMC Helper Functions */
|
|
u_int8_t bmc_read(struct ipmi_softc *, int);
|
|
void bmc_write(struct ipmi_softc *, int, u_int8_t);
|
|
int bmc_io_wait(struct ipmi_softc *, struct ipmi_iowait *);
|
|
|
|
void bt_buildmsg(struct ipmi_cmd *);
|
|
void cmn_buildmsg(struct ipmi_cmd *);
|
|
|
|
int getbits(u_int8_t *, int, int);
|
|
int ipmi_sensor_type(int, int, int, int);
|
|
|
|
void ipmi_refresh_sensors(struct ipmi_softc *sc);
|
|
int ipmi_map_regs(struct ipmi_softc *sc, struct ipmi_attach_args *ia);
|
|
void ipmi_unmap_regs(struct ipmi_softc *);
|
|
|
|
int ipmi_sensor_status(struct ipmi_softc *, struct ipmi_sensor *,
|
|
u_int8_t *);
|
|
|
|
int add_child_sensors(struct ipmi_softc *, u_int8_t *, int, int, int,
|
|
int, int, int, const char *);
|
|
|
|
void ipmi_create_thread(void *);
|
|
void ipmi_poll_thread(void *);
|
|
|
|
int kcs_probe(struct ipmi_softc *);
|
|
int kcs_reset(struct ipmi_softc *);
|
|
int kcs_sendmsg(struct ipmi_cmd *);
|
|
int kcs_recvmsg(struct ipmi_cmd *);
|
|
|
|
int bt_probe(struct ipmi_softc *);
|
|
int bt_reset(struct ipmi_softc *);
|
|
int bt_sendmsg(struct ipmi_cmd *);
|
|
int bt_recvmsg(struct ipmi_cmd *);
|
|
|
|
int smic_probe(struct ipmi_softc *);
|
|
int smic_reset(struct ipmi_softc *);
|
|
int smic_sendmsg(struct ipmi_cmd *);
|
|
int smic_recvmsg(struct ipmi_cmd *);
|
|
|
|
struct ipmi_if kcs_if = {
|
|
"KCS",
|
|
IPMI_IF_KCS_NREGS,
|
|
cmn_buildmsg,
|
|
kcs_sendmsg,
|
|
kcs_recvmsg,
|
|
kcs_reset,
|
|
kcs_probe,
|
|
IPMI_MSG_DATASND,
|
|
IPMI_MSG_DATARCV,
|
|
};
|
|
|
|
struct ipmi_if smic_if = {
|
|
"SMIC",
|
|
IPMI_IF_SMIC_NREGS,
|
|
cmn_buildmsg,
|
|
smic_sendmsg,
|
|
smic_recvmsg,
|
|
smic_reset,
|
|
smic_probe,
|
|
IPMI_MSG_DATASND,
|
|
IPMI_MSG_DATARCV,
|
|
};
|
|
|
|
struct ipmi_if bt_if = {
|
|
"BT",
|
|
IPMI_IF_BT_NREGS,
|
|
bt_buildmsg,
|
|
bt_sendmsg,
|
|
bt_recvmsg,
|
|
bt_reset,
|
|
bt_probe,
|
|
IPMI_BTMSG_DATASND,
|
|
IPMI_BTMSG_DATARCV,
|
|
};
|
|
|
|
struct ipmi_if *ipmi_get_if(int);
|
|
|
|
struct ipmi_if *
|
|
ipmi_get_if(int iftype)
|
|
{
|
|
switch (iftype) {
|
|
case IPMI_IF_KCS:
|
|
return (&kcs_if);
|
|
case IPMI_IF_SMIC:
|
|
return (&smic_if);
|
|
case IPMI_IF_BT:
|
|
return (&bt_if);
|
|
}
|
|
|
|
return (NULL);
|
|
}
|
|
|
|
/*
|
|
* BMC Helper Functions
|
|
*/
|
|
u_int8_t
|
|
bmc_read(struct ipmi_softc *sc, int offset)
|
|
{
|
|
if (sc->sc_if_iosize == 4)
|
|
return (bus_space_read_4(sc->sc_iot, sc->sc_ioh,
|
|
offset * sc->sc_if_iospacing));
|
|
else
|
|
return (bus_space_read_1(sc->sc_iot, sc->sc_ioh,
|
|
offset * sc->sc_if_iospacing));
|
|
}
|
|
|
|
void
|
|
bmc_write(struct ipmi_softc *sc, int offset, u_int8_t val)
|
|
{
|
|
if (sc->sc_if_iosize == 4)
|
|
bus_space_write_4(sc->sc_iot, sc->sc_ioh,
|
|
offset * sc->sc_if_iospacing, val);
|
|
else
|
|
bus_space_write_1(sc->sc_iot, sc->sc_ioh,
|
|
offset * sc->sc_if_iospacing, val);
|
|
}
|
|
|
|
int
|
|
bmc_io_wait(struct ipmi_softc *sc, struct ipmi_iowait *a)
|
|
{
|
|
volatile u_int8_t v;
|
|
int count = 5000000; /* == 5s XXX can be shorter */
|
|
|
|
while (count--) {
|
|
v = bmc_read(sc, a->offset);
|
|
if ((v & a->mask) == a->value)
|
|
return v;
|
|
|
|
delay(1);
|
|
}
|
|
|
|
dbg_printf(1, "%s: bmc_io_wait fails : *v=%.2x m=%.2x b=%.2x %s\n",
|
|
DEVNAME(sc), v, a->mask, a->value, a->lbl);
|
|
return (-1);
|
|
|
|
}
|
|
|
|
#define RSSA_MASK 0xff
|
|
#define LUN_MASK 0x3
|
|
#define NETFN_LUN(nf,ln) (((nf) << 2) | ((ln) & LUN_MASK))
|
|
|
|
/*
|
|
* BT interface
|
|
*/
|
|
#define _BT_CTRL_REG 0
|
|
#define BT_CLR_WR_PTR (1L << 0)
|
|
#define BT_CLR_RD_PTR (1L << 1)
|
|
#define BT_HOST2BMC_ATN (1L << 2)
|
|
#define BT_BMC2HOST_ATN (1L << 3)
|
|
#define BT_EVT_ATN (1L << 4)
|
|
#define BT_HOST_BUSY (1L << 6)
|
|
#define BT_BMC_BUSY (1L << 7)
|
|
|
|
#define BT_READY (BT_HOST_BUSY|BT_HOST2BMC_ATN|BT_BMC2HOST_ATN)
|
|
|
|
#define _BT_DATAIN_REG 1
|
|
#define _BT_DATAOUT_REG 1
|
|
|
|
#define _BT_INTMASK_REG 2
|
|
#define BT_IM_HIRQ_PEND (1L << 1)
|
|
#define BT_IM_SCI_EN (1L << 2)
|
|
#define BT_IM_SMI_EN (1L << 3)
|
|
#define BT_IM_NMI2SMI (1L << 4)
|
|
|
|
int bt_read(struct ipmi_softc *, int);
|
|
int bt_write(struct ipmi_softc *, int, uint8_t);
|
|
|
|
int
|
|
bt_read(struct ipmi_softc *sc, int reg)
|
|
{
|
|
return bmc_read(sc, reg);
|
|
}
|
|
|
|
int
|
|
bt_write(struct ipmi_softc *sc, int reg, uint8_t data)
|
|
{
|
|
struct ipmi_iowait a;
|
|
|
|
a.offset = _BT_CTRL_REG;
|
|
a.mask = BT_BMC_BUSY;
|
|
a.value = 0;
|
|
a.lbl = "bt_write";
|
|
if (bmc_io_wait(sc, &a) < 0)
|
|
return (-1);
|
|
|
|
bmc_write(sc, reg, data);
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
bt_sendmsg(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
struct ipmi_iowait a;
|
|
int i;
|
|
|
|
bt_write(sc, _BT_CTRL_REG, BT_CLR_WR_PTR);
|
|
for (i = 0; i < c->c_txlen; i++)
|
|
bt_write(sc, _BT_DATAOUT_REG, sc->sc_buf[i]);
|
|
|
|
bt_write(sc, _BT_CTRL_REG, BT_HOST2BMC_ATN);
|
|
a.offset = _BT_CTRL_REG;
|
|
a.mask = BT_HOST2BMC_ATN | BT_BMC_BUSY;
|
|
a.value = 0;
|
|
a.lbl = "bt_sendwait";
|
|
if (bmc_io_wait(sc, &a) < 0)
|
|
return (-1);
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
bt_recvmsg(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
struct ipmi_iowait a;
|
|
u_int8_t len, v, i, j;
|
|
|
|
a.offset = _BT_CTRL_REG;
|
|
a.mask = BT_BMC2HOST_ATN;
|
|
a.value = BT_BMC2HOST_ATN;
|
|
a.lbl = "bt_recvwait";
|
|
if (bmc_io_wait(sc, &a) < 0)
|
|
return (-1);
|
|
|
|
bt_write(sc, _BT_CTRL_REG, BT_HOST_BUSY);
|
|
bt_write(sc, _BT_CTRL_REG, BT_BMC2HOST_ATN);
|
|
bt_write(sc, _BT_CTRL_REG, BT_CLR_RD_PTR);
|
|
len = bt_read(sc, _BT_DATAIN_REG);
|
|
for (i = IPMI_BTMSG_NFLN, j = 0; i <= len; i++) {
|
|
v = bt_read(sc, _BT_DATAIN_REG);
|
|
if (i != IPMI_BTMSG_SEQ)
|
|
*(sc->sc_buf + j++) = v;
|
|
}
|
|
bt_write(sc, _BT_CTRL_REG, BT_HOST_BUSY);
|
|
c->c_rxlen = len - 1;
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
bt_reset(struct ipmi_softc *sc)
|
|
{
|
|
return (-1);
|
|
}
|
|
|
|
int
|
|
bt_probe(struct ipmi_softc *sc)
|
|
{
|
|
u_int8_t rv;
|
|
|
|
rv = bmc_read(sc, _BT_CTRL_REG);
|
|
rv &= BT_HOST_BUSY;
|
|
rv |= BT_CLR_WR_PTR|BT_CLR_RD_PTR|BT_BMC2HOST_ATN|BT_HOST2BMC_ATN;
|
|
bmc_write(sc, _BT_CTRL_REG, rv);
|
|
|
|
rv = bmc_read(sc, _BT_INTMASK_REG);
|
|
rv &= BT_IM_SCI_EN|BT_IM_SMI_EN|BT_IM_NMI2SMI;
|
|
rv |= BT_IM_HIRQ_PEND;
|
|
bmc_write(sc, _BT_INTMASK_REG, rv);
|
|
|
|
#if 0
|
|
printf("bt_probe: %2x\n", v);
|
|
printf(" WR : %2x\n", v & BT_CLR_WR_PTR);
|
|
printf(" RD : %2x\n", v & BT_CLR_RD_PTR);
|
|
printf(" H2B : %2x\n", v & BT_HOST2BMC_ATN);
|
|
printf(" B2H : %2x\n", v & BT_BMC2HOST_ATN);
|
|
printf(" EVT : %2x\n", v & BT_EVT_ATN);
|
|
printf(" HBSY : %2x\n", v & BT_HOST_BUSY);
|
|
printf(" BBSY : %2x\n", v & BT_BMC_BUSY);
|
|
#endif
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* SMIC interface
|
|
*/
|
|
#define _SMIC_DATAIN_REG 0
|
|
#define _SMIC_DATAOUT_REG 0
|
|
|
|
#define _SMIC_CTRL_REG 1
|
|
#define SMS_CC_GET_STATUS 0x40
|
|
#define SMS_CC_START_TRANSFER 0x41
|
|
#define SMS_CC_NEXT_TRANSFER 0x42
|
|
#define SMS_CC_END_TRANSFER 0x43
|
|
#define SMS_CC_START_RECEIVE 0x44
|
|
#define SMS_CC_NEXT_RECEIVE 0x45
|
|
#define SMS_CC_END_RECEIVE 0x46
|
|
#define SMS_CC_TRANSFER_ABORT 0x47
|
|
|
|
#define SMS_SC_READY 0xc0
|
|
#define SMS_SC_WRITE_START 0xc1
|
|
#define SMS_SC_WRITE_NEXT 0xc2
|
|
#define SMS_SC_WRITE_END 0xc3
|
|
#define SMS_SC_READ_START 0xc4
|
|
#define SMS_SC_READ_NEXT 0xc5
|
|
#define SMS_SC_READ_END 0xc6
|
|
|
|
#define _SMIC_FLAG_REG 2
|
|
#define SMIC_BUSY (1L << 0)
|
|
#define SMIC_SMS_ATN (1L << 2)
|
|
#define SMIC_EVT_ATN (1L << 3)
|
|
#define SMIC_SMI (1L << 4)
|
|
#define SMIC_TX_DATA_RDY (1L << 6)
|
|
#define SMIC_RX_DATA_RDY (1L << 7)
|
|
|
|
int smic_wait(struct ipmi_softc *, u_int8_t, u_int8_t, const char *);
|
|
int smic_write_cmd_data(struct ipmi_softc *, u_int8_t, const u_int8_t *);
|
|
int smic_read_data(struct ipmi_softc *, u_int8_t *);
|
|
|
|
int
|
|
smic_wait(struct ipmi_softc *sc, u_int8_t mask, u_int8_t val, const char *lbl)
|
|
{
|
|
struct ipmi_iowait a;
|
|
int v;
|
|
|
|
/* Wait for expected flag bits */
|
|
a.offset = _SMIC_FLAG_REG;
|
|
a.mask = mask;
|
|
a.value = val;
|
|
a.lbl = "smicwait";
|
|
v = bmc_io_wait(sc, &a);
|
|
if (v < 0)
|
|
return (-1);
|
|
|
|
/* Return current status */
|
|
v = bmc_read(sc, _SMIC_CTRL_REG);
|
|
dbg_printf(99, "smic_wait = %.2x\n", v);
|
|
return (v);
|
|
}
|
|
|
|
int
|
|
smic_write_cmd_data(struct ipmi_softc *sc, u_int8_t cmd, const u_int8_t *data)
|
|
{
|
|
int sts, v;
|
|
|
|
dbg_printf(50, "smic_wcd: %.2x %.2x\n", cmd, data ? *data : -1);
|
|
sts = smic_wait(sc, SMIC_TX_DATA_RDY | SMIC_BUSY, SMIC_TX_DATA_RDY,
|
|
"smic_write_cmd_data ready");
|
|
if (sts < 0)
|
|
return (sts);
|
|
|
|
bmc_write(sc, _SMIC_CTRL_REG, cmd);
|
|
if (data)
|
|
bmc_write(sc, _SMIC_DATAOUT_REG, *data);
|
|
|
|
/* Toggle BUSY bit, then wait for busy bit to clear */
|
|
v = bmc_read(sc, _SMIC_FLAG_REG);
|
|
bmc_write(sc, _SMIC_FLAG_REG, v | SMIC_BUSY);
|
|
|
|
return (smic_wait(sc, SMIC_BUSY, 0, "smic_write_cmd_data busy"));
|
|
}
|
|
|
|
int
|
|
smic_read_data(struct ipmi_softc *sc, u_int8_t *data)
|
|
{
|
|
int sts;
|
|
|
|
sts = smic_wait(sc, SMIC_RX_DATA_RDY | SMIC_BUSY, SMIC_RX_DATA_RDY,
|
|
"smic_read_data");
|
|
if (sts >= 0) {
|
|
*data = bmc_read(sc, _SMIC_DATAIN_REG);
|
|
dbg_printf(50, "smic_readdata: %.2x\n", *data);
|
|
}
|
|
return (sts);
|
|
}
|
|
|
|
#define ErrStat(a,b) if (a) printf(b);
|
|
|
|
int
|
|
smic_sendmsg(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
int sts, idx;
|
|
|
|
sts = smic_write_cmd_data(sc, SMS_CC_START_TRANSFER, &sc->sc_buf[0]);
|
|
ErrStat(sts != SMS_SC_WRITE_START, "wstart");
|
|
for (idx = 1; idx < c->c_txlen - 1; idx++) {
|
|
sts = smic_write_cmd_data(sc, SMS_CC_NEXT_TRANSFER,
|
|
&sc->sc_buf[idx]);
|
|
ErrStat(sts != SMS_SC_WRITE_NEXT, "write");
|
|
}
|
|
sts = smic_write_cmd_data(sc, SMS_CC_END_TRANSFER, &sc->sc_buf[idx]);
|
|
if (sts != SMS_SC_WRITE_END) {
|
|
dbg_printf(50, "smic_sendmsg %d/%d = %.2x\n", idx, c->c_txlen, sts);
|
|
return (-1);
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
smic_recvmsg(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
int sts, idx;
|
|
|
|
c->c_rxlen = 0;
|
|
sts = smic_wait(sc, SMIC_RX_DATA_RDY, SMIC_RX_DATA_RDY, "smic_recvmsg");
|
|
if (sts < 0)
|
|
return (-1);
|
|
|
|
sts = smic_write_cmd_data(sc, SMS_CC_START_RECEIVE, NULL);
|
|
ErrStat(sts != SMS_SC_READ_START, "rstart");
|
|
for (idx = 0;; ) {
|
|
sts = smic_read_data(sc, &sc->sc_buf[idx++]);
|
|
if (sts != SMS_SC_READ_START && sts != SMS_SC_READ_NEXT)
|
|
break;
|
|
smic_write_cmd_data(sc, SMS_CC_NEXT_RECEIVE, NULL);
|
|
}
|
|
ErrStat(sts != SMS_SC_READ_END, "rend");
|
|
|
|
c->c_rxlen = idx;
|
|
|
|
sts = smic_write_cmd_data(sc, SMS_CC_END_RECEIVE, NULL);
|
|
if (sts != SMS_SC_READY) {
|
|
dbg_printf(50, "smic_recvmsg %d/%d = %.2x\n", idx, c->c_maxrxlen, sts);
|
|
return (-1);
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
smic_reset(struct ipmi_softc *sc)
|
|
{
|
|
return (-1);
|
|
}
|
|
|
|
int
|
|
smic_probe(struct ipmi_softc *sc)
|
|
{
|
|
/* Flag register should not be 0xFF on a good system */
|
|
if (bmc_read(sc, _SMIC_FLAG_REG) == 0xFF)
|
|
return (-1);
|
|
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* KCS interface
|
|
*/
|
|
#define _KCS_DATAIN_REGISTER 0
|
|
#define _KCS_DATAOUT_REGISTER 0
|
|
#define KCS_READ_NEXT 0x68
|
|
|
|
#define _KCS_COMMAND_REGISTER 1
|
|
#define KCS_GET_STATUS 0x60
|
|
#define KCS_WRITE_START 0x61
|
|
#define KCS_WRITE_END 0x62
|
|
|
|
#define _KCS_STATUS_REGISTER 1
|
|
#define KCS_OBF (1L << 0)
|
|
#define KCS_IBF (1L << 1)
|
|
#define KCS_SMS_ATN (1L << 2)
|
|
#define KCS_CD (1L << 3)
|
|
#define KCS_OEM1 (1L << 4)
|
|
#define KCS_OEM2 (1L << 5)
|
|
#define KCS_STATE_MASK 0xc0
|
|
#define KCS_IDLE_STATE 0x00
|
|
#define KCS_READ_STATE 0x40
|
|
#define KCS_WRITE_STATE 0x80
|
|
#define KCS_ERROR_STATE 0xC0
|
|
|
|
int kcs_wait(struct ipmi_softc *, u_int8_t, u_int8_t, const char *);
|
|
int kcs_write_cmd(struct ipmi_softc *, u_int8_t);
|
|
int kcs_write_data(struct ipmi_softc *, u_int8_t);
|
|
int kcs_read_data(struct ipmi_softc *, u_int8_t *);
|
|
|
|
int
|
|
kcs_wait(struct ipmi_softc *sc, u_int8_t mask, u_int8_t value, const char *lbl)
|
|
{
|
|
struct ipmi_iowait a;
|
|
int v;
|
|
|
|
a.offset = _KCS_STATUS_REGISTER;
|
|
a.mask = mask;
|
|
a.value = value;
|
|
a.lbl = lbl;
|
|
v = bmc_io_wait(sc, &a);
|
|
if (v < 0)
|
|
return (v);
|
|
|
|
/* Check if output buffer full, read dummy byte */
|
|
if ((v & (KCS_OBF | KCS_STATE_MASK)) == (KCS_OBF | KCS_WRITE_STATE))
|
|
bmc_read(sc, _KCS_DATAIN_REGISTER);
|
|
|
|
/* Check for error state */
|
|
if ((v & KCS_STATE_MASK) == KCS_ERROR_STATE) {
|
|
bmc_write(sc, _KCS_COMMAND_REGISTER, KCS_GET_STATUS);
|
|
while (bmc_read(sc, _KCS_STATUS_REGISTER) & KCS_IBF)
|
|
continue;
|
|
printf("%s: error code: %x\n", DEVNAME(sc),
|
|
bmc_read(sc, _KCS_DATAIN_REGISTER));
|
|
}
|
|
|
|
return (v & KCS_STATE_MASK);
|
|
}
|
|
|
|
int
|
|
kcs_write_cmd(struct ipmi_softc *sc, u_int8_t cmd)
|
|
{
|
|
/* ASSERT: IBF and OBF are clear */
|
|
dbg_printf(50, "kcswritecmd: %.2x\n", cmd);
|
|
bmc_write(sc, _KCS_COMMAND_REGISTER, cmd);
|
|
|
|
return (kcs_wait(sc, KCS_IBF, 0, "write_cmd"));
|
|
}
|
|
|
|
int
|
|
kcs_write_data(struct ipmi_softc *sc, u_int8_t data)
|
|
{
|
|
/* ASSERT: IBF and OBF are clear */
|
|
dbg_printf(50, "kcswritedata: %.2x\n", data);
|
|
bmc_write(sc, _KCS_DATAOUT_REGISTER, data);
|
|
|
|
return (kcs_wait(sc, KCS_IBF, 0, "write_data"));
|
|
}
|
|
|
|
int
|
|
kcs_read_data(struct ipmi_softc *sc, u_int8_t * data)
|
|
{
|
|
int sts;
|
|
|
|
sts = kcs_wait(sc, KCS_IBF | KCS_OBF, KCS_OBF, "read_data");
|
|
if (sts != KCS_READ_STATE)
|
|
return (sts);
|
|
|
|
/* ASSERT: OBF is set read data, request next byte */
|
|
*data = bmc_read(sc, _KCS_DATAIN_REGISTER);
|
|
bmc_write(sc, _KCS_DATAOUT_REGISTER, KCS_READ_NEXT);
|
|
|
|
dbg_printf(50, "kcsreaddata: %.2x\n", *data);
|
|
|
|
return (sts);
|
|
}
|
|
|
|
/* Exported KCS functions */
|
|
int
|
|
kcs_sendmsg(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
int idx, sts;
|
|
|
|
/* ASSERT: IBF is clear */
|
|
dbg_dump(50, "kcs sendmsg", c->c_txlen, sc->sc_buf);
|
|
sts = kcs_write_cmd(sc, KCS_WRITE_START);
|
|
for (idx = 0; idx < c->c_txlen; idx++) {
|
|
if (idx == c->c_txlen - 1)
|
|
sts = kcs_write_cmd(sc, KCS_WRITE_END);
|
|
|
|
if (sts != KCS_WRITE_STATE)
|
|
break;
|
|
|
|
sts = kcs_write_data(sc, sc->sc_buf[idx]);
|
|
}
|
|
if (sts != KCS_READ_STATE) {
|
|
dbg_printf(1, "kcs sendmsg = %d/%d <%.2x>\n", idx, c->c_txlen, sts);
|
|
dbg_dump(1, "kcs_sendmsg", c->c_txlen, sc->sc_buf);
|
|
return (-1);
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
kcs_recvmsg(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
int idx, sts;
|
|
|
|
for (idx = 0; idx < c->c_maxrxlen; idx++) {
|
|
sts = kcs_read_data(sc, &sc->sc_buf[idx]);
|
|
if (sts != KCS_READ_STATE)
|
|
break;
|
|
}
|
|
sts = kcs_wait(sc, KCS_IBF, 0, "recv");
|
|
c->c_rxlen = idx;
|
|
if (sts != KCS_IDLE_STATE) {
|
|
dbg_printf(1, "kcs recvmsg = %d/%d <%.2x>\n", idx, c->c_maxrxlen, sts);
|
|
return (-1);
|
|
}
|
|
|
|
dbg_dump(50, "kcs recvmsg", idx, sc->sc_buf);
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
kcs_reset(struct ipmi_softc *sc)
|
|
{
|
|
return (-1);
|
|
}
|
|
|
|
int
|
|
kcs_probe(struct ipmi_softc *sc)
|
|
{
|
|
u_int8_t v;
|
|
|
|
v = bmc_read(sc, _KCS_STATUS_REGISTER);
|
|
if ((v & KCS_STATE_MASK) == KCS_ERROR_STATE)
|
|
return (1);
|
|
#if 0
|
|
printf("kcs_probe: %2x\n", v);
|
|
printf(" STS: %2x\n", v & KCS_STATE_MASK);
|
|
printf(" ATN: %2x\n", v & KCS_SMS_ATN);
|
|
printf(" C/D: %2x\n", v & KCS_CD);
|
|
printf(" IBF: %2x\n", v & KCS_IBF);
|
|
printf(" OBF: %2x\n", v & KCS_OBF);
|
|
#endif
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* IPMI code
|
|
*/
|
|
#define READ_SMS_BUFFER 0x37
|
|
#define WRITE_I2C 0x50
|
|
|
|
#define GET_MESSAGE_CMD 0x33
|
|
#define SEND_MESSAGE_CMD 0x34
|
|
|
|
#define IPMB_CHANNEL_NUMBER 0
|
|
|
|
#define PUBLIC_BUS 0
|
|
|
|
#define MIN_I2C_PACKET_SIZE 3
|
|
#define MIN_IMB_PACKET_SIZE 7 /* one byte for cksum */
|
|
|
|
#define MIN_BTBMC_REQ_SIZE 4
|
|
#define MIN_BTBMC_RSP_SIZE 5
|
|
#define MIN_BMC_REQ_SIZE 2
|
|
#define MIN_BMC_RSP_SIZE 3
|
|
|
|
#define BMC_SA 0x20 /* BMC/ESM3 */
|
|
#define FPC_SA 0x22 /* front panel */
|
|
#define BP_SA 0xC0 /* Primary Backplane */
|
|
#define BP2_SA 0xC2 /* Secondary Backplane */
|
|
#define PBP_SA 0xC4 /* Peripheral Backplane */
|
|
#define DRAC_SA 0x28 /* DRAC-III */
|
|
#define DRAC3_SA 0x30 /* DRAC-III */
|
|
#define BMC_LUN 0
|
|
#define SMS_LUN 2
|
|
|
|
struct ipmi_request {
|
|
u_int8_t rsSa;
|
|
u_int8_t rsLun;
|
|
u_int8_t netFn;
|
|
u_int8_t cmd;
|
|
u_int8_t data_len;
|
|
u_int8_t *data;
|
|
};
|
|
|
|
struct ipmi_response {
|
|
u_int8_t cCode;
|
|
u_int8_t data_len;
|
|
u_int8_t *data;
|
|
};
|
|
|
|
struct ipmi_bmc_request {
|
|
u_int8_t bmc_nfLn;
|
|
u_int8_t bmc_cmd;
|
|
u_int8_t bmc_data_len;
|
|
u_int8_t bmc_data[1];
|
|
};
|
|
|
|
struct ipmi_bmc_response {
|
|
u_int8_t bmc_nfLn;
|
|
u_int8_t bmc_cmd;
|
|
u_int8_t bmc_cCode;
|
|
u_int8_t bmc_data_len;
|
|
u_int8_t bmc_data[1];
|
|
};
|
|
|
|
struct cfdriver ipmi_cd = {
|
|
NULL, "ipmi", DV_DULL
|
|
};
|
|
|
|
void
|
|
dumpb(const char *lbl, int len, const u_int8_t *data)
|
|
{
|
|
int idx;
|
|
|
|
printf("%s: ", lbl);
|
|
for (idx = 0; idx < len; idx++)
|
|
printf("%.2x ", data[idx]);
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
/*
|
|
* bt_buildmsg builds an IPMI message from a nfLun, cmd, and data
|
|
* This is used by BT protocol
|
|
*/
|
|
void
|
|
bt_buildmsg(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
u_int8_t *buf = sc->sc_buf;
|
|
|
|
buf[IPMI_BTMSG_LEN] = c->c_txlen + (IPMI_BTMSG_DATASND - 1);
|
|
buf[IPMI_BTMSG_NFLN] = NETFN_LUN(c->c_netfn, c->c_rslun);
|
|
buf[IPMI_BTMSG_SEQ] = sc->sc_btseq++;
|
|
buf[IPMI_BTMSG_CMD] = c->c_cmd;
|
|
if (c->c_txlen && c->c_data)
|
|
memcpy(buf + IPMI_BTMSG_DATASND, c->c_data, c->c_txlen);
|
|
}
|
|
|
|
/*
|
|
* cmn_buildmsg builds an IPMI message from a nfLun, cmd, and data
|
|
* This is used by both SMIC and KCS protocols
|
|
*/
|
|
void
|
|
cmn_buildmsg(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
u_int8_t *buf = sc->sc_buf;
|
|
|
|
buf[IPMI_MSG_NFLN] = NETFN_LUN(c->c_netfn, c->c_rslun);
|
|
buf[IPMI_MSG_CMD] = c->c_cmd;
|
|
if (c->c_txlen && c->c_data)
|
|
memcpy(buf + IPMI_MSG_DATASND, c->c_data, c->c_txlen);
|
|
}
|
|
|
|
/* Send an IPMI command */
|
|
int
|
|
ipmi_sendcmd(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
int rc = -1;
|
|
|
|
dbg_printf(50, "ipmi_sendcmd: rssa=%.2x nfln=%.2x cmd=%.2x len=%.2x\n",
|
|
c->c_rssa, NETFN_LUN(c->c_netfn, c->c_rslun), c->c_cmd, c->c_txlen);
|
|
dbg_dump(10, " send", c->c_txlen, c->c_data);
|
|
if (c->c_rssa != BMC_SA) {
|
|
#if 0
|
|
sc->sc_if->buildmsg(c);
|
|
pI2C->bus = (sc->if_ver == 0x09) ?
|
|
PUBLIC_BUS :
|
|
IPMB_CHANNEL_NUMBER;
|
|
|
|
imbreq->rsSa = rssa;
|
|
imbreq->nfLn = NETFN_LUN(netfn, rslun);
|
|
imbreq->cSum1 = -(imbreq->rsSa + imbreq->nfLn);
|
|
imbreq->rqSa = BMC_SA;
|
|
imbreq->seqLn = NETFN_LUN(sc->imb_seq++, SMS_LUN);
|
|
imbreq->cmd = cmd;
|
|
if (txlen)
|
|
memcpy(imbreq->data, data, txlen);
|
|
/* Set message checksum */
|
|
imbreq->data[txlen] = cksum8(&imbreq->rqSa, txlen + 3);
|
|
#endif
|
|
goto done;
|
|
} else
|
|
sc->sc_if->buildmsg(c);
|
|
|
|
c->c_txlen += sc->sc_if->datasnd;
|
|
rc = sc->sc_if->sendmsg(c);
|
|
|
|
done:
|
|
return (rc);
|
|
}
|
|
|
|
/* Receive an IPMI command */
|
|
int
|
|
ipmi_recvcmd(struct ipmi_cmd *c)
|
|
{
|
|
struct ipmi_softc *sc = c->c_sc;
|
|
u_int8_t *buf = sc->sc_buf, rc = 0;
|
|
|
|
/* Receive message from interface, copy out result data */
|
|
c->c_maxrxlen += sc->sc_if->datarcv;
|
|
if (sc->sc_if->recvmsg(c) ||
|
|
c->c_rxlen < sc->sc_if->datarcv) {
|
|
return (-1);
|
|
}
|
|
|
|
c->c_rxlen -= sc->sc_if->datarcv;
|
|
if (c->c_rxlen > 0 && c->c_data)
|
|
memcpy(c->c_data, buf + sc->sc_if->datarcv, c->c_rxlen);
|
|
|
|
rc = buf[IPMI_MSG_CCODE];
|
|
#ifdef IPMI_DEBUG
|
|
if (rc != 0)
|
|
dbg_printf(1, "ipmi_recvcmd: nfln=%.2x cmd=%.2x err=%.2x\n",
|
|
buf[IPMI_MSG_NFLN], buf[IPMI_MSG_CMD], buf[IPMI_MSG_CCODE]);
|
|
#endif
|
|
|
|
dbg_printf(50, "ipmi_recvcmd: nfln=%.2x cmd=%.2x err=%.2x len=%.2x\n",
|
|
buf[IPMI_MSG_NFLN], buf[IPMI_MSG_CMD], buf[IPMI_MSG_CCODE],
|
|
c->c_rxlen);
|
|
dbg_dump(10, " recv", c->c_rxlen, c->c_data);
|
|
|
|
return (rc);
|
|
}
|
|
|
|
void
|
|
ipmi_cmd(struct ipmi_cmd *c)
|
|
{
|
|
if (cold || panicstr != NULL)
|
|
ipmi_cmd_poll(c);
|
|
else
|
|
ipmi_cmd_wait(c);
|
|
}
|
|
|
|
void
|
|
ipmi_cmd_poll(struct ipmi_cmd *c)
|
|
{
|
|
if ((c->c_ccode = ipmi_sendcmd(c)))
|
|
printf("%s: sendcmd fails\n", DEVNAME(c->c_sc));
|
|
else
|
|
c->c_ccode = ipmi_recvcmd(c);
|
|
}
|
|
|
|
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_nsec(c, PWAIT, "ipmicmd", INFSLP);
|
|
|
|
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 */
|
|
int
|
|
get_sdr_partial(struct ipmi_softc *sc, u_int16_t recordId, u_int16_t reserveId,
|
|
u_int8_t offset, u_int8_t length, void *buffer, u_int16_t *nxtRecordId)
|
|
{
|
|
u_int8_t cmd[IPMI_GET_WDOG_MAX + 255]; /* 8 + max of length */
|
|
int len;
|
|
|
|
((u_int16_t *) cmd)[0] = reserveId;
|
|
((u_int16_t *) cmd)[1] = recordId;
|
|
cmd[4] = offset;
|
|
cmd[5] = length;
|
|
|
|
struct ipmi_cmd c;
|
|
c.c_sc = sc;
|
|
c.c_rssa = BMC_SA;
|
|
c.c_rslun = BMC_LUN;
|
|
c.c_netfn = STORAGE_NETFN;
|
|
c.c_cmd = STORAGE_GET_SDR;
|
|
c.c_txlen = IPMI_SET_WDOG_MAX;
|
|
c.c_rxlen = 0;
|
|
c.c_maxrxlen = 8 + length;
|
|
c.c_data = cmd;
|
|
ipmi_cmd(&c);
|
|
len = c.c_rxlen;
|
|
|
|
if (nxtRecordId)
|
|
*nxtRecordId = *(uint16_t *) cmd;
|
|
if (len > 2)
|
|
memcpy(buffer, cmd + 2, len - 2);
|
|
else
|
|
return (1);
|
|
|
|
return (0);
|
|
}
|
|
|
|
int maxsdrlen = 0x10;
|
|
|
|
/* Read an entire SDR; pass to add sensor */
|
|
int
|
|
get_sdr(struct ipmi_softc *sc, u_int16_t recid, u_int16_t *nxtrec)
|
|
{
|
|
u_int16_t resid = 0;
|
|
int len, sdrlen, offset;
|
|
u_int8_t *psdr;
|
|
struct sdrhdr shdr;
|
|
|
|
/* Reserve SDR */
|
|
struct ipmi_cmd c;
|
|
c.c_sc = sc;
|
|
c.c_rssa = BMC_SA;
|
|
c.c_rslun = BMC_LUN;
|
|
c.c_netfn = STORAGE_NETFN;
|
|
c.c_cmd = STORAGE_RESERVE_SDR;
|
|
c.c_txlen = 0;
|
|
c.c_maxrxlen = sizeof(resid);
|
|
c.c_rxlen = 0;
|
|
c.c_data = &resid;
|
|
ipmi_cmd(&c);
|
|
|
|
/* Get SDR Header */
|
|
if (get_sdr_partial(sc, recid, resid, 0, sizeof shdr, &shdr, nxtrec)) {
|
|
printf("%s: get header fails\n", DEVNAME(sc));
|
|
return (1);
|
|
}
|
|
/* Allocate space for entire SDR Length of SDR in header does not
|
|
* include header length */
|
|
sdrlen = sizeof(shdr) + shdr.record_length;
|
|
psdr = malloc(sdrlen, M_DEVBUF, M_NOWAIT);
|
|
if (psdr == NULL)
|
|
return (1);
|
|
|
|
memcpy(psdr, &shdr, sizeof(shdr));
|
|
|
|
/* Read SDR Data maxsdrlen bytes at a time */
|
|
for (offset = sizeof(shdr); offset < sdrlen; offset += maxsdrlen) {
|
|
len = sdrlen - offset;
|
|
if (len > maxsdrlen)
|
|
len = maxsdrlen;
|
|
|
|
if (get_sdr_partial(sc, recid, resid, offset, len,
|
|
psdr + offset, NULL)) {
|
|
printf("%s: get chunk: %d,%d fails\n", DEVNAME(sc),
|
|
offset, len);
|
|
free(psdr, M_DEVBUF, sdrlen);
|
|
return (1);
|
|
}
|
|
}
|
|
|
|
/* Add SDR to sensor list, if not wanted, free buffer */
|
|
if (add_sdr_sensor(sc, psdr, sdrlen) == 0)
|
|
free(psdr, M_DEVBUF, sdrlen);
|
|
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
getbits(u_int8_t *bytes, int bitpos, int bitlen)
|
|
{
|
|
int v;
|
|
int mask;
|
|
|
|
bitpos += bitlen - 1;
|
|
for (v = 0; bitlen--;) {
|
|
v <<= 1;
|
|
mask = 1L << (bitpos & 7);
|
|
if (bytes[bitpos >> 3] & mask)
|
|
v |= 1;
|
|
bitpos--;
|
|
}
|
|
|
|
return (v);
|
|
}
|
|
|
|
/* Decode IPMI sensor name */
|
|
int
|
|
ipmi_sensor_name(char *name, int len, u_int8_t typelen, u_int8_t *bits,
|
|
int bitslen)
|
|
{
|
|
int i, slen;
|
|
char bcdplus[] = "0123456789 -.:,_";
|
|
|
|
slen = typelen & 0x1F;
|
|
switch (typelen >> 6) {
|
|
case IPMI_NAME_UNICODE:
|
|
//unicode
|
|
break;
|
|
|
|
case IPMI_NAME_BCDPLUS:
|
|
/* Characters are encoded in 4-bit BCDPLUS */
|
|
if (len < slen * 2 + 1)
|
|
slen = (len >> 1) - 1;
|
|
if (slen > bitslen)
|
|
return (0);
|
|
for (i = 0; i < slen; i++) {
|
|
*(name++) = bcdplus[bits[i] >> 4];
|
|
*(name++) = bcdplus[bits[i] & 0xF];
|
|
}
|
|
break;
|
|
|
|
case IPMI_NAME_ASCII6BIT:
|
|
/* Characters are encoded in 6-bit ASCII
|
|
* 0x00 - 0x3F maps to 0x20 - 0x5F */
|
|
/* XXX: need to calculate max len: slen = 3/4 * len */
|
|
if (len < slen + 1)
|
|
slen = len - 1;
|
|
if (slen * 6 / 8 > bitslen)
|
|
return (0);
|
|
for (i = 0; i < slen * 8; i += 6) {
|
|
*(name++) = getbits(bits, i, 6) + ' ';
|
|
}
|
|
break;
|
|
|
|
case IPMI_NAME_ASCII8BIT:
|
|
/* Characters are 8-bit ascii */
|
|
if (len < slen + 1)
|
|
slen = len - 1;
|
|
if (slen > bitslen)
|
|
return (0);
|
|
while (slen--)
|
|
*(name++) = *(bits++);
|
|
break;
|
|
}
|
|
*name = 0;
|
|
|
|
return (1);
|
|
}
|
|
|
|
/* Calculate val * 10^exp */
|
|
long
|
|
ipow(long val, int exp)
|
|
{
|
|
while (exp > 0) {
|
|
val *= 10;
|
|
exp--;
|
|
}
|
|
|
|
while (exp < 0) {
|
|
val /= 10;
|
|
exp++;
|
|
}
|
|
|
|
return (val);
|
|
}
|
|
|
|
/* Sign extend a n-bit value */
|
|
long
|
|
signextend(unsigned long val, int bits)
|
|
{
|
|
long msk = (1L << (bits-1))-1;
|
|
|
|
return (-(val & ~msk) | val);
|
|
}
|
|
|
|
/* Convert IPMI reading from sensor factors */
|
|
long
|
|
ipmi_convert(u_int8_t v, struct sdrtype1 *s1, long adj)
|
|
{
|
|
int16_t M, B;
|
|
int8_t K1, K2;
|
|
long val;
|
|
|
|
/* Calculate linear reading variables */
|
|
M = signextend((((short)(s1->m_tolerance & 0xC0)) << 2) + s1->m, 10);
|
|
B = signextend((((short)(s1->b_accuracy & 0xC0)) << 2) + s1->b, 10);
|
|
K1 = signextend(s1->rbexp & 0xF, 4);
|
|
K2 = signextend(s1->rbexp >> 4, 4);
|
|
|
|
/* Calculate sensor reading:
|
|
* y = L((M * v + (B * 10^K1)) * 10^(K2+adj)
|
|
*
|
|
* This commutes out to:
|
|
* y = L(M*v * 10^(K2+adj) + B * 10^(K1+K2+adj)); */
|
|
val = ipow(M * v, K2 + adj) + ipow(B, K1 + K2 + adj);
|
|
|
|
/* Linearization function: y = f(x) 0 : y = x 1 : y = ln(x) 2 : y =
|
|
* log10(x) 3 : y = log2(x) 4 : y = e^x 5 : y = 10^x 6 : y = 2^x 7 : y
|
|
* = 1/x 8 : y = x^2 9 : y = x^3 10 : y = square root(x) 11 : y = cube
|
|
* root(x) */
|
|
return (val);
|
|
}
|
|
|
|
int
|
|
ipmi_sensor_status(struct ipmi_softc *sc, struct ipmi_sensor *psensor,
|
|
u_int8_t *reading)
|
|
{
|
|
struct sdrtype1 *s1 = (struct sdrtype1 *)psensor->i_sdr;
|
|
int etype;
|
|
|
|
/* Get reading of sensor */
|
|
switch (psensor->i_sensor.type) {
|
|
case SENSOR_TEMP:
|
|
psensor->i_sensor.value = ipmi_convert(reading[0], s1, 6);
|
|
psensor->i_sensor.value += 273150000;
|
|
break;
|
|
|
|
case SENSOR_VOLTS_DC:
|
|
case SENSOR_VOLTS_AC:
|
|
case SENSOR_AMPS:
|
|
case SENSOR_WATTS:
|
|
psensor->i_sensor.value = ipmi_convert(reading[0], s1, 6);
|
|
break;
|
|
|
|
case SENSOR_FANRPM:
|
|
psensor->i_sensor.value = ipmi_convert(reading[0], s1, 0);
|
|
if (((s1->units1>>3)&0x7) == 0x3)
|
|
psensor->i_sensor.value *= 60; // RPS -> RPM
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
/* Return Sensor Status */
|
|
etype = (psensor->etype << 8) + psensor->stype;
|
|
switch (etype) {
|
|
case IPMI_SENSOR_TYPE_TEMP:
|
|
case IPMI_SENSOR_TYPE_VOLT:
|
|
case IPMI_SENSOR_TYPE_CURRENT:
|
|
case IPMI_SENSOR_TYPE_FAN:
|
|
/* non-recoverable threshold */
|
|
if (reading[2] & ((1 << 5) | (1 << 2)))
|
|
return (SENSOR_S_CRIT);
|
|
/* critical threshold */
|
|
else if (reading[2] & ((1 << 4) | (1 << 1)))
|
|
return (SENSOR_S_CRIT);
|
|
/* non-critical threshold */
|
|
else if (reading[2] & ((1 << 3) | (1 << 0)))
|
|
return (SENSOR_S_WARN);
|
|
break;
|
|
|
|
case IPMI_SENSOR_TYPE_INTRUSION:
|
|
psensor->i_sensor.value = (reading[2] & 1) ? 1 : 0;
|
|
if (reading[2] & 0x1)
|
|
return (SENSOR_S_CRIT);
|
|
break;
|
|
|
|
case IPMI_SENSOR_TYPE_PWRSUPPLY:
|
|
/* Reading: 1 = present+powered, 0 = otherwise */
|
|
psensor->i_sensor.value = (reading[2] & 1) ? 1 : 0;
|
|
if (reading[2] & 0x10) {
|
|
/* XXX: Need sysctl type for Power Supply types
|
|
* ok: power supply installed && powered
|
|
* warn: power supply installed && !powered
|
|
* crit: power supply !installed
|
|
*/
|
|
return (SENSOR_S_CRIT);
|
|
}
|
|
if (reading[2] & 0x08) {
|
|
/* Power supply AC lost */
|
|
return (SENSOR_S_WARN);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return (SENSOR_S_OK);
|
|
}
|
|
|
|
int
|
|
read_sensor(struct ipmi_softc *sc, struct ipmi_sensor *psensor)
|
|
{
|
|
struct sdrtype1 *s1 = (struct sdrtype1 *) psensor->i_sdr;
|
|
u_int8_t data[8];
|
|
int rv = -1;
|
|
|
|
memset(data, 0, sizeof(data));
|
|
data[0] = psensor->i_num;
|
|
|
|
struct ipmi_cmd c;
|
|
c.c_sc = sc;
|
|
c.c_rssa = s1->owner_id;
|
|
c.c_rslun = s1->owner_lun;
|
|
c.c_netfn = SE_NETFN;
|
|
c.c_cmd = SE_GET_SENSOR_READING;
|
|
c.c_txlen = 1;
|
|
c.c_maxrxlen = sizeof(data);
|
|
c.c_rxlen = 0;
|
|
c.c_data = data;
|
|
ipmi_cmd(&c);
|
|
|
|
if (c.c_ccode != 0) {
|
|
dbg_printf(1, "sensor reading command for %s failed: %.2x\n",
|
|
psensor->i_sensor.desc, c.c_ccode);
|
|
return (rv);
|
|
}
|
|
dbg_printf(10, "values=%.2x %.2x %.2x %.2x %s\n",
|
|
data[0],data[1],data[2],data[3], psensor->i_sensor.desc);
|
|
psensor->i_sensor.flags &= ~SENSOR_FINVALID;
|
|
if ((data[1] & IPMI_INVALID_SENSOR) ||
|
|
((data[1] & IPMI_DISABLED_SENSOR) == 0 && data[0] == 0))
|
|
psensor->i_sensor.flags |= SENSOR_FINVALID;
|
|
psensor->i_sensor.status = ipmi_sensor_status(sc, psensor, data);
|
|
rv = 0;
|
|
return (rv);
|
|
}
|
|
|
|
int
|
|
ipmi_sensor_type(int type, int ext_type, int units2, int entity)
|
|
{
|
|
switch (units2) {
|
|
case IPMI_UNIT_TYPE_AMPS:
|
|
return (SENSOR_AMPS);
|
|
|
|
case IPMI_UNIT_TYPE_RPM:
|
|
return (SENSOR_FANRPM);
|
|
|
|
/* XXX sensors framework distinguishes AC/DC but ipmi does not */
|
|
case IPMI_UNIT_TYPE_VOLTS:
|
|
return (SENSOR_VOLTS_DC);
|
|
|
|
case IPMI_UNIT_TYPE_WATTS:
|
|
return (SENSOR_WATTS);
|
|
}
|
|
|
|
switch (ext_type << 8L | type) {
|
|
case IPMI_SENSOR_TYPE_TEMP:
|
|
return (SENSOR_TEMP);
|
|
|
|
case IPMI_SENSOR_TYPE_PWRSUPPLY:
|
|
if (entity == IPMI_ENTITY_PWRSUPPLY)
|
|
return (SENSOR_INDICATOR);
|
|
break;
|
|
|
|
case IPMI_SENSOR_TYPE_INTRUSION:
|
|
return (SENSOR_INDICATOR);
|
|
}
|
|
|
|
return (-1);
|
|
}
|
|
|
|
/* Add Sensor to BSD Sysctl interface */
|
|
int
|
|
add_sdr_sensor(struct ipmi_softc *sc, u_int8_t *psdr, int sdrlen)
|
|
{
|
|
int rc;
|
|
struct sdrtype1 *s1 = (struct sdrtype1 *)psdr;
|
|
struct sdrtype2 *s2 = (struct sdrtype2 *)psdr;
|
|
char name[64];
|
|
|
|
switch (s1->sdrhdr.record_type) {
|
|
case IPMI_SDR_TYPEFULL:
|
|
rc = ipmi_sensor_name(name, sizeof(name), s1->typelen,
|
|
s1->name, sdrlen - (int)offsetof(struct sdrtype1, name));
|
|
if (rc == 0)
|
|
return (0);
|
|
rc = add_child_sensors(sc, psdr, 1, s1->sensor_num,
|
|
s1->sensor_type, s1->event_code, 0, s1->entity_id, name);
|
|
break;
|
|
|
|
case IPMI_SDR_TYPECOMPACT:
|
|
rc = ipmi_sensor_name(name, sizeof(name), s2->typelen,
|
|
s2->name, sdrlen - (int)offsetof(struct sdrtype2, name));
|
|
if (rc == 0)
|
|
return (0);
|
|
rc = add_child_sensors(sc, psdr, s2->share1 & 0xF,
|
|
s2->sensor_num, s2->sensor_type, s2->event_code,
|
|
s2->share2 & 0x7F, s2->entity_id, name);
|
|
break;
|
|
|
|
default:
|
|
return (0);
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
int
|
|
add_child_sensors(struct ipmi_softc *sc, u_int8_t *psdr, int count,
|
|
int sensor_num, int sensor_type, int ext_type, int sensor_base,
|
|
int entity, const char *name)
|
|
{
|
|
int typ, idx, rc = 0;
|
|
struct ipmi_sensor *psensor;
|
|
struct sdrtype1 *s1 = (struct sdrtype1 *)psdr;
|
|
|
|
typ = ipmi_sensor_type(sensor_type, ext_type, s1->units2, entity);
|
|
if (typ == -1) {
|
|
dbg_printf(5, "Unknown sensor type:%.2x et:%.2x sn:%.2x "
|
|
"units2:%u name:%s\n", sensor_type, ext_type, sensor_num,
|
|
s1->units2, name);
|
|
return 0;
|
|
}
|
|
for (idx = 0; idx < count; idx++) {
|
|
psensor = malloc(sizeof(*psensor), M_DEVBUF, M_NOWAIT | M_ZERO);
|
|
if (psensor == NULL)
|
|
break;
|
|
|
|
/* Initialize BSD Sensor info */
|
|
psensor->i_sdr = psdr;
|
|
psensor->i_num = sensor_num + idx;
|
|
psensor->stype = sensor_type;
|
|
psensor->etype = ext_type;
|
|
psensor->i_sensor.type = typ;
|
|
if (count > 1)
|
|
snprintf(psensor->i_sensor.desc,
|
|
sizeof(psensor->i_sensor.desc),
|
|
"%s - %d", name, sensor_base + idx);
|
|
else
|
|
strlcpy(psensor->i_sensor.desc, name,
|
|
sizeof(psensor->i_sensor.desc));
|
|
|
|
dbg_printf(5, "add sensor:%.4x %.2x:%d ent:%.2x:%.2x %s\n",
|
|
s1->sdrhdr.record_id, s1->sensor_type,
|
|
typ, s1->entity_id, s1->entity_instance,
|
|
psensor->i_sensor.desc);
|
|
if (read_sensor(sc, psensor) == 0) {
|
|
SLIST_INSERT_HEAD(&ipmi_sensor_list, psensor, list);
|
|
sensor_attach(&sc->sc_sensordev, &psensor->i_sensor);
|
|
dbg_printf(5, " reading: %lld [%s]\n",
|
|
psensor->i_sensor.value,
|
|
psensor->i_sensor.desc);
|
|
rc = 1;
|
|
} else
|
|
free(psensor, M_DEVBUF, sizeof(*psensor));
|
|
}
|
|
|
|
return (rc);
|
|
}
|
|
|
|
/* Handle IPMI Timer - reread sensor values */
|
|
void
|
|
ipmi_refresh_sensors(struct ipmi_softc *sc)
|
|
{
|
|
if (SLIST_EMPTY(&ipmi_sensor_list))
|
|
return;
|
|
|
|
sc->current_sensor = SLIST_NEXT(sc->current_sensor, list);
|
|
if (sc->current_sensor == NULL)
|
|
sc->current_sensor = SLIST_FIRST(&ipmi_sensor_list);
|
|
|
|
if (read_sensor(sc, sc->current_sensor)) {
|
|
dbg_printf(1, "%s: error reading: %s\n", DEVNAME(sc),
|
|
sc->current_sensor->i_sensor.desc);
|
|
return;
|
|
}
|
|
}
|
|
|
|
int
|
|
ipmi_map_regs(struct ipmi_softc *sc, struct ipmi_attach_args *ia)
|
|
{
|
|
if (sc->sc_if && sc->sc_if->nregs == 0)
|
|
return (0);
|
|
|
|
sc->sc_if = ipmi_get_if(ia->iaa_if_type);
|
|
if (sc->sc_if == NULL)
|
|
return (-1);
|
|
|
|
if (ia->iaa_if_iotype == 'i')
|
|
sc->sc_iot = ia->iaa_iot;
|
|
else
|
|
sc->sc_iot = ia->iaa_memt;
|
|
|
|
sc->sc_if_rev = ia->iaa_if_rev;
|
|
sc->sc_if_iosize = ia->iaa_if_iosize;
|
|
sc->sc_if_iospacing = ia->iaa_if_iospacing;
|
|
if (bus_space_map(sc->sc_iot, ia->iaa_if_iobase,
|
|
sc->sc_if->nregs * sc->sc_if_iospacing,
|
|
0, &sc->sc_ioh)) {
|
|
printf("%s: bus_space_map(%lx %lx %x 0 %p) failed\n",
|
|
DEVNAME(sc),
|
|
(unsigned long)sc->sc_iot, ia->iaa_if_iobase,
|
|
sc->sc_if->nregs * sc->sc_if_iospacing, &sc->sc_ioh);
|
|
return (-1);
|
|
}
|
|
return (0);
|
|
}
|
|
|
|
void
|
|
ipmi_unmap_regs(struct ipmi_softc *sc)
|
|
{
|
|
if (sc->sc_if->nregs > 0) {
|
|
bus_space_unmap(sc->sc_iot, sc->sc_ioh,
|
|
sc->sc_if->nregs * sc->sc_if_iospacing);
|
|
}
|
|
}
|
|
|
|
void
|
|
ipmi_poll_thread(void *arg)
|
|
{
|
|
struct ipmi_thread *thread = arg;
|
|
struct ipmi_softc *sc = thread->sc;
|
|
u_int16_t rec;
|
|
|
|
/* Scan SDRs, add sensors */
|
|
for (rec = 0; rec != 0xFFFF;) {
|
|
if (get_sdr(sc, rec, &rec)) {
|
|
ipmi_unmap_regs(sc);
|
|
printf("%s: no SDRs IPMI disabled\n", DEVNAME(sc));
|
|
goto done;
|
|
}
|
|
tsleep_nsec(sc, PWAIT, "ipmirun", MSEC_TO_NSEC(1));
|
|
}
|
|
|
|
/* initialize sensor list for thread */
|
|
if (SLIST_EMPTY(&ipmi_sensor_list))
|
|
goto done;
|
|
else
|
|
sc->current_sensor = SLIST_FIRST(&ipmi_sensor_list);
|
|
|
|
strlcpy(sc->sc_sensordev.xname, sc->sc_dev.dv_xname,
|
|
sizeof(sc->sc_sensordev.xname));
|
|
sensordev_install(&sc->sc_sensordev);
|
|
|
|
while (thread->running) {
|
|
ipmi_refresh_sensors(sc);
|
|
tsleep_nsec(thread, PWAIT, "ipmi_poll",
|
|
SEC_TO_NSEC(SENSOR_REFRESH_RATE));
|
|
}
|
|
|
|
done:
|
|
kthread_exit(0);
|
|
}
|
|
|
|
void
|
|
ipmi_create_thread(void *arg)
|
|
{
|
|
struct ipmi_softc *sc = arg;
|
|
|
|
if (kthread_create(ipmi_poll_thread, sc->sc_thread, NULL,
|
|
DEVNAME(sc)) != 0) {
|
|
printf("%s: unable to create run thread, ipmi disabled\n",
|
|
DEVNAME(sc));
|
|
return;
|
|
}
|
|
}
|
|
|
|
void
|
|
ipmi_attach_common(struct ipmi_softc *sc, struct ipmi_attach_args *ia)
|
|
{
|
|
struct ipmi_cmd *c = &sc->sc_ioctl.cmd;
|
|
|
|
/* Map registers */
|
|
ipmi_map_regs(sc, ia);
|
|
|
|
sc->sc_thread = malloc(sizeof(struct ipmi_thread), M_DEVBUF, M_NOWAIT);
|
|
if (sc->sc_thread == NULL) {
|
|
printf(": unable to allocate thread\n");
|
|
return;
|
|
}
|
|
sc->sc_thread->sc = sc;
|
|
sc->sc_thread->running = 1;
|
|
|
|
/* Setup threads */
|
|
kthread_create_deferred(ipmi_create_thread, sc);
|
|
|
|
printf(": version %d.%d interface %s",
|
|
ia->iaa_if_rev >> 4, ia->iaa_if_rev & 0xF, sc->sc_if->name);
|
|
if (sc->sc_if->nregs > 0)
|
|
printf(" %sbase 0x%lx/%x spacing %d",
|
|
ia->iaa_if_iotype == 'i' ? "io" : "mem", ia->iaa_if_iobase,
|
|
ia->iaa_if_iospacing * sc->sc_if->nregs,
|
|
ia->iaa_if_iospacing);
|
|
if (ia->iaa_if_irq != -1)
|
|
printf(" irq %d", ia->iaa_if_irq);
|
|
printf("\n");
|
|
|
|
/* setup flag to exclude iic */
|
|
ipmi_enabled = 1;
|
|
|
|
/* Setup Watchdog timer */
|
|
sc->sc_wdog_period = 0;
|
|
task_set(&sc->sc_wdog_tickle_task, ipmi_watchdog_tickle, sc);
|
|
wdog_register(ipmi_watchdog, sc);
|
|
|
|
rw_init(&sc->sc_ioctl.lock, DEVNAME(sc));
|
|
sc->sc_ioctl.req.msgid = -1;
|
|
c->c_sc = sc;
|
|
c->c_ccode = -1;
|
|
|
|
sc->sc_cmd_taskq = taskq_create("ipmicmd", 1, IPL_MPFLOOR,
|
|
TASKQ_MPSAFE);
|
|
}
|
|
|
|
int
|
|
ipmi_activate(struct device *self, int act)
|
|
{
|
|
switch (act) {
|
|
case DVACT_POWERDOWN:
|
|
wdog_shutdown(self);
|
|
break;
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
struct ipmi_softc *
|
|
ipmilookup(dev_t dev)
|
|
{
|
|
return (struct ipmi_softc *)device_lookup(&ipmi_cd, minor(dev));
|
|
}
|
|
|
|
int
|
|
ipmiopen(dev_t dev, int flags, int mode, struct proc *p)
|
|
{
|
|
struct ipmi_softc *sc = ipmilookup(dev);
|
|
|
|
if (sc == NULL)
|
|
return (ENXIO);
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
ipmiclose(dev_t dev, int flags, int mode, struct proc *p)
|
|
{
|
|
struct ipmi_softc *sc = ipmilookup(dev);
|
|
|
|
if (sc == NULL)
|
|
return (ENXIO);
|
|
return (0);
|
|
}
|
|
|
|
int
|
|
ipmiioctl(dev_t dev, u_long cmd, caddr_t data, int flag, struct proc *proc)
|
|
{
|
|
struct ipmi_softc *sc = ipmilookup(dev);
|
|
struct ipmi_req *req = (struct ipmi_req *)data;
|
|
struct ipmi_recv *recv = (struct ipmi_recv *)data;
|
|
struct ipmi_cmd *c = &sc->sc_ioctl.cmd;
|
|
int iv;
|
|
int len;
|
|
u_char ccode;
|
|
int rc = 0;
|
|
|
|
if (sc == NULL)
|
|
return (ENXIO);
|
|
|
|
rw_enter_write(&sc->sc_ioctl.lock);
|
|
|
|
c->c_maxrxlen = sizeof(sc->sc_ioctl.buf);
|
|
c->c_data = sc->sc_ioctl.buf;
|
|
|
|
switch (cmd) {
|
|
case IPMICTL_SEND_COMMAND:
|
|
if (req->msgid == -1) {
|
|
rc = EINVAL;
|
|
goto reset;
|
|
}
|
|
if (sc->sc_ioctl.req.msgid != -1) {
|
|
rc = EBUSY;
|
|
goto reset;
|
|
}
|
|
len = req->msg.data_len;
|
|
if (len < 0) {
|
|
rc = EINVAL;
|
|
goto reset;
|
|
}
|
|
if (len > c->c_maxrxlen) {
|
|
rc = E2BIG;
|
|
goto reset;
|
|
}
|
|
sc->sc_ioctl.req = *req;
|
|
c->c_ccode = -1;
|
|
rc = copyin(req->msg.data, c->c_data, len);
|
|
if (rc != 0)
|
|
goto reset;
|
|
KASSERT(c->c_ccode == -1);
|
|
|
|
/* Execute a command synchronously. */
|
|
c->c_netfn = req->msg.netfn;
|
|
c->c_cmd = req->msg.cmd;
|
|
c->c_txlen = req->msg.data_len;
|
|
c->c_rxlen = 0;
|
|
ipmi_cmd(c);
|
|
break;
|
|
case IPMICTL_RECEIVE_MSG_TRUNC:
|
|
case IPMICTL_RECEIVE_MSG:
|
|
if (sc->sc_ioctl.req.msgid == -1) {
|
|
rc = EINVAL;
|
|
goto reset;
|
|
}
|
|
if (c->c_ccode == -1) {
|
|
rc = EAGAIN;
|
|
goto reset;
|
|
}
|
|
ccode = c->c_ccode & 0xff;
|
|
rc = copyout(&ccode, recv->msg.data, 1);
|
|
if (rc != 0)
|
|
goto reset;
|
|
|
|
/* Return a command result. */
|
|
recv->recv_type = IPMI_RESPONSE_RECV_TYPE;
|
|
recv->msgid = sc->sc_ioctl.req.msgid;
|
|
recv->msg.netfn = sc->sc_ioctl.req.msg.netfn;
|
|
recv->msg.cmd = sc->sc_ioctl.req.msg.cmd;
|
|
recv->msg.data_len = c->c_rxlen + 1;
|
|
|
|
rc = copyout(c->c_data, recv->msg.data + 1, c->c_rxlen);
|
|
/* Always reset state after command completion. */
|
|
goto reset;
|
|
case IPMICTL_SET_MY_ADDRESS_CMD:
|
|
iv = *(int *)data;
|
|
if (iv < 0 || iv > RSSA_MASK) {
|
|
rc = EINVAL;
|
|
goto reset;
|
|
}
|
|
c->c_rssa = iv;
|
|
break;
|
|
case IPMICTL_GET_MY_ADDRESS_CMD:
|
|
*(int *)data = c->c_rssa;
|
|
break;
|
|
case IPMICTL_SET_MY_LUN_CMD:
|
|
iv = *(int *)data;
|
|
if (iv < 0 || iv > LUN_MASK) {
|
|
rc = EINVAL;
|
|
goto reset;
|
|
}
|
|
c->c_rslun = iv;
|
|
break;
|
|
case IPMICTL_GET_MY_LUN_CMD:
|
|
*(int *)data = c->c_rslun;
|
|
break;
|
|
case IPMICTL_SET_GETS_EVENTS_CMD:
|
|
break;
|
|
case IPMICTL_REGISTER_FOR_CMD:
|
|
case IPMICTL_UNREGISTER_FOR_CMD:
|
|
default:
|
|
break;
|
|
}
|
|
done:
|
|
rw_exit_write(&sc->sc_ioctl.lock);
|
|
return (rc);
|
|
reset:
|
|
sc->sc_ioctl.req.msgid = -1;
|
|
c->c_ccode = -1;
|
|
goto done;
|
|
}
|
|
|
|
#define MIN_PERIOD 10
|
|
|
|
int
|
|
ipmi_watchdog(void *arg, int period)
|
|
{
|
|
struct ipmi_softc *sc = arg;
|
|
|
|
if (sc->sc_wdog_period == period) {
|
|
if (period != 0) {
|
|
struct task *t;
|
|
int res;
|
|
|
|
t = &sc->sc_wdog_tickle_task;
|
|
(void)task_del(systq, t);
|
|
res = task_add(systq, t);
|
|
KASSERT(res == 1);
|
|
}
|
|
return (period);
|
|
}
|
|
|
|
if (period < MIN_PERIOD && period > 0)
|
|
period = MIN_PERIOD;
|
|
sc->sc_wdog_period = period;
|
|
ipmi_watchdog_set(sc);
|
|
printf("%s: watchdog %sabled\n", DEVNAME(sc),
|
|
(period == 0) ? "dis" : "en");
|
|
return (period);
|
|
}
|
|
|
|
void
|
|
ipmi_watchdog_tickle(void *arg)
|
|
{
|
|
struct ipmi_softc *sc = arg;
|
|
struct ipmi_cmd c;
|
|
|
|
c.c_sc = sc;
|
|
c.c_rssa = BMC_SA;
|
|
c.c_rslun = BMC_LUN;
|
|
c.c_netfn = APP_NETFN;
|
|
c.c_cmd = APP_RESET_WATCHDOG;
|
|
c.c_txlen = 0;
|
|
c.c_maxrxlen = 0;
|
|
c.c_rxlen = 0;
|
|
c.c_data = NULL;
|
|
ipmi_cmd(&c);
|
|
}
|
|
|
|
void
|
|
ipmi_watchdog_set(void *arg)
|
|
{
|
|
struct ipmi_softc *sc = arg;
|
|
uint8_t wdog[IPMI_GET_WDOG_MAX];
|
|
struct ipmi_cmd c;
|
|
|
|
c.c_sc = sc;
|
|
c.c_rssa = BMC_SA;
|
|
c.c_rslun = BMC_LUN;
|
|
c.c_netfn = APP_NETFN;
|
|
c.c_cmd = APP_GET_WATCHDOG_TIMER;
|
|
c.c_txlen = 0;
|
|
c.c_maxrxlen = IPMI_GET_WDOG_MAX;
|
|
c.c_rxlen = 0;
|
|
c.c_data = wdog;
|
|
ipmi_cmd(&c);
|
|
|
|
/* Period is 10ths/sec */
|
|
uint16_t timo = htole16(sc->sc_wdog_period * 10);
|
|
|
|
memcpy(&wdog[IPMI_SET_WDOG_TIMOL], &timo, 2);
|
|
wdog[IPMI_SET_WDOG_TIMER] &= ~IPMI_WDOG_DONTSTOP;
|
|
wdog[IPMI_SET_WDOG_TIMER] |= (sc->sc_wdog_period == 0) ?
|
|
0 : IPMI_WDOG_DONTSTOP;
|
|
wdog[IPMI_SET_WDOG_ACTION] &= ~IPMI_WDOG_MASK;
|
|
wdog[IPMI_SET_WDOG_ACTION] |= (sc->sc_wdog_period == 0) ?
|
|
IPMI_WDOG_DISABLED : IPMI_WDOG_REBOOT;
|
|
|
|
c.c_cmd = APP_SET_WATCHDOG_TIMER;
|
|
c.c_txlen = IPMI_SET_WDOG_MAX;
|
|
c.c_maxrxlen = 0;
|
|
c.c_rxlen = 0;
|
|
c.c_data = wdog;
|
|
ipmi_cmd(&c);
|
|
}
|
|
|
|
#if defined(__amd64__) || defined(__i386__)
|
|
|
|
#include <dev/isa/isareg.h>
|
|
#include <dev/isa/isavar.h>
|
|
|
|
/*
|
|
* Format of SMBIOS IPMI Flags
|
|
*
|
|
* bit0: interrupt trigger mode (1=level, 0=edge)
|
|
* bit1: interrupt polarity (1=active high, 0=active low)
|
|
* bit2: reserved
|
|
* bit3: address LSB (1=odd,0=even)
|
|
* bit4: interrupt (1=specified, 0=not specified)
|
|
* bit5: reserved
|
|
* bit6/7: register spacing (1,4,2,err)
|
|
*/
|
|
#define SMIPMI_FLAG_IRQLVL (1L << 0)
|
|
#define SMIPMI_FLAG_IRQEN (1L << 3)
|
|
#define SMIPMI_FLAG_ODDOFFSET (1L << 4)
|
|
#define SMIPMI_FLAG_IFSPACING(x) (((x)>>6)&0x3)
|
|
#define IPMI_IOSPACING_BYTE 0
|
|
#define IPMI_IOSPACING_WORD 2
|
|
#define IPMI_IOSPACING_DWORD 1
|
|
|
|
struct dmd_ipmi {
|
|
u_int8_t dmd_sig[4]; /* Signature 'IPMI' */
|
|
u_int8_t dmd_i2c_address; /* Address of BMC */
|
|
u_int8_t dmd_nvram_address; /* Address of NVRAM */
|
|
u_int8_t dmd_if_type; /* IPMI Interface Type */
|
|
u_int8_t dmd_if_rev; /* IPMI Interface Revision */
|
|
} __packed;
|
|
|
|
void *scan_sig(long, long, int, int, const void *);
|
|
|
|
void ipmi_smbios_probe(struct smbios_ipmi *, struct ipmi_attach_args *);
|
|
int ipmi_match(struct device *, void *, void *);
|
|
void ipmi_attach(struct device *, struct device *, void *);
|
|
|
|
const struct cfattach ipmi_ca = {
|
|
sizeof(struct ipmi_softc), ipmi_match, ipmi_attach,
|
|
NULL, ipmi_activate
|
|
};
|
|
|
|
int
|
|
ipmi_match(struct device *parent, void *match, void *aux)
|
|
{
|
|
struct ipmi_softc *sc;
|
|
struct ipmi_attach_args *ia = aux;
|
|
struct cfdata *cf = match;
|
|
u_int8_t cmd[32];
|
|
int rv = 0;
|
|
|
|
if (strcmp(ia->iaa_name, cf->cf_driver->cd_name))
|
|
return (0);
|
|
|
|
/* XXX local softc is wrong wrong wrong */
|
|
sc = malloc(sizeof(*sc), M_TEMP, M_WAITOK | M_ZERO);
|
|
strlcpy(sc->sc_dev.dv_xname, "ipmi0", sizeof(sc->sc_dev.dv_xname));
|
|
|
|
/* Map registers */
|
|
if (ipmi_map_regs(sc, ia) == 0) {
|
|
sc->sc_if->probe(sc);
|
|
|
|
/* Identify BMC device early to detect lying bios */
|
|
struct ipmi_cmd c;
|
|
c.c_sc = sc;
|
|
c.c_rssa = BMC_SA;
|
|
c.c_rslun = BMC_LUN;
|
|
c.c_netfn = APP_NETFN;
|
|
c.c_cmd = APP_GET_DEVICE_ID;
|
|
c.c_txlen = 0;
|
|
c.c_maxrxlen = sizeof(cmd);
|
|
c.c_rxlen = 0;
|
|
c.c_data = cmd;
|
|
ipmi_cmd(&c);
|
|
|
|
dbg_dump(1, "bmc data", c.c_rxlen, cmd);
|
|
rv = 1; /* GETID worked, we got IPMI */
|
|
ipmi_unmap_regs(sc);
|
|
}
|
|
|
|
free(sc, M_TEMP, sizeof(*sc));
|
|
|
|
return (rv);
|
|
}
|
|
|
|
void
|
|
ipmi_attach(struct device *parent, struct device *self, void *aux)
|
|
{
|
|
ipmi_attach_common((struct ipmi_softc *)self, aux);
|
|
}
|
|
|
|
/* Scan memory for signature */
|
|
void *
|
|
scan_sig(long start, long end, int skip, int len, const void *data)
|
|
{
|
|
void *va;
|
|
|
|
while (start < end) {
|
|
va = ISA_HOLE_VADDR(start);
|
|
if (memcmp(va, data, len) == 0)
|
|
return (va);
|
|
|
|
start += skip;
|
|
}
|
|
|
|
return (NULL);
|
|
}
|
|
|
|
void
|
|
ipmi_smbios_probe(struct smbios_ipmi *pipmi, struct ipmi_attach_args *ia)
|
|
{
|
|
|
|
dbg_printf(1, "ipmi_smbios_probe: %02x %02x %02x %02x %08llx %02x "
|
|
"%02x\n",
|
|
pipmi->smipmi_if_type,
|
|
pipmi->smipmi_if_rev,
|
|
pipmi->smipmi_i2c_address,
|
|
pipmi->smipmi_nvram_address,
|
|
pipmi->smipmi_base_address,
|
|
pipmi->smipmi_base_flags,
|
|
pipmi->smipmi_irq);
|
|
|
|
ia->iaa_if_type = pipmi->smipmi_if_type;
|
|
ia->iaa_if_rev = pipmi->smipmi_if_rev;
|
|
ia->iaa_if_irq = (pipmi->smipmi_base_flags & SMIPMI_FLAG_IRQEN) ?
|
|
pipmi->smipmi_irq : -1;
|
|
ia->iaa_if_irqlvl = (pipmi->smipmi_base_flags & SMIPMI_FLAG_IRQLVL) ?
|
|
IST_LEVEL : IST_EDGE;
|
|
ia->iaa_if_iosize = 1;
|
|
|
|
switch (SMIPMI_FLAG_IFSPACING(pipmi->smipmi_base_flags)) {
|
|
case IPMI_IOSPACING_BYTE:
|
|
ia->iaa_if_iospacing = 1;
|
|
break;
|
|
|
|
case IPMI_IOSPACING_DWORD:
|
|
ia->iaa_if_iospacing = 4;
|
|
break;
|
|
|
|
case IPMI_IOSPACING_WORD:
|
|
ia->iaa_if_iospacing = 2;
|
|
break;
|
|
|
|
default:
|
|
ia->iaa_if_iospacing = 1;
|
|
printf("ipmi: unknown register spacing\n");
|
|
}
|
|
|
|
/* Calculate base address (PCI BAR format) */
|
|
if (pipmi->smipmi_base_address & 0x1) {
|
|
ia->iaa_if_iotype = 'i';
|
|
ia->iaa_if_iobase = pipmi->smipmi_base_address & ~0x1;
|
|
} else {
|
|
ia->iaa_if_iotype = 'm';
|
|
ia->iaa_if_iobase = pipmi->smipmi_base_address & ~0xF;
|
|
}
|
|
if (pipmi->smipmi_base_flags & SMIPMI_FLAG_ODDOFFSET)
|
|
ia->iaa_if_iobase++;
|
|
|
|
if (pipmi->smipmi_base_flags == 0x7f) {
|
|
/* IBM 325 eServer workaround */
|
|
ia->iaa_if_iospacing = 1;
|
|
ia->iaa_if_iobase = pipmi->smipmi_base_address;
|
|
ia->iaa_if_iotype = 'i';
|
|
return;
|
|
}
|
|
}
|
|
|
|
int
|
|
ipmi_probe(void *aux)
|
|
{
|
|
struct ipmi_attach_args *ia = aux;
|
|
struct dmd_ipmi *pipmi;
|
|
struct smbtable tbl;
|
|
|
|
tbl.cookie = 0;
|
|
if (smbios_find_table(SMBIOS_TYPE_IPMIDEV, &tbl))
|
|
ipmi_smbios_probe(tbl.tblhdr, ia);
|
|
else {
|
|
pipmi = (struct dmd_ipmi *)scan_sig(0xC0000L, 0xFFFFFL, 16, 4,
|
|
"IPMI");
|
|
/* XXX hack to find Dell PowerEdge 8450 */
|
|
if (pipmi == NULL) {
|
|
/* no IPMI found */
|
|
return (0);
|
|
}
|
|
|
|
/* we have an IPMI signature, fill in attach arg structure */
|
|
ia->iaa_if_type = pipmi->dmd_if_type;
|
|
ia->iaa_if_rev = pipmi->dmd_if_rev;
|
|
}
|
|
|
|
return (1);
|
|
}
|
|
|
|
#endif
|