Blame src/plugins/bmc/bmc.c

Packit d14fb6
/*
Packit d14fb6
 * Copyright (c) 2004 Sun Microsystems, Inc.  All Rights Reserved.
Packit d14fb6
 * 
Packit d14fb6
 * Redistribution and use in source and binary forms, with or without
Packit d14fb6
 * modification, are permitted provided that the following conditions
Packit d14fb6
 * are met:
Packit d14fb6
 * 
Packit d14fb6
 * Redistribution of source code must retain the above copyright
Packit d14fb6
 * notice, this list of conditions and the following disclaimer.
Packit d14fb6
 * 
Packit d14fb6
 * Redistribution in binary form must reproduce the above copyright
Packit d14fb6
 * notice, this list of conditions and the following disclaimer in the
Packit d14fb6
 * documentation and/or other materials provided with the distribution.
Packit d14fb6
 * 
Packit d14fb6
 * Neither the name of Sun Microsystems, Inc. or the names of
Packit d14fb6
 * contributors may be used to endorse or promote products derived
Packit d14fb6
 * from this software without specific prior written permission.
Packit d14fb6
 * 
Packit d14fb6
 * This software is provided "AS IS," without a warranty of any kind.
Packit d14fb6
 * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES,
Packit d14fb6
 * INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A
Packit d14fb6
 * PARTICULAR PURPOSE OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED.
Packit d14fb6
 * SUN MICROSYSTEMS, INC. ("SUN") AND ITS LICENSORS SHALL NOT BE LIABLE
Packit d14fb6
 * FOR ANY DAMAGES SUFFERED BY LICENSEE AS A RESULT OF USING, MODIFYING
Packit d14fb6
 * OR DISTRIBUTING THIS SOFTWARE OR ITS DERIVATIVES.  IN NO EVENT WILL
Packit d14fb6
 * SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST REVENUE, PROFIT OR DATA,
Packit d14fb6
 * OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, INCIDENTAL OR
Packit d14fb6
 * PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY OF
Packit d14fb6
 * LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE,
Packit d14fb6
 * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
Packit d14fb6
 */
Packit d14fb6
Packit d14fb6
Packit d14fb6
/*
Packit d14fb6
 * interface routines between ipmitool and the bmc kernel driver
Packit d14fb6
 */
Packit d14fb6
Packit d14fb6
#include <stdio.h>
Packit d14fb6
#include <fcntl.h>
Packit d14fb6
#include <unistd.h>
Packit d14fb6
#include <sys/ioctl.h>
Packit d14fb6
#include <errno.h>
Packit d14fb6
#include <stdlib.h>
Packit d14fb6
#include <string.h>
Packit d14fb6
#include <sys/types.h>
Packit d14fb6
#include <sys/stropts.h>
Packit d14fb6
#include <stddef.h>
Packit d14fb6
#include <stropts.h>
Packit d14fb6
Packit d14fb6
#include <ipmitool/ipmi.h>
Packit d14fb6
#include <ipmitool/ipmi_intf.h>
Packit d14fb6
#include "bmc_intf.h"
Packit d14fb6
Packit d14fb6
#include "bmc.h"
Packit d14fb6
Packit d14fb6
static int	curr_seq;
Packit d14fb6
static int bmc_method(int fd, int *if_type);
Packit d14fb6
struct ipmi_rs *(*sendrecv_fn)(struct ipmi_intf *, struct ipmi_rq *) = NULL;
Packit d14fb6
extern int	verbose;
Packit d14fb6
Packit d14fb6
static void dump_request(bmc_req_t *request);
Packit d14fb6
static void dump_response(bmc_rsp_t *response);
Packit d14fb6
static struct ipmi_rs *ipmi_bmc_send_cmd_ioctl(struct ipmi_intf *intf,
Packit d14fb6
	struct ipmi_rq *req);
Packit d14fb6
static struct ipmi_rs *ipmi_bmc_send_cmd_putmsg(struct ipmi_intf *intf,
Packit d14fb6
	struct ipmi_rq *req);
Packit d14fb6
Packit d14fb6
#define	MESSAGE_BUFSIZE 1024
Packit d14fb6
Packit d14fb6
struct ipmi_intf ipmi_bmc_intf = {
Packit d14fb6
	name:		"bmc",
Packit d14fb6
	desc:		"IPMI v2.0 BMC interface",
Packit d14fb6
	open:		ipmi_bmc_open,
Packit d14fb6
	close:		ipmi_bmc_close,
Packit d14fb6
	sendrecv:	ipmi_bmc_send_cmd};
Packit d14fb6
Packit d14fb6
void
Packit d14fb6
ipmi_bmc_close(struct ipmi_intf *intf)
Packit d14fb6
{
Packit d14fb6
	if (intf && intf->fd >= 0)
Packit d14fb6
		close(intf->fd);
Packit d14fb6
Packit d14fb6
	intf->opened = 0;
Packit d14fb6
	intf->manufacturer_id = IPMI_OEM_UNKNOWN;
Packit d14fb6
	intf->fd = -1;
Packit d14fb6
}
Packit d14fb6
Packit d14fb6
int
Packit d14fb6
ipmi_bmc_open(struct ipmi_intf *intf)
Packit d14fb6
{
Packit d14fb6
	int method;
Packit d14fb6
Packit d14fb6
	if (!intf)
Packit d14fb6
                return -1;
Packit d14fb6
Packit d14fb6
	/* Open local device */
Packit d14fb6
	intf->fd = open(BMC_DEV, O_RDWR);
Packit d14fb6
Packit d14fb6
	if (intf->fd <= 0) {
Packit d14fb6
		perror("Could not open bmc device");
Packit d14fb6
		return (-1);
Packit d14fb6
	}
Packit d14fb6
	curr_seq = 0;
Packit d14fb6
Packit d14fb6
	intf->opened = 1;
Packit d14fb6
Packit d14fb6
	if (bmc_method(intf->fd, &method) < 0) {
Packit d14fb6
		perror("Could not determine bmc messaging interface");
Packit d14fb6
		return (-1);
Packit d14fb6
	}
Packit d14fb6
Packit d14fb6
	sendrecv_fn = (method == BMC_PUTMSG_METHOD) ?
Packit d14fb6
	    ipmi_bmc_send_cmd_putmsg : ipmi_bmc_send_cmd_ioctl;
Packit d14fb6
Packit d14fb6
	intf->manufacturer_id = ipmi_get_oem(intf);
Packit d14fb6
	return (intf->fd);
Packit d14fb6
}
Packit d14fb6
Packit d14fb6
struct ipmi_rs *
Packit d14fb6
ipmi_bmc_send_cmd(struct ipmi_intf *intf, struct ipmi_rq *req)
Packit d14fb6
{
Packit d14fb6
	/* If not already opened open the device or network connection */
Packit d14fb6
	if (!intf->opened && intf->open && intf->open(intf) < 0)
Packit d14fb6
		return NULL;
Packit d14fb6
Packit d14fb6
	/* sendrecv_fn cannot be NULL at this point */
Packit d14fb6
	return ((*sendrecv_fn)(intf, req));
Packit d14fb6
}
Packit d14fb6
Packit d14fb6
static struct ipmi_rs *
Packit d14fb6
ipmi_bmc_send_cmd_ioctl(struct ipmi_intf *intf, struct ipmi_rq *req)
Packit d14fb6
{
Packit d14fb6
	struct strioctl istr;
Packit d14fb6
	static struct bmc_reqrsp reqrsp;
Packit d14fb6
	static struct ipmi_rs rsp;
Packit d14fb6
Packit d14fb6
	memset(&reqrsp, 0, sizeof (reqrsp));
Packit d14fb6
	reqrsp.req.fn = req->msg.netfn;
Packit d14fb6
	reqrsp.req.lun = 0;
Packit d14fb6
	reqrsp.req.cmd = req->msg.cmd;
Packit d14fb6
	reqrsp.req.datalength = req->msg.data_len;
Packit d14fb6
	memcpy(reqrsp.req.data, req->msg.data, req->msg.data_len);
Packit d14fb6
	reqrsp.rsp.datalength = RECV_MAX_PAYLOAD_SIZE;
Packit d14fb6
Packit d14fb6
	istr.ic_cmd = IOCTL_IPMI_KCS_ACTION;
Packit d14fb6
	istr.ic_timout = 0;
Packit d14fb6
	istr.ic_dp = (char *)&reqrsp;
Packit d14fb6
	istr.ic_len = sizeof (struct bmc_reqrsp);
Packit d14fb6
Packit d14fb6
	if (verbose) {
Packit d14fb6
		printf("--\n");
Packit d14fb6
		dump_request(&reqrsp.req);
Packit d14fb6
		printf("--\n");
Packit d14fb6
	}
Packit d14fb6
Packit d14fb6
	if (ioctl(intf->fd, I_STR, &istr) < 0) {
Packit d14fb6
		perror("BMC IOCTL: I_STR");
Packit d14fb6
		return (NULL);
Packit d14fb6
	}
Packit d14fb6
Packit d14fb6
	if (verbose > 2) {
Packit d14fb6
		dump_response(&reqrsp.rsp);
Packit d14fb6
		printf("--\n");
Packit d14fb6
	}
Packit d14fb6
Packit d14fb6
	memset(&rsp, 0, sizeof (struct ipmi_rs));
Packit d14fb6
	rsp.ccode = reqrsp.rsp.ccode;
Packit d14fb6
	rsp.data_len = reqrsp.rsp.datalength;
Packit d14fb6
Packit d14fb6
	/* Decrement for sizeof lun, cmd and ccode */
Packit d14fb6
	rsp.data_len -= 3;
Packit d14fb6
Packit d14fb6
	if (!rsp.ccode && (rsp.data_len > 0))
Packit d14fb6
		memcpy(rsp.data, reqrsp.rsp.data, rsp.data_len);
Packit d14fb6
Packit d14fb6
	return (&rsp;;
Packit d14fb6
}
Packit d14fb6
Packit d14fb6
static struct ipmi_rs *
Packit d14fb6
ipmi_bmc_send_cmd_putmsg(struct ipmi_intf *intf, struct ipmi_rq *req)
Packit d14fb6
{
Packit d14fb6
	struct strbuf sb;
Packit d14fb6
	int flags = 0;
Packit d14fb6
	static uint32_t msg_seq = 0;
Packit d14fb6
Packit d14fb6
	/*
Packit d14fb6
	 * The length of the message structure is equal to the size of the
Packit d14fb6
	 * bmc_req_t structure, PLUS any additional data space in excess of
Packit d14fb6
	 * the data space already reserved in the data member + <n> for
Packit d14fb6
	 * the rest of the members in the bmc_msg_t structure.
Packit d14fb6
	 */
Packit d14fb6
	int msgsz = offsetof(bmc_msg_t, msg) + sizeof(bmc_req_t) +
Packit d14fb6
		((req->msg.data_len > SEND_MAX_PAYLOAD_SIZE) ?
Packit d14fb6
			(req->msg.data_len - SEND_MAX_PAYLOAD_SIZE) : 0);
Packit d14fb6
	bmc_msg_t *msg = malloc(msgsz);
Packit d14fb6
	bmc_req_t *request = (bmc_req_t *)&msg->msg[0];
Packit d14fb6
	bmc_rsp_t *response;
Packit d14fb6
	static struct ipmi_rs rsp;
Packit d14fb6
	struct ipmi_rs *ret = NULL;
Packit d14fb6
Packit d14fb6
	msg->m_type = BMC_MSG_REQUEST;
Packit d14fb6
	msg->m_id = msg_seq++;
Packit d14fb6
	request->fn = req->msg.netfn;
Packit d14fb6
	request->lun = 0;
Packit d14fb6
	request->cmd = req->msg.cmd;
Packit d14fb6
	request->datalength = req->msg.data_len;
Packit d14fb6
	memcpy(request->data, req->msg.data, req->msg.data_len);
Packit d14fb6
Packit d14fb6
	sb.len = msgsz;
Packit d14fb6
	sb.buf = (unsigned char *)msg;
Packit d14fb6
Packit d14fb6
	if (verbose) {
Packit d14fb6
		printf("--\n");
Packit d14fb6
		dump_request(request);
Packit d14fb6
		printf("--\n");
Packit d14fb6
	}
Packit d14fb6
Packit d14fb6
	if (putmsg(intf->fd, NULL, &sb, 0) < 0) {
Packit d14fb6
		perror("BMC putmsg: ");
Packit d14fb6
		free(msg);
Packit d14fb6
		msg = NULL;
Packit d14fb6
		return (NULL);
Packit d14fb6
	}
Packit d14fb6
Packit d14fb6
	free(msg);
Packit d14fb6
	msg = NULL;
Packit d14fb6
Packit d14fb6
	sb.buf = malloc(MESSAGE_BUFSIZE);
Packit d14fb6
	sb.maxlen = MESSAGE_BUFSIZE;
Packit d14fb6
Packit d14fb6
	if (getmsg(intf->fd, NULL, &sb, &flags) < 0) {
Packit d14fb6
		perror("BMC getmsg: ");
Packit d14fb6
		free(sb.buf);
Packit d14fb6
		sb.buf = NULL;
Packit d14fb6
		return (NULL);
Packit d14fb6
	}
Packit d14fb6
Packit d14fb6
	msg = (bmc_msg_t *)sb.buf;
Packit d14fb6
Packit d14fb6
	if (verbose > 3) {
Packit d14fb6
		printf("Got msg (id 0x%x) type 0x%x\n", msg->m_id, msg->m_type);
Packit d14fb6
	}
Packit d14fb6
Packit d14fb6
Packit d14fb6
	/* Did we get an error back from the stream? */
Packit d14fb6
	switch (msg->m_type) {
Packit d14fb6
Packit d14fb6
	case BMC_MSG_RESPONSE:
Packit d14fb6
		response = (bmc_rsp_t *)&msg->msg[0];
Packit d14fb6
Packit d14fb6
		if (verbose > 2) {
Packit d14fb6
			dump_response(response);
Packit d14fb6
			printf("--\n");
Packit d14fb6
		}
Packit d14fb6
Packit d14fb6
		memset(&rsp, 0, sizeof (struct ipmi_rs));
Packit d14fb6
		rsp.ccode = response->ccode;
Packit d14fb6
		rsp.data_len = response->datalength;
Packit d14fb6
Packit d14fb6
		if (!rsp.ccode && (rsp.data_len > 0))
Packit d14fb6
			memcpy(rsp.data, response->data, rsp.data_len);
Packit d14fb6
Packit d14fb6
		ret = &rsp;
Packit d14fb6
		break;
Packit d14fb6
Packit d14fb6
	case BMC_MSG_ERROR:
Packit d14fb6
		/* In case of an error, msg->msg[0] has the error code */
Packit d14fb6
		printf("bmc_send_cmd: %s\n", strerror(msg->msg[0]));
Packit d14fb6
		break;
Packit d14fb6
Packit d14fb6
	}
Packit d14fb6
	
Packit d14fb6
	free(sb.buf);
Packit d14fb6
	sb.buf = NULL;
Packit d14fb6
	return (ret);
Packit d14fb6
}
Packit d14fb6
Packit d14fb6
/*
Packit d14fb6
 * Determine which interface to use.  Returns the interface method
Packit d14fb6
 * to use.
Packit d14fb6
 */
Packit d14fb6
static int
Packit d14fb6
bmc_method(int fd, int *if_type)
Packit d14fb6
{
Packit d14fb6
	struct strioctl istr;
Packit d14fb6
	int retval = 0;
Packit d14fb6
	uint8_t method = BMC_PUTMSG_METHOD;
Packit d14fb6
Packit d14fb6
	istr.ic_cmd = IOCTL_IPMI_INTERFACE_METHOD;
Packit d14fb6
	istr.ic_timout = 0;
Packit d14fb6
	istr.ic_dp = (uint8_t *)&method;
Packit d14fb6
	istr.ic_len = 1;
Packit d14fb6
Packit d14fb6
	/*
Packit d14fb6
	 * If the ioctl doesn't exist, we should get an EINVAL back.
Packit d14fb6
	 * Bail out on any other error.
Packit d14fb6
	 */
Packit d14fb6
	if (ioctl(fd, I_STR, &istr) < 0) {
Packit d14fb6
Packit d14fb6
		if (errno != EINVAL)
Packit d14fb6
			retval = -1;
Packit d14fb6
		else
Packit d14fb6
			method = BMC_IOCTL_METHOD;
Packit d14fb6
	}
Packit d14fb6
Packit d14fb6
	if (retval == 0)
Packit d14fb6
		*if_type = method;
Packit d14fb6
Packit d14fb6
	return (retval);
Packit d14fb6
}
Packit d14fb6
Packit d14fb6
static void
Packit d14fb6
dump_request(bmc_req_t *request)
Packit d14fb6
{
Packit d14fb6
	int i;
Packit d14fb6
Packit d14fb6
	printf("BMC req.fn         : 0x%x\n", request->fn);
Packit d14fb6
	printf("BMC req.lun        : 0x%x\n", request->lun);
Packit d14fb6
	printf("BMC req.cmd        : 0x%x\n", request->cmd);
Packit d14fb6
	printf("BMC req.datalength : 0x%x\n", request->datalength);
Packit d14fb6
	printf("BMC req.data       : ");
Packit d14fb6
Packit d14fb6
	if (request->datalength > 0) {
Packit d14fb6
		for (i = 0; i < request->datalength; i++)
Packit d14fb6
			printf("0x%x ", request->data[i]);
Packit d14fb6
	} else {
Packit d14fb6
		printf("<NONE>");
Packit d14fb6
	}
Packit d14fb6
	printf("\n");
Packit d14fb6
}
Packit d14fb6
Packit d14fb6
static void
Packit d14fb6
dump_response(bmc_rsp_t *response)
Packit d14fb6
{
Packit d14fb6
	int i;
Packit d14fb6
Packit d14fb6
	printf("BMC rsp.fn         : 0x%x\n", response->fn);
Packit d14fb6
	printf("BMC rsp.lun        : 0x%x\n", response->lun);
Packit d14fb6
	printf("BMC rsp.cmd        : 0x%x\n", response->cmd);
Packit d14fb6
	printf("BMC rsp.ccode      : 0x%x\n", response->ccode);
Packit d14fb6
	printf("BMC rsp.datalength : 0x%x\n", response->datalength);
Packit d14fb6
	printf("BMC rsp.data       : ");
Packit d14fb6
Packit d14fb6
	if (response->datalength > 0) {
Packit d14fb6
		for (i = 0; i < response->datalength; i++)
Packit d14fb6
			printf("0x%x ", response->data[i]);
Packit d14fb6
	} else {
Packit d14fb6
		printf("<NONE>");
Packit d14fb6
	}
Packit d14fb6
	printf("\n");
Packit d14fb6
}