Skip to content

Commit

Permalink
ipmi: Add BMC firmware version to device tree
Browse files Browse the repository at this point in the history
BMC Get device ID command gives BMC firmware version details. Lets add this
to device tree. User space tools will use this information to display BMC
version details.

Stewart,
  I have added bmc information under /ibm,firmware-version node as its firmware
  version. But may be we should add new node (/bmc/firmware). So that we can
  keep BMC related information separately. Let me know your thoughts on this.

Signed-off-by: Vasant Hegde <hegdevasant@linux.vnet.ibm.com>
Signed-off-by: Stewart Smith <stewart@linux.ibm.com>
  • Loading branch information
Vasant Hegde authored and stewartsmith committed May 9, 2018
1 parent c5bff43 commit 13878e5
Show file tree
Hide file tree
Showing 4 changed files with 126 additions and 1 deletion.
2 changes: 1 addition & 1 deletion hw/ipmi/Makefile.inc
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
SUBDIRS += hw/ipmi

IPMI_OBJS = ipmi-rtc.o ipmi-power.o ipmi-fru.o ipmi-sel.o
IPMI_OBJS += ipmi-watchdog.o ipmi-sensor.o ipmi-attn.o
IPMI_OBJS += ipmi-watchdog.o ipmi-sensor.o ipmi-attn.o ipmi-info.o

IPMI = hw/ipmi/built-in.a
$(IPMI): $(IPMI_OBJS:%=hw/ipmi/%)
112 changes: 112 additions & 0 deletions hw/ipmi/ipmi-info.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,112 @@
/* Copyright 2018 IBM Corp.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
* implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

#include <device.h>
#include <skiboot.h>
#include <stdlib.h>
#include <ipmi.h>
#include <mem_region-malloc.h>
#include <opal.h>
#include <timebase.h>

/*
* Respones data from IPMI Get device ID command (As defined in
* Section 20.1 Get Device ID Command - IPMI standard spec).
*/
struct ipmi_dev_id {
uint8_t dev_id;
uint8_t dev_revision;
uint8_t fw_rev1;
uint8_t fw_rev2;
uint8_t ipmi_ver;
uint8_t add_dev_support;
uint8_t manufactur_id[3];
uint8_t product_id[2];
uint8_t aux_fw_rev[4];
};
static struct ipmi_dev_id *ipmi_dev_id;

/* Got response from BMC? */
static bool bmc_info_waiting = false;
static bool bmc_info_valid = false;

/* This will free ipmi_dev_id structure */
void ipmi_dt_add_bmc_info(void)
{
char buf[8];
struct dt_node *dt_fw_version;

while (bmc_info_waiting)
time_wait_ms(5);

if (!bmc_info_valid)
return;

dt_fw_version = dt_find_by_name(dt_root, "ibm,firmware-versions");
if (!dt_fw_version) {
free(ipmi_dev_id);
return;
}

memset(buf, 0, sizeof(buf));
snprintf(buf, sizeof(buf), "%x.%02x",
ipmi_dev_id->fw_rev1, ipmi_dev_id->fw_rev2);
dt_add_property_string(dt_fw_version, "bmc-firmware-version", buf);

free(ipmi_dev_id);
}

static void ipmi_get_bmc_info_resp(struct ipmi_msg *msg)
{
bmc_info_waiting = false;

if (msg->cc != IPMI_CC_NO_ERROR) {
prlog(PR_ERR, "IPMI: IPMI_BMC_GET_DEVICE_ID cmd returned error"
" [rc : 0x%x]\n", msg->data[0]);
return;
}

bmc_info_valid = true;
memcpy(ipmi_dev_id, msg->data, msg->resp_size);
ipmi_free_msg(msg);
}

int ipmi_get_bmc_info_request(void)
{
int rc;
struct ipmi_msg *msg;

ipmi_dev_id = zalloc(sizeof(struct ipmi_dev_id));
assert(ipmi_dev_id);

msg = ipmi_mkmsg(IPMI_DEFAULT_INTERFACE, IPMI_BMC_GET_DEVICE_ID,
ipmi_get_bmc_info_resp, NULL, NULL,
0, sizeof(struct ipmi_dev_id));
if (!msg)
return OPAL_NO_MEM;

msg->error = ipmi_get_bmc_info_resp;
prlog(PR_INFO, "IPMI: Requesting IPMI_BMC_GET_DEVICE_ID\n");
rc = ipmi_queue_msg(msg);
if (rc) {
prlog(PR_ERR, "IPMI: Failed to queue IPMI_BMC_GET_DEVICE_ID\n");
ipmi_free_msg(msg);
return rc;
}

bmc_info_waiting = true;
return rc;
}
7 changes: 7 additions & 0 deletions include/ipmi.h
Original file line number Diff line number Diff line change
Expand Up @@ -109,6 +109,7 @@
#define IPMI_GET_SEL_TIME IPMI_CODE(IPMI_NETFN_STORAGE, 0x48)
#define IPMI_SET_SEL_TIME IPMI_CODE(IPMI_NETFN_STORAGE, 0x49)
#define IPMI_CHASSIS_CONTROL IPMI_CODE(IPMI_NETFN_CHASSIS, 0x02)
#define IPMI_BMC_GET_DEVICE_ID IPMI_CODE(IPMI_NETFN_APP, 0x01)
#define IPMI_SET_POWER_STATE IPMI_CODE(IPMI_NETFN_APP, 0x06)
#define IPMI_GET_POWER_STATE IPMI_CODE(IPMI_NETFN_APP, 0x07)
#define IPMI_RESET_WDT IPMI_CODE(IPMI_NETFN_APP, 0x22)
Expand Down Expand Up @@ -278,4 +279,10 @@ int ipmi_set_boot_count(void);
/* Terminate immediate */
void __attribute__((noreturn)) ipmi_terminate(const char *msg);

/* Get BMC firmware info */
extern int ipmi_get_bmc_info_request(void);

/* Add BMC firmware info to device tree */
extern void ipmi_dt_add_bmc_info(void);

#endif
6 changes: 6 additions & 0 deletions platforms/astbmc/common.c
Original file line number Diff line number Diff line change
Expand Up @@ -137,6 +137,9 @@ void astbmc_init(void)
/* Preload PNOR VERSION section */
flash_fw_version_preload();

/* Request BMC information */
ipmi_get_bmc_info_request();

/* As soon as IPMI is up, inform BMC we are in "S0" */
ipmi_set_power_state(IPMI_PWR_SYS_S0_WORKING, IPMI_PWR_NOCHANGE);

Expand All @@ -150,6 +153,9 @@ void astbmc_init(void)

/* Add ibm,firmware-versions node */
flash_dt_add_fw_version();

/* Add BMC firmware info to device tree */
ipmi_dt_add_bmc_info();
}

int64_t astbmc_ipmi_power_down(uint64_t request)
Expand Down

0 comments on commit 13878e5

Please sign in to comment.