freebsd-skq/sys/dev/ipmi/ipmi.c
ambrisko b7dae28d4a Add an OpenIPMI mostly compatible driver. This driver was developed
to work with ipmitools.  It works with other tools that have an OpenIPMI
driver interface.  The port will need to get updated to used this.
I have not implemented the IPMB mode yet so ioctl's for that don't
really do much otherwise it should work like the OpenIPMI version.
The ipmi.h definitions was derived from the ipmitool header file.
The bus attachments are done for smbios and pci/smbios.  Differences
in bus probe order for modules/static are delt with.  ACPI attachment
should be done.

This drivers registers with the watchdod(4) interface

Work to do:
     - BT interface
     - IPMB mode

This has been tested on Dell PE2850, PE2650 & PE850 with i386 & amd64
kernel.

I will link this into the build on next week.

Tom Rhodes, helped me with the man page.

Sponsored by:   IronPort Systems Inc.
Inspired from:  ipmitool & Linux
2006-02-10 20:51:35 +00:00

1097 lines
25 KiB
C

/*-
* Copyright (c) 2006 IronPort Systems Inc. <ambrisko@ironport.com>
* 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 AUTHOR 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 AUTHOR 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/cdefs.h>
__FBSDID("$FreeBSD$");
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/malloc.h>
#include <sys/poll.h>
#include <sys/selinfo.h>
#include <sys/disk.h>
#include <sys/module.h>
#include <sys/bus.h>
#include <machine/bus.h>
#include <machine/resource.h>
#include <machine/clock.h>
#include <sys/rman.h>
#include <sys/watchdog.h>
#include <sys/sysctl.h>
#ifdef LOCAL_MODULE
#include <ipmi.h>
#include <ipmivars.h>
#else
#include <sys/ipmi.h>
#include <dev/ipmi/ipmivars.h>
#endif
struct ipmi_done_list {
u_char *data;
int channel;
int msgid;
int len;
TAILQ_ENTRY(ipmi_done_list) list;
};
#define MAX_TIMEOUT 3 * hz
static int ipmi_wait_for_ibf(device_t, int);
static int ipmi_wait_for_obf(device_t, int);
static void ipmi_clear_obf(device_t, int);
static void ipmi_error(device_t);
static void ipmi_check_read(device_t);
static int ipmi_write(device_t, u_char *, int);
static void ipmi_wait_for_tx_okay(device_t);
static void ipmi_wait_for_rx_okay(device_t);
static void ipmi_wait_for_not_busy(device_t);
static void ipmi_set_busy(device_t);
static int ipmi_ready_to_read(device_t);
#ifdef IPMB
static int ipmi_handle_attn(device_t dev);
static int ipmi_ipmb_checksum(u_char, int);
static int ipmi_ipmb_send_message(device_t, u_char, u_char, u_char,
u_char, u_char, int)
#endif
static d_ioctl_t ipmi_ioctl;
static d_poll_t ipmi_poll;
static d_open_t ipmi_open;
static d_close_t ipmi_close;
int ipmi_attached = 0;
#define IPMI_MINOR 0
static int on = 1;
SYSCTL_NODE(_hw, OID_AUTO, ipmi, CTLFLAG_RD, 0, "IPMI driver parameters");
SYSCTL_INT(_hw_ipmi, OID_AUTO, on, CTLFLAG_RW,
&on, 0, "");
static struct cdevsw ipmi_cdevsw = {
.d_version = D_VERSION,
.d_flags = D_NEEDGIANT,
.d_open = ipmi_open,
.d_close = ipmi_close,
.d_ioctl = ipmi_ioctl,
.d_poll = ipmi_poll,
.d_name = "ipmi",
};
MALLOC_DEFINE(M_IPMI, "ipmi", "ipmi");
static int
ipmi_open(struct cdev *dev, int flags, int fmt, struct thread *td)
{
struct ipmi_softc *sc;
if (!on)
return ENOENT;
sc = dev->si_drv1;
if (sc->ipmi_refcnt) {
return EBUSY;
}
sc->ipmi_refcnt = 1;
return 0;
}
static int
ipmi_poll(struct cdev *dev, int poll_events, struct thread *td)
{
struct ipmi_softc *sc;
int revents = 0;
sc = dev->si_drv1;
ipmi_check_read(sc->ipmi_dev);
if (poll_events & (POLLIN | POLLRDNORM)) {
if (!TAILQ_EMPTY(&sc->ipmi_done))
revents |= poll_events & (POLLIN | POLLRDNORM);
if (TAILQ_EMPTY(&sc->ipmi_done) && sc->ipmi_requests == 0) {
revents |= POLLERR;
}
}
if (revents == 0) {
if (poll_events & (POLLIN | POLLRDNORM))
selrecord(td, &sc->ipmi_select);
}
return revents;
}
static int
ipmi_close(struct cdev *dev, int flags, int fmt, struct thread *td)
{
struct ipmi_softc *sc;
int error = 0;
sc = dev->si_drv1;
sc->ipmi_refcnt = 0;
return error;
}
#ifdef IPMB
static int
ipmi_ipmb_checksum(u_char *data, int len)
{
u_char sum = 0;
for (; len; len--) {
sum += *data++;
}
return -sum;
}
static int
ipmi_ipmb_send_message(device_t dev, u_char channel, u_char netfn,
u_char command, u_char seq, u_char *data, int data_len)
{
u_char *temp;
struct ipmi_softc *sc = device_get_softc(dev);
int error;
u_char slave_addr = 0x52;
temp = malloc(data_len + 10, M_IPMI, M_WAITOK);
bzero(temp, data_len + 10);
temp[0] = IPMI_APP_REQUEST << 2;
temp[1] = IPMI_SEND_MSG;
temp[2] = channel;
temp[3] = slave_addr;
temp[4] = netfn << 2;
temp[5] = ipmi_ipmb_check_sum(&temp[3], 2);
temp[6] = sc->ipmi_address;
temp[7] = seq << 2 | sc->ipmi_lun;
temp[8] = command;
bcopy(data, &temp[9], data_len);
temp[data_len + 9] = ipmi_ipmb_check(&temp[6], data_len + 3);
ipmi_write(sc->ipmi_dev, temp, data_len + 9);
free(temp, M_IPMI);
while (!ipmi_ready_to_read(dev))
DELAY(1000);
temp = malloc(IPMI_MAX_RX, M_IPMI, M_WAITOK);
bzero(temp, IPMI_MAX_RX);
error = ipmi_read(dev, temp, IPMI_MAX_RX);
free(temp, M_IPMI);
return error;
}
static int
ipmi_handle_attn(device_t dev)
{
u_char temp[IPMI_MAX_RX];
struct ipmi_softc *sc = device_get_softc(dev);
int error;
device_printf(sc->ipmi_dev, "BMC has a message\n");
temp[0] = IPMI_APP_REQUEST << 2;
temp[1] = IPMI_GET_MSG_FLAGS;
ipmi_write(sc->ipmi_dev, temp, 2);
while (!ipmi_ready_to_read(dev))
DELAY(1000);
bzero(temp, IPMI_MAX_RX);
error = ipmi_read(dev, temp, IPMI_MAX_RX);
if (temp[2] == 0) {
if (temp[3] & IPMI_MSG_BUFFER_FULL) {
device_printf(sc->ipmi_dev, "message buffer full");
}
if (temp[3] & IPMI_WDT_PRE_TIMEOUT) {
device_printf(sc->ipmi_dev,
"watchdog about to go off");
}
if (temp[3] & IPMI_MSG_AVAILABLE) {
temp[0] = IPMI_APP_REQUEST << 2;
temp[1] = IPMI_GET_MSG;
ipmi_write(sc->ipmi_dev, temp, 2);
while (!ipmi_ready_to_read(dev))
DELAY(1000);
bzero(temp, IPMI_MAX_RX);
error = ipmi_read(dev, temp, IPMI_MAX_RX);
device_printf(sc->ipmi_dev, "throw out message ");
dump_buf(temp, 16);
}
} else
return -1;
return error;
}
#endif
static int
ipmi_ready_to_read(device_t dev)
{
struct ipmi_softc *sc = device_get_softc(dev);
int status, flags;
if (sc->ipmi_bios_info.smic_mode) {
flags = INB(sc, sc->ipmi_smic_flags);
#ifdef IPMB
if (flags & SMIC_STATUS_SMS_ATN) {
ipmi_handle_attn(dev);
return 0;
}
#endif
if (flags & SMIC_STATUS_RX_RDY)
return 1;
} else if (sc->ipmi_bios_info.kcs_mode) {
status = INB(sc, sc->ipmi_kcs_status_reg);
#ifdef IPMB
if (status & KCS_STATUS_SMS_ATN) {
ipmi_handle_attn(dev);
return 0;
}
#endif
if (status & KCS_STATUS_OBF)
return 1;
} else {
device_printf(dev,"Unsupported mode\n");
}
return 0;
}
void
ipmi_intr(void *arg) {
device_t dev = arg;
ipmi_check_read(dev);
}
static void
ipmi_check_read(device_t dev){
struct ipmi_softc *sc = device_get_softc(dev);
struct ipmi_done_list *item;
int status;
u_char *temp;
if (!sc->ipmi_requests)
return;
untimeout((timeout_t *)ipmi_check_read, dev, sc->ipmi_timeout_handle);
if(ipmi_ready_to_read(dev)) {
sc->ipmi_requests--;
temp = malloc(IPMI_MAX_RX, M_IPMI, M_WAITOK);
bzero(temp, IPMI_MAX_RX);
status = ipmi_read(dev, temp, IPMI_MAX_RX);
item = malloc(sizeof(struct ipmi_done_list), M_IPMI, M_WAITOK);
bzero(item, sizeof(struct ipmi_done_list));
item->data = temp;
item->len = status;
if (ticks - sc->ipmi_timestamp > MAX_TIMEOUT) {
device_printf(dev, "read timeout when ready\n");
TAILQ_INSERT_TAIL(&sc->ipmi_done, item, list);
selwakeup(&sc->ipmi_select);
} else if (status) {
TAILQ_INSERT_TAIL(&sc->ipmi_done, item, list);
selwakeup(&sc->ipmi_select);
}
} else {
if (ticks - sc->ipmi_timestamp > MAX_TIMEOUT) {
sc->ipmi_requests--;
device_printf(dev, "read timeout when not ready\n");
temp = malloc(IPMI_MAX_RX, M_IPMI, M_WAITOK);
bzero(temp, IPMI_MAX_RX);
sc->ipmi_busy = 0;
wakeup(&sc->ipmi_busy);
status = -1;
item = malloc(sizeof(struct ipmi_done_list),
M_IPMI, M_WAITOK);
bzero(item, sizeof(struct ipmi_done_list));
item->data = temp;
item->len = status;
TAILQ_INSERT_TAIL(&sc->ipmi_done, item, list);
selwakeup(&sc->ipmi_select);
}
}
if (sc->ipmi_requests)
sc->ipmi_timeout_handle
= timeout((timeout_t *)ipmi_check_read, dev, hz/30);
}
static int
ipmi_ioctl(struct cdev *dev __unused, u_long cmd, caddr_t data,
int flags, struct thread *td)
{
struct ipmi_softc *sc;
struct ipmi_req *req = (struct ipmi_req *)data;
struct ipmi_recv *recv = (struct ipmi_recv *)data;
struct ipmi_addr addr;
struct ipmi_done_list *item;
u_char *temp;
int error, len;
sc = dev->si_drv1;
switch (cmd) {
case IPMICTL_SEND_COMMAND:
/* clear out old stuff in queue of stuff done */
while((item = TAILQ_FIRST(&sc->ipmi_done))) {
TAILQ_REMOVE(&sc->ipmi_done, item, list);
free(item->data, M_IPMI);
free(item, M_IPMI);
}
error = copyin(req->addr, &addr, sizeof(addr));
temp = malloc(req->msg.data_len + 2, M_IPMI, M_WAITOK);
if (temp == NULL) {
return ENOMEM;
}
temp[0] = req->msg.netfn << 2;
temp[1] = req->msg.cmd;
error = copyin(req->msg.data, &temp[2],
req->msg.data_len);
if (error != 0) {
free(temp, M_IPMI);
return error;
}
error = ipmi_write(sc->ipmi_dev,
temp, req->msg.data_len + 2);
free(temp, M_IPMI);
if (error != 1)
return EIO;
sc->ipmi_requests++;
sc->ipmi_timestamp = ticks;
ipmi_check_read(sc->ipmi_dev);
return 0;
case IPMICTL_RECEIVE_MSG_TRUNC:
case IPMICTL_RECEIVE_MSG:
item = TAILQ_FIRST(&sc->ipmi_done);
if (!item) {
return EAGAIN;
}
error = copyin(recv->addr, &addr, sizeof(addr));
if (error != 0)
return error;
TAILQ_REMOVE(&sc->ipmi_done, item, list);
addr.channel = IPMI_BMC_CHANNEL;
recv->recv_type = IPMI_RESPONSE_RECV_TYPE;
recv->msgid = item->msgid;
recv->msg.netfn = item->data[0] >> 2;
recv->msg.cmd = item->data[1];
error = len = item->len;
len -= 2;
if (len < 0)
len = 1;
if (recv->msg.data_len < len && cmd == IPMICTL_RECEIVE_MSG) {
TAILQ_INSERT_HEAD(&sc->ipmi_done, item, list);
return EMSGSIZE;
}
len = min(recv->msg.data_len, len);
recv->msg.data_len = len;
error = copyout(&addr, recv->addr,sizeof(addr));
if (error == 0)
error = copyout(&item->data[2], recv->msg.data, len);
free(item->data, M_IPMI);
free(item, M_IPMI);
if (error != 0)
return error;
return 0;
case IPMICTL_SET_MY_ADDRESS_CMD:
sc->ipmi_address = *(int*)data;
return 0;
case IPMICTL_GET_MY_ADDRESS_CMD:
*(int*)data = sc->ipmi_address;
return 0;
case IPMICTL_SET_MY_LUN_CMD:
sc->ipmi_lun = *(int*)data & 0x3;
return 0;
case IPMICTL_GET_MY_LUN_CMD:
*(int*)data = sc->ipmi_lun;
return 0;
case IPMICTL_SET_GETS_EVENTS_CMD:
/*
device_printf(sc->ipmi_dev,
"IPMICTL_SET_GETS_EVENTS_CMD NA\n");
*/
return 0;
case IPMICTL_REGISTER_FOR_CMD:
case IPMICTL_UNREGISTER_FOR_CMD:
return EOPNOTSUPP;
}
device_printf(sc->ipmi_dev, "Unknown IOCTL %lX\n", cmd);
return ENOIOCTL;
}
static int
ipmi_wait_for_ibf(device_t dev, int state) {
struct ipmi_softc *sc = device_get_softc(dev);
int status, start = ticks;
int first = 1;
if (state == 0) {
/* WAIT FOR IBF = 0 */
do {
if (first)
first =0;
else
DELAY(100);
status = INB(sc, sc->ipmi_kcs_status_reg);
} while (ticks - start < MAX_TIMEOUT
&& status & KCS_STATUS_IBF);
} else {
/* WAIT FOR IBF = 1 */
do {
if (first)
first =0;
else
DELAY(100);
status = INB(sc, sc->ipmi_kcs_status_reg);
} while (ticks - start < MAX_TIMEOUT
&& !(status & KCS_STATUS_IBF));
}
return status;
}
static int
ipmi_wait_for_obf(device_t dev, int state) {
struct ipmi_softc *sc = device_get_softc(dev);
int status, start = ticks;
int first = 1;
if (state == 0) {
/* WAIT FOR OBF = 0 */
do {
if (first)
first = 0;
else
DELAY(100);
status = INB(sc, sc->ipmi_kcs_status_reg);
} while (ticks - start < MAX_TIMEOUT
&& status & KCS_STATUS_OBF);
} else {
/* WAIT FOR OBF = 1 */
do {
if (first)
first =0;
else
DELAY(100);
status = INB(sc, sc->ipmi_kcs_status_reg);
} while (ticks - start < MAX_TIMEOUT
&& !(status & KCS_STATUS_OBF));
}
return status;
}
static void
ipmi_clear_obf(device_t dev, int status) {
struct ipmi_softc *sc = device_get_softc(dev);
int data;
/* Clear OBF */
if (status & KCS_STATUS_OBF) {
data = INB(sc, sc->ipmi_kcs_data_out_reg);
}
}
static void
ipmi_error(device_t dev) {
struct ipmi_softc *sc = device_get_softc(dev);
int status, data = 0;
int retry = 0;
for(;;){
status = ipmi_wait_for_ibf(dev, 0);
/* ABORT */
OUTB(sc, sc->ipmi_kcs_command_reg,
KCS_CONTROL_GET_STATUS_ABORT);
/* Wait for IBF = 0 */
status = ipmi_wait_for_ibf(dev, 0);
/* Clear OBF */
ipmi_clear_obf(dev, status);
if (status & KCS_STATUS_OBF) {
data = INB(sc, sc->ipmi_kcs_data_out_reg);
device_printf(dev, "Data %x\n", data);
}
/* 0x00 to DATA_IN */
OUTB(sc, sc->ipmi_kcs_data_in_reg, 0x00);
/* Wait for IBF = 0 */
status = ipmi_wait_for_ibf(dev, 0);
if (KCS_STATUS_STATE(status) == KCS_STATUS_STATE_READ) {
/* Wait for OBF = 1 */
status = ipmi_wait_for_obf(dev, 1);
/* Read error status */
data = INB(sc, sc->ipmi_kcs_data_out_reg);
/* Write READ into Data_in */
OUTB(sc, sc->ipmi_kcs_data_in_reg, KCS_DATA_IN_READ);
/* Wait for IBF = 0 */
status = ipmi_wait_for_ibf(dev, 0);
}
/* IDLE STATE */
if (KCS_STATUS_STATE(status) == KCS_STATUS_STATE_IDLE) {
/* Wait for OBF = 1 */
status = ipmi_wait_for_obf(dev, 1);
/* Clear OBF */
ipmi_clear_obf(dev, status);
break;
}
retry++;
if (retry > 2) {
device_printf(dev, "Retry exhausted %x\n", retry);
break;
}
}
}
static void
ipmi_wait_for_tx_okay(device_t dev) {
struct ipmi_softc *sc = device_get_softc(dev);
int flags;
do {
flags = INB(sc, sc->ipmi_smic_flags);
} while(!flags & SMIC_STATUS_TX_RDY);
}
static void
ipmi_wait_for_rx_okay(device_t dev) {
struct ipmi_softc *sc = device_get_softc(dev);
int flags;
do {
flags = INB(sc, sc->ipmi_smic_flags);
} while(!flags & SMIC_STATUS_RX_RDY);
}
static void
ipmi_wait_for_not_busy(device_t dev) {
struct ipmi_softc *sc = device_get_softc(dev);
int flags;
do {
flags = INB(sc, sc->ipmi_smic_flags);
} while(flags & SMIC_STATUS_BUSY);
}
static void
ipmi_set_busy(device_t dev) {
struct ipmi_softc *sc = device_get_softc(dev);
int flags;
flags = INB(sc, sc->ipmi_smic_flags);
flags |= SMIC_STATUS_BUSY;
OUTB(sc, sc->ipmi_smic_flags, flags);
}
int
ipmi_read(device_t dev, u_char *bytes, int len){
struct ipmi_softc *sc = device_get_softc(dev);
int status, flags, data, i = -1, error;
bzero(bytes, len);
if (sc->ipmi_bios_info.smic_mode) {
ipmi_wait_for_not_busy(dev);
do {
flags = INB(sc, sc->ipmi_smic_flags);
} while(!flags & SMIC_STATUS_RX_RDY);
OUTB(sc, sc->ipmi_smic_ctl_sts, SMIC_CC_SMS_RD_START);
ipmi_wait_for_rx_okay(dev);
ipmi_set_busy(dev);
ipmi_wait_for_not_busy(dev);
status = INB(sc, sc->ipmi_smic_ctl_sts);
if (status != SMIC_SC_SMS_RD_START) {
error = INB(sc, sc->ipmi_smic_data);
device_printf(dev, "Read did not start %x %x\n",
status, error);
sc->ipmi_busy = 0;
return -1;
}
for (i = -1; ; len--) {
i++;
data = INB(sc, sc->ipmi_smic_data);
if (len > 0)
*bytes++ = data;
else {
device_printf(dev, "Read short %x\n", data);
break;
}
do {
flags = INB(sc, sc->ipmi_smic_flags);
} while(!flags & SMIC_STATUS_RX_RDY);
OUTB(sc, sc->ipmi_smic_ctl_sts, SMIC_CC_SMS_RD_NEXT);
ipmi_wait_for_rx_okay(dev);
ipmi_set_busy(dev);
ipmi_wait_for_not_busy(dev);
status = INB(sc, sc->ipmi_smic_ctl_sts);
if (status == SMIC_SC_SMS_RD_NEXT) {
continue;
} else if (status == SMIC_SC_SMS_RD_END) {
break;
} else {
device_printf(dev, "Read did not next %x\n",
status);
}
}
i++;
data = INB(sc, sc->ipmi_smic_data);
if (len > 0)
*bytes++ = data;
else
device_printf(dev, "Read short %x\n", data);
OUTB(sc, sc->ipmi_smic_ctl_sts, SMIC_CC_SMS_RD_END);
i++;
} else if (sc->ipmi_bios_info.kcs_mode) {
for (i = -1; ; len--) {
/* Wait for IBF = 0 */
status = ipmi_wait_for_ibf(dev, 0);
/* Read State */
if (KCS_STATUS_STATE(status)
== KCS_STATUS_STATE_READ) {
i++;
/* Wait for OBF = 1 */
status = ipmi_wait_for_obf(dev, 1);
/* Read Data_out */
data = INB(sc, sc->ipmi_kcs_data_out_reg);
if (len > 0)
*bytes++ = data;
else {
device_printf(dev, "Read short %x byte %d\n", data, i);
break;
}
/* Write READ into Data_in */
OUTB(sc, sc->ipmi_kcs_data_in_reg,
KCS_DATA_IN_READ);
/* Idle State */
} else if (KCS_STATUS_STATE(status)
== KCS_STATUS_STATE_IDLE) {
i++;
/* Wait for OBF = 1*/
status = ipmi_wait_for_obf(dev, 1);
/* Read Dummy */
data = INB(sc, sc->ipmi_kcs_data_out_reg);
break;
/* error state */
} else {
device_printf(dev,
"read status error %x byte %d\n",
status, i);
sc->ipmi_busy = 0;
ipmi_error(dev);
return -1;
}
}
} else {
device_printf(dev, "Unsupported mode\n");
}
sc->ipmi_busy = 0;
wakeup(&sc->ipmi_busy);
return i;
}
static int
ipmi_write(device_t dev, u_char *bytes, int len){
struct ipmi_softc *sc = device_get_softc(dev);
int status, flags, retry;
while(sc->ipmi_busy){
status = tsleep(&sc->ipmi_busy, PCATCH, "ipmi", 0);
if (status)
return status;
}
sc->ipmi_busy = 1;
if (sc->ipmi_bios_info.smic_mode) {
ipmi_wait_for_not_busy(dev);
OUTB(sc, sc->ipmi_smic_ctl_sts, SMIC_CC_SMS_WR_START);
ipmi_wait_for_tx_okay(dev);
OUTB(sc, sc->ipmi_smic_data, *bytes++);
len--;
ipmi_set_busy(dev);
ipmi_wait_for_not_busy(dev);
status = INB(sc, sc->ipmi_smic_ctl_sts);
if (status != SMIC_SC_SMS_WR_START) {
device_printf(dev, "Write did not start %x\n",status);
sc->ipmi_busy = 0;
return -1;
}
for(len--; len; len--) {
OUTB(sc, sc->ipmi_smic_ctl_sts, SMIC_CC_SMS_WR_NEXT);
ipmi_wait_for_tx_okay(dev);
OUTB(sc, sc->ipmi_smic_data, *bytes++);
ipmi_set_busy(dev);
ipmi_wait_for_not_busy(dev);
status = INB(sc, sc->ipmi_smic_ctl_sts);
if (status != SMIC_SC_SMS_WR_NEXT) {
device_printf(dev, "Write did not next %x\n",
status);
sc->ipmi_busy = 0;
return -1;
}
}
do {
flags = INB(sc, sc->ipmi_smic_flags);
} while(!flags & SMIC_STATUS_TX_RDY);
OUTB(sc, sc->ipmi_smic_ctl_sts, SMIC_CC_SMS_WR_END);
ipmi_wait_for_tx_okay(dev);
OUTB(sc, sc->ipmi_smic_data, *bytes);
ipmi_set_busy(dev);
ipmi_wait_for_not_busy(dev);
status = INB(sc, sc->ipmi_smic_ctl_sts);
if (status != SMIC_SC_SMS_WR_END) {
device_printf(dev, "Write did not end %x\n",status);
return -1;
}
} else if (sc->ipmi_bios_info.kcs_mode) {
for (retry = 0; retry < 10; retry++) {
/* Wait for IBF = 0 */
status = ipmi_wait_for_ibf(dev, 0);
/* Clear OBF */
ipmi_clear_obf(dev, status);
/* Write start to command */
OUTB(sc, sc->ipmi_kcs_command_reg,
KCS_CONTROL_WRITE_START);
/* Wait for IBF = 0 */
status = ipmi_wait_for_ibf(dev, 0);
if (KCS_STATUS_STATE(status) == KCS_STATUS_STATE_WRITE)
break;
DELAY(1000000);
}
for(len--; len; len--) {
if (KCS_STATUS_STATE(status)
!= KCS_STATUS_STATE_WRITE) {
/* error state */
device_printf(dev, "status error %x\n",status);
ipmi_error(dev);
sc->ipmi_busy = 0;
return -1;
break;
} else {
/* Clear OBF */
ipmi_clear_obf(dev, status);
/* Data to Data */
OUTB(sc, sc->ipmi_kcs_data_out_reg, *bytes++);
/* Wait for IBF = 0 */
status = ipmi_wait_for_ibf(dev, 0);
if (KCS_STATUS_STATE(status)
!= KCS_STATUS_STATE_WRITE) {
device_printf(dev, "status error %x\n"
,status);
ipmi_error(dev);
return -1;
} else {
/* Clear OBF */
ipmi_clear_obf(dev, status);
}
}
}
/* Write end to command */
OUTB(sc, sc->ipmi_kcs_command_reg, KCS_CONTROL_WRITE_END);
/* Wait for IBF = 0 */
status = ipmi_wait_for_ibf(dev, 0);
if (KCS_STATUS_STATE(status) != KCS_STATUS_STATE_WRITE) {
/* error state */
device_printf(dev, "status error %x\n",status);
ipmi_error(dev);
sc->ipmi_busy = 0;
return -1;
} else {
/* Clear OBF */
ipmi_clear_obf(dev, status);
OUTB(sc, sc->ipmi_kcs_data_out_reg, *bytes++);
}
} else {
device_printf(dev, "Unsupported mode\n");
}
sc->ipmi_busy = 2;
return 1;
}
/*
* Watchdog event handler.
*/
static void
ipmi_set_watchdog(device_t dev, int sec) {
u_char *temp;
int s;
temp = malloc(IPMI_MAX_RX, M_IPMI, M_WAITOK);
temp[0] = IPMI_APP_REQUEST << 2;
if (sec) {
temp[1] = IPMI_SET_WDOG;
temp[2] = IPMI_SET_WD_TIMER_DONT_STOP
| IPMI_SET_WD_TIMER_SMS_OS;
temp[3] = IPMI_SET_WD_ACTION_RESET;
temp[4] = 0;
temp[5] = 0; /* Timer use */
temp[6] = (sec * 10) & 0xff;
temp[7] = (sec * 10) / 2550;
} else {
temp[1] = IPMI_SET_WDOG;
temp[2] = IPMI_SET_WD_TIMER_SMS_OS;
temp[3] = 0;
temp[4] = 0;
temp[5] = 0; /* Timer use */
temp[6] = 0;
temp[7] = 0;
}
s = splhigh();
ipmi_write(dev, temp, 8);
while (!ipmi_ready_to_read(dev))
DELAY(1000);
bzero(temp, IPMI_MAX_RX);
ipmi_read(dev, temp, IPMI_MAX_RX);
if (sec) {
temp[0] = IPMI_APP_REQUEST << 2;
temp[1] = IPMI_RESET_WDOG;
ipmi_write(dev, temp, 2);
while (!ipmi_ready_to_read(dev))
DELAY(1000);
bzero(temp, IPMI_MAX_RX);
ipmi_read(dev, temp, IPMI_MAX_RX);
}
splx(s);
free(temp, M_IPMI);
/*
dump_watchdog(dev);
*/
}
static void
ipmi_wd_event(void *arg, unsigned int cmd, int *error)
{
struct ipmi_softc *sc = arg;
unsigned int timeout;
/* disable / enable */
if (!(cmd & WD_ACTIVE)) {
ipmi_set_watchdog(sc->ipmi_dev, 0);
*error = 0;
return;
}
cmd &= WD_INTERVAL;
/* convert from power-of-to-ns to WDT ticks */
if (cmd >= 64) {
*error = EINVAL;
return;
}
timeout = ((uint64_t)1 << cmd) / 1800000000;
/* reload */
ipmi_set_watchdog(sc->ipmi_dev, timeout);
*error = 0;
}
int
ipmi_attach(device_t dev)
{
struct ipmi_softc *sc = device_get_softc(dev);
u_char temp[1024];
int i;
int status;
int unit;
TAILQ_INIT(&sc->ipmi_done);
sc->ipmi_address = IPMI_BMC_SLAVE_ADDR;
sc->ipmi_lun = IPMI_BMC_SMS_LUN;
temp[0] = IPMI_APP_REQUEST << 2;
temp[1] = IPMI_GET_DEVICE_ID;
ipmi_write(dev, temp, 2);
while (!ipmi_ready_to_read(dev))
DELAY(1000);
bzero(temp, sizeof(temp));
ipmi_read(dev, temp, sizeof(temp));
device_printf(dev, "IPMI device rev. %d, firmware rev. %d.%d, "
"version %d.%d\n",
temp[4] & 0x0f,
temp[5] & 0x0f, temp[7],
temp[7] & 0x0f, temp[7] >> 4);
temp[0] = IPMI_APP_REQUEST << 2;
temp[1] = IPMI_CLEAR_FLAGS;
temp[2] = 8;
ipmi_write(dev, temp, 3);
while (!ipmi_ready_to_read(dev))
DELAY(1000);
bzero(temp, sizeof(temp));
ipmi_read(dev, temp, sizeof(temp));
if (temp[2] == 0xc0) {
device_printf(dev, "Clear flags is busy\n");
}
if (temp[2] == 0xc1) {
device_printf(dev, "Clear flags illegal\n");
}
for(i = 0; i < 8; i++){
temp[0] = IPMI_APP_REQUEST << 2;
temp[1] = IPMI_GET_CHANNEL_INFO;
temp[2] = i;
ipmi_write(dev, temp, 3);
while (!ipmi_ready_to_read(dev))
DELAY(1000);
bzero(temp, sizeof(temp));
ipmi_read(dev, temp, sizeof(temp));
if (temp[2]) {
break;
}
}
device_printf(dev, "Number of channels %d\n", i);
/* probe for watchdog */
bzero(temp, sizeof(temp));
temp[0] = IPMI_APP_REQUEST << 2;
temp[1] = IPMI_GET_WDOG;
status = ipmi_write(dev, temp, 2);
while (!ipmi_ready_to_read(dev))
DELAY(1000);
bzero(temp, sizeof(temp));
ipmi_read(dev, temp, sizeof(temp));
if (temp[0] == 0x1c && temp[2] == 0x00) {
device_printf(dev, "Attached watchdog\n");
/* register the watchdog event handler */
sc->ipmi_ev_tag = EVENTHANDLER_REGISTER(watchdog_list,
ipmi_wd_event, sc, 0);
}
unit = device_get_unit(sc->ipmi_dev);
/* force device to be ipmi0 since that is what ipmitool expects */
sc->ipmi_dev_t = make_dev(&ipmi_cdevsw, unit, UID_ROOT, GID_OPERATOR,
0660, "ipmi%d", 0);
sc->ipmi_dev_t->si_drv1 = sc;
ipmi_attached = 1;
return 0;
}
int
ipmi_detach(device_t dev)
{
struct ipmi_softc *sc;
sc = device_get_softc(dev);
if (sc->ipmi_requests)
untimeout((timeout_t *)ipmi_check_read, dev,
sc->ipmi_timeout_handle);
destroy_dev(sc->ipmi_dev_t);
return 0;
}
#ifdef DEBUG
static void
dump_buf(u_char *data, int len){
char buf[20];
char line[1024];
char temp[30];
int count = 0;
int i=0;
printf("Address %p len %d\n", data, len);
if (len > 256)
len = 256;
line[0] = '\000';
for (; len > 0; len--, data++) {
sprintf(temp, "%02x ", *data);
strcat(line, temp);
if (*data >= ' ' && *data <= '~')
buf[count] = *data;
else if (*data >= 'A' && *data <= 'Z')
buf[count] = *data;
else
buf[count] = '.';
if (++count == 16) {
buf[count] = '\000';
count = 0;
printf(" %3x %s %s\n", i, line, buf);
i+=16;
line[0] = '\000';
}
}
buf[count] = '\000';
for (; count != 16; count++) {
strcat(line, " ");
}
printf(" %3x %s %s\n", i, line, buf);
}
#endif