谈到BMC,绕不开的是ipmi,ipmi是BMC通过系统总线与其它部件进行通信,而BMC中支持IPMI的硬件通道有4个,参见我的上一篇博文,ipmi之kcs。Open BMC开发系列(八)ipmi之kcs通道_大余里,大余的干货分享-CSDN博客
ipmi的系统交互图如下:
一、什么是IPMI
IPMI就是智能平台管理接口(Intelligent Platform Management Interface)原本是一种Intel架构的企业系统的周边设备所采用的一种工业标准。IPMI亦是一个开放的免费标准,用户无需支付额外的费用即可使用此标准。
IPMI 能够横跨不同的操作系统、固件和硬件平台,可以智能的监视、控制和自动回报大量服务器的运作状况,以降低服务器系统成本。
二、IPMI的协议报文
IPMI的协议报文并不复杂,具体如下:
2.1 请求报文
我们主要关注的字段有三个:
网络功能码:netFn - 这个通常用于查找目标设备模块,类似网络路由的功能
命令字:command - 这个交互的具体指令类型。
数据:发送的数据,即交互消息体。
2.2 响应报文
响应消息我们
网络功能码:netFn - 这个通常用于查找目标设备模块,类似网络路由的功能
命令字:command - 这个交互的具体指令类型。
完成码:Completion Code 这个相当返回值
数据:返回的数据
三、IPMI的编码
open bmc的IPMI编码并不困难,再主框架下,主要分两部分:
1,定义好对应的ipmi的netfun和command,根据ipmi协议报文定义来回的交互报文格式。
例如:我需要定义BIOS与BMC的交互报文,基本消息流程如下:
定义send消息的请求:
netfuc:0x3c comand 0x01
data:
第一个字节:xxx
第二个字节:xxx
...
第m个字节以后:字符消息
2,注册IPMI的netFun和command以及对那个的回调函数。
再代码分支phosphor-ipmi-host中的sensorhandle.cpp中的
void register_netfn_sen_functions()
{
// <Platform Event Message>
ipmi_register_callback(NETFUN_SENSOR, IPMI_CMD_PLATFORM_EVENT, nullptr,
ipmicmdPlatformEvent, PRIVILEGE_OPERATOR);
...
//新增send msg的注册函数
ipmi_register_callback(ipmi::prioOpenBmcBase, 0x3c, 0x01,
ipmi::Privilege::User, ipmiHandleSendMsg)
}
3,填充对应的ipmi的具体回调内容。
// 函数参数需要根据请求消息的数据类型进行定义,比如说第一个字段是1个字节定义uinn8_t
// 第二个字段是4个字节,可以定义uint32_t
// 如果出现变成的字段定义vec<uint8_t>即可
// 返回值的类型参数也是根据定义返回消息的数据类型进行匹配,
// 数据类型一定要严格匹配,否则bmc会找不到对应回调函数,导致消息无法处理
ipmi::RspType<uint8_t > ipmiHandleSendMsg(
uint8_t first,
uint32_t second,
...// 根据数据格式填充寒素参数
std::vector<uint8_t> vecMailData)
{
// todo 处理send消息。
}
IPMI还有许多细节,但对于BMC的应用层而言,搞懂这些就能开始玩IPMI,别把IPMI想得太复杂,自己吓自己,没啥东西。上手还是可以很快的,具体的代码逻辑,可以有空慢慢撸。
最后:
点赞是美德,
关注是缘分,
收藏是肯定,
打赏你随意,
你的鼓励是我世界善的一部分,爱你们!