xref: /openbmc/phosphor-networkd/src/ncsi_util.cpp (revision b18114584743c85f3d6180a61d524edea595e46c)
1 #include "ncsi_util.hpp"
2 
3 #include <linux/ncsi.h>
4 #include <netlink/genl/ctrl.h>
5 #include <netlink/genl/genl.h>
6 #include <netlink/netlink.h>
7 
8 #include <phosphor-logging/lg2.hpp>
9 #include <stdplus/numeric/str.hpp>
10 #include <stdplus/str/buf.hpp>
11 
12 #include <iomanip>
13 #include <iostream>
14 #include <vector>
15 
16 namespace phosphor
17 {
18 namespace network
19 {
20 namespace ncsi
21 {
22 
23 using CallBack = int (*)(struct nl_msg* msg, void* arg);
24 
25 static stdplus::StrBuf toHexStr(std::span<const uint8_t> c) noexcept
26 {
27     stdplus::StrBuf ret;
28     if (c.empty())
29     {
30         return ret;
31     }
32     stdplus::IntToStr<16, uint8_t> its;
33     auto oit = ret.append(c.size() * 3 - 1);
34     auto cit = c.begin();
35     oit = its(oit, *cit++, 2);
36     for (; cit != c.end(); ++cit)
37     {
38         *oit++ = ' ';
39         oit = its(oit, *cit, 2);
40     }
41     return ret;
42 }
43 
44 namespace internal
45 {
46 
47 struct NCSIPacketHeader
48 {
49     uint8_t MCID;
50     uint8_t revision;
51     uint8_t reserved;
52     uint8_t id;
53     uint8_t type;
54     uint8_t channel;
55     uint16_t length;
56     uint32_t rsvd[2];
57 };
58 
59 class Command
60 {
61   public:
62     Command() = delete;
63     ~Command() = default;
64     Command(const Command&) = delete;
65     Command& operator=(const Command&) = delete;
66     Command(Command&&) = default;
67     Command& operator=(Command&&) = default;
68     Command(
69         int c, int nc = DEFAULT_VALUE,
70         std::span<const unsigned char> p = std::span<const unsigned char>()) :
71         cmd(c),
72         ncsi_cmd(nc), payload(p)
73     {}
74 
75     int cmd;
76     int ncsi_cmd;
77     std::span<const unsigned char> payload;
78 };
79 
80 using nlMsgPtr = std::unique_ptr<nl_msg, decltype(&::nlmsg_free)>;
81 using nlSocketPtr = std::unique_ptr<nl_sock, decltype(&::nl_socket_free)>;
82 
83 CallBack infoCallBack = [](struct nl_msg* msg, void* arg) {
84     using namespace phosphor::network::ncsi;
85     auto nlh = nlmsg_hdr(msg);
86 
87     struct nlattr* tb[NCSI_ATTR_MAX + 1] = {nullptr};
88     struct nla_policy ncsiPolicy[NCSI_ATTR_MAX + 1] = {
89         {NLA_UNSPEC, 0, 0}, {NLA_U32, 0, 0}, {NLA_NESTED, 0, 0},
90         {NLA_U32, 0, 0},    {NLA_U32, 0, 0},
91     };
92 
93     struct nlattr* packagetb[NCSI_PKG_ATTR_MAX + 1] = {nullptr};
94     struct nla_policy packagePolicy[NCSI_PKG_ATTR_MAX + 1] = {
95         {NLA_UNSPEC, 0, 0}, {NLA_NESTED, 0, 0}, {NLA_U32, 0, 0},
96         {NLA_FLAG, 0, 0},   {NLA_NESTED, 0, 0},
97     };
98 
99     struct nlattr* channeltb[NCSI_CHANNEL_ATTR_MAX + 1] = {nullptr};
100     struct nla_policy channelPolicy[NCSI_CHANNEL_ATTR_MAX + 1] = {
101         {NLA_UNSPEC, 0, 0}, {NLA_NESTED, 0, 0}, {NLA_U32, 0, 0},
102         {NLA_FLAG, 0, 0},   {NLA_NESTED, 0, 0}, {NLA_UNSPEC, 0, 0},
103     };
104 
105     *(int*)arg = 0;
106 
107     auto ret = genlmsg_parse(nlh, 0, tb, NCSI_ATTR_MAX, ncsiPolicy);
108     if (!tb[NCSI_ATTR_PACKAGE_LIST])
109     {
110         lg2::error("No Packages");
111         return -1;
112     }
113 
114     auto attrTgt = static_cast<nlattr*>(nla_data(tb[NCSI_ATTR_PACKAGE_LIST]));
115     if (!attrTgt)
116     {
117         lg2::error("Package list attribute is null");
118         return -1;
119     }
120 
121     auto rem = nla_len(tb[NCSI_ATTR_PACKAGE_LIST]);
122     nla_for_each_nested(attrTgt, tb[NCSI_ATTR_PACKAGE_LIST], rem)
123     {
124         ret = nla_parse_nested(packagetb, NCSI_PKG_ATTR_MAX, attrTgt,
125                                packagePolicy);
126         if (ret < 0)
127         {
128             lg2::error("Failed to parse package nested");
129             return -1;
130         }
131 
132         if (packagetb[NCSI_PKG_ATTR_ID])
133         {
134             auto attrID = nla_get_u32(packagetb[NCSI_PKG_ATTR_ID]);
135             lg2::debug("Package has id : {ATTR_ID}", "ATTR_ID", lg2::hex,
136                        attrID);
137         }
138         else
139         {
140             lg2::debug("Package with no id");
141         }
142 
143         if (packagetb[NCSI_PKG_ATTR_FORCED])
144         {
145             lg2::debug("This package is forced");
146         }
147 
148         auto channelListTarget = static_cast<nlattr*>(
149             nla_data(packagetb[NCSI_PKG_ATTR_CHANNEL_LIST]));
150 
151         auto channelrem = nla_len(packagetb[NCSI_PKG_ATTR_CHANNEL_LIST]);
152         nla_for_each_nested(channelListTarget,
153                             packagetb[NCSI_PKG_ATTR_CHANNEL_LIST], channelrem)
154         {
155             ret = nla_parse_nested(channeltb, NCSI_CHANNEL_ATTR_MAX,
156                                    channelListTarget, channelPolicy);
157             if (ret < 0)
158             {
159                 lg2::error("Failed to parse channel nested");
160                 return -1;
161             }
162 
163             if (channeltb[NCSI_CHANNEL_ATTR_ID])
164             {
165                 auto channel = nla_get_u32(channeltb[NCSI_CHANNEL_ATTR_ID]);
166                 if (channeltb[NCSI_CHANNEL_ATTR_ACTIVE])
167                 {
168                     lg2::debug("Channel Active : {CHANNEL}", "CHANNEL",
169                                lg2::hex, channel);
170                 }
171                 else
172                 {
173                     lg2::debug("Channel Not Active : {CHANNEL}", "CHANNEL",
174                                lg2::hex, channel);
175                 }
176 
177                 if (channeltb[NCSI_CHANNEL_ATTR_FORCED])
178                 {
179                     lg2::debug("Channel is forced");
180                 }
181             }
182             else
183             {
184                 lg2::debug("Channel with no ID");
185             }
186 
187             if (channeltb[NCSI_CHANNEL_ATTR_VERSION_MAJOR])
188             {
189                 auto major =
190                     nla_get_u32(channeltb[NCSI_CHANNEL_ATTR_VERSION_MAJOR]);
191                 lg2::debug("Channel Major Version : {CHANNEL_MAJOR_VERSION}",
192                            "CHANNEL_MAJOR_VERSION", lg2::hex, major);
193             }
194             if (channeltb[NCSI_CHANNEL_ATTR_VERSION_MINOR])
195             {
196                 auto minor =
197                     nla_get_u32(channeltb[NCSI_CHANNEL_ATTR_VERSION_MINOR]);
198                 lg2::debug("Channel Minor Version : {CHANNEL_MINOR_VERSION}",
199                            "CHANNEL_MINOR_VERSION", lg2::hex, minor);
200             }
201             if (channeltb[NCSI_CHANNEL_ATTR_VERSION_STR])
202             {
203                 auto str =
204                     nla_get_string(channeltb[NCSI_CHANNEL_ATTR_VERSION_STR]);
205                 lg2::debug("Channel Version Str : {CHANNEL_VERSION_STR}",
206                            "CHANNEL_VERSION_STR", str);
207             }
208             if (channeltb[NCSI_CHANNEL_ATTR_LINK_STATE])
209             {
210                 auto link =
211                     nla_get_u32(channeltb[NCSI_CHANNEL_ATTR_LINK_STATE]);
212                 lg2::debug("Channel Link State : {LINK_STATE}", "LINK_STATE",
213                            lg2::hex, link);
214             }
215             if (channeltb[NCSI_CHANNEL_ATTR_VLAN_LIST])
216             {
217                 lg2::debug("Active Vlan ids");
218                 auto vids = channeltb[NCSI_CHANNEL_ATTR_VLAN_LIST];
219                 auto vid = static_cast<nlattr*>(nla_data(vids));
220                 auto len = nla_len(vids);
221                 while (nla_ok(vid, len))
222                 {
223                     auto id = nla_get_u16(vid);
224                     lg2::debug("VID : {VLAN_ID}", "VLAN_ID", id);
225                     vid = nla_next(vid, &len);
226                 }
227             }
228         }
229     }
230     return (int)NL_SKIP;
231 };
232 
233 CallBack sendCallBack = [](struct nl_msg* msg, void* arg) {
234     using namespace phosphor::network::ncsi;
235     auto nlh = nlmsg_hdr(msg);
236     struct nlattr* tb[NCSI_ATTR_MAX + 1] = {nullptr};
237     static struct nla_policy ncsiPolicy[NCSI_ATTR_MAX + 1] = {
238         {NLA_UNSPEC, 0, 0}, {NLA_U32, 0, 0}, {NLA_NESTED, 0, 0},
239         {NLA_U32, 0, 0},    {NLA_U32, 0, 0}, {NLA_BINARY, 0, 0},
240         {NLA_FLAG, 0, 0},   {NLA_U32, 0, 0}, {NLA_U32, 0, 0},
241     };
242 
243     *(int*)arg = 0;
244 
245     auto ret = genlmsg_parse(nlh, 0, tb, NCSI_ATTR_MAX, ncsiPolicy);
246     if (ret)
247     {
248         lg2::error("Failed to parse package");
249         return ret;
250     }
251 
252     if (tb[NCSI_ATTR_DATA] == nullptr)
253     {
254         lg2::error("Response: No data");
255         return -1;
256     }
257 
258     auto data_len = nla_len(tb[NCSI_ATTR_DATA]) - sizeof(NCSIPacketHeader);
259     unsigned char* data = (unsigned char*)nla_data(tb[NCSI_ATTR_DATA]) +
260                           sizeof(NCSIPacketHeader);
261 
262     // Dump the response to stdout. Enhancement: option to save response data
263     auto str = toHexStr(std::span<const unsigned char>(data, data_len));
264     lg2::debug("Response {DATA_LEN} bytes: {DATA}", "DATA_LEN", data_len,
265                "DATA", str);
266 
267     return 0;
268 };
269 
270 int applyCmd(int ifindex, const Command& cmd, int package = DEFAULT_VALUE,
271              int channel = DEFAULT_VALUE, int flags = NONE,
272              CallBack function = nullptr)
273 {
274     int cb_ret = 0;
275     nlSocketPtr socket(nl_socket_alloc(), &::nl_socket_free);
276     if (socket == nullptr)
277     {
278         lg2::error("Unable to allocate memory for the socket");
279         return -ENOMEM;
280     }
281 
282     auto ret = genl_connect(socket.get());
283     if (ret < 0)
284     {
285         lg2::error("Failed to open the socket , RC : {RC}", "RC", ret);
286         return ret;
287     }
288 
289     auto driverID = genl_ctrl_resolve(socket.get(), "NCSI");
290     if (driverID < 0)
291     {
292         lg2::error("Failed to resolve, RC : {RC}", "RC", ret);
293         return driverID;
294     }
295 
296     nlMsgPtr msg(nlmsg_alloc(), &::nlmsg_free);
297     if (msg == nullptr)
298     {
299         lg2::error("Unable to allocate memory for the message");
300         return -ENOMEM;
301     }
302 
303     auto msgHdr = genlmsg_put(msg.get(), 0, 0, driverID, 0, flags, cmd.cmd, 0);
304     if (!msgHdr)
305     {
306         lg2::error("Unable to add the netlink headers , COMMAND : {COMMAND}",
307                    "COMMAND", cmd.cmd);
308         return -ENOMEM;
309     }
310 
311     if (package != DEFAULT_VALUE)
312     {
313         ret = nla_put_u32(msg.get(), ncsi_nl_attrs::NCSI_ATTR_PACKAGE_ID,
314                           package);
315         if (ret < 0)
316         {
317             lg2::error("Failed to set the attribute , RC : {RC} PACKAGE "
318                        "{PACKAGE}",
319                        "RC", ret, "PACKAGE", lg2::hex, package);
320             return ret;
321         }
322     }
323 
324     if (channel != DEFAULT_VALUE)
325     {
326         ret = nla_put_u32(msg.get(), ncsi_nl_attrs::NCSI_ATTR_CHANNEL_ID,
327                           channel);
328         if (ret < 0)
329         {
330             lg2::error("Failed to set the attribute , RC : {RC} CHANNEL : "
331                        "{CHANNEL}",
332                        "RC", ret, "CHANNEL", lg2::hex, channel);
333             return ret;
334         }
335     }
336 
337     ret = nla_put_u32(msg.get(), ncsi_nl_attrs::NCSI_ATTR_IFINDEX, ifindex);
338     if (ret < 0)
339     {
340         lg2::error("Failed to set the attribute , RC : {RC} INTERFACE : "
341                    "{INTERFACE}",
342                    "RC", ret, "INTERFACE", lg2::hex, ifindex);
343         return ret;
344     }
345 
346     if (cmd.ncsi_cmd != DEFAULT_VALUE)
347     {
348         std::vector<unsigned char> pl(sizeof(NCSIPacketHeader) +
349                                       cmd.payload.size());
350         NCSIPacketHeader* hdr = (NCSIPacketHeader*)pl.data();
351 
352         std::copy(cmd.payload.begin(), cmd.payload.end(),
353                   pl.begin() + sizeof(NCSIPacketHeader));
354 
355         hdr->type = cmd.ncsi_cmd;
356         hdr->length = htons(cmd.payload.size());
357 
358         ret = nla_put(msg.get(), ncsi_nl_attrs::NCSI_ATTR_DATA, pl.size(),
359                       pl.data());
360         if (ret < 0)
361         {
362             lg2::error("Failed to set the data attribute, RC : {RC}", "RC",
363                        ret);
364             return ret;
365         }
366 
367         nl_socket_disable_seq_check(socket.get());
368     }
369 
370     if (function)
371     {
372         cb_ret = 1;
373 
374         // Add a callback function to the socket
375         nl_socket_modify_cb(socket.get(), NL_CB_VALID, NL_CB_CUSTOM, function,
376                             &cb_ret);
377     }
378 
379     ret = nl_send_auto(socket.get(), msg.get());
380     if (ret < 0)
381     {
382         lg2::error("Failed to send the message , RC : {RC}", "RC", ret);
383         return ret;
384     }
385 
386     do
387     {
388         ret = nl_recvmsgs_default(socket.get());
389         if (ret < 0)
390         {
391             lg2::error("Failed to receive the message , RC : {RC}", "RC", ret);
392             break;
393         }
394     } while (cb_ret);
395 
396     return ret;
397 }
398 
399 } // namespace internal
400 
401 int sendOemCommand(int ifindex, int package, int channel,
402                    std::span<const unsigned char> payload)
403 {
404     constexpr auto cmd = 0x50;
405 
406     lg2::debug("Send OEM Command, CHANNEL : {CHANNEL} , PACKAGE : {PACKAGE}, "
407                "INTERFACE_INDEX: {INTERFACE_INDEX}",
408                "CHANNEL", lg2::hex, channel, "PACKAGE", lg2::hex, package,
409                "INTERFACE_INDEX", lg2::hex, ifindex);
410     if (!payload.empty())
411     {
412         lg2::debug("Payload: {PAYLOAD}", "PAYLOAD", toHexStr(payload));
413     }
414 
415     return internal::applyCmd(
416         ifindex,
417         internal::Command(ncsi_nl_commands::NCSI_CMD_SEND_CMD, cmd, payload),
418         package, channel, NONE, internal::sendCallBack);
419 }
420 
421 int setChannel(int ifindex, int package, int channel)
422 {
423     lg2::debug(
424         "Set CHANNEL : {CHANNEL} , PACKAGE : {PACKAGE}, INTERFACE_INDEX: "
425         "{INTERFACE_INDEX}",
426         "CHANNEL", lg2::hex, channel, "PACKAGE", lg2::hex, package,
427         "INTERFACE_INDEX", lg2::hex, ifindex);
428     return internal::applyCmd(
429         ifindex, internal::Command(ncsi_nl_commands::NCSI_CMD_SET_INTERFACE),
430         package, channel);
431 }
432 
433 int clearInterface(int ifindex)
434 {
435     lg2::debug("ClearInterface , INTERFACE_INDEX : {INTERFACE_INDEX}",
436                "INTERFACE_INDEX", lg2::hex, ifindex);
437     return internal::applyCmd(
438         ifindex, internal::Command(ncsi_nl_commands::NCSI_CMD_CLEAR_INTERFACE));
439 }
440 
441 int getInfo(int ifindex, int package)
442 {
443     lg2::debug(
444         "Get Info , PACKAGE : {PACKAGE}, INTERFACE_INDEX: {INTERFACE_INDEX}",
445         "PACKAGE", lg2::hex, package, "INTERFACE_INDEX", lg2::hex, ifindex);
446     if (package == DEFAULT_VALUE)
447     {
448         return internal::applyCmd(
449             ifindex, internal::Command(ncsi_nl_commands::NCSI_CMD_PKG_INFO),
450             package, DEFAULT_VALUE, NLM_F_DUMP, internal::infoCallBack);
451     }
452     else
453     {
454         return internal::applyCmd(ifindex, ncsi_nl_commands::NCSI_CMD_PKG_INFO,
455                                   package, DEFAULT_VALUE, NONE,
456                                   internal::infoCallBack);
457     }
458 }
459 
460 } // namespace ncsi
461 } // namespace network
462 } // namespace phosphor
463