Refactoring of IPMI Transport Handler
diff --git a/transporthandler.C b/transporthandler.C
index 0fd25f9..3b4cf07 100644
--- a/transporthandler.C
+++ b/transporthandler.C
@@ -2,6 +2,7 @@
#include <string.h>
#include <stdint.h>
#include <arpa/inet.h>
+#include <string>
#include "ipmid-api.h"
#include "ipmid.H"
@@ -18,19 +19,109 @@
const char *obj = "/org/openbmc/NetworkManager/Interface";
const char *ifc = "org.openbmc.NetworkManager";
+const char *nwinterface = "eth0";
+
const int SIZE_MAC = 18; //xx:xx:xx:xx:xx:xx
-const int SIZE_LAN_PARM = 16; //xxx.xxx.xxx.xxx
-char cur_ipaddr [16] = "";
-char cur_netmask [16] = "";
-char cur_gateway [16] = "";
+char new_ipaddr [INET_ADDRSTRLEN] = "";
+char new_netmask [INET_ADDRSTRLEN] = "";
+char new_gateway [INET_ADDRSTRLEN] = "";
-char new_ipaddr [16] = "";
-char new_netmask [16] = "";
-char new_gateway [16] = "";
+const uint8_t SET_COMPLETE = 0;
+const uint8_t SET_IN_PROGRESS = 1;
+const uint8_t SET_COMMIT_WRITE = 2; //Optional
+const uint8_t SET_IN_PROGRESS_RESERVED = 3; //Reserved
+
+// Status of Set-In-Progress Parameter (# 0)
+uint8_t lan_set_in_progress = SET_COMPLETE;
+
+
void register_netfn_transport_functions() __attribute__((constructor));
+// Helper Function to get IP Address/NetMask/Gateway from Network Manager or Cache
+// based on Set-In-Progress State
+ipmi_ret_t getNetworkData(uint8_t lan_param, uint8_t * data)
+{
+ sd_bus *bus = ipmid_get_sd_bus_connection();
+ sd_bus_message *reply = NULL;
+ sd_bus_error error = SD_BUS_ERROR_NULL;
+ int family;
+ unsigned char prefixlen;
+ char* ipaddr = NULL;
+ unsigned long mask = 0xFFFFFFFF;
+ char* gateway = NULL;
+ int r = 0;
+ ipmi_ret_t rc = IPMI_CC_OK;
+
+ r = sd_bus_call_method(bus, app, obj, ifc, "GetAddress4", &error,
+ &reply, "s", nwinterface);
+ if(r < 0)
+ {
+ fprintf(stderr, "Failed to call Get Method: %s\n", strerror(-r));
+ rc = IPMI_CC_UNSPECIFIED_ERROR;
+ goto cleanup;
+ }
+
+ r = sd_bus_message_read(reply, "iyss", &family, &prefixlen, &ipaddr, &gateway);
+ if(r < 0)
+ {
+ fprintf(stderr, "Failed to get a response: %s\n", strerror(-rc));
+ rc = IPMI_CC_RESPONSE_ERROR;
+ goto cleanup;
+ }
+
+ printf("N/W data from HW %s:%d:%s:%s\n", family==AF_INET?"IPv4":"IPv6", prefixlen, ipaddr,gateway);
+ printf("N/W data from Cache: %s:%s:%s\n", new_ipaddr, new_netmask, new_gateway);
+
+ if(lan_param == LAN_PARM_IP)
+ {
+ if(lan_set_in_progress == SET_COMPLETE)
+ {
+ std::string ipaddrstr(ipaddr);
+ inet_pton(AF_INET, ipaddrstr.c_str(),(void *)data);
+ }
+ else if(lan_set_in_progress == SET_IN_PROGRESS)
+ {
+ inet_pton(AF_INET, new_ipaddr, (void *)data);
+ }
+ }
+ else if(lan_param == LAN_PARM_SUBNET)
+ {
+ if(lan_set_in_progress == SET_COMPLETE)
+ {
+ mask = htonl(mask<<(32-prefixlen));
+ memcpy(data, &mask, 4);
+ }
+ else if(lan_set_in_progress == SET_IN_PROGRESS)
+ {
+ inet_pton(AF_INET, new_netmask, (void *)data);
+ }
+ }
+ else if(lan_param == LAN_PARM_GATEWAY)
+ {
+ if(lan_set_in_progress == SET_COMPLETE)
+ {
+ std::string gatewaystr(gateway);
+ inet_pton(AF_INET, gatewaystr.c_str(), (void *)data);
+ }
+ else if(lan_set_in_progress == SET_IN_PROGRESS)
+ {
+ inet_pton(AF_INET, new_gateway,(void *)data);
+ }
+ }
+ else
+ {
+ rc = IPMI_CC_PARM_OUT_OF_RANGE;
+ }
+
+cleanup:
+ sd_bus_error_free(&error);
+ reply = sd_bus_message_unref(reply);
+
+ return rc;
+}
+
ipmi_ret_t ipmi_transport_wildcard(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
ipmi_request_t request, ipmi_response_t response,
ipmi_data_len_t data_len, ipmi_context_t context)
@@ -69,12 +160,12 @@
if (reqptr->parameter == LAN_PARM_IP)
{
- snprintf(new_ipaddr, SIZE_LAN_PARM, "%d.%d.%d.%d",
+ snprintf(new_ipaddr, INET_ADDRSTRLEN, "%d.%d.%d.%d",
reqptr->data[0], reqptr->data[1], reqptr->data[2], reqptr->data[3]);
}
else if (reqptr->parameter == LAN_PARM_MAC)
{
- char mac[SIZE_MAC];
+ char mac[SIZE_MAC];
snprintf(mac, SIZE_MAC, "%02x:%02x:%02x:%02x:%02x:%02x",
reqptr->data[0],
@@ -84,76 +175,62 @@
reqptr->data[4],
reqptr->data[5]);
- r = sd_bus_call_method(bus,app,obj,ifc,"SetHwAddress",
- &error, &reply, "s", mac);
- if (r < 0) {
- fprintf(stderr, "Failed to call method: %s\n", strerror(-r));
+ r = sd_bus_call_method(bus, app, obj, ifc, "SetHwAddress", &error,
+ &reply, "ss", nwinterface, mac);
+ if(r < 0)
+ {
+ fprintf(stderr, "Failed to call the method: %s\n", strerror(-r));
+ rc = IPMI_CC_UNSPECIFIED_ERROR;
}
}
else if (reqptr->parameter == LAN_PARM_SUBNET)
{
- snprintf(new_netmask, SIZE_LAN_PARM, "%d.%d.%d.%d",
+ snprintf(new_netmask, INET_ADDRSTRLEN, "%d.%d.%d.%d",
reqptr->data[0], reqptr->data[1], reqptr->data[2], reqptr->data[3]);
}
else if (reqptr->parameter == LAN_PARM_GATEWAY)
{
- snprintf(new_gateway, SIZE_LAN_PARM, "%d.%d.%d.%d",
+ snprintf(new_gateway, INET_ADDRSTRLEN, "%d.%d.%d.%d",
reqptr->data[0], reqptr->data[1], reqptr->data[2], reqptr->data[3]);
}
- else if (reqptr->parameter == LAN_PARM_INPROGRESS) // Apply config
+ else if (reqptr->parameter == LAN_PARM_INPROGRESS)
{
- if (!strcmp(new_ipaddr, "") || !strcmp (new_netmask, "") || !strcmp (new_gateway, ""))
+ if(reqptr->data[0] == SET_COMPLETE) // Set Complete
{
- fprintf(stderr,"ERROR: Incomplete LAN Parameters\n");
- return -1;
- }
+ lan_set_in_progress = SET_COMPLETE;
+ // Apply the IP settings once IP Address, Netmask and Gateway is set
+ if (!strcmp(new_ipaddr, "") || !strcmp (new_netmask, "") || !strcmp (new_gateway, ""))
+ {
+ printf("ERROR: Incomplete LAN Parameters\n");
+ }
+ else
+ {
- if (strcmp(cur_ipaddr, ""))
- {
- r = sd_bus_call_method(bus, // On the System Bus
- app, // Service to contact
- obj, // Object path
- ifc, // Interface name
- "DelAddress4", // Method to be called
- &error, // object to return error
- &reply, // Response message on success
- "ssss", // input message (dev,ip,nm,gw)
- "eth0",
- cur_ipaddr,
- cur_netmask,
- cur_gateway);
+ r = sd_bus_call_method(bus, // On the System Bus
+ app, // Service to contact
+ obj, // Object path
+ ifc, // Interface name
+ "SetAddress4", // Method to be called
+ &error, // object to return error
+ &reply, // Response message on success
+ "ssss", // input message (Interface, IP Address, Netmask, Gateway)
+ nwinterface, // eth0
+ new_ipaddr,
+ new_netmask,
+ new_gateway);
+ if(r < 0)
+ {
+ fprintf(stderr, "Failed to set network data %s:%s:%s %s\n", new_ipaddr, new_netmask, new_gateway, error.message);
+ rc = IPMI_CC_UNSPECIFIED_ERROR;
+ }
+ memset(new_ipaddr, 0, INET_ADDRSTRLEN);
+ memset(new_netmask, 0, INET_ADDRSTRLEN);
+ memset(new_gateway, 0, INET_ADDRSTRLEN);
+ }
}
-
- if(r < 0)
+ else if(reqptr->data[0] == SET_IN_PROGRESS) // Set In Progress
{
- fprintf(stderr, "Failed to remove existing IP %s: %s\n", cur_ipaddr, error.message);
- goto finish;
- }
-
- sd_bus_error_free(&error);
- reply = sd_bus_message_unref(reply);
-
- r = sd_bus_call_method(bus, // On the System Bus
- app, // Service to contact
- obj, // Object path
- ifc, // Interface name
- "AddAddress4", // Method to be called
- &error, // object to return error
- &reply, // Response message on success
- "ssss", // input message (dev,ip,nm,gw)
- "eth0",
- new_ipaddr,
- new_netmask,
- new_gateway);
- if(r < 0)
- {
- fprintf(stderr, "Failed to set IP %s: %s\n", new_ipaddr, error.message);
- }
- else
- {
- strcpy (cur_ipaddr, new_ipaddr);
- strcpy (cur_netmask, new_netmask);
- strcpy (cur_gateway, new_gateway);
+ lan_set_in_progress = SET_IN_PROGRESS;
}
}
else
@@ -162,12 +239,10 @@
rc = IPMI_CC_PARM_NOT_SUPPORTED;
}
-finish:
- // Clenaup the resources allocated reply and error
sd_bus_error_free(&error);
reply = sd_bus_message_unref(reply);
- return (r < 0) ? -1 : rc;
+ return rc;
}
struct get_lan_t {
@@ -188,13 +263,7 @@
sd_bus_error error = SD_BUS_ERROR_NULL;
int r = 0;
const uint8_t current_revision = 0x11; // Current rev per IPMI Spec 2.0
-
- int family;
- unsigned char prefixlen;
- unsigned char scope;
- unsigned int flags;
- char *saddr = NULL;
- int i = 0;
+ int i = 0;
printf("IPMI GET_LAN\n");
@@ -213,108 +282,66 @@
if (reqptr->parameter == LAN_PARM_INPROGRESS)
{
- uint8_t buf[] = {current_revision,0};
+ uint8_t buf[] = {current_revision, lan_set_in_progress};
*data_len = sizeof(buf);
memcpy(response, &buf, *data_len);
- return IPMI_CC_OK;
}
else if (reqptr->parameter == LAN_PARM_AUTHSUPPORT)
{
uint8_t buf[] = {current_revision,0x04};
*data_len = sizeof(buf);
memcpy(response, &buf, *data_len);
- return IPMI_CC_OK;
}
else if (reqptr->parameter == LAN_PARM_AUTHENABLES)
{
uint8_t buf[] = {current_revision,0x04,0x04,0x04,0x04,0x04};
*data_len = sizeof(buf);
memcpy(response, &buf, *data_len);
- return IPMI_CC_OK;
}
- else if (reqptr->parameter == LAN_PARM_IP)
+ else if ((reqptr->parameter == LAN_PARM_IP) || (reqptr->parameter == LAN_PARM_SUBNET) || (reqptr->parameter == LAN_PARM_GATEWAY))
{
- const char* device = "eth0";
- uint8_t buf[5]; // Size of expected IPMI response msg
+ uint8_t buf[5];
- r = sd_bus_call_method(bus,app,obj,ifc,"GetAddress4",
- &error, &reply, "s", device);
- if (r < 0)
+ *data_len = sizeof(current_revision);
+ memcpy(buf, ¤t_revision, *data_len);
+
+ if(getNetworkData(reqptr->parameter, &buf[1]) == IPMI_CC_OK)
{
- fprintf(stderr, "Failed to call method: %s\n", strerror(-r));
- rc = -1;
- goto finish;
+ *data_len = sizeof(buf);
+ memcpy(response, &buf, *data_len);
}
- r = sd_bus_message_enter_container (reply, 'a', "(iyyus)");
- if(r < 0)
+ else
{
- fprintf(stderr, "Failed to parse response message:[%s]\n", strerror(-rc));
- rc = -1;
- goto finish;
+ rc = IPMI_CC_UNSPECIFIED_ERROR;
}
- r = sd_bus_message_read(reply, "(iyyus)", &family, &prefixlen, &scope, &flags, &saddr);
- if (r < 0)
- {
- fprintf(stderr, "Failed to receive response: %s\n", strerror(-r));
- rc = -1;
- goto finish;
- }
-
- printf("%s:%d:%d:%d:%s\n", family==AF_INET?"IPv4":"IPv6", prefixlen, scope, flags, saddr);
-
- memcpy((void*)&buf[0], ¤t_revision, 1);
-
- // Parse IP address
- char *tokptr = NULL;
- char* digit = strtok_r(saddr, ".", &tokptr);
- if (digit == NULL)
- {
- fprintf(stderr, "Unexpected IP format: %s", saddr);
- rc = IPMI_CC_RESPONSE_ERROR;
- goto finish;
- }
- i = 0;
- while (digit != NULL)
- {
- int resp_byte = strtoul(digit, NULL, 10);
- memcpy((void*)&buf[i+1], &resp_byte, 1);
- i++;
- digit = strtok_r(NULL, ".", &tokptr);
- }
-
- *data_len = sizeof(buf);
- memcpy(response, &buf, *data_len);
-
- rc = IPMI_CC_OK;
}
else if (reqptr->parameter == LAN_PARM_MAC)
{
//string to parse: link/ether xx:xx:xx:xx:xx:xx
-
- const char* device = "eth0";
- uint8_t buf[7];
+ uint8_t buf[7];
char *eaddr1 = NULL;
- r = sd_bus_call_method(bus,app,obj,ifc,"GetHwAddress",
- &error, &reply, "s", device);
- if (r < 0)
+ r = sd_bus_call_method(bus, app, obj, ifc, "GetHwAddress", &error,
+ &reply, "s", nwinterface);
+ if(r < 0)
{
- fprintf(stderr, "Failed to call GetHwAddress: %s\n", strerror(-r));
- rc = -1;
- goto finish;
+ fprintf(stderr, "Failed to call Get Method: %s\n", strerror(-r));
+ rc = IPMI_CC_UNSPECIFIED_ERROR;
+ goto cleanup;
}
+
r = sd_bus_message_read(reply, "s", &eaddr1);
if (r < 0)
{
fprintf(stderr, "Failed to get a response: %s", strerror(-r));
- rc = IPMI_CC_RESPONSE_ERROR;
- goto finish;
+ rc = IPMI_CC_UNSPECIFIED_ERROR;
+ goto cleanup;
}
if (eaddr1 == NULL)
{
fprintf(stderr, "Failed to get a valid response: %s", strerror(-r));
- rc = IPMI_CC_RESPONSE_ERROR;
- goto finish;
+ rc = IPMI_CC_UNSPECIFIED_ERROR;
+ goto cleanup;
}
memcpy((void*)&buf[0], ¤t_revision, 1);
@@ -325,7 +352,7 @@
{
fprintf(stderr, "Unexpected MAC format: %s", eaddr1);
rc = IPMI_CC_RESPONSE_ERROR;
- goto finish;
+ goto cleanup;
}
i=0;
@@ -339,8 +366,6 @@
*data_len = sizeof(buf);
memcpy(response, &buf, *data_len);
-
- rc = IPMI_CC_OK;
}
else
{
@@ -348,7 +373,7 @@
rc = IPMI_CC_PARM_NOT_SUPPORTED;
}
-finish:
+cleanup:
sd_bus_error_free(&error);
reply = sd_bus_message_unref(reply);