#include <stddef.h>
#include <strings.h>
#include <libipmi.h>
#include "ipmi_impl.h"
typedef struct ipmi_cmd_lan_get_config {
DECL_BITFIELD3(
ilgc_number :4,
__reserved :3,
ilgc_revonly :1);
uint8_t ilgc_param;
uint8_t ilgc_set;
uint8_t ilgc_block;
} ipmi_cmd_lan_get_config_t;
typedef struct ipmi_cmd_lan_set_config {
DECL_BITFIELD2(
ilsc_number :4,
__reserved :4);
uint8_t ilsc_param;
uint8_t ilsc_data[18];
} ipmi_cmd_lan_set_config_t;
#define IPMI_LAN_SET_LEN(dlen) \
(offsetof(ipmi_cmd_lan_set_config_t, ilsc_data) + (dlen))
#define IPMI_LAN_PARAM_SET_IN_PROGRESS 0
#define IPMI_LAN_PARAM_IP_ADDR 3
#define IPMI_LAN_PARAM_IP_SOURCE 4
#define IPMI_LAN_PARAM_MAC_ADDR 5
#define IPMI_LAN_PARAM_SUBNET_MASK 6
#define IPMI_LAN_PARAM_GATEWAY_ADDR 12
#define IPMI_LAN_SET_COMPLETE 0x0
#define IPMI_LAN_SET_INPROGRESS 0x1
#define IPMI_LAN_SET_COMMIT 0x2
typedef struct ipmi_lan_entry {
int ile_param;
int ile_mask;
int ile_set;
int ile_block;
size_t ile_offset;
size_t ile_len;
} ipmi_lan_entry_t;
static ipmi_lan_entry_t ipmi_lan_table[] = {
{ IPMI_LAN_PARAM_IP_ADDR, IPMI_LAN_SET_IPADDR, 0, 0,
offsetof(ipmi_lan_config_t, ilc_ipaddr), sizeof (uint32_t) },
{ IPMI_LAN_PARAM_IP_SOURCE, IPMI_LAN_SET_IPADDR_SOURCE, 0, 0,
offsetof(ipmi_lan_config_t, ilc_ipaddr_source), sizeof (uint8_t) },
{ IPMI_LAN_PARAM_MAC_ADDR, IPMI_LAN_SET_MACADDR, 0, 0,
offsetof(ipmi_lan_config_t, ilc_macaddr), 6 * sizeof (uint8_t) },
{ IPMI_LAN_PARAM_SUBNET_MASK, IPMI_LAN_SET_SUBNET, 0, 0,
offsetof(ipmi_lan_config_t, ilc_subnet), sizeof (uint32_t) },
{ IPMI_LAN_PARAM_GATEWAY_ADDR, IPMI_LAN_SET_GATEWAY_ADDR, 0, 0,
offsetof(ipmi_lan_config_t, ilc_gateway_addr), sizeof (uint32_t) }
};
#define IPMI_LAN_NENTRIES \
(sizeof (ipmi_lan_table) / sizeof (ipmi_lan_table[0]))
static int
ipmi_lan_get_param(ipmi_handle_t *ihp, int channel, int param, int set,
int block, void *data, size_t len)
{
ipmi_cmd_t cmd, *rsp;
ipmi_cmd_lan_get_config_t lcmd = { 0 };
lcmd.ilgc_number = channel;
lcmd.ilgc_param = param;
lcmd.ilgc_set = set;
lcmd.ilgc_block = block;
cmd.ic_netfn = IPMI_NETFN_TRANSPORT;
cmd.ic_lun = 0;
cmd.ic_cmd = IPMI_CMD_GET_LAN_CONFIG;
cmd.ic_data = &lcmd;
cmd.ic_dlen = sizeof (lcmd);
if ((rsp = ipmi_send(ihp, &cmd)) == NULL) {
switch (ihp->ih_completion) {
case 0x80:
(void) ipmi_set_error(ihp, EIPMI_BADPARAM, NULL);
break;
}
return (-1);
}
if (rsp->ic_dlen < len + 1)
return (ipmi_set_error(ihp, EIPMI_BAD_RESPONSE_LENGTH, NULL));
bcopy((uint8_t *)rsp->ic_data + 1, data, len);
return (0);
}
int
ipmi_lan_get_config(ipmi_handle_t *ihp, int channel, ipmi_lan_config_t *cfgp)
{
uint8_t set;
int i;
ipmi_lan_entry_t *lep;
if (ipmi_lan_get_param(ihp, channel, IPMI_LAN_PARAM_SET_IN_PROGRESS, 0,
0, &set, sizeof (set)) != 0)
return (-1);
if (set & IPMI_LAN_SET_INPROGRESS)
cfgp->ilc_set_in_progress = B_TRUE;
else
cfgp->ilc_set_in_progress = B_FALSE;
for (i = 0; i < IPMI_LAN_NENTRIES; i++) {
lep = &ipmi_lan_table[i];
if (ipmi_lan_get_param(ihp, channel, lep->ile_param,
lep->ile_set, lep->ile_block,
(char *)cfgp + lep->ile_offset, lep->ile_len) != 0)
return (-1);
}
return (0);
}
static int
ipmi_lan_set_param(ipmi_handle_t *ihp, int channel, int param, void *data,
size_t len)
{
ipmi_cmd_t cmd;
ipmi_cmd_lan_set_config_t lcmd = { 0 };
lcmd.ilsc_number = channel;
lcmd.ilsc_param = param;
bcopy(data, lcmd.ilsc_data, len);
cmd.ic_netfn = IPMI_NETFN_TRANSPORT;
cmd.ic_lun = 0;
cmd.ic_cmd = IPMI_CMD_SET_LAN_CONFIG;
cmd.ic_data = &lcmd;
cmd.ic_dlen = IPMI_LAN_SET_LEN(len);
if (ipmi_send(ihp, &cmd) == NULL) {
switch (ihp->ih_completion) {
case 0x80:
(void) ipmi_set_error(ihp, EIPMI_BADPARAM, NULL);
break;
case 0x81:
(void) ipmi_set_error(ihp, EIPMI_BUSY, NULL);
break;
case 0x82:
(void) ipmi_set_error(ihp, EIPMI_READONLY, NULL);
break;
case 0x83:
(void) ipmi_set_error(ihp, EIPMI_WRITEONLY, NULL);
break;
}
return (-1);
}
return (0);
}
int
ipmi_lan_set_config(ipmi_handle_t *ihp, int channel, ipmi_lan_config_t *cfgp,
int mask)
{
uint8_t set;
int i;
ipmi_lan_entry_t *lep;
set = IPMI_LAN_SET_COMPLETE;
if (ipmi_lan_set_param(ihp, channel, IPMI_LAN_PARAM_SET_IN_PROGRESS,
&set, sizeof (set)) != 0)
return (-1);
set = IPMI_LAN_SET_INPROGRESS;
if (ipmi_lan_set_param(ihp, channel, IPMI_LAN_PARAM_SET_IN_PROGRESS,
&set, sizeof (set)) != 0)
return (-1);
for (i = 0; i < IPMI_LAN_NENTRIES; i++) {
lep = &ipmi_lan_table[i];
if (!(lep->ile_mask & mask))
continue;
if (ipmi_lan_set_param(ihp, channel, lep->ile_param,
(char *)cfgp + lep->ile_offset, lep->ile_len) != 0) {
if (mask != IPMI_LAN_SET_IPADDR_SOURCE ||
cfgp->ilc_ipaddr_source != IPMI_LAN_SRC_DHCP ||
ihp->ih_completion != 0xC3)
return (-1);
}
}
set = IPMI_LAN_SET_COMPLETE;
if (ipmi_lan_set_param(ihp, channel, IPMI_LAN_PARAM_SET_IN_PROGRESS,
&set, sizeof (set)) != 0)
return (-1);
return (0);
}