xref: /openbmc/intel-ipmi-oem/src/oemcommands.cpp (revision 4843746a466f09f4592752e666eb0d15cabf54ee)
1 /*
2 // Copyright (c) 2018 Intel Corporation
3 //
4 // Licensed under the Apache License, Version 2.0 (the "License");
5 // you may not use this file except in compliance with the License.
6 // You may obtain a copy of the License at
7 //
8 //      http://www.apache.org/licenses/LICENSE-2.0
9 //
10 // Unless required by applicable law or agreed to in writing, software
11 // distributed under the License is distributed on an "AS IS" BASIS,
12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 // See the License for the specific language governing permissions and
14 // limitations under the License.
15 */
16 
17 #include "types.hpp"
18 #include "xyz/openbmc_project/Common/error.hpp"
19 #include "xyz/openbmc_project/Led/Physical/server.hpp"
20 
21 #include <openssl/crypto.h>
22 #include <systemd/sd-journal.h>
23 
24 #include <appcommands.hpp>
25 #include <boost/container/flat_map.hpp>
26 #include <boost/process/child.hpp>
27 #include <boost/process/io.hpp>
28 #include <com/intel/Control/OCOTShutdownPolicy/server.hpp>
29 #include <commandutils.hpp>
30 #include <gpiod.hpp>
31 #include <ipmid/api.hpp>
32 #include <ipmid/utils.hpp>
33 #include <nlohmann/json.hpp>
34 #include <oemcommands.hpp>
35 #include <phosphor-logging/log.hpp>
36 #include <sdbusplus/bus.hpp>
37 #include <sdbusplus/message/types.hpp>
38 #include <xyz/openbmc_project/Chassis/Control/NMISource/server.hpp>
39 #include <xyz/openbmc_project/Control/Boot/Mode/server.hpp>
40 #include <xyz/openbmc_project/Control/Boot/Source/server.hpp>
41 #include <xyz/openbmc_project/Control/PowerSupplyRedundancy/server.hpp>
42 #include <xyz/openbmc_project/Control/Security/RestrictionMode/server.hpp>
43 #include <xyz/openbmc_project/Control/Security/SpecialMode/server.hpp>
44 
45 #include <array>
46 #include <filesystem>
47 #include <fstream>
48 #include <iostream>
49 #include <regex>
50 #include <set>
51 #include <string>
52 #include <variant>
53 #include <vector>
54 
55 namespace ipmi
56 {
57 static void registerOEMFunctions() __attribute__((constructor));
58 
59 static constexpr size_t maxFRUStringLength = 0x3F;
60 
61 static constexpr auto ethernetIntf =
62     "xyz.openbmc_project.Network.EthernetInterface";
63 static constexpr auto networkIPIntf = "xyz.openbmc_project.Network.IP";
64 static constexpr auto networkService = "xyz.openbmc_project.Network";
65 static constexpr auto networkRoot = "/xyz/openbmc_project/network";
66 
67 static constexpr const char* oemNmiSourceIntf =
68     "xyz.openbmc_project.Chassis.Control.NMISource";
69 static constexpr const char* oemNmiSourceObjPath =
70     "/xyz/openbmc_project/Chassis/Control/NMISource";
71 static constexpr const char* oemNmiBmcSourceObjPathProp = "BMCSource";
72 static constexpr const char* oemNmiEnabledObjPathProp = "Enabled";
73 
74 static constexpr const char* dimmOffsetFile = "/var/lib/ipmi/ipmi_dimms.json";
75 static constexpr const char* multiNodeObjPath =
76     "/xyz/openbmc_project/MultiNode/Status";
77 static constexpr const char* multiNodeIntf =
78     "xyz.openbmc_project.Chassis.MultiNode";
79 
80 enum class NmiSource : uint8_t
81 {
82     none = 0,
83     frontPanelButton = 1,
84     watchdog = 2,
85     chassisCmd = 3,
86     memoryError = 4,
87     pciBusError = 5,
88     pch = 6,
89     chipset = 7,
90 };
91 
92 enum class SpecialUserIndex : uint8_t
93 {
94     rootUser = 0,
95     atScaleDebugUser = 1
96 };
97 
98 static constexpr const char* restricionModeService =
99     "xyz.openbmc_project.RestrictionMode.Manager";
100 static constexpr const char* restricionModeBasePath =
101     "/xyz/openbmc_project/control/security/restriction_mode";
102 static constexpr const char* restricionModeIntf =
103     "xyz.openbmc_project.Control.Security.RestrictionMode";
104 static constexpr const char* restricionModeProperty = "RestrictionMode";
105 
106 static constexpr const char* specialModeService =
107     "xyz.openbmc_project.SpecialMode";
108 static constexpr const char* specialModeBasePath =
109     "/xyz/openbmc_project/security/special_mode";
110 static constexpr const char* specialModeIntf =
111     "xyz.openbmc_project.Security.SpecialMode";
112 static constexpr const char* specialModeProperty = "SpecialMode";
113 
114 static constexpr const char* dBusPropertyIntf =
115     "org.freedesktop.DBus.Properties";
116 static constexpr const char* dBusPropertyGetMethod = "Get";
117 static constexpr const char* dBusPropertySetMethod = "Set";
118 
119 // return code: 0 successful
getChassisSerialNumber(sdbusplus::bus_t & bus,std::string & serial)120 int8_t getChassisSerialNumber(sdbusplus::bus_t& bus, std::string& serial)
121 {
122     std::string objpath = "/xyz/openbmc_project/FruDevice";
123     std::string intf = "xyz.openbmc_project.FruDeviceManager";
124     std::string service = getService(bus, intf, objpath);
125     ObjectValueTree valueTree = getManagedObjects(bus, service, "/");
126     if (valueTree.empty())
127     {
128         phosphor::logging::log<phosphor::logging::level::ERR>(
129             "No object implements interface",
130             phosphor::logging::entry("INTF=%s", intf.c_str()));
131         return -1;
132     }
133 
134     for (const auto& item : valueTree)
135     {
136         auto interface = item.second.find("xyz.openbmc_project.FruDevice");
137         if (interface == item.second.end())
138         {
139             continue;
140         }
141 
142         auto property = interface->second.find("CHASSIS_SERIAL_NUMBER");
143         if (property == interface->second.end())
144         {
145             continue;
146         }
147 
148         try
149         {
150             Value variant = property->second;
151             std::string& result = std::get<std::string>(variant);
152             if (result.size() > maxFRUStringLength)
153             {
154                 phosphor::logging::log<phosphor::logging::level::ERR>(
155                     "FRU serial number exceed maximum length");
156                 return -1;
157             }
158             serial = result;
159             return 0;
160         }
161         catch (const std::bad_variant_access& e)
162         {
163             phosphor::logging::log<phosphor::logging::level::ERR>(e.what());
164             return -1;
165         }
166     }
167     return -1;
168 }
169 
170 namespace mailbox
171 {
172 static uint8_t bus = 4;
173 static std::string i2cBus = "/dev/i2c-" + std::to_string(bus);
174 static uint8_t targetAddr = 56;
175 static constexpr auto systemRoot = "/xyz/openbmc_project/inventory/system";
176 static constexpr auto sessionIntf = "xyz.openbmc_project.Configuration.PFR";
177 const std::string match = "Baseboard/PFR";
178 static bool i2cConfigLoaded = false;
179 // Command register for UFM provisioning/access commands; read/write allowed
180 // from CPU/BMC.
181 static const constexpr uint8_t provisioningCommand = 0x0b;
182 // Trigger register for the command set in the previous offset.
183 static const constexpr uint8_t triggerCommand = 0x0c;
184 // Set 0x0c to 0x05 to execute command specified at “UFM/Provisioning Command”
185 // register
186 static const constexpr uint8_t flushRead = 0x05;
187 // FIFO read registers
188 std::set<uint8_t> readFifoReg = {0x08, 0x0C, 0x0D, 0x13};
189 
190 // UFM Read FIFO
191 static const constexpr uint8_t readFifo = 0x0e;
192 
193 enum registerType : uint8_t
194 {
195     singleByteRegister = 0,
196     fifoReadRegister,
197 
198 };
199 
loadPfrConfig(ipmi::Context::ptr & ctx,bool & i2cConfigLoaded)200 void loadPfrConfig(ipmi::Context::ptr& ctx, bool& i2cConfigLoaded)
201 {
202     ipmi::ObjectTree objectTree;
203 
204     boost::system::error_code ec = ipmi::getAllDbusObjects(
205         ctx, systemRoot, sessionIntf, match, objectTree);
206 
207     if (ec)
208     {
209         phosphor::logging::log<phosphor::logging::level::ERR>(
210             "Failed to fetch PFR object from dbus",
211             phosphor::logging::entry("INTERFACE=%s", sessionIntf),
212             phosphor::logging::entry("ERROR=%s", ec.message().c_str()));
213 
214         return;
215     }
216 
217     for (auto& softObject : objectTree)
218     {
219         const std::string& objPath = softObject.first;
220         const std::string& serviceName = softObject.second.begin()->first;
221         // PFR object found.. check for PFR support
222         ipmi::PropertyMap result;
223 
224         ec = ipmi::getAllDbusProperties(ctx, serviceName, objPath, sessionIntf,
225                                         result);
226 
227         if (ec)
228         {
229             phosphor::logging::log<phosphor::logging::level::ERR>(
230                 "Failed to fetch pfr properties",
231                 phosphor::logging::entry("ERROR=%s", ec.message().c_str()));
232             return;
233         }
234 
235         const uint64_t* i2cBusNum = nullptr;
236         const uint64_t* address = nullptr;
237 
238         for (const auto& [propName, propVariant] : result)
239         {
240             if (propName == "Address")
241             {
242                 address = std::get_if<uint64_t>(&propVariant);
243             }
244             else if (propName == "Bus")
245             {
246                 i2cBusNum = std::get_if<uint64_t>(&propVariant);
247             }
248         }
249 
250         if ((address == nullptr) || (i2cBusNum == nullptr))
251         {
252             phosphor::logging::log<phosphor::logging::level::ERR>(
253                 "Unable to read the pfr properties");
254             return;
255         }
256 
257         bus = static_cast<int>(*i2cBusNum);
258         i2cBus = "/dev/i2c-" + std::to_string(bus);
259         targetAddr = static_cast<int>(*address);
260 
261         i2cConfigLoaded = true;
262     }
263 }
264 
writefifo(const uint8_t cmdReg,const uint8_t val)265 void writefifo(const uint8_t cmdReg, const uint8_t val)
266 {
267     // Based on the spec, writing cmdReg to address val on this device, will
268     // trigger the write FIFO operation.
269     std::vector<uint8_t> writeData = {cmdReg, val};
270     std::vector<uint8_t> readBuf{};
271     ipmi::Cc retI2C =
272         ipmi::i2cWriteRead(i2cBus, targetAddr, writeData, readBuf);
273     if (retI2C)
274     {
275         phosphor::logging::log<phosphor::logging::level::ERR>(
276             "i2cWriteRead returns non-zero");
277     }
278 }
279 
280 } // namespace mailbox
281 
ipmiOEMGetBmcVersionString()282 ipmi::RspType<std::string> ipmiOEMGetBmcVersionString()
283 {
284     static std::string version{};
285     if (version.empty())
286     {
287         std::regex expr{"^VERSION_ID=(.*)$"};
288         static constexpr auto osReleasePath{"/etc/os-release"};
289         std::ifstream ifs(osReleasePath);
290         if (!ifs.is_open())
291         {
292             version = "os-release not present";
293         }
294         std::string line{};
295         while (std::getline(ifs, line))
296         {
297             std::smatch sm;
298             if (regex_match(line, sm, expr))
299             {
300                 if (sm.size() == 2)
301                 {
302                     std::string v = sm[1].str();
303                     // remove the quotes
304                     v.erase(std::remove(v.begin(), v.end(), '\"'), v.end());
305                     version = v;
306                     break;
307                 }
308             }
309         }
310         ifs.close();
311         if (version.empty())
312         {
313             version = "VERSION_ID not present";
314         }
315     }
316     return ipmi::responseSuccess(version);
317 }
318 
319 // Returns the Chassis Identifier (serial #)
ipmiOEMGetChassisIdentifier(ipmi_netfn_t,ipmi_cmd_t,ipmi_request_t,ipmi_response_t response,ipmi_data_len_t dataLen,ipmi_context_t)320 ipmi_ret_t ipmiOEMGetChassisIdentifier(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t,
321                                        ipmi_response_t response,
322                                        ipmi_data_len_t dataLen, ipmi_context_t)
323 {
324     std::string serial;
325     if (*dataLen != 0) // invalid request if there are extra parameters
326     {
327         *dataLen = 0;
328         return IPMI_CC_REQ_DATA_LEN_INVALID;
329     }
330     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
331     if (getChassisSerialNumber(*dbus, serial) == 0)
332     {
333         *dataLen = serial.size(); // length will never exceed response length
334                                   // as it is checked in getChassisSerialNumber
335         char* resp = static_cast<char*>(response);
336         serial.copy(resp, *dataLen);
337         return IPMI_CC_OK;
338     }
339     *dataLen = 0;
340     return IPMI_CC_RESPONSE_ERROR;
341 }
342 
ipmiOEMSetSystemGUID(ipmi_netfn_t,ipmi_cmd_t,ipmi_request_t request,ipmi_response_t,ipmi_data_len_t dataLen,ipmi_context_t)343 ipmi_ret_t ipmiOEMSetSystemGUID(ipmi_netfn_t, ipmi_cmd_t,
344                                 ipmi_request_t request, ipmi_response_t,
345                                 ipmi_data_len_t dataLen, ipmi_context_t)
346 {
347     static constexpr size_t safeBufferLength = 50;
348     char buf[safeBufferLength] = {0};
349     GUIDData* Data = reinterpret_cast<GUIDData*>(request);
350 
351     if (*dataLen != sizeof(GUIDData)) // 16bytes
352     {
353         *dataLen = 0;
354         return IPMI_CC_REQ_DATA_LEN_INVALID;
355     }
356 
357     *dataLen = 0;
358 
359     snprintf(
360         buf, safeBufferLength,
361         "%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x",
362         Data->timeLow4, Data->timeLow3, Data->timeLow2, Data->timeLow1,
363         Data->timeMid2, Data->timeMid1, Data->timeHigh2, Data->timeHigh1,
364         Data->clock2, Data->clock1, Data->node6, Data->node5, Data->node4,
365         Data->node3, Data->node2, Data->node1);
366     // UUID is in RFC4122 format. Ex: 61a39523-78f2-11e5-9862-e6402cfc3223
367     std::string guid = buf;
368 
369     std::string objpath = "/xyz/openbmc_project/control/host0/systemGUID";
370     std::string intf = "xyz.openbmc_project.Common.UUID";
371     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
372     std::string service = getService(*dbus, intf, objpath);
373     setDbusProperty(*dbus, service, objpath, intf, "UUID", guid);
374     return IPMI_CC_OK;
375 }
376 
377 ipmi::RspType<>
ipmiOEMDisableBMCSystemReset(bool disableResetOnSMI,uint7_t reserved1)378     ipmiOEMDisableBMCSystemReset(bool disableResetOnSMI, uint7_t reserved1)
379 {
380     if (reserved1)
381     {
382         return ipmi::responseInvalidFieldRequest();
383     }
384 
385     std::shared_ptr<sdbusplus::asio::connection> busp = getSdBus();
386 
387     try
388     {
389         auto service =
390             ipmi::getService(*busp, bmcResetDisablesIntf, bmcResetDisablesPath);
391         ipmi::setDbusProperty(*busp, service, bmcResetDisablesPath,
392                               bmcResetDisablesIntf, "ResetOnSMI",
393                               !disableResetOnSMI);
394     }
395     catch (const std::exception& e)
396     {
397         phosphor::logging::log<phosphor::logging::level::ERR>(
398             "Failed to set BMC reset disables",
399             phosphor::logging::entry("EXCEPTION=%s", e.what()));
400         return ipmi::responseUnspecifiedError();
401     }
402 
403     return ipmi::responseSuccess();
404 }
405 
406 ipmi::RspType<bool,   // disableResetOnSMI
407               uint7_t // reserved
408               >
ipmiOEMGetBMCResetDisables()409     ipmiOEMGetBMCResetDisables()
410 {
411     bool disableResetOnSMI = true;
412 
413     std::shared_ptr<sdbusplus::asio::connection> busp = getSdBus();
414     try
415     {
416         auto service =
417             ipmi::getService(*busp, bmcResetDisablesIntf, bmcResetDisablesPath);
418         Value variant =
419             ipmi::getDbusProperty(*busp, service, bmcResetDisablesPath,
420                                   bmcResetDisablesIntf, "ResetOnSMI");
421         disableResetOnSMI = !std::get<bool>(variant);
422     }
423     catch (const std::exception& e)
424     {
425         phosphor::logging::log<phosphor::logging::level::ERR>(
426             "Failed to get BMC reset disables",
427             phosphor::logging::entry("EXCEPTION=%s", e.what()));
428         return ipmi::responseUnspecifiedError();
429     }
430 
431     return ipmi::responseSuccess(disableResetOnSMI, 0);
432 }
433 
ipmiOEMSetBIOSID(ipmi_netfn_t,ipmi_cmd_t,ipmi_request_t request,ipmi_response_t response,ipmi_data_len_t dataLen,ipmi_context_t)434 ipmi_ret_t ipmiOEMSetBIOSID(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t request,
435                             ipmi_response_t response, ipmi_data_len_t dataLen,
436                             ipmi_context_t)
437 {
438     DeviceInfo* data = reinterpret_cast<DeviceInfo*>(request);
439 
440     if ((*dataLen < 2ul) || (*dataLen != (1ul + data->biosIDLength)))
441     {
442         *dataLen = 0;
443         return IPMI_CC_REQ_DATA_LEN_INVALID;
444     }
445     std::string idString((char*)data->biosId, data->biosIDLength);
446     for (auto idChar : idString)
447     {
448         if (!std::isprint(static_cast<unsigned char>(idChar)))
449         {
450             phosphor::logging::log<phosphor::logging::level::ERR>(
451                 "BIOS ID contains non printable character");
452             return IPMI_CC_INVALID_FIELD_REQUEST;
453         }
454     }
455 
456     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
457     std::string service = getService(*dbus, biosVersionIntf, biosActiveObjPath);
458     setDbusProperty(*dbus, service, biosActiveObjPath, biosVersionIntf,
459                     biosVersionProp, idString);
460     uint8_t* bytesWritten = static_cast<uint8_t*>(response);
461     *bytesWritten =
462         data->biosIDLength; // how many bytes are written into storage
463     *dataLen = 1;
464     return IPMI_CC_OK;
465 }
466 
getActiveHSCSoftwareVersionInfo(std::string & hscVersion,size_t hscNumber)467 bool getActiveHSCSoftwareVersionInfo(std::string& hscVersion, size_t hscNumber)
468 {
469     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
470     try
471     {
472         std::string hsbpObjPath =
473             "/xyz/openbmc_project/software/HSBP_" + std::to_string(hscNumber);
474         auto service = getService(*dbus, biosVersionIntf, hsbpObjPath);
475         Value hscVersionValue =
476             getDbusProperty(*dbus, "xyz.openbmc_project.HsbpManager",
477                             hsbpObjPath, biosVersionIntf, "Version");
478         hscVersion = std::get<std::string>(hscVersionValue);
479     }
480     catch (const sdbusplus::exception_t& e)
481     {
482         phosphor::logging::log<phosphor::logging::level::INFO>(
483             "Failed to retrieve HSBP version information",
484             phosphor::logging::entry("HSBP Number=%d", hscNumber));
485         return false;
486     }
487     return true;
488 }
489 
getHscVerInfo(ipmi::Context::ptr &,uint8_t & hsc0Major,uint8_t & hsc0Minor,uint8_t & hsc1Major,uint8_t & hsc1Minor,uint8_t & hsc2Major,uint8_t & hsc2Minor)490 bool getHscVerInfo(ipmi::Context::ptr&, uint8_t& hsc0Major, uint8_t& hsc0Minor,
491                    uint8_t& hsc1Major, uint8_t& hsc1Minor, uint8_t& hsc2Major,
492                    uint8_t& hsc2Minor)
493 {
494     std::string hscVersion;
495     std::array<uint8_t, 6> hscVersions{0};
496 
497     for (size_t hscNumber = 1; hscNumber <= 3; hscNumber++)
498     {
499         if (!getActiveHSCSoftwareVersionInfo(hscVersion, hscNumber))
500         {
501             continue;
502         }
503         std::regex pattern1("(\\d+?).(\\d+?).(\\d+?)");
504         constexpr size_t matchedPhosphor = 4;
505         std::smatch results;
506         // hscVersion = BOOT_VER.FPGA_VER.SECURITY_REVISION (Example: 00.02.01)
507         if (std::regex_match(hscVersion, results, pattern1))
508         {
509             // Major version is FPGA_VER and Minor version is SECURITY_REV
510             if (results.size() == matchedPhosphor)
511             {
512                 int index = (hscNumber - 1) * 2;
513                 hscVersions[index] =
514                     static_cast<uint8_t>(std::stoi(results[2]));
515                 hscVersions[index + 1] =
516                     static_cast<uint8_t>(std::stoi(results[3]));
517             }
518         }
519     }
520     hsc0Major = hscVersions[0];
521     hsc0Minor = hscVersions[1];
522     hsc1Major = hscVersions[2];
523     hsc1Minor = hscVersions[3];
524     hsc2Major = hscVersions[4];
525     hsc2Minor = hscVersions[5];
526     return true;
527 }
528 
getSwVerInfo(ipmi::Context::ptr & ctx,uint8_t & bmcMajor,uint8_t & bmcMinor,uint8_t & meMajor,uint8_t & meMinor)529 bool getSwVerInfo(ipmi::Context::ptr& ctx, uint8_t& bmcMajor, uint8_t& bmcMinor,
530                   uint8_t& meMajor, uint8_t& meMinor)
531 {
532     // step 1 : get BMC Major and Minor numbers from its DBUS property
533     std::string bmcVersion;
534     if (getActiveSoftwareVersionInfo(ctx, versionPurposeBMC, bmcVersion))
535     {
536         return false;
537     }
538 
539     std::optional<MetaRevision> rev = convertIntelVersion(bmcVersion);
540     if (rev.has_value())
541     {
542         MetaRevision revision = rev.value();
543         bmcMajor = revision.major;
544 
545         revision.minor = (revision.minor > 99 ? 99 : revision.minor);
546         bmcMinor = revision.minor % 10 + (revision.minor / 10) * 16;
547     }
548 
549     // step 2 : get ME Major and Minor numbers from its DBUS property
550     std::string meVersion;
551     if (getActiveSoftwareVersionInfo(ctx, versionPurposeME, meVersion))
552     {
553         return false;
554     }
555     std::regex pattern1("(\\d+?).(\\d+?).(\\d+?).(\\d+?).(\\d+?)");
556     constexpr size_t matchedPhosphor = 6;
557     std::smatch results;
558     if (std::regex_match(meVersion, results, pattern1))
559     {
560         if (results.size() == matchedPhosphor)
561         {
562             meMajor = static_cast<uint8_t>(std::stoi(results[1]));
563             meMinor = static_cast<uint8_t>(
564                 std::stoi(results[2]) << 4 | std::stoi(results[3]));
565         }
566     }
567     return true;
568 }
569 
570 ipmi::RspType<
571     std::variant<std::string,
572                  std::tuple<uint8_t, std::array<uint8_t, 2>,
573                             std::array<uint8_t, 2>, std::array<uint8_t, 2>,
574                             std::array<uint8_t, 2>, std::array<uint8_t, 2>>,
575                  std::tuple<uint8_t, std::array<uint8_t, 2>>>>
ipmiOEMGetDeviceInfo(ipmi::Context::ptr & ctx,uint8_t entityType,std::optional<uint8_t> countToRead,std::optional<uint8_t> offset)576     ipmiOEMGetDeviceInfo(ipmi::Context::ptr& ctx, uint8_t entityType,
577                          std::optional<uint8_t> countToRead,
578                          std::optional<uint8_t> offset)
579 {
580     if (entityType > static_cast<uint8_t>(OEMDevEntityType::sdrVer))
581     {
582         return ipmi::responseInvalidFieldRequest();
583     }
584 
585     // handle OEM command items
586     switch (OEMDevEntityType(entityType))
587     {
588         case OEMDevEntityType::biosId:
589         {
590             // Byte 2&3, Only used with selecting BIOS
591             if (!countToRead || !offset)
592             {
593                 return ipmi::responseReqDataLenInvalid();
594             }
595 
596             std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
597             std::string service =
598                 getService(*dbus, biosVersionIntf, biosActiveObjPath);
599             try
600             {
601                 Value variant =
602                     getDbusProperty(*dbus, service, biosActiveObjPath,
603                                     biosVersionIntf, biosVersionProp);
604                 std::string& idString = std::get<std::string>(variant);
605                 if (*offset >= idString.size())
606                 {
607                     return ipmi::responseParmOutOfRange();
608                 }
609                 size_t length = 0;
610                 if (*countToRead > (idString.size() - *offset))
611                 {
612                     length = idString.size() - *offset;
613                 }
614                 else
615                 {
616                     length = *countToRead;
617                 }
618 
619                 std::string readBuf = {0};
620                 readBuf.resize(length);
621                 std::copy_n(idString.begin() + *offset, length,
622                             (readBuf.begin()));
623                 return ipmi::responseSuccess(readBuf);
624             }
625             catch (const std::bad_variant_access& e)
626             {
627                 return ipmi::responseUnspecifiedError();
628             }
629         }
630         break;
631 
632         case OEMDevEntityType::devVer:
633         {
634             // Byte 2&3, Only used with selecting BIOS
635             if (countToRead || offset)
636             {
637                 return ipmi::responseReqDataLenInvalid();
638             }
639 
640             constexpr const size_t verLen = 2;
641             constexpr const size_t verTotalLen = 10;
642             std::array<uint8_t, verLen> bmcBuf = {0xff, 0xff};
643             std::array<uint8_t, verLen> hsc0Buf = {0xff, 0xff};
644             std::array<uint8_t, verLen> hsc1Buf = {0xff, 0xff};
645             std::array<uint8_t, verLen> meBuf = {0xff, 0xff};
646             std::array<uint8_t, verLen> hsc2Buf = {0xff, 0xff};
647             // data0/1: BMC version number; data6/7: ME version number
648             if (!getSwVerInfo(ctx, bmcBuf[0], bmcBuf[1], meBuf[0], meBuf[1]))
649             {
650                 return ipmi::responseUnspecifiedError();
651             }
652             if (!getHscVerInfo(ctx, hsc0Buf[0], hsc0Buf[1], hsc1Buf[0],
653                                hsc1Buf[1], hsc2Buf[0], hsc2Buf[1]))
654             {
655                 return ipmi::responseUnspecifiedError();
656             }
657             return ipmi::responseSuccess(
658                 std::tuple<
659                     uint8_t, std::array<uint8_t, verLen>,
660                     std::array<uint8_t, verLen>, std::array<uint8_t, verLen>,
661                     std::array<uint8_t, verLen>, std::array<uint8_t, verLen>>{
662                     verTotalLen, bmcBuf, hsc0Buf, hsc1Buf, meBuf, hsc2Buf});
663         }
664         break;
665 
666         case OEMDevEntityType::sdrVer:
667         {
668             // Byte 2&3, Only used with selecting BIOS
669             if (countToRead || offset)
670             {
671                 return ipmi::responseReqDataLenInvalid();
672             }
673 
674             constexpr const size_t sdrLen = 2;
675             std::array<uint8_t, sdrLen> readBuf = {0x01, 0x0};
676             return ipmi::responseSuccess(
677                 std::tuple<uint8_t, std::array<uint8_t, sdrLen>>{sdrLen,
678                                                                  readBuf});
679         }
680         break;
681 
682         default:
683             return ipmi::responseInvalidFieldRequest();
684     }
685 }
686 
ipmiOEMGetAICFRU(ipmi_netfn_t,ipmi_cmd_t,ipmi_request_t,ipmi_response_t response,ipmi_data_len_t dataLen,ipmi_context_t)687 ipmi_ret_t ipmiOEMGetAICFRU(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t,
688                             ipmi_response_t response, ipmi_data_len_t dataLen,
689                             ipmi_context_t)
690 {
691     if (*dataLen != 0)
692     {
693         *dataLen = 0;
694         return IPMI_CC_REQ_DATA_LEN_INVALID;
695     }
696 
697     *dataLen = 1;
698     uint8_t* res = reinterpret_cast<uint8_t*>(response);
699     // temporary fix. We don't support AIC FRU now. Just tell BIOS that no
700     // AIC is available so that BIOS will not timeout repeatly which leads to
701     // slow booting.
702     *res = 0; // Byte1=Count of SlotPosition/FruID records.
703     return IPMI_CC_OK;
704 }
705 
ipmiOEMGetPowerRestoreDelay(ipmi_netfn_t,ipmi_cmd_t,ipmi_request_t,ipmi_response_t response,ipmi_data_len_t dataLen,ipmi_context_t)706 ipmi_ret_t ipmiOEMGetPowerRestoreDelay(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t,
707                                        ipmi_response_t response,
708                                        ipmi_data_len_t dataLen, ipmi_context_t)
709 {
710     GetPowerRestoreDelayRes* resp =
711         reinterpret_cast<GetPowerRestoreDelayRes*>(response);
712 
713     if (*dataLen != 0)
714     {
715         *dataLen = 0;
716         return IPMI_CC_REQ_DATA_LEN_INVALID;
717     }
718 
719     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
720     std::string service =
721         getService(*dbus, powerRestoreDelayIntf, powerRestoreDelayObjPath);
722     Value variant =
723         getDbusProperty(*dbus, service, powerRestoreDelayObjPath,
724                         powerRestoreDelayIntf, powerRestoreDelayProp);
725 
726     uint64_t val = std::get<uint64_t>(variant);
727     val /= 1000000UL;
728     uint16_t delay = val;
729     resp->byteLSB = delay;
730     resp->byteMSB = delay >> 8;
731 
732     *dataLen = sizeof(GetPowerRestoreDelayRes);
733 
734     return IPMI_CC_OK;
735 }
736 
bcdToDec(uint8_t val)737 static uint8_t bcdToDec(uint8_t val)
738 {
739     return ((val / 16 * 10) + (val % 16));
740 }
741 
742 // Allows an update utility or system BIOS to send the status of an embedded
743 // firmware update attempt to the BMC. After received, BMC will create a logging
744 // record.
ipmiOEMSendEmbeddedFwUpdStatus(uint8_t status,uint8_t target,uint8_t majorRevision,uint8_t minorRevision,uint32_t auxInfo)745 ipmi::RspType<> ipmiOEMSendEmbeddedFwUpdStatus(
746     uint8_t status, uint8_t target, uint8_t majorRevision,
747     uint8_t minorRevision, uint32_t auxInfo)
748 {
749     std::string firmware;
750     int instance = (target & targetInstanceMask) >> targetInstanceShift;
751     target = (target & selEvtTargetMask) >> selEvtTargetShift;
752 
753     /* make sure the status is 0, 1, or 2 as per the spec */
754     if (status > 2)
755     {
756         return ipmi::response(ipmi::ccInvalidFieldRequest);
757     }
758     /* make sure the target is 0, 1, 2, or 4 as per the spec */
759     if (target > 4 || target == 3)
760     {
761         return ipmi::response(ipmi::ccInvalidFieldRequest);
762     }
763     /*orignal OEM command is to record OEM SEL.
764     But openbmc does not support OEM SEL, so we redirect it to redfish event
765     logging. */
766     std::string buildInfo;
767     std::string action;
768     switch (FWUpdateTarget(target))
769     {
770         case FWUpdateTarget::targetBMC:
771             firmware = "BMC";
772             buildInfo = "major: " + std::to_string(majorRevision) + " minor: " +
773                         std::to_string(bcdToDec(minorRevision)) + // BCD encoded
774                         " BuildID: " + std::to_string(auxInfo);
775             buildInfo += std::to_string(auxInfo);
776             break;
777         case FWUpdateTarget::targetBIOS:
778             firmware = "BIOS";
779             buildInfo =
780                 "major: " +
781                 std::to_string(bcdToDec(majorRevision)) + // BCD encoded
782                 " minor: " +
783                 std::to_string(bcdToDec(minorRevision)) + // BCD encoded
784                 " ReleaseNumber: " +                      // ASCII encoded
785                 std::to_string(static_cast<uint8_t>(auxInfo >> 0) - '0') +
786                 std::to_string(static_cast<uint8_t>(auxInfo >> 8) - '0') +
787                 std::to_string(static_cast<uint8_t>(auxInfo >> 16) - '0') +
788                 std::to_string(static_cast<uint8_t>(auxInfo >> 24) - '0');
789             break;
790         case FWUpdateTarget::targetME:
791             firmware = "ME";
792             buildInfo =
793                 "major: " + std::to_string(majorRevision) + " minor1: " +
794                 std::to_string(bcdToDec(minorRevision)) + // BCD encoded
795                 " minor2: " +
796                 std::to_string(bcdToDec(static_cast<uint8_t>(auxInfo >> 0))) +
797                 " build1: " +
798                 std::to_string(bcdToDec(static_cast<uint8_t>(auxInfo >> 8))) +
799                 " build2: " +
800                 std::to_string(bcdToDec(static_cast<uint8_t>(auxInfo >> 16)));
801             break;
802         case FWUpdateTarget::targetOEMEWS:
803             firmware = "EWS";
804             buildInfo = "major: " + std::to_string(majorRevision) + " minor: " +
805                         std::to_string(bcdToDec(minorRevision)) + // BCD encoded
806                         " BuildID: " + std::to_string(auxInfo);
807             break;
808     }
809 
810     static const std::string openBMCMessageRegistryVersion("0.1");
811     std::string redfishMsgID = "OpenBMC." + openBMCMessageRegistryVersion;
812 
813     switch (status)
814     {
815         case 0x0:
816             action = "update started";
817             redfishMsgID += ".FirmwareUpdateStarted";
818             break;
819         case 0x1:
820             action = "update completed successfully";
821             redfishMsgID += ".FirmwareUpdateCompleted";
822             break;
823         case 0x2:
824             action = "update failure";
825             redfishMsgID += ".FirmwareUpdateFailed";
826             break;
827         default:
828             action = "unknown";
829             break;
830     }
831 
832     std::string firmwareInstanceStr =
833         firmware + " instance: " + std::to_string(instance);
834     std::string message("[firmware update] " + firmwareInstanceStr +
835                         " status: <" + action + "> " + buildInfo);
836 
837     sd_journal_send("MESSAGE=%s", message.c_str(), "PRIORITY=%i", LOG_INFO,
838                     "REDFISH_MESSAGE_ID=%s", redfishMsgID.c_str(),
839                     "REDFISH_MESSAGE_ARGS=%s,%s", firmwareInstanceStr.c_str(),
840                     buildInfo.c_str(), NULL);
841     return ipmi::responseSuccess();
842 }
843 
ipmiOEMSlotIpmb(ipmi::Context::ptr & ctx,uint6_t reserved1,uint2_t slotNumber,uint3_t baseBoardSlotNum,uint3_t riserSlotNum,uint2_t reserved2,uint8_t targetAddr,uint8_t netFn,uint8_t cmd,std::optional<std::vector<uint8_t>> writeData)844 ipmi::RspType<uint8_t, std::vector<uint8_t>> ipmiOEMSlotIpmb(
845     ipmi::Context::ptr& ctx, uint6_t reserved1, uint2_t slotNumber,
846     uint3_t baseBoardSlotNum, [[maybe_unused]] uint3_t riserSlotNum,
847     uint2_t reserved2, uint8_t targetAddr, uint8_t netFn, uint8_t cmd,
848     std::optional<std::vector<uint8_t>> writeData)
849 {
850     if (reserved1 || reserved2)
851     {
852         return ipmi::responseInvalidFieldRequest();
853     }
854 
855     boost::system::error_code ec;
856     using ipmbResponse = std::tuple<int, uint8_t, uint8_t, uint8_t, uint8_t,
857                                     std::vector<uint8_t>>;
858     ipmbResponse res = ctx->bus->yield_method_call<ipmbResponse>(
859         ctx->yield, ec, "xyz.openbmc_project.Ipmi.Channel.Ipmb",
860         "/xyz/openbmc_project/Ipmi/Channel/Ipmb", "org.openbmc.Ipmb",
861         "SlotIpmbRequest", static_cast<uint8_t>(slotNumber),
862         static_cast<uint8_t>(baseBoardSlotNum), targetAddr, netFn, cmd,
863         *writeData);
864     if (ec)
865     {
866         phosphor::logging::log<phosphor::logging::level::ERR>(
867             "Failed to call dbus method SlotIpmbRequest");
868         return ipmi::responseUnspecifiedError();
869     }
870 
871     std::vector<uint8_t> dataReceived{};
872     int status = -1;
873     uint8_t resNetFn = 0, resLun = 0, resCmd = 0, cc = 0;
874 
875     std::tie(status, resNetFn, resLun, resCmd, cc, dataReceived) = res;
876 
877     if (status)
878     {
879         phosphor::logging::log<phosphor::logging::level::ERR>(
880             "Failed to get response from SlotIpmbRequest");
881         return ipmi::responseResponseError();
882     }
883     return ipmi::responseSuccess(cc, dataReceived);
884 }
885 
ipmiOEMSetPowerRestoreDelay(ipmi_netfn_t,ipmi_cmd_t,ipmi_request_t request,ipmi_response_t,ipmi_data_len_t dataLen,ipmi_context_t)886 ipmi_ret_t ipmiOEMSetPowerRestoreDelay(ipmi_netfn_t, ipmi_cmd_t,
887                                        ipmi_request_t request, ipmi_response_t,
888                                        ipmi_data_len_t dataLen, ipmi_context_t)
889 {
890     SetPowerRestoreDelayReq* data =
891         reinterpret_cast<SetPowerRestoreDelayReq*>(request);
892     uint16_t delay = 0;
893 
894     if (*dataLen != sizeof(SetPowerRestoreDelayReq))
895     {
896         *dataLen = 0;
897         return IPMI_CC_REQ_DATA_LEN_INVALID;
898     }
899     delay = data->byteMSB;
900     delay = (delay << 8) | data->byteLSB;
901     uint64_t val = delay * 1000000;
902     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
903     std::string service =
904         getService(*dbus, powerRestoreDelayIntf, powerRestoreDelayObjPath);
905     setDbusProperty(*dbus, service, powerRestoreDelayObjPath,
906                     powerRestoreDelayIntf, powerRestoreDelayProp, val);
907     *dataLen = 0;
908 
909     return IPMI_CC_OK;
910 }
911 
cpuPresent(const std::string & cpuName)912 static bool cpuPresent(const std::string& cpuName)
913 {
914     static constexpr const char* cpuPresencePathPrefix =
915         "/xyz/openbmc_project/inventory/system/chassis/motherboard/";
916     static constexpr const char* cpuPresenceIntf =
917         "xyz.openbmc_project.Inventory.Item";
918     std::string cpuPresencePath = cpuPresencePathPrefix + cpuName;
919     std::shared_ptr<sdbusplus::asio::connection> busp = getSdBus();
920     try
921     {
922         auto service =
923             ipmi::getService(*busp, cpuPresenceIntf, cpuPresencePath);
924 
925         ipmi::Value result = ipmi::getDbusProperty(
926             *busp, service, cpuPresencePath, cpuPresenceIntf, "Present");
927         return std::get<bool>(result);
928     }
929     catch (const std::exception& e)
930     {
931         phosphor::logging::log<phosphor::logging::level::INFO>(
932             "Cannot find processor presence",
933             phosphor::logging::entry("NAME=%s", cpuName.c_str()));
934         return false;
935     }
936 }
937 
938 ipmi::RspType<bool,    // IERR Reset Enabled
939               bool,    // ERR2 Reset Enabled
940               bool,    // MCERR Reset Enabled
941               uint5_t, // reserved
942               uint8_t, // reserved, returns 0x3F
943               uint6_t, // CPU1 IERR Count
944               uint2_t, // CPU1 Status
945               uint6_t, // CPU2 IERR Count
946               uint2_t, // CPU2 Status
947               uint6_t, // CPU3 IERR Count
948               uint2_t, // CPU3 Status
949               uint6_t, // CPU4 IERR Count
950               uint2_t, // CPU4 Status
951               uint8_t  // Crashdump Count
952               >
ipmiOEMGetProcessorErrConfig()953     ipmiOEMGetProcessorErrConfig()
954 {
955     bool resetOnIERR = false;
956     bool resetOnERR2 = false;
957     bool resetOnMCERR = false;
958     uint6_t cpu1IERRCount = 0;
959     uint6_t cpu2IERRCount = 0;
960     uint6_t cpu3IERRCount = 0;
961     uint6_t cpu4IERRCount = 0;
962     uint8_t crashdumpCount = 0;
963     uint2_t cpu1Status = cpuPresent("CPU_1")
964                              ? types::enum_cast<uint8_t>(CPUStatus::enabled)
965                              : types::enum_cast<uint8_t>(CPUStatus::notPresent);
966     uint2_t cpu2Status = cpuPresent("CPU_2")
967                              ? types::enum_cast<uint8_t>(CPUStatus::enabled)
968                              : types::enum_cast<uint8_t>(CPUStatus::notPresent);
969     uint2_t cpu3Status = cpuPresent("CPU_3")
970                              ? types::enum_cast<uint8_t>(CPUStatus::enabled)
971                              : types::enum_cast<uint8_t>(CPUStatus::notPresent);
972     uint2_t cpu4Status = cpuPresent("CPU_4")
973                              ? types::enum_cast<uint8_t>(CPUStatus::enabled)
974                              : types::enum_cast<uint8_t>(CPUStatus::notPresent);
975 
976     std::shared_ptr<sdbusplus::asio::connection> busp = getSdBus();
977     try
978     {
979         auto service = ipmi::getService(*busp, processorErrConfigIntf,
980                                         processorErrConfigObjPath);
981 
982         ipmi::PropertyMap result = ipmi::getAllDbusProperties(
983             *busp, service, processorErrConfigObjPath, processorErrConfigIntf);
984         resetOnIERR = std::get<bool>(result.at("ResetOnIERR"));
985         resetOnERR2 = std::get<bool>(result.at("ResetOnERR2"));
986         resetOnMCERR = std::get<bool>(result.at("ResetOnMCERR"));
987         cpu1IERRCount = std::get<uint8_t>(result.at("ErrorCountCPU1"));
988         cpu2IERRCount = std::get<uint8_t>(result.at("ErrorCountCPU2"));
989         cpu3IERRCount = std::get<uint8_t>(result.at("ErrorCountCPU3"));
990         cpu4IERRCount = std::get<uint8_t>(result.at("ErrorCountCPU4"));
991         crashdumpCount = std::get<uint8_t>(result.at("CrashdumpCount"));
992     }
993     catch (const std::exception& e)
994     {
995         phosphor::logging::log<phosphor::logging::level::ERR>(
996             "Failed to fetch processor error config",
997             phosphor::logging::entry("ERROR=%s", e.what()));
998         return ipmi::responseUnspecifiedError();
999     }
1000 
1001     return ipmi::responseSuccess(
1002         resetOnIERR, resetOnERR2, resetOnMCERR, 0, 0x3F, cpu1IERRCount,
1003         cpu1Status, cpu2IERRCount, cpu2Status, cpu3IERRCount, cpu3Status,
1004         cpu4IERRCount, cpu4Status, crashdumpCount);
1005 }
1006 
ipmiOEMSetProcessorErrConfig(bool resetOnIERR,bool resetOnERR2,bool resetOnMCERR,uint5_t reserved1,uint8_t reserved2,std::optional<bool> clearCPUErrorCount,std::optional<bool> clearCrashdumpCount,std::optional<uint6_t> reserved3)1007 ipmi::RspType<> ipmiOEMSetProcessorErrConfig(
1008     bool resetOnIERR, bool resetOnERR2, bool resetOnMCERR, uint5_t reserved1,
1009     uint8_t reserved2, std::optional<bool> clearCPUErrorCount,
1010     std::optional<bool> clearCrashdumpCount, std::optional<uint6_t> reserved3)
1011 {
1012     if (reserved1 || reserved2)
1013     {
1014         return ipmi::responseInvalidFieldRequest();
1015     }
1016 
1017     std::shared_ptr<sdbusplus::asio::connection> busp = getSdBus();
1018 
1019     try
1020     {
1021         if (reserved3.value_or(0))
1022         {
1023             return ipmi::responseInvalidFieldRequest();
1024         }
1025         auto service = ipmi::getService(*busp, processorErrConfigIntf,
1026                                         processorErrConfigObjPath);
1027         ipmi::setDbusProperty(*busp, service, processorErrConfigObjPath,
1028                               processorErrConfigIntf, "ResetOnIERR",
1029                               resetOnIERR);
1030         ipmi::setDbusProperty(*busp, service, processorErrConfigObjPath,
1031                               processorErrConfigIntf, "ResetOnERR2",
1032                               resetOnERR2);
1033         ipmi::setDbusProperty(*busp, service, processorErrConfigObjPath,
1034                               processorErrConfigIntf, "ResetOnMCERR",
1035                               resetOnMCERR);
1036         if (clearCPUErrorCount.value_or(false))
1037         {
1038             ipmi::setDbusProperty(*busp, service, processorErrConfigObjPath,
1039                                   processorErrConfigIntf, "ErrorCountCPU1",
1040                                   static_cast<uint8_t>(0));
1041             ipmi::setDbusProperty(*busp, service, processorErrConfigObjPath,
1042                                   processorErrConfigIntf, "ErrorCountCPU2",
1043                                   static_cast<uint8_t>(0));
1044             ipmi::setDbusProperty(*busp, service, processorErrConfigObjPath,
1045                                   processorErrConfigIntf, "ErrorCountCPU3",
1046                                   static_cast<uint8_t>(0));
1047             ipmi::setDbusProperty(*busp, service, processorErrConfigObjPath,
1048                                   processorErrConfigIntf, "ErrorCountCPU4",
1049                                   static_cast<uint8_t>(0));
1050         }
1051         if (clearCrashdumpCount.value_or(false))
1052         {
1053             ipmi::setDbusProperty(*busp, service, processorErrConfigObjPath,
1054                                   processorErrConfigIntf, "CrashdumpCount",
1055                                   static_cast<uint8_t>(0));
1056         }
1057     }
1058     catch (const std::exception& e)
1059     {
1060         phosphor::logging::log<phosphor::logging::level::ERR>(
1061             "Failed to set processor error config",
1062             phosphor::logging::entry("EXCEPTION=%s", e.what()));
1063         return ipmi::responseUnspecifiedError();
1064     }
1065 
1066     return ipmi::responseSuccess();
1067 }
1068 
ipmiOEMGetShutdownPolicy(ipmi_netfn_t,ipmi_cmd_t,ipmi_request_t,ipmi_response_t response,ipmi_data_len_t dataLen,ipmi_context_t)1069 ipmi_ret_t ipmiOEMGetShutdownPolicy(ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t,
1070                                     ipmi_response_t response,
1071                                     ipmi_data_len_t dataLen, ipmi_context_t)
1072 {
1073     GetOEMShutdownPolicyRes* resp =
1074         reinterpret_cast<GetOEMShutdownPolicyRes*>(response);
1075 
1076     if (*dataLen != 0)
1077     {
1078         phosphor::logging::log<phosphor::logging::level::ERR>(
1079             "oem_get_shutdown_policy: invalid input len!");
1080         *dataLen = 0;
1081         return IPMI_CC_REQ_DATA_LEN_INVALID;
1082     }
1083 
1084     *dataLen = 0;
1085 
1086     try
1087     {
1088         std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1089         std::string service =
1090             getService(*dbus, oemShutdownPolicyIntf, oemShutdownPolicyObjPath);
1091         Value variant = getDbusProperty(
1092             *dbus, service, oemShutdownPolicyObjPath, oemShutdownPolicyIntf,
1093             oemShutdownPolicyObjPathProp);
1094 
1095         if (sdbusplus::com::intel::Control::server::OCOTShutdownPolicy::
1096                 convertPolicyFromString(std::get<std::string>(variant)) ==
1097             sdbusplus::com::intel::Control::server::OCOTShutdownPolicy::Policy::
1098                 NoShutdownOnOCOT)
1099         {
1100             resp->policy = 0;
1101         }
1102         else if (sdbusplus::com::intel::Control::server::OCOTShutdownPolicy::
1103                      convertPolicyFromString(std::get<std::string>(variant)) ==
1104                  sdbusplus::com::intel::Control::server::OCOTShutdownPolicy::
1105                      Policy::ShutdownOnOCOT)
1106         {
1107             resp->policy = 1;
1108         }
1109         else
1110         {
1111             phosphor::logging::log<phosphor::logging::level::ERR>(
1112                 "oem_set_shutdown_policy: invalid property!",
1113                 phosphor::logging::entry(
1114                     "PROP=%s", std::get<std::string>(variant).c_str()));
1115             return IPMI_CC_UNSPECIFIED_ERROR;
1116         }
1117         // TODO needs to check if it is multi-node products,
1118         // policy is only supported on node 3/4
1119         resp->policySupport = shutdownPolicySupported;
1120     }
1121     catch (const sdbusplus::exception_t& e)
1122     {
1123         phosphor::logging::log<phosphor::logging::level::ERR>(e.description());
1124         return IPMI_CC_UNSPECIFIED_ERROR;
1125     }
1126 
1127     *dataLen = sizeof(GetOEMShutdownPolicyRes);
1128     return IPMI_CC_OK;
1129 }
1130 
ipmiOEMSetShutdownPolicy(ipmi_netfn_t,ipmi_cmd_t,ipmi_request_t request,ipmi_response_t,ipmi_data_len_t dataLen,ipmi_context_t)1131 ipmi_ret_t ipmiOEMSetShutdownPolicy(ipmi_netfn_t, ipmi_cmd_t,
1132                                     ipmi_request_t request, ipmi_response_t,
1133                                     ipmi_data_len_t dataLen, ipmi_context_t)
1134 {
1135     uint8_t* req = reinterpret_cast<uint8_t*>(request);
1136     sdbusplus::com::intel::Control::server::OCOTShutdownPolicy::Policy policy =
1137         sdbusplus::com::intel::Control::server::OCOTShutdownPolicy::Policy::
1138             NoShutdownOnOCOT;
1139 
1140     // TODO needs to check if it is multi-node products,
1141     // policy is only supported on node 3/4
1142     if (*dataLen != 1)
1143     {
1144         phosphor::logging::log<phosphor::logging::level::ERR>(
1145             "oem_set_shutdown_policy: invalid input len!");
1146         *dataLen = 0;
1147         return IPMI_CC_REQ_DATA_LEN_INVALID;
1148     }
1149 
1150     *dataLen = 0;
1151     if ((*req != noShutdownOnOCOT) && (*req != shutdownOnOCOT))
1152     {
1153         phosphor::logging::log<phosphor::logging::level::ERR>(
1154             "oem_set_shutdown_policy: invalid input!");
1155         return IPMI_CC_INVALID_FIELD_REQUEST;
1156     }
1157 
1158     if (*req == noShutdownOnOCOT)
1159     {
1160         policy = sdbusplus::com::intel::Control::server::OCOTShutdownPolicy::
1161             Policy::NoShutdownOnOCOT;
1162     }
1163     else
1164     {
1165         policy = sdbusplus::com::intel::Control::server::OCOTShutdownPolicy::
1166             Policy::ShutdownOnOCOT;
1167     }
1168 
1169     try
1170     {
1171         std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1172         std::string service =
1173             getService(*dbus, oemShutdownPolicyIntf, oemShutdownPolicyObjPath);
1174         setDbusProperty(
1175             *dbus, service, oemShutdownPolicyObjPath, oemShutdownPolicyIntf,
1176             oemShutdownPolicyObjPathProp,
1177             sdbusplus::com::intel::Control::server::convertForMessage(policy));
1178     }
1179     catch (const sdbusplus::exception_t& e)
1180     {
1181         phosphor::logging::log<phosphor::logging::level::ERR>(e.description());
1182         return IPMI_CC_UNSPECIFIED_ERROR;
1183     }
1184 
1185     return IPMI_CC_OK;
1186 }
1187 
1188 /** @brief implementation for check the DHCP or not in IPv4
1189  *  @param[in] Channel - Channel number
1190  *  @returns true or false.
1191  */
isDHCPEnabled(uint8_t Channel)1192 static bool isDHCPEnabled(uint8_t Channel)
1193 {
1194     try
1195     {
1196         auto ethdevice = getChannelName(Channel);
1197         if (ethdevice.empty())
1198         {
1199             return false;
1200         }
1201         auto ethIP = ethdevice + "/ipv4";
1202         std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1203         auto ethernetObj =
1204             getDbusObject(*dbus, networkIPIntf, networkRoot, ethIP);
1205         auto value = getDbusProperty(*dbus, networkService, ethernetObj.first,
1206                                      networkIPIntf, "Origin");
1207         if (std::get<std::string>(value) ==
1208             "xyz.openbmc_project.Network.IP.AddressOrigin.DHCP")
1209         {
1210             return true;
1211         }
1212         else
1213         {
1214             return false;
1215         }
1216     }
1217     catch (const sdbusplus::exception_t& e)
1218     {
1219         phosphor::logging::log<phosphor::logging::level::ERR>(e.description());
1220         return true;
1221     }
1222 }
1223 
1224 /** @brief implementes for check the DHCP or not in IPv6
1225  *  @param[in] Channel - Channel number
1226  *  @returns true or false.
1227  */
isDHCPIPv6Enabled(uint8_t Channel)1228 static bool isDHCPIPv6Enabled(uint8_t Channel)
1229 {
1230     try
1231     {
1232         auto ethdevice = getChannelName(Channel);
1233         if (ethdevice.empty())
1234         {
1235             return false;
1236         }
1237         auto ethIP = ethdevice + "/ipv6";
1238         std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1239         auto objectInfo =
1240             getDbusObject(*dbus, networkIPIntf, networkRoot, ethIP);
1241         auto properties = getAllDbusProperties(*dbus, objectInfo.second,
1242                                                objectInfo.first, networkIPIntf);
1243         if (std::get<std::string>(properties["Origin"]) ==
1244             "xyz.openbmc_project.Network.IP.AddressOrigin.DHCP")
1245         {
1246             return true;
1247         }
1248         else
1249         {
1250             return false;
1251         }
1252     }
1253     catch (const sdbusplus::exception_t& e)
1254     {
1255         phosphor::logging::log<phosphor::logging::level::ERR>(e.description());
1256         return true;
1257     }
1258 }
1259 
1260 /** @brief implementes the creating of default new user
1261  *  @param[in] userName - new username in 16 bytes.
1262  *  @param[in] userPassword - new password in 20 bytes
1263  *  @returns ipmi completion code.
1264  */
ipmiOEMSetUser2Activation(std::array<uint8_t,ipmi::ipmiMaxUserName> & userName,const SecureBuffer & userPassword)1265 ipmi::RspType<> ipmiOEMSetUser2Activation(
1266     std::array<uint8_t, ipmi::ipmiMaxUserName>& userName,
1267     const SecureBuffer& userPassword)
1268 {
1269     if (userPassword.size() != ipmi::maxIpmi20PasswordSize)
1270     {
1271         return ipmi::responseReqDataLenInvalid();
1272     }
1273     bool userState = false;
1274     // Check for System Interface not exist and LAN should be static
1275     for (uint8_t channel = 0; channel < maxIpmiChannels; channel++)
1276     {
1277         ChannelInfo chInfo{};
1278         try
1279         {
1280             getChannelInfo(channel, chInfo);
1281         }
1282         catch (const sdbusplus::exception_t& e)
1283         {
1284             phosphor::logging::log<phosphor::logging::level::ERR>(
1285                 "ipmiOEMSetUser2Activation: Failed to get Channel Info",
1286                 phosphor::logging::entry("MSG: %s", e.description()));
1287             return ipmi::response(ipmi::ccUnspecifiedError);
1288         }
1289         if (chInfo.mediumType ==
1290             static_cast<uint8_t>(EChannelMediumType::systemInterface))
1291         {
1292             phosphor::logging::log<phosphor::logging::level::ERR>(
1293                 "ipmiOEMSetUser2Activation: system interface  exist .");
1294             return ipmi::response(ipmi::ccCommandNotAvailable);
1295         }
1296         else
1297         {
1298             if (chInfo.mediumType ==
1299                 static_cast<uint8_t>(EChannelMediumType::lan8032))
1300             {
1301                 if (isDHCPIPv6Enabled(channel) || isDHCPEnabled(channel))
1302                 {
1303                     phosphor::logging::log<phosphor::logging::level::ERR>(
1304                         "ipmiOEMSetUser2Activation: DHCP enabled .");
1305                     return ipmi::response(ipmi::ccCommandNotAvailable);
1306                 }
1307             }
1308         }
1309     }
1310     uint8_t maxChUsers = 0, enabledUsers = 0, fixedUsers = 0;
1311     if (ipmi::ccSuccess ==
1312         ipmiUserGetAllCounts(maxChUsers, enabledUsers, fixedUsers))
1313     {
1314         if (enabledUsers > 1)
1315         {
1316             phosphor::logging::log<phosphor::logging::level::ERR>(
1317                 "ipmiOEMSetUser2Activation: more than one user is enabled.");
1318             return ipmi::response(ipmi::ccCommandNotAvailable);
1319         }
1320         // Check the user 2 is enabled or not
1321         ipmiUserCheckEnabled(ipmiDefaultUserId, userState);
1322         if (userState == true)
1323         {
1324             phosphor::logging::log<phosphor::logging::level::ERR>(
1325                 "ipmiOEMSetUser2Activation: user 2 already enabled .");
1326             return ipmi::response(ipmi::ccCommandNotAvailable);
1327         }
1328     }
1329     else
1330     {
1331         return ipmi::response(ipmi::ccUnspecifiedError);
1332     }
1333 
1334 #if BYTE_ORDER == LITTLE_ENDIAN
1335     PrivAccess privAccess = {PRIVILEGE_ADMIN, true, true, true, 0};
1336 #endif
1337 #if BYTE_ORDER == BIG_ENDIAN
1338     PrivAccess privAccess = {0, true, true, true, PRIVILEGE_ADMIN};
1339 #endif
1340 
1341     // ipmiUserSetUserName correctly handles char*, possibly non-null
1342     // terminated strings using ipmiMaxUserName size
1343     size_t nameLen = strnlen(reinterpret_cast<const char*>(userName.data()),
1344                              sizeof(userName));
1345     const std::string userNameRaw(
1346         reinterpret_cast<const char*>(userName.data()), nameLen);
1347 
1348     if (ipmi::ccSuccess == ipmiUserSetUserName(ipmiDefaultUserId, userNameRaw))
1349     {
1350         if (ipmi::ccSuccess ==
1351             ipmiUserSetUserPassword(
1352                 ipmiDefaultUserId,
1353                 reinterpret_cast<const char*>(userPassword.data())))
1354         {
1355             if (ipmi::ccSuccess ==
1356                 ipmiUserSetPrivilegeAccess(
1357                     ipmiDefaultUserId,
1358                     static_cast<uint8_t>(ipmi::EChannelID::chanLan1),
1359                     privAccess, true))
1360             {
1361                 phosphor::logging::log<phosphor::logging::level::INFO>(
1362                     "ipmiOEMSetUser2Activation: user created successfully ");
1363 
1364                 return ipmi::responseSuccess();
1365             }
1366         }
1367         // we need to delete  the default user id which added in this command as
1368         // password / priv setting is failed.
1369         ipmiUserSetUserName(ipmiDefaultUserId, static_cast<std::string>(""));
1370         phosphor::logging::log<phosphor::logging::level::ERR>(
1371             "ipmiOEMSetUser2Activation: password / priv setting is failed.");
1372     }
1373     else
1374     {
1375         phosphor::logging::log<phosphor::logging::level::ERR>(
1376             "ipmiOEMSetUser2Activation: Setting username failed.");
1377     }
1378 
1379     return ipmi::response(ipmi::ccCommandNotAvailable);
1380 }
1381 
1382 /** @brief implementes executing the linux command
1383  *  @param[in] linux command
1384  *  @returns status
1385  */
1386 
executeCmd(const char * path)1387 static uint8_t executeCmd(const char* path)
1388 {
1389     boost::process::child execProg(path);
1390     execProg.wait();
1391 
1392     int retCode = execProg.exit_code();
1393     if (retCode)
1394     {
1395         return ipmi::ccUnspecifiedError;
1396     }
1397     return ipmi::ccSuccess;
1398 }
1399 
1400 /** @brief implementes ASD Security event logging
1401  *  @param[in] Event message string
1402  *  @param[in] Event Severity
1403  *  @returns status
1404  */
1405 
atScaleDebugEventlog(std::string msg,int severity)1406 static void atScaleDebugEventlog(std::string msg, int severity)
1407 {
1408     std::string eventStr = "OpenBMC.0.1." + msg;
1409     sd_journal_send("MESSAGE=Security Event: %s", eventStr.c_str(),
1410                     "PRIORITY=%i", severity, "REDFISH_MESSAGE_ID=%s",
1411                     eventStr.c_str(), NULL);
1412 }
1413 
1414 /** @brief implementes setting password for special user
1415  *  @param[in] specialUserIndex
1416  *  @param[in] userPassword - new password in 20 bytes
1417  *  @returns ipmi completion code.
1418  */
ipmiOEMSetSpecialUserPassword(ipmi::Context::ptr & ctx,uint8_t specialUserIndex,std::vector<uint8_t> userPassword)1419 ipmi::RspType<> ipmiOEMSetSpecialUserPassword(ipmi::Context::ptr& ctx,
1420                                               uint8_t specialUserIndex,
1421                                               std::vector<uint8_t> userPassword)
1422 {
1423     ChannelInfo chInfo;
1424     ipmi_ret_t status = ipmi::ccSuccess;
1425 
1426     try
1427     {
1428         getChannelInfo(ctx->channel, chInfo);
1429     }
1430     catch (const sdbusplus::exception_t& e)
1431     {
1432         phosphor::logging::log<phosphor::logging::level::ERR>(
1433             "ipmiOEMSetSpecialUserPassword: Failed to get Channel Info",
1434             phosphor::logging::entry("MSG: %s", e.description()));
1435         return ipmi::responseUnspecifiedError();
1436     }
1437     if (chInfo.mediumType !=
1438         static_cast<uint8_t>(EChannelMediumType::systemInterface))
1439     {
1440         phosphor::logging::log<phosphor::logging::level::ERR>(
1441             "ipmiOEMSetSpecialUserPassword: Error - supported only in KCS "
1442             "interface");
1443         return ipmi::responseCommandNotAvailable();
1444     }
1445 
1446     // 0 for root user  and 1 for AtScaleDebug is allowed
1447     if (specialUserIndex >
1448         static_cast<uint8_t>(SpecialUserIndex::atScaleDebugUser))
1449     {
1450         phosphor::logging::log<phosphor::logging::level::ERR>(
1451             "ipmiOEMSetSpecialUserPassword: Invalid user account");
1452         return ipmi::responseParmOutOfRange();
1453     }
1454     if (userPassword.size() != 0)
1455     {
1456         constexpr uint8_t minPasswordSizeRequired = 6;
1457         SecureString passwd;
1458         if (userPassword.size() < minPasswordSizeRequired ||
1459             userPassword.size() > ipmi::maxIpmi20PasswordSize)
1460         {
1461             OPENSSL_cleanse(userPassword.data(), userPassword.size());
1462             return ipmi::responseReqDataLenInvalid();
1463         }
1464         passwd.assign(reinterpret_cast<const char*>(userPassword.data()),
1465                       userPassword.size());
1466         // Clear sensitive data
1467         OPENSSL_cleanse(userPassword.data(), userPassword.size());
1468         if (specialUserIndex ==
1469             static_cast<uint8_t>(SpecialUserIndex::atScaleDebugUser))
1470         {
1471             status = ipmiSetSpecialUserPassword("asdbg", passwd);
1472 
1473             atScaleDebugEventlog("AtScaleDebugSpecialUserEnabled", LOG_CRIT);
1474         }
1475         else
1476         {
1477             status = ipmiSetSpecialUserPassword("root", passwd);
1478         }
1479         return ipmi::response(status);
1480     }
1481     else
1482     {
1483         if (specialUserIndex ==
1484             static_cast<uint8_t>(SpecialUserIndex::rootUser))
1485         {
1486             status = executeCmd("passwd -d root");
1487         }
1488         else
1489         {
1490             status = executeCmd("passwd -d asdbg");
1491 
1492             if (status == 0)
1493             {
1494                 atScaleDebugEventlog("AtScaleDebugSpecialUserDisabled",
1495                                      LOG_INFO);
1496             }
1497         }
1498         return ipmi::response(status);
1499     }
1500 }
1501 
1502 namespace ledAction
1503 {
1504 using namespace sdbusplus::xyz::openbmc_project::Led::server;
1505 std::map<Physical::Action, uint8_t> actionDbusToIpmi = {
1506     {Physical::Action::Off, 0},
1507     {Physical::Action::On, 2},
1508     {Physical::Action::Blink, 1}};
1509 
1510 std::map<uint8_t, std::string> offsetObjPath = {
1511     {2, statusAmberObjPath}, {4, statusGreenObjPath}, {6, identifyLEDObjPath}};
1512 
1513 } // namespace ledAction
1514 
getLEDState(sdbusplus::bus_t & bus,const std::string & intf,const std::string & objPath,uint8_t & state)1515 int8_t getLEDState(sdbusplus::bus_t& bus, const std::string& intf,
1516                    const std::string& objPath, uint8_t& state)
1517 {
1518     try
1519     {
1520         std::string service = getService(bus, intf, objPath);
1521         Value stateValue =
1522             getDbusProperty(bus, service, objPath, intf, "State");
1523         std::string strState = std::get<std::string>(stateValue);
1524         state = ledAction::actionDbusToIpmi.at(
1525             sdbusplus::xyz::openbmc_project::Led::server::Physical::
1526                 convertActionFromString(strState));
1527     }
1528     catch (const sdbusplus::exception_t& e)
1529     {
1530         phosphor::logging::log<phosphor::logging::level::ERR>(e.what());
1531         return -1;
1532     }
1533     return 0;
1534 }
1535 
ipmiOEMGetLEDStatus()1536 ipmi::RspType<uint8_t> ipmiOEMGetLEDStatus()
1537 {
1538     uint8_t ledstate = 0;
1539     phosphor::logging::log<phosphor::logging::level::DEBUG>("GET led status");
1540     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1541     for (auto it = ledAction::offsetObjPath.begin();
1542          it != ledAction::offsetObjPath.end(); ++it)
1543     {
1544         uint8_t state = 0;
1545         if (getLEDState(*dbus, ledIntf, it->second, state) == -1)
1546         {
1547             phosphor::logging::log<phosphor::logging::level::ERR>(
1548                 "oem_get_led_status: fail to get ID LED status!");
1549             return ipmi::responseUnspecifiedError();
1550         }
1551         ledstate |= state << it->first;
1552     }
1553     return ipmi::responseSuccess(ledstate);
1554 }
1555 
ipmiOEMCfgHostSerialPortSpeed(ipmi_netfn_t,ipmi_cmd_t,ipmi_request_t request,ipmi_response_t response,ipmi_data_len_t dataLen,ipmi_context_t)1556 ipmi_ret_t ipmiOEMCfgHostSerialPortSpeed(
1557     ipmi_netfn_t, ipmi_cmd_t, ipmi_request_t request, ipmi_response_t response,
1558     ipmi_data_len_t dataLen, ipmi_context_t)
1559 {
1560     CfgHostSerialReq* req = reinterpret_cast<CfgHostSerialReq*>(request);
1561     uint8_t* resp = reinterpret_cast<uint8_t*>(response);
1562 
1563     if (*dataLen == 0)
1564     {
1565         phosphor::logging::log<phosphor::logging::level::ERR>(
1566             "CfgHostSerial: invalid input len!",
1567             phosphor::logging::entry("LEN=%d", *dataLen));
1568         return IPMI_CC_REQ_DATA_LEN_INVALID;
1569     }
1570 
1571     switch (req->command)
1572     {
1573         case getHostSerialCfgCmd:
1574         {
1575             if (*dataLen != 1)
1576             {
1577                 phosphor::logging::log<phosphor::logging::level::ERR>(
1578                     "CfgHostSerial: invalid input len!");
1579                 *dataLen = 0;
1580                 return IPMI_CC_REQ_DATA_LEN_INVALID;
1581             }
1582 
1583             *dataLen = 0;
1584 
1585             boost::process::ipstream is;
1586             std::vector<std::string> data;
1587             std::string line;
1588             boost::process::child c1(fwGetEnvCmd, "-n", fwHostSerailCfgEnvName,
1589                                      boost::process::std_out > is);
1590 
1591             while (c1.running() && std::getline(is, line) && !line.empty())
1592             {
1593                 data.push_back(line);
1594             }
1595 
1596             c1.wait();
1597             if (c1.exit_code())
1598             {
1599                 phosphor::logging::log<phosphor::logging::level::ERR>(
1600                     "CfgHostSerial:: error on execute",
1601                     phosphor::logging::entry("EXECUTE=%s", fwSetEnvCmd));
1602                 // Using the default value
1603                 *resp = 0;
1604             }
1605             else
1606             {
1607                 if (data.size() != 1)
1608                 {
1609                     phosphor::logging::log<phosphor::logging::level::ERR>(
1610                         "CfgHostSerial:: error on read env");
1611                     return IPMI_CC_UNSPECIFIED_ERROR;
1612                 }
1613                 try
1614                 {
1615                     unsigned long tmp = std::stoul(data[0]);
1616                     if (tmp > std::numeric_limits<uint8_t>::max())
1617                     {
1618                         throw std::out_of_range("Out of range");
1619                     }
1620                     *resp = static_cast<uint8_t>(tmp);
1621                 }
1622                 catch (const std::invalid_argument& e)
1623                 {
1624                     phosphor::logging::log<phosphor::logging::level::ERR>(
1625                         "invalid config ",
1626                         phosphor::logging::entry("ERR=%s", e.what()));
1627                     return IPMI_CC_UNSPECIFIED_ERROR;
1628                 }
1629                 catch (const std::out_of_range& e)
1630                 {
1631                     phosphor::logging::log<phosphor::logging::level::ERR>(
1632                         "out_of_range config ",
1633                         phosphor::logging::entry("ERR=%s", e.what()));
1634                     return IPMI_CC_UNSPECIFIED_ERROR;
1635                 }
1636             }
1637 
1638             *dataLen = 1;
1639             break;
1640         }
1641         case setHostSerialCfgCmd:
1642         {
1643             if (*dataLen != sizeof(CfgHostSerialReq))
1644             {
1645                 phosphor::logging::log<phosphor::logging::level::ERR>(
1646                     "CfgHostSerial: invalid input len!");
1647                 *dataLen = 0;
1648                 return IPMI_CC_REQ_DATA_LEN_INVALID;
1649             }
1650 
1651             *dataLen = 0;
1652 
1653             if (req->parameter > HostSerialCfgParamMax)
1654             {
1655                 phosphor::logging::log<phosphor::logging::level::ERR>(
1656                     "CfgHostSerial: invalid input!");
1657                 return IPMI_CC_INVALID_FIELD_REQUEST;
1658             }
1659 
1660             boost::process::child c1(fwSetEnvCmd, fwHostSerailCfgEnvName,
1661                                      std::to_string(req->parameter));
1662 
1663             c1.wait();
1664             if (c1.exit_code())
1665             {
1666                 phosphor::logging::log<phosphor::logging::level::ERR>(
1667                     "CfgHostSerial:: error on execute",
1668                     phosphor::logging::entry("EXECUTE=%s", fwGetEnvCmd));
1669                 return IPMI_CC_UNSPECIFIED_ERROR;
1670             }
1671             break;
1672         }
1673         default:
1674             phosphor::logging::log<phosphor::logging::level::ERR>(
1675                 "CfgHostSerial: invalid input!");
1676             *dataLen = 0;
1677             return IPMI_CC_INVALID_FIELD_REQUEST;
1678     }
1679 
1680     return IPMI_CC_OK;
1681 }
1682 
1683 constexpr const char* thermalModeInterface =
1684     "xyz.openbmc_project.Control.ThermalMode";
1685 constexpr const char* thermalModePath =
1686     "/xyz/openbmc_project/control/thermal_mode";
1687 
getFanProfileInterface(sdbusplus::bus_t & bus,boost::container::flat_map<std::string,ipmi::DbusVariant> & resp)1688 bool getFanProfileInterface(
1689     sdbusplus::bus_t& bus,
1690     boost::container::flat_map<std::string, ipmi::DbusVariant>& resp)
1691 {
1692     auto call = bus.new_method_call(settingsBusName, thermalModePath, PROP_INTF,
1693                                     "GetAll");
1694     call.append(thermalModeInterface);
1695     try
1696     {
1697         auto data = bus.call(call);
1698         data.read(resp);
1699     }
1700     catch (const sdbusplus::exception_t& e)
1701     {
1702         phosphor::logging::log<phosphor::logging::level::ERR>(
1703             "getFanProfileInterface: can't get thermal mode!",
1704             phosphor::logging::entry("ERR=%s", e.what()));
1705         return false;
1706     }
1707     return true;
1708 }
1709 
1710 /**@brief implements the OEM set fan config.
1711  * @param selectedFanProfile - fan profile to enable
1712  * @param reserved1
1713  * @param performanceMode - Performance/Acoustic mode
1714  * @param reserved2
1715  * @param setPerformanceMode - set Performance/Acoustic mode
1716  * @param setFanProfile - set fan profile
1717  *
1718  * @return IPMI completion code.
1719  **/
ipmiOEMSetFanConfig(uint8_t selectedFanProfile,uint2_t reserved1,bool performanceMode,uint3_t reserved2,bool setPerformanceMode,bool setFanProfile,std::optional<uint8_t> dimmGroupId,std::optional<uint32_t> dimmPresenceBitmap)1720 ipmi::RspType<> ipmiOEMSetFanConfig(
1721     [[maybe_unused]] uint8_t selectedFanProfile, uint2_t reserved1,
1722     bool performanceMode, uint3_t reserved2, bool setPerformanceMode,
1723     [[maybe_unused]] bool setFanProfile, std::optional<uint8_t> dimmGroupId,
1724     [[maybe_unused]] std::optional<uint32_t> dimmPresenceBitmap)
1725 {
1726     if (reserved1 || reserved2)
1727     {
1728         return ipmi::responseInvalidFieldRequest();
1729     }
1730 
1731     if (dimmGroupId)
1732     {
1733         if (*dimmGroupId >= maxCPUNum)
1734         {
1735             return ipmi::responseInvalidFieldRequest();
1736         }
1737         if (!cpuPresent("cpu" + std::to_string(*dimmGroupId)))
1738         {
1739             return ipmi::responseInvalidFieldRequest();
1740         }
1741     }
1742 
1743     // todo: tell bios to only send first 2 bytes
1744     boost::container::flat_map<std::string, ipmi::DbusVariant> profileData;
1745     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1746     if (!getFanProfileInterface(*dbus, profileData))
1747     {
1748         return ipmi::responseUnspecifiedError();
1749     }
1750 
1751     std::vector<std::string>* supported =
1752         std::get_if<std::vector<std::string>>(&profileData["Supported"]);
1753     if (supported == nullptr)
1754     {
1755         return ipmi::responseInvalidFieldRequest();
1756     }
1757     std::string mode;
1758     if (setPerformanceMode)
1759     {
1760         if (performanceMode)
1761         {
1762             if (std::find(supported->begin(), supported->end(),
1763                           "Performance") != supported->end())
1764             {
1765                 mode = "Performance";
1766             }
1767         }
1768         else
1769         {
1770             if (std::find(supported->begin(), supported->end(), "Acoustic") !=
1771                 supported->end())
1772             {
1773                 mode = "Acoustic";
1774             }
1775         }
1776         if (mode.empty())
1777         {
1778             return ipmi::responseInvalidFieldRequest();
1779         }
1780 
1781         try
1782         {
1783             setDbusProperty(*dbus, settingsBusName, thermalModePath,
1784                             thermalModeInterface, "Current", mode);
1785         }
1786         catch (const sdbusplus::exception_t& e)
1787         {
1788             phosphor::logging::log<phosphor::logging::level::ERR>(
1789                 "ipmiOEMSetFanConfig: can't set thermal mode!",
1790                 phosphor::logging::entry("EXCEPTION=%s", e.what()));
1791             return ipmi::responseResponseError();
1792         }
1793     }
1794 
1795     return ipmi::responseSuccess();
1796 }
1797 
1798 ipmi::RspType<uint8_t, // profile support map
1799               uint8_t, // fan control profile enable
1800               uint8_t, // flags
1801               uint32_t // dimm presence bit map
1802               >
ipmiOEMGetFanConfig(uint8_t dimmGroupId)1803     ipmiOEMGetFanConfig(uint8_t dimmGroupId)
1804 {
1805     if (dimmGroupId >= maxCPUNum)
1806     {
1807         return ipmi::responseInvalidFieldRequest();
1808     }
1809 
1810     bool cpuStatus = cpuPresent("cpu" + std::to_string(dimmGroupId));
1811 
1812     if (!cpuStatus)
1813     {
1814         return ipmi::responseInvalidFieldRequest();
1815     }
1816 
1817     boost::container::flat_map<std::string, ipmi::DbusVariant> profileData;
1818 
1819     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1820     if (!getFanProfileInterface(*dbus, profileData))
1821     {
1822         return ipmi::responseResponseError();
1823     }
1824 
1825     std::string* current = std::get_if<std::string>(&profileData["Current"]);
1826 
1827     if (current == nullptr)
1828     {
1829         phosphor::logging::log<phosphor::logging::level::ERR>(
1830             "ipmiOEMGetFanConfig: can't get current mode!");
1831         return ipmi::responseResponseError();
1832     }
1833     bool performance = (*current == "Performance");
1834 
1835     uint8_t flags = 0;
1836     if (performance)
1837     {
1838         flags |= 1 << 2;
1839     }
1840 
1841     constexpr uint8_t fanControlDefaultProfile = 0x80;
1842     constexpr uint8_t fanControlProfileState = 0x00;
1843     constexpr uint32_t dimmPresenceBitmap = 0x00;
1844 
1845     return ipmi::responseSuccess(fanControlDefaultProfile,
1846                                  fanControlProfileState, flags,
1847                                  dimmPresenceBitmap);
1848 }
1849 constexpr const char* cfmLimitSettingPath =
1850     "/xyz/openbmc_project/control/cfm_limit";
1851 constexpr const char* cfmLimitIface = "xyz.openbmc_project.Control.CFMLimit";
1852 constexpr const size_t legacyExitAirSensorNumber = 0x2e;
1853 constexpr const size_t legacyPCHSensorNumber = 0x22;
1854 constexpr const char* exitAirPathName = "Exit_Air";
1855 constexpr const char* pchPathName = "SSB_Temp";
1856 constexpr const char* pidConfigurationIface =
1857     "xyz.openbmc_project.Configuration.Pid";
1858 
getConfigPath(const std::string & name)1859 static std::string getConfigPath(const std::string& name)
1860 {
1861     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1862     auto method =
1863         dbus->new_method_call("xyz.openbmc_project.ObjectMapper",
1864                               "/xyz/openbmc_project/object_mapper",
1865                               "xyz.openbmc_project.ObjectMapper", "GetSubTree");
1866 
1867     method.append("/", 0, std::array<const char*, 1>{pidConfigurationIface});
1868     std::string path;
1869     GetSubTreeType resp;
1870     try
1871     {
1872         auto reply = dbus->call(method);
1873         reply.read(resp);
1874     }
1875     catch (const sdbusplus::exception_t&)
1876     {
1877         phosphor::logging::log<phosphor::logging::level::ERR>(
1878             "ipmiOEMGetFscParameter: mapper error");
1879     };
1880     auto config =
1881         std::find_if(resp.begin(), resp.end(), [&name](const auto& pair) {
1882             return pair.first.find(name) != std::string::npos;
1883         });
1884     if (config != resp.end())
1885     {
1886         path = std::move(config->first);
1887     }
1888     return path;
1889 }
1890 
1891 // flat map to make alphabetical
getPidConfigs()1892 static boost::container::flat_map<std::string, PropertyMap> getPidConfigs()
1893 {
1894     boost::container::flat_map<std::string, PropertyMap> ret;
1895     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1896     auto method =
1897         dbus->new_method_call("xyz.openbmc_project.ObjectMapper",
1898                               "/xyz/openbmc_project/object_mapper",
1899                               "xyz.openbmc_project.ObjectMapper", "GetSubTree");
1900 
1901     method.append("/", 0, std::array<const char*, 1>{pidConfigurationIface});
1902     GetSubTreeType resp;
1903 
1904     try
1905     {
1906         auto reply = dbus->call(method);
1907         reply.read(resp);
1908     }
1909     catch (const sdbusplus::exception_t&)
1910     {
1911         phosphor::logging::log<phosphor::logging::level::ERR>(
1912             "getFanConfigPaths: mapper error");
1913     };
1914     for (const auto& [path, objects] : resp)
1915     {
1916         if (objects.empty())
1917         {
1918             continue; // should be impossible
1919         }
1920 
1921         try
1922         {
1923             ret.emplace(path,
1924                         getAllDbusProperties(*dbus, objects[0].first, path,
1925                                              pidConfigurationIface));
1926         }
1927         catch (const sdbusplus::exception_t& e)
1928         {
1929             phosphor::logging::log<phosphor::logging::level::ERR>(
1930                 "getPidConfigs: can't get DbusProperties!",
1931                 phosphor::logging::entry("ERR=%s", e.what()));
1932         }
1933     }
1934     return ret;
1935 }
1936 
ipmiOEMGetFanSpeedOffset(void)1937 ipmi::RspType<uint8_t> ipmiOEMGetFanSpeedOffset(void)
1938 {
1939     boost::container::flat_map<std::string, PropertyMap> data = getPidConfigs();
1940     if (data.empty())
1941     {
1942         return ipmi::responseResponseError();
1943     }
1944     uint8_t minOffset = std::numeric_limits<uint8_t>::max();
1945     for (const auto& [_, pid] : data)
1946     {
1947         auto findClass = pid.find("Class");
1948         if (findClass == pid.end())
1949         {
1950             phosphor::logging::log<phosphor::logging::level::ERR>(
1951                 "ipmiOEMGetFscParameter: found illegal pid "
1952                 "configurations");
1953             return ipmi::responseResponseError();
1954         }
1955         std::string type = std::get<std::string>(findClass->second);
1956         if (type == "fan")
1957         {
1958             auto findOutLimit = pid.find("OutLimitMin");
1959             if (findOutLimit == pid.end())
1960             {
1961                 phosphor::logging::log<phosphor::logging::level::ERR>(
1962                     "ipmiOEMGetFscParameter: found illegal pid "
1963                     "configurations");
1964                 return ipmi::responseResponseError();
1965             }
1966             // get the min out of all the offsets
1967             minOffset = std::min(
1968                 minOffset,
1969                 static_cast<uint8_t>(std::get<double>(findOutLimit->second)));
1970         }
1971     }
1972     if (minOffset == std::numeric_limits<uint8_t>::max())
1973     {
1974         phosphor::logging::log<phosphor::logging::level::ERR>(
1975             "ipmiOEMGetFscParameter: found no fan configurations!");
1976         return ipmi::responseResponseError();
1977     }
1978 
1979     return ipmi::responseSuccess(minOffset);
1980 }
1981 
ipmiOEMSetFanSpeedOffset(uint8_t offset)1982 ipmi::RspType<> ipmiOEMSetFanSpeedOffset(uint8_t offset)
1983 {
1984     constexpr uint8_t maxFanSpeedOffset = 100;
1985     if (offset > maxFanSpeedOffset)
1986     {
1987         phosphor::logging::log<phosphor::logging::level::ERR>(
1988             "ipmiOEMSetFanSpeedOffset: fan offset greater than limit");
1989         return ipmi::responseInvalidFieldRequest();
1990     }
1991     boost::container::flat_map<std::string, PropertyMap> data = getPidConfigs();
1992     if (data.empty())
1993     {
1994         phosphor::logging::log<phosphor::logging::level::ERR>(
1995             "ipmiOEMSetFanSpeedOffset: found no pid configurations!");
1996         return ipmi::responseResponseError();
1997     }
1998 
1999     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
2000     bool found = false;
2001     for (const auto& [path, pid] : data)
2002     {
2003         auto findClass = pid.find("Class");
2004         if (findClass == pid.end())
2005         {
2006             phosphor::logging::log<phosphor::logging::level::ERR>(
2007                 "ipmiOEMSetFanSpeedOffset: found illegal pid "
2008                 "configurations");
2009             return ipmi::responseResponseError();
2010         }
2011         std::string type = std::get<std::string>(findClass->second);
2012         if (type == "fan")
2013         {
2014             auto findOutLimit = pid.find("OutLimitMin");
2015             if (findOutLimit == pid.end())
2016             {
2017                 phosphor::logging::log<phosphor::logging::level::ERR>(
2018                     "ipmiOEMSetFanSpeedOffset: found illegal pid "
2019                     "configurations");
2020                 return ipmi::responseResponseError();
2021             }
2022             ipmi::setDbusProperty(*dbus, "xyz.openbmc_project.EntityManager",
2023                                   path, pidConfigurationIface, "OutLimitMin",
2024                                   static_cast<double>(offset));
2025             found = true;
2026         }
2027     }
2028     if (!found)
2029     {
2030         phosphor::logging::log<phosphor::logging::level::ERR>(
2031             "ipmiOEMSetFanSpeedOffset: set no fan offsets");
2032         return ipmi::responseResponseError();
2033     }
2034 
2035     return ipmi::responseSuccess();
2036 }
2037 
ipmiOEMSetFscParameter(uint8_t command,uint8_t param1,uint8_t param2)2038 ipmi::RspType<> ipmiOEMSetFscParameter(uint8_t command, uint8_t param1,
2039                                        uint8_t param2)
2040 {
2041     constexpr const size_t disableLimiting = 0x0;
2042 
2043     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
2044     if (command == static_cast<uint8_t>(setFscParamFlags::tcontrol))
2045     {
2046         std::string pathName;
2047         if (param1 == legacyExitAirSensorNumber)
2048         {
2049             pathName = exitAirPathName;
2050         }
2051         else if (param1 == legacyPCHSensorNumber)
2052         {
2053             pathName = pchPathName;
2054         }
2055         else
2056         {
2057             return ipmi::responseParmOutOfRange();
2058         }
2059         std::string path = getConfigPath(pathName);
2060         ipmi::setDbusProperty(*dbus, "xyz.openbmc_project.EntityManager", path,
2061                               pidConfigurationIface, "SetPoint",
2062                               static_cast<double>(param2));
2063         return ipmi::responseSuccess();
2064     }
2065     else if (command == static_cast<uint8_t>(setFscParamFlags::cfm))
2066     {
2067         uint16_t cfm = param1 | (static_cast<uint16_t>(param2) << 8);
2068 
2069         // must be greater than 50 based on eps
2070         if (cfm < 50 && cfm != disableLimiting)
2071         {
2072             return ipmi::responseParmOutOfRange();
2073         }
2074 
2075         try
2076         {
2077             ipmi::setDbusProperty(*dbus, settingsBusName, cfmLimitSettingPath,
2078                                   cfmLimitIface, "Limit",
2079                                   static_cast<double>(cfm));
2080         }
2081         catch (const sdbusplus::exception_t& e)
2082         {
2083             phosphor::logging::log<phosphor::logging::level::ERR>(
2084                 "ipmiOEMSetFscParameter: can't set cfm setting!",
2085                 phosphor::logging::entry("ERR=%s", e.what()));
2086             return ipmi::responseResponseError();
2087         }
2088         return ipmi::responseSuccess();
2089     }
2090     else if (command == static_cast<uint8_t>(setFscParamFlags::maxPwm))
2091     {
2092         uint8_t requestedDomainMask = param1;
2093         boost::container::flat_map data = getPidConfigs();
2094         if (data.empty())
2095         {
2096             phosphor::logging::log<phosphor::logging::level::ERR>(
2097                 "ipmiOEMSetFscParameter: found no pid configurations!");
2098             return ipmi::responseResponseError();
2099         }
2100         size_t count = 0;
2101         for (const auto& [path, pid] : data)
2102         {
2103             auto findClass = pid.find("Class");
2104             if (findClass == pid.end())
2105             {
2106                 phosphor::logging::log<phosphor::logging::level::ERR>(
2107                     "ipmiOEMSetFscParameter: found illegal pid "
2108                     "configurations");
2109                 return ipmi::responseResponseError();
2110             }
2111             std::string type = std::get<std::string>(findClass->second);
2112             if (type == "fan")
2113             {
2114                 if (requestedDomainMask & (1 << count))
2115                 {
2116                     ipmi::setDbusProperty(
2117                         *dbus, "xyz.openbmc_project.EntityManager", path,
2118                         pidConfigurationIface, "OutLimitMax",
2119                         static_cast<double>(param2));
2120                 }
2121                 count++;
2122             }
2123         }
2124         return ipmi::responseSuccess();
2125     }
2126     else
2127     {
2128         // todo other command parts possibly
2129         // tcontrol is handled in peci now
2130         // fan speed offset not implemented yet
2131         // domain pwm limit not implemented
2132         return ipmi::responseParmOutOfRange();
2133     }
2134 }
2135 
2136 ipmi::RspType<
2137     std::variant<uint8_t, std::array<uint8_t, 2>, std::array<uint16_t, 2>>>
ipmiOEMGetFscParameter(uint8_t command,std::optional<uint8_t> param)2138     ipmiOEMGetFscParameter(uint8_t command, std::optional<uint8_t> param)
2139 {
2140     constexpr uint8_t legacyDefaultSetpoint = -128;
2141 
2142     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
2143     if (command == static_cast<uint8_t>(setFscParamFlags::tcontrol))
2144     {
2145         if (!param)
2146         {
2147             return ipmi::responseReqDataLenInvalid();
2148         }
2149 
2150         std::string pathName;
2151 
2152         if (*param == legacyExitAirSensorNumber)
2153         {
2154             pathName = exitAirPathName;
2155         }
2156         else if (*param == legacyPCHSensorNumber)
2157         {
2158             pathName = pchPathName;
2159         }
2160         else
2161         {
2162             return ipmi::responseParmOutOfRange();
2163         }
2164 
2165         uint8_t setpoint = legacyDefaultSetpoint;
2166         std::string path = getConfigPath(pathName);
2167         if (path.size())
2168         {
2169             Value val = ipmi::getDbusProperty(
2170                 *dbus, "xyz.openbmc_project.EntityManager", path,
2171                 pidConfigurationIface, "SetPoint");
2172             setpoint = std::floor(std::get<double>(val) + 0.5);
2173         }
2174 
2175         // old implementation used to return the "default" and current, we
2176         // don't make the default readily available so just make both the
2177         // same
2178 
2179         return ipmi::responseSuccess(
2180             std::array<uint8_t, 2>{setpoint, setpoint});
2181     }
2182     else if (command == static_cast<uint8_t>(setFscParamFlags::maxPwm))
2183     {
2184         constexpr const size_t maxDomainCount = 8;
2185 
2186         if (!param)
2187         {
2188             return ipmi::responseReqDataLenInvalid();
2189         }
2190         uint8_t requestedDomain = *param;
2191         if (requestedDomain >= maxDomainCount)
2192         {
2193             return ipmi::responseInvalidFieldRequest();
2194         }
2195 
2196         boost::container::flat_map data = getPidConfigs();
2197         if (data.empty())
2198         {
2199             phosphor::logging::log<phosphor::logging::level::ERR>(
2200                 "ipmiOEMGetFscParameter: found no pid configurations!");
2201             return ipmi::responseResponseError();
2202         }
2203         size_t count = 0;
2204         for (const auto& [_, pid] : data)
2205         {
2206             auto findClass = pid.find("Class");
2207             if (findClass == pid.end())
2208             {
2209                 phosphor::logging::log<phosphor::logging::level::ERR>(
2210                     "ipmiOEMGetFscParameter: found illegal pid "
2211                     "configurations");
2212                 return ipmi::responseResponseError();
2213             }
2214             std::string type = std::get<std::string>(findClass->second);
2215             if (type == "fan")
2216             {
2217                 if (requestedDomain == count)
2218                 {
2219                     auto findOutLimit = pid.find("OutLimitMax");
2220                     if (findOutLimit == pid.end())
2221                     {
2222                         phosphor::logging::log<phosphor::logging::level::ERR>(
2223                             "ipmiOEMGetFscParameter: found illegal pid "
2224                             "configurations");
2225                         return ipmi::responseResponseError();
2226                     }
2227 
2228                     return ipmi::responseSuccess(
2229                         static_cast<uint8_t>(std::floor(
2230                             std::get<double>(findOutLimit->second) + 0.5)));
2231                 }
2232                 else
2233                 {
2234                     count++;
2235                 }
2236             }
2237         }
2238 
2239         return ipmi::responseInvalidFieldRequest();
2240     }
2241     else if (command == static_cast<uint8_t>(setFscParamFlags::cfm))
2242     {
2243         /*
2244         DataLen should be 1, but host is sending us an extra bit. As the
2245         previous behavior didn't seem to prevent this, ignore the check for
2246         now.
2247 
2248         if (param)
2249         {
2250             phosphor::logging::log<phosphor::logging::level::ERR>(
2251                 "ipmiOEMGetFscParameter: invalid input len!");
2252             return IPMI_CC_REQ_DATA_LEN_INVALID;
2253         }
2254         */
2255         Value cfmLimit;
2256         Value cfmMaximum;
2257         try
2258         {
2259             cfmLimit = ipmi::getDbusProperty(*dbus, settingsBusName,
2260                                              cfmLimitSettingPath, cfmLimitIface,
2261                                              "Limit");
2262             cfmMaximum = ipmi::getDbusProperty(
2263                 *dbus, "xyz.openbmc_project.ExitAirTempSensor",
2264                 "/xyz/openbmc_project/control/MaxCFM", cfmLimitIface, "Limit");
2265         }
2266         catch (const sdbusplus::exception_t& e)
2267         {
2268             phosphor::logging::log<phosphor::logging::level::ERR>(
2269                 "ipmiOEMGetFscParameter: can't get cfm setting!",
2270                 phosphor::logging::entry("ERR=%s", e.what()));
2271             return ipmi::responseResponseError();
2272         }
2273 
2274         double cfmMax = std::get<double>(cfmMaximum);
2275         double cfmLim = std::get<double>(cfmLimit);
2276 
2277         cfmLim = std::floor(cfmLim + 0.5);
2278         cfmMax = std::floor(cfmMax + 0.5);
2279         uint16_t cfmLimResp = static_cast<uint16_t>(cfmLim);
2280         uint16_t cfmMaxResp = static_cast<uint16_t>(cfmMax);
2281 
2282         return ipmi::responseSuccess(
2283             std::array<uint16_t, 2>{cfmLimResp, cfmMaxResp});
2284     }
2285 
2286     else
2287     {
2288         // todo other command parts possibly
2289         // domain pwm limit not implemented
2290         return ipmi::responseParmOutOfRange();
2291     }
2292 }
2293 
2294 using crConfigVariant = ipmi::DbusVariant;
2295 
setCRConfig(ipmi::Context::ptr & ctx,const std::string & property,const crConfigVariant & value,std::chrono::microseconds timeout=ipmi::IPMI_DBUS_TIMEOUT)2296 int setCRConfig(ipmi::Context::ptr& ctx, const std::string& property,
2297                 const crConfigVariant& value,
2298                 [[maybe_unused]] std::chrono::microseconds timeout =
2299                     ipmi::IPMI_DBUS_TIMEOUT)
2300 {
2301     boost::system::error_code ec;
2302     ctx->bus->yield_method_call<void>(
2303         ctx->yield, ec, "xyz.openbmc_project.PSURedundancy",
2304         "/xyz/openbmc_project/control/power_supply_redundancy",
2305         "org.freedesktop.DBus.Properties", "Set",
2306         "xyz.openbmc_project.Control.PowerSupplyRedundancy", property, value);
2307     if (ec)
2308     {
2309         phosphor::logging::log<phosphor::logging::level::ERR>(
2310             "Failed to set dbus property to cold redundancy");
2311         return -1;
2312     }
2313 
2314     return 0;
2315 }
2316 
getCRConfig(ipmi::Context::ptr & ctx,const std::string & property,crConfigVariant & value,const std::string & service="xyz.openbmc_project.PSURedundancy",std::chrono::microseconds timeout=ipmi::IPMI_DBUS_TIMEOUT)2317 int getCRConfig(
2318     ipmi::Context::ptr& ctx, const std::string& property,
2319     crConfigVariant& value,
2320     const std::string& service = "xyz.openbmc_project.PSURedundancy",
2321     [[maybe_unused]] std::chrono::microseconds timeout =
2322         ipmi::IPMI_DBUS_TIMEOUT)
2323 {
2324     boost::system::error_code ec;
2325     value = ctx->bus->yield_method_call<crConfigVariant>(
2326         ctx->yield, ec, service,
2327         "/xyz/openbmc_project/control/power_supply_redundancy",
2328         "org.freedesktop.DBus.Properties", "Get",
2329         "xyz.openbmc_project.Control.PowerSupplyRedundancy", property);
2330     if (ec)
2331     {
2332         phosphor::logging::log<phosphor::logging::level::ERR>(
2333             "Failed to get dbus property to cold redundancy");
2334         return -1;
2335     }
2336     return 0;
2337 }
2338 
getPSUCount(void)2339 uint8_t getPSUCount(void)
2340 {
2341     std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
2342     ipmi::Value num;
2343     try
2344     {
2345         num = ipmi::getDbusProperty(
2346             *dbus, "xyz.openbmc_project.PSURedundancy",
2347             "/xyz/openbmc_project/control/power_supply_redundancy",
2348             "xyz.openbmc_project.Control.PowerSupplyRedundancy", "PSUNumber");
2349     }
2350     catch (const sdbusplus::exception_t& e)
2351     {
2352         phosphor::logging::log<phosphor::logging::level::ERR>(
2353             "Failed to get PSUNumber property from dbus interface");
2354         return 0;
2355     }
2356     uint8_t* pNum = std::get_if<uint8_t>(&num);
2357     if (!pNum)
2358     {
2359         phosphor::logging::log<phosphor::logging::level::ERR>(
2360             "Error to get PSU Number");
2361         return 0;
2362     }
2363     return *pNum;
2364 }
2365 
validateCRAlgo(std::vector<uint8_t> & conf,uint8_t num)2366 bool validateCRAlgo(std::vector<uint8_t>& conf, uint8_t num)
2367 {
2368     if (conf.size() < num)
2369     {
2370         phosphor::logging::log<phosphor::logging::level::ERR>(
2371             "Invalid PSU Ranking");
2372         return false;
2373     }
2374     std::set<uint8_t> confSet;
2375     for (uint8_t i = 0; i < num; i++)
2376     {
2377         if (conf[i] > num)
2378         {
2379             phosphor::logging::log<phosphor::logging::level::ERR>(
2380                 "PSU Ranking is larger than current PSU number");
2381             return false;
2382         }
2383         confSet.emplace(conf[i]);
2384     }
2385 
2386     if (confSet.size() != num)
2387     {
2388         phosphor::logging::log<phosphor::logging::level::ERR>(
2389             "duplicate PSU Ranking");
2390         return false;
2391     }
2392     return true;
2393 }
2394 
2395 enum class crParameter
2396 {
2397     crStatus = 0,
2398     crFeature = 1,
2399     rotationFeature = 2,
2400     rotationAlgo = 3,
2401     rotationPeriod = 4,
2402     numOfPSU = 5,
2403     rotationRankOrderEffective = 6
2404 };
2405 
2406 constexpr ipmi::Cc ccParameterNotSupported = 0x80;
2407 static const constexpr uint32_t oneDay = 0x15180;
2408 static const constexpr uint32_t oneMonth = 0xf53700;
2409 static const constexpr uint8_t userSpecific = 0x01;
2410 static const constexpr uint8_t crSetCompleted = 0;
ipmiOEMSetCRConfig(ipmi::Context::ptr & ctx,uint8_t parameter,ipmi::message::Payload & payload)2411 ipmi::RspType<uint8_t> ipmiOEMSetCRConfig(
2412     ipmi::Context::ptr& ctx, uint8_t parameter, ipmi::message::Payload& payload)
2413 {
2414     switch (static_cast<crParameter>(parameter))
2415     {
2416         case crParameter::rotationFeature:
2417         {
2418             uint8_t param1;
2419             if (payload.unpack(param1) || !payload.fullyUnpacked())
2420             {
2421                 return ipmi::responseReqDataLenInvalid();
2422             }
2423             // Rotation Enable can only be true or false
2424             if (param1 > 1)
2425             {
2426                 return ipmi::responseInvalidFieldRequest();
2427             }
2428             if (setCRConfig(ctx, "RotationEnabled", static_cast<bool>(param1)))
2429             {
2430                 return ipmi::responseResponseError();
2431             }
2432             break;
2433         }
2434         case crParameter::rotationAlgo:
2435         {
2436             // Rotation Algorithm can only be 0-BMC Specific or 1-User Specific
2437             std::string algoName;
2438             uint8_t param1;
2439             if (payload.unpack(param1))
2440             {
2441                 return ipmi::responseReqDataLenInvalid();
2442             }
2443             switch (param1)
2444             {
2445                 case 0:
2446                     algoName = "xyz.openbmc_project.Control."
2447                                "PowerSupplyRedundancy.Algo.bmcSpecific";
2448                     break;
2449                 case 1:
2450                     algoName = "xyz.openbmc_project.Control."
2451                                "PowerSupplyRedundancy.Algo.userSpecific";
2452                     break;
2453                 default:
2454                     return ipmi::responseInvalidFieldRequest();
2455             }
2456             if (setCRConfig(ctx, "RotationAlgorithm", algoName))
2457             {
2458                 return ipmi::responseResponseError();
2459             }
2460 
2461             uint8_t numberOfPSU = getPSUCount();
2462             if (!numberOfPSU)
2463             {
2464                 return ipmi::responseResponseError();
2465             }
2466             std::vector<uint8_t> rankOrder;
2467 
2468             if (param1 == userSpecific)
2469             {
2470                 if (payload.unpack(rankOrder) || !payload.fullyUnpacked())
2471                 {
2472                     ipmi::responseReqDataLenInvalid();
2473                 }
2474                 if (rankOrder.size() != numberOfPSU)
2475                 {
2476                     return ipmi::responseReqDataLenInvalid();
2477                 }
2478 
2479                 if (!validateCRAlgo(rankOrder, numberOfPSU))
2480                 {
2481                     return ipmi::responseInvalidFieldRequest();
2482                 }
2483             }
2484             else
2485             {
2486                 if (rankOrder.size() > 0)
2487                 {
2488                     return ipmi::responseReqDataLenInvalid();
2489                 }
2490                 for (uint8_t i = 1; i <= numberOfPSU; i++)
2491                 {
2492                     rankOrder.emplace_back(i);
2493                 }
2494             }
2495             if (setCRConfig(ctx, "RotationRankOrder", rankOrder))
2496             {
2497                 return ipmi::responseResponseError();
2498             }
2499             break;
2500         }
2501         case crParameter::rotationPeriod:
2502         {
2503             // Minimum Rotation period is  One day (86400 seconds) and Max
2504             // Rotation Period is 6 month (0xf53700 seconds)
2505             uint32_t period;
2506             if (payload.unpack(period) || !payload.fullyUnpacked())
2507             {
2508                 return ipmi::responseReqDataLenInvalid();
2509             }
2510             if ((period < oneDay) || (period > oneMonth))
2511             {
2512                 return ipmi::responseInvalidFieldRequest();
2513             }
2514             if (setCRConfig(ctx, "PeriodOfRotation", period))
2515             {
2516                 return ipmi::responseResponseError();
2517             }
2518             break;
2519         }
2520         default:
2521         {
2522             return ipmi::response(ccParameterNotSupported);
2523         }
2524     }
2525 
2526     return ipmi::responseSuccess(crSetCompleted);
2527 }
2528 
2529 ipmi::RspType<uint8_t, std::variant<uint8_t, uint32_t, std::vector<uint8_t>>>
ipmiOEMGetCRConfig(ipmi::Context::ptr & ctx,uint8_t parameter)2530     ipmiOEMGetCRConfig(ipmi::Context::ptr& ctx, uint8_t parameter)
2531 {
2532     crConfigVariant value;
2533     switch (static_cast<crParameter>(parameter))
2534     {
2535         case crParameter::crStatus:
2536         {
2537             if (getCRConfig(ctx, "ColdRedundancyStatus", value))
2538             {
2539                 return ipmi::responseResponseError();
2540             }
2541             std::string* pStatus = std::get_if<std::string>(&value);
2542             if (!pStatus)
2543             {
2544                 phosphor::logging::log<phosphor::logging::level::ERR>(
2545                     "Error to get ColdRedundancyStatus property");
2546                 return ipmi::responseResponseError();
2547             }
2548             namespace server = sdbusplus::xyz::openbmc_project::Control::server;
2549             auto status =
2550                 server::PowerSupplyRedundancy::convertStatusFromString(
2551                     *pStatus);
2552             switch (status)
2553             {
2554                 case server::PowerSupplyRedundancy::Status::inProgress:
2555                     return ipmi::responseSuccess(parameter,
2556                                                  static_cast<uint8_t>(1));
2557 
2558                 case server::PowerSupplyRedundancy::Status::completed:
2559                     return ipmi::responseSuccess(parameter,
2560                                                  static_cast<uint8_t>(0));
2561                 default:
2562                     phosphor::logging::log<phosphor::logging::level::ERR>(
2563                         "Error to get valid status");
2564                     return ipmi::responseResponseError();
2565             }
2566         }
2567         case crParameter::crFeature:
2568         {
2569             if (getCRConfig(ctx, "PowerSupplyRedundancyEnabled", value))
2570             {
2571                 return ipmi::responseResponseError();
2572             }
2573             bool* pResponse = std::get_if<bool>(&value);
2574             if (!pResponse)
2575             {
2576                 phosphor::logging::log<phosphor::logging::level::ERR>(
2577                     "Error to get PowerSupplyRedundancyEnabled property");
2578                 return ipmi::responseResponseError();
2579             }
2580 
2581             return ipmi::responseSuccess(parameter,
2582                                          static_cast<uint8_t>(*pResponse));
2583         }
2584         case crParameter::rotationFeature:
2585         {
2586             if (getCRConfig(ctx, "RotationEnabled", value))
2587             {
2588                 return ipmi::responseResponseError();
2589             }
2590             bool* pResponse = std::get_if<bool>(&value);
2591             if (!pResponse)
2592             {
2593                 phosphor::logging::log<phosphor::logging::level::ERR>(
2594                     "Error to get RotationEnabled property");
2595                 return ipmi::responseResponseError();
2596             }
2597             return ipmi::responseSuccess(parameter,
2598                                          static_cast<uint8_t>(*pResponse));
2599         }
2600         case crParameter::rotationAlgo:
2601         {
2602             if (getCRConfig(ctx, "RotationAlgorithm", value))
2603             {
2604                 return ipmi::responseResponseError();
2605             }
2606 
2607             std::string* pAlgo = std::get_if<std::string>(&value);
2608             if (!pAlgo)
2609             {
2610                 phosphor::logging::log<phosphor::logging::level::ERR>(
2611                     "Error to get RotationAlgorithm property");
2612                 return ipmi::responseResponseError();
2613             }
2614             std::vector<uint8_t> response;
2615             namespace server = sdbusplus::xyz::openbmc_project::Control::server;
2616             auto algo =
2617                 server::PowerSupplyRedundancy::convertAlgoFromString(*pAlgo);
2618 
2619             switch (algo)
2620             {
2621                 case server::PowerSupplyRedundancy::Algo::bmcSpecific:
2622                     response.push_back(0);
2623                     break;
2624                 case server::PowerSupplyRedundancy::Algo::userSpecific:
2625                     response.push_back(1);
2626                     break;
2627                 default:
2628                     phosphor::logging::log<phosphor::logging::level::ERR>(
2629                         "Error to get valid algo");
2630                     return ipmi::responseResponseError();
2631             }
2632 
2633             if (getCRConfig(ctx, "RotationRankOrder", value))
2634             {
2635                 return ipmi::responseResponseError();
2636             }
2637             std::vector<uint8_t>* pResponse =
2638                 std::get_if<std::vector<uint8_t>>(&value);
2639             if (!pResponse)
2640             {
2641                 phosphor::logging::log<phosphor::logging::level::ERR>(
2642                     "Error to get RotationRankOrder property");
2643                 return ipmi::responseResponseError();
2644             }
2645 
2646             std::copy(pResponse->begin(), pResponse->end(),
2647                       std::back_inserter(response));
2648 
2649             return ipmi::responseSuccess(parameter, response);
2650         }
2651         case crParameter::rotationPeriod:
2652         {
2653             if (getCRConfig(ctx, "PeriodOfRotation", value))
2654             {
2655                 return ipmi::responseResponseError();
2656             }
2657             uint32_t* pResponse = std::get_if<uint32_t>(&value);
2658             if (!pResponse)
2659             {
2660                 phosphor::logging::log<phosphor::logging::level::ERR>(
2661                     "Error to get RotationAlgorithm property");
2662                 return ipmi::responseResponseError();
2663             }
2664             return ipmi::responseSuccess(parameter, *pResponse);
2665         }
2666         case crParameter::numOfPSU:
2667         {
2668             uint8_t numberOfPSU = getPSUCount();
2669             if (!numberOfPSU)
2670             {
2671                 return ipmi::responseResponseError();
2672             }
2673             return ipmi::responseSuccess(parameter, numberOfPSU);
2674         }
2675         case crParameter::rotationRankOrderEffective:
2676         {
2677             if (getCRConfig(ctx, "RotationRankOrder", value,
2678                             "xyz.openbmc_project.PSURedundancy"))
2679             {
2680                 return ipmi::responseResponseError();
2681             }
2682             std::vector<uint8_t>* pResponse =
2683                 std::get_if<std::vector<uint8_t>>(&value);
2684             if (!pResponse)
2685             {
2686                 phosphor::logging::log<phosphor::logging::level::ERR>(
2687                     "Error to get effective RotationRankOrder property");
2688                 return ipmi::responseResponseError();
2689             }
2690             return ipmi::responseSuccess(parameter, *pResponse);
2691         }
2692         default:
2693         {
2694             return ipmi::response(ccParameterNotSupported);
2695         }
2696     }
2697 }
2698 
ipmiOEMSetFaultIndication(uint8_t sourceId,uint8_t faultType,uint8_t faultState,uint8_t faultGroup,std::array<uint8_t,8> & ledStateData)2699 ipmi::RspType<> ipmiOEMSetFaultIndication(
2700     uint8_t sourceId, uint8_t faultType, uint8_t faultState, uint8_t faultGroup,
2701     std::array<uint8_t, 8>& ledStateData)
2702 {
2703     constexpr auto maxFaultType = static_cast<size_t>(RemoteFaultType::max);
2704     static const std::array<std::string, maxFaultType> faultNames = {
2705         "faultFan",       "faultTemp",     "faultPower",
2706         "faultDriveSlot", "faultSoftware", "faultMemory"};
2707 
2708     constexpr uint8_t maxFaultSource = 0x4;
2709     constexpr uint8_t skipLEDs = 0xFF;
2710     constexpr uint8_t pinSize = 64;
2711     constexpr uint8_t groupSize = 16;
2712     constexpr uint8_t groupNum = 5; // 4 for fault memory, 1 for faultFan
2713 
2714     // same pin names need to be defined in dts file
2715     static const std::array<std::array<std::string, groupSize>, groupNum>
2716         faultLedPinNames = {{
2717             "LED_CPU1_CH1_DIMM1_FAULT",
2718             "LED_CPU1_CH1_DIMM2_FAULT",
2719             "LED_CPU1_CH2_DIMM1_FAULT",
2720             "LED_CPU1_CH2_DIMM2_FAULT",
2721             "LED_CPU1_CH3_DIMM1_FAULT",
2722             "LED_CPU1_CH3_DIMM2_FAULT",
2723             "LED_CPU1_CH4_DIMM1_FAULT",
2724             "LED_CPU1_CH4_DIMM2_FAULT",
2725             "LED_CPU1_CH5_DIMM1_FAULT",
2726             "LED_CPU1_CH5_DIMM2_FAULT",
2727             "LED_CPU1_CH6_DIMM1_FAULT",
2728             "LED_CPU1_CH6_DIMM2_FAULT",
2729             "",
2730             "",
2731             "",
2732             "", // end of group1
2733             "LED_CPU2_CH1_DIMM1_FAULT",
2734             "LED_CPU2_CH1_DIMM2_FAULT",
2735             "LED_CPU2_CH2_DIMM1_FAULT",
2736             "LED_CPU2_CH2_DIMM2_FAULT",
2737             "LED_CPU2_CH3_DIMM1_FAULT",
2738             "LED_CPU2_CH3_DIMM2_FAULT",
2739             "LED_CPU2_CH4_DIMM1_FAULT",
2740             "LED_CPU2_CH4_DIMM2_FAULT",
2741             "LED_CPU2_CH5_DIMM1_FAULT",
2742             "LED_CPU2_CH5_DIMM2_FAULT",
2743             "LED_CPU2_CH6_DIMM1_FAULT",
2744             "LED_CPU2_CH6_DIMM2_FAULT",
2745             "",
2746             "",
2747             "",
2748             "", // endof group2
2749             "LED_CPU3_CH1_DIMM1_FAULT",
2750             "LED_CPU3_CH1_DIMM2_FAULT",
2751             "LED_CPU3_CH2_DIMM1_FAULT",
2752             "LED_CPU3_CH2_DIMM2_FAULT",
2753             "LED_CPU3_CH3_DIMM1_FAULT",
2754             "LED_CPU3_CH3_DIMM2_FAULT",
2755             "LED_CPU3_CH4_DIMM1_FAULT",
2756             "LED_CPU3_CH4_DIMM2_FAULT",
2757             "LED_CPU3_CH5_DIMM1_FAULT",
2758             "LED_CPU3_CH5_DIMM2_FAULT",
2759             "LED_CPU3_CH6_DIMM1_FAULT",
2760             "LED_CPU3_CH6_DIMM2_FAULT",
2761             "",
2762             "",
2763             "",
2764             "", // end of group3
2765             "LED_CPU4_CH1_DIMM1_FAULT",
2766             "LED_CPU4_CH1_DIMM2_FAULT",
2767             "LED_CPU4_CH2_DIMM1_FAULT",
2768             "LED_CPU4_CH2_DIMM2_FAULT",
2769             "LED_CPU4_CH3_DIMM1_FAULT",
2770             "LED_CPU4_CH3_DIMM2_FAULT",
2771             "LED_CPU4_CH4_DIMM1_FAULT",
2772             "LED_CPU4_CH4_DIMM2_FAULT",
2773             "LED_CPU4_CH5_DIMM1_FAULT",
2774             "LED_CPU4_CH5_DIMM2_FAULT",
2775             "LED_CPU4_CH6_DIMM1_FAULT",
2776             "LED_CPU4_CH6_DIMM2_FAULT",
2777             "",
2778             "",
2779             "",
2780             "", // end of group4
2781             "LED_FAN1_FAULT",
2782             "LED_FAN2_FAULT",
2783             "LED_FAN3_FAULT",
2784             "LED_FAN4_FAULT",
2785             "LED_FAN5_FAULT",
2786             "LED_FAN6_FAULT",
2787             "LED_FAN7_FAULT",
2788             "LED_FAN8_FAULT",
2789             "",
2790             "",
2791             "",
2792             "",
2793             "",
2794             "",
2795             "",
2796             "" // end of group5
2797         }};
2798 
2799     // Validate the source, fault type --
2800     // (Byte 1) sourceId: Unspecified, Hot-Swap Controller 0, Hot-Swap
2801     // Controller 1, BIOS (Byte 2) fault type: fan, temperature, power,
2802     // driveslot, software, memory (Byte 3) FaultState: OK, Degraded,
2803     // Non-Critical, Critical, Non-Recoverable, (Byte 4) is faultGroup,
2804     // definition differs based on fault type (Byte 2)
2805     //          Type Fan=> Group: 0=FanGroupID, FF-not used
2806     //                  Byte 5-11 00h, not used
2807     //                  Byte12 FanLedState [7:0]-Fans 7:0
2808     //          Type Memory=> Group: 0 = DIMM GroupID, FF-not used
2809     //                  Byte 5:12 - DIMM LED state (64bit field, LS Byte first)
2810     //                  [63:48] = CPU4 channels 7:0, 2 bits per channel
2811     //                  [47:32] = CPU3 channels 7:0, 2 bits per channel
2812     //                  [31:16] = CPU2 channels 7:0, 2 bits per channel
2813     //                  [15:0] =  CPU1 channels 7:0, 2 bits per channel
2814     //          Type Other=> Component Fault LED Group ID, not used set to 0xFF
2815     //                  Byte[5:12]: reserved 0x00h
2816     if ((sourceId >= maxFaultSource) ||
2817         (faultType >= static_cast<int8_t>(RemoteFaultType::max)) ||
2818         (faultState >= static_cast<int8_t>(RemoteFaultState::maxFaultState)) ||
2819         (faultGroup >= static_cast<int8_t>(DimmFaultType::maxFaultGroup)))
2820     {
2821         return ipmi::responseParmOutOfRange();
2822     }
2823 
2824     size_t pinGroupOffset = 0;
2825     size_t pinGroupMax = pinSize / groupSize;
2826     if (RemoteFaultType::fan == RemoteFaultType(faultType))
2827     {
2828         pinGroupOffset = 4;
2829         pinGroupMax = groupNum - pinSize / groupSize;
2830     }
2831 
2832     switch (RemoteFaultType(faultType))
2833     {
2834         case (RemoteFaultType::fan):
2835         case (RemoteFaultType::memory):
2836         {
2837             if (faultGroup == skipLEDs)
2838             {
2839                 return ipmi::responseSuccess();
2840             }
2841             // calculate led state bit filed count, each byte has 8bits
2842             // the maximum bits will be 8 * 8 bits
2843             constexpr uint8_t size = sizeof(ledStateData) * 8;
2844 
2845             // assemble ledState
2846             uint64_t ledState = 0;
2847             bool hasError = false;
2848             for (size_t i = 0; i < sizeof(ledStateData); i++)
2849             {
2850                 ledState = (uint64_t)(ledState << 8);
2851                 ledState = (uint64_t)(ledState | (uint64_t)ledStateData[i]);
2852             }
2853             std::bitset<size> ledStateBits(ledState);
2854 
2855             for (size_t group = 0; group < pinGroupMax; group++)
2856             {
2857                 for (int i = 0; i < groupSize; i++)
2858                 { // skip non-existing pins
2859                     if (0 == faultLedPinNames[group + pinGroupOffset][i].size())
2860                     {
2861                         continue;
2862                     }
2863 
2864                     gpiod::line line = gpiod::find_line(
2865                         faultLedPinNames[group + pinGroupOffset][i]);
2866                     if (!line)
2867                     {
2868                         phosphor::logging::log<phosphor::logging::level::ERR>(
2869                             "Not Find Led Gpio Device!",
2870                             phosphor::logging::entry(
2871                                 "DEVICE=%s",
2872                                 faultLedPinNames[group + pinGroupOffset][i]
2873                                     .c_str()));
2874                         hasError = true;
2875                         continue;
2876                     }
2877 
2878                     bool activeHigh =
2879                         (line.active_state() == gpiod::line::ACTIVE_HIGH);
2880                     try
2881                     {
2882                         line.request(
2883                             {"faultLed", gpiod::line_request::DIRECTION_OUTPUT,
2884                              activeHigh
2885                                  ? 0
2886                                  : gpiod::line_request::FLAG_ACTIVE_LOW});
2887                         line.set_value(ledStateBits[i + group * groupSize]);
2888                     }
2889                     catch (const std::system_error&)
2890                     {
2891                         phosphor::logging::log<phosphor::logging::level::ERR>(
2892                             "Error write Led Gpio Device!",
2893                             phosphor::logging::entry(
2894                                 "DEVICE=%s",
2895                                 faultLedPinNames[group + pinGroupOffset][i]
2896                                     .c_str()));
2897                         hasError = true;
2898                         continue;
2899                     }
2900                 } // for int i
2901             }
2902             if (hasError)
2903             {
2904                 return ipmi::responseResponseError();
2905             }
2906             break;
2907         }
2908         default:
2909         {
2910             // now only support two fault types
2911             return ipmi::responseParmOutOfRange();
2912         }
2913     } // switch
2914     return ipmi::responseSuccess();
2915 }
2916 
ipmiOEMReadBoardProductId()2917 ipmi::RspType<uint8_t> ipmiOEMReadBoardProductId()
2918 {
2919     uint8_t prodId = 0;
2920     try
2921     {
2922         std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
2923         const DbusObjectInfo& object = getDbusObject(
2924             *dbus, "xyz.openbmc_project.Inventory.Item.Board",
2925             "/xyz/openbmc_project/inventory/system/board/", "Baseboard");
2926         const Value& propValue = getDbusProperty(
2927             *dbus, object.second, object.first,
2928             "xyz.openbmc_project.Inventory.Item.Board.Motherboard",
2929             "ProductId");
2930         prodId = static_cast<uint8_t>(std::get<uint64_t>(propValue));
2931     }
2932     catch (const std::exception& e)
2933     {
2934         phosphor::logging::log<phosphor::logging::level::ERR>(
2935             "ipmiOEMReadBoardProductId: Product ID read failed!",
2936             phosphor::logging::entry("ERR=%s", e.what()));
2937     }
2938     return ipmi::responseSuccess(prodId);
2939 }
2940 
2941 /** @brief implements the get security mode command
2942  *  @param ctx - ctx pointer
2943  *
2944  *  @returns IPMI completion code with following data
2945  *   - restriction mode value - As specified in
2946  * xyz.openbmc_project.Control.Security.RestrictionMode.interface.yaml
2947  *   - special mode value - As specified in
2948  * xyz.openbmc_project.Control.Security.SpecialMode.interface.yaml
2949  */
ipmiGetSecurityMode(ipmi::Context::ptr & ctx)2950 ipmi::RspType<uint8_t, uint8_t> ipmiGetSecurityMode(ipmi::Context::ptr& ctx)
2951 {
2952     namespace securityNameSpace =
2953         sdbusplus::xyz::openbmc_project::Control::Security::server;
2954     uint8_t restrictionModeValue = 0;
2955     uint8_t specialModeValue = 0;
2956 
2957     boost::system::error_code ec;
2958     auto varRestrMode = ctx->bus->yield_method_call<ipmi::DbusVariant>(
2959         ctx->yield, ec, restricionModeService, restricionModeBasePath,
2960         dBusPropertyIntf, dBusPropertyGetMethod, restricionModeIntf,
2961         restricionModeProperty);
2962     if (ec)
2963     {
2964         phosphor::logging::log<phosphor::logging::level::ERR>(
2965             "ipmiGetSecurityMode: failed to get RestrictionMode property",
2966             phosphor::logging::entry("ERROR=%s", ec.message().c_str()));
2967         return ipmi::responseUnspecifiedError();
2968     }
2969     restrictionModeValue = static_cast<uint8_t>(
2970         securityNameSpace::RestrictionMode::convertModesFromString(
2971             std::get<std::string>(varRestrMode)));
2972     auto varSpecialMode = ctx->bus->yield_method_call<ipmi::DbusVariant>(
2973         ctx->yield, ec, specialModeService, specialModeBasePath,
2974         dBusPropertyIntf, dBusPropertyGetMethod, specialModeIntf,
2975         specialModeProperty);
2976     if (ec)
2977     {
2978         phosphor::logging::log<phosphor::logging::level::ERR>(
2979             "ipmiGetSecurityMode: failed to get SpecialMode property",
2980             phosphor::logging::entry("ERROR=%s", ec.message().c_str()));
2981         // fall through, let us not worry about SpecialMode property, which is
2982         // not required in user scenario
2983     }
2984     else
2985     {
2986         specialModeValue = static_cast<uint8_t>(
2987             securityNameSpace::SpecialMode::convertModesFromString(
2988                 std::get<std::string>(varSpecialMode)));
2989     }
2990     return ipmi::responseSuccess(restrictionModeValue, specialModeValue);
2991 }
2992 
2993 /** @brief implements the set security mode command
2994  *  Command allows to upgrade the restriction mode and won't allow
2995  *  to downgrade from system interface
2996  *  @param ctx - ctx pointer
2997  *  @param restrictionMode - restriction mode value to be set.
2998  *
2999  *  @returns IPMI completion code
3000  */
ipmiSetSecurityMode(ipmi::Context::ptr & ctx,uint8_t restrictionMode,std::optional<uint8_t> specialMode)3001 ipmi::RspType<> ipmiSetSecurityMode(ipmi::Context::ptr& ctx,
3002                                     uint8_t restrictionMode,
3003                                     std::optional<uint8_t> specialMode)
3004 {
3005 #ifndef BMC_VALIDATION_UNSECURE_FEATURE
3006     if (specialMode)
3007     {
3008         return ipmi::responseReqDataLenInvalid();
3009     }
3010 #endif
3011     namespace securityNameSpace =
3012         sdbusplus::xyz::openbmc_project::Control::Security::server;
3013 
3014     ChannelInfo chInfo;
3015     if (getChannelInfo(ctx->channel, chInfo) != ccSuccess)
3016     {
3017         phosphor::logging::log<phosphor::logging::level::ERR>(
3018             "ipmiSetSecurityMode: Failed to get Channel Info",
3019             phosphor::logging::entry("CHANNEL=%d", ctx->channel));
3020         return ipmi::responseUnspecifiedError();
3021     }
3022     auto reqMode =
3023         static_cast<securityNameSpace::RestrictionMode::Modes>(restrictionMode);
3024 
3025     if ((reqMode < securityNameSpace::RestrictionMode::Modes::Provisioning) ||
3026         (reqMode >
3027          securityNameSpace::RestrictionMode::Modes::ProvisionedHostDisabled))
3028     {
3029         return ipmi::responseInvalidFieldRequest();
3030     }
3031 
3032     boost::system::error_code ec;
3033     auto varRestrMode = ctx->bus->yield_method_call<ipmi::DbusVariant>(
3034         ctx->yield, ec, restricionModeService, restricionModeBasePath,
3035         dBusPropertyIntf, dBusPropertyGetMethod, restricionModeIntf,
3036         restricionModeProperty);
3037     if (ec)
3038     {
3039         phosphor::logging::log<phosphor::logging::level::ERR>(
3040             "ipmiSetSecurityMode: failed to get RestrictionMode property",
3041             phosphor::logging::entry("ERROR=%s", ec.message().c_str()));
3042         return ipmi::responseUnspecifiedError();
3043     }
3044     auto currentRestrictionMode =
3045         securityNameSpace::RestrictionMode::convertModesFromString(
3046             std::get<std::string>(varRestrMode));
3047 
3048     if (chInfo.mediumType !=
3049             static_cast<uint8_t>(EChannelMediumType::lan8032) &&
3050         currentRestrictionMode > reqMode)
3051     {
3052         phosphor::logging::log<phosphor::logging::level::ERR>(
3053             "ipmiSetSecurityMode - Downgrading security mode not supported "
3054             "through system interface",
3055             phosphor::logging::entry(
3056                 "CUR_MODE=%d", static_cast<uint8_t>(currentRestrictionMode)),
3057             phosphor::logging::entry("REQ_MODE=%d", restrictionMode));
3058         return ipmi::responseCommandNotAvailable();
3059     }
3060 
3061     ec.clear();
3062     ctx->bus->yield_method_call<>(
3063         ctx->yield, ec, restricionModeService, restricionModeBasePath,
3064         dBusPropertyIntf, dBusPropertySetMethod, restricionModeIntf,
3065         restricionModeProperty,
3066         static_cast<ipmi::DbusVariant>(
3067             securityNameSpace::convertForMessage(reqMode)));
3068 
3069     if (ec)
3070     {
3071         phosphor::logging::log<phosphor::logging::level::ERR>(
3072             "ipmiSetSecurityMode: failed to set RestrictionMode property",
3073             phosphor::logging::entry("ERROR=%s", ec.message().c_str()));
3074         return ipmi::responseUnspecifiedError();
3075     }
3076 
3077 #ifdef BMC_VALIDATION_UNSECURE_FEATURE
3078     if (specialMode)
3079     {
3080         constexpr uint8_t mfgMode = 0x01;
3081         // Manufacturing mode is reserved. So can't enable this mode.
3082         if (specialMode.value() == mfgMode)
3083         {
3084             phosphor::logging::log<phosphor::logging::level::INFO>(
3085                 "ipmiSetSecurityMode: Can't enable Manufacturing mode");
3086             return ipmi::responseInvalidFieldRequest();
3087         }
3088 
3089         ec.clear();
3090         ctx->bus->yield_method_call<>(
3091             ctx->yield, ec, specialModeService, specialModeBasePath,
3092             dBusPropertyIntf, dBusPropertySetMethod, specialModeIntf,
3093             specialModeProperty,
3094             static_cast<ipmi::DbusVariant>(securityNameSpace::convertForMessage(
3095                 static_cast<securityNameSpace::SpecialMode::Modes>(
3096                     specialMode.value()))));
3097 
3098         if (ec)
3099         {
3100             phosphor::logging::log<phosphor::logging::level::ERR>(
3101                 "ipmiSetSecurityMode: failed to set SpecialMode property",
3102                 phosphor::logging::entry("ERROR=%s", ec.message().c_str()));
3103             return ipmi::responseUnspecifiedError();
3104         }
3105     }
3106 #endif
3107     return ipmi::responseSuccess();
3108 }
3109 
3110 ipmi::RspType<uint8_t /* restore status */>
ipmiRestoreConfiguration(const std::array<uint8_t,3> & clr,uint8_t cmd)3111     ipmiRestoreConfiguration(const std::array<uint8_t, 3>& clr, uint8_t cmd)
3112 {
3113     static constexpr std::array<uint8_t, 3> expClr = {'C', 'L', 'R'};
3114 
3115     if (clr != expClr)
3116     {
3117         return ipmi::responseInvalidFieldRequest();
3118     }
3119     constexpr uint8_t cmdStatus = 0;
3120     constexpr uint8_t cmdDefaultRestore = 0xaa;
3121     constexpr uint8_t cmdFullRestore = 0xbb;
3122     constexpr uint8_t cmdFormat = 0xcc;
3123 
3124     constexpr const char* restoreOpFname = "/tmp/.rwfs/.restore_op";
3125 
3126     switch (cmd)
3127     {
3128         case cmdStatus:
3129             break;
3130         case cmdDefaultRestore:
3131         case cmdFullRestore:
3132         case cmdFormat:
3133         {
3134             // write file to rwfs root
3135             int value = (cmd - 1) & 0x03; // map aa, bb, cc => 1, 2, 3
3136             std::ofstream restoreFile(restoreOpFname);
3137             if (!restoreFile)
3138             {
3139                 return ipmi::responseUnspecifiedError();
3140             }
3141             restoreFile << value << "\n";
3142 
3143             phosphor::logging::log<phosphor::logging::level::WARNING>(
3144                 "Restore to default will be performed on next BMC boot",
3145                 phosphor::logging::entry("ACTION=0x%0X", cmd));
3146 
3147             break;
3148         }
3149         default:
3150             return ipmi::responseInvalidFieldRequest();
3151     }
3152 
3153     constexpr uint8_t restorePending = 0;
3154     constexpr uint8_t restoreComplete = 1;
3155 
3156     uint8_t restoreStatus = std::filesystem::exists(restoreOpFname)
3157                                 ? restorePending
3158                                 : restoreComplete;
3159     return ipmi::responseSuccess(restoreStatus);
3160 }
3161 
ipmiOEMGetNmiSource(void)3162 ipmi::RspType<uint8_t> ipmiOEMGetNmiSource(void)
3163 {
3164     uint8_t bmcSource;
3165     namespace nmi = sdbusplus::xyz::openbmc_project::Chassis::Control::server;
3166 
3167     try
3168     {
3169         std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
3170         std::string service =
3171             getService(*dbus, oemNmiSourceIntf, oemNmiSourceObjPath);
3172         Value variant =
3173             getDbusProperty(*dbus, service, oemNmiSourceObjPath,
3174                             oemNmiSourceIntf, oemNmiBmcSourceObjPathProp);
3175 
3176         switch (nmi::NMISource::convertBMCSourceSignalFromString(
3177             std::get<std::string>(variant)))
3178         {
3179             case nmi::NMISource::BMCSourceSignal::None:
3180                 bmcSource = static_cast<uint8_t>(NmiSource::none);
3181                 break;
3182             case nmi::NMISource::BMCSourceSignal::FrontPanelButton:
3183                 bmcSource = static_cast<uint8_t>(NmiSource::frontPanelButton);
3184                 break;
3185             case nmi::NMISource::BMCSourceSignal::Watchdog:
3186                 bmcSource = static_cast<uint8_t>(NmiSource::watchdog);
3187                 break;
3188             case nmi::NMISource::BMCSourceSignal::ChassisCmd:
3189                 bmcSource = static_cast<uint8_t>(NmiSource::chassisCmd);
3190                 break;
3191             case nmi::NMISource::BMCSourceSignal::MemoryError:
3192                 bmcSource = static_cast<uint8_t>(NmiSource::memoryError);
3193                 break;
3194             case nmi::NMISource::BMCSourceSignal::PciBusError:
3195                 bmcSource = static_cast<uint8_t>(NmiSource::pciBusError);
3196                 break;
3197             case nmi::NMISource::BMCSourceSignal::PCH:
3198                 bmcSource = static_cast<uint8_t>(NmiSource::pch);
3199                 break;
3200             case nmi::NMISource::BMCSourceSignal::Chipset:
3201                 bmcSource = static_cast<uint8_t>(NmiSource::chipset);
3202                 break;
3203             default:
3204                 phosphor::logging::log<phosphor::logging::level::ERR>(
3205                     "NMI source: invalid property!",
3206                     phosphor::logging::entry(
3207                         "PROP=%s", std::get<std::string>(variant).c_str()));
3208                 return ipmi::responseResponseError();
3209         }
3210     }
3211     catch (const sdbusplus::exception_t& e)
3212     {
3213         phosphor::logging::log<phosphor::logging::level::ERR>(e.what());
3214         return ipmi::responseResponseError();
3215     }
3216 
3217     return ipmi::responseSuccess(bmcSource);
3218 }
3219 
ipmiOEMSetNmiSource(uint8_t sourceId)3220 ipmi::RspType<> ipmiOEMSetNmiSource(uint8_t sourceId)
3221 {
3222     namespace nmi = sdbusplus::xyz::openbmc_project::Chassis::Control::server;
3223 
3224     nmi::NMISource::BMCSourceSignal bmcSourceSignal =
3225         nmi::NMISource::BMCSourceSignal::None;
3226 
3227     switch (NmiSource(sourceId))
3228     {
3229         case NmiSource::none:
3230             bmcSourceSignal = nmi::NMISource::BMCSourceSignal::None;
3231             break;
3232         case NmiSource::frontPanelButton:
3233             bmcSourceSignal = nmi::NMISource::BMCSourceSignal::FrontPanelButton;
3234             break;
3235         case NmiSource::watchdog:
3236             bmcSourceSignal = nmi::NMISource::BMCSourceSignal::Watchdog;
3237             break;
3238         case NmiSource::chassisCmd:
3239             bmcSourceSignal = nmi::NMISource::BMCSourceSignal::ChassisCmd;
3240             break;
3241         case NmiSource::memoryError:
3242             bmcSourceSignal = nmi::NMISource::BMCSourceSignal::MemoryError;
3243             break;
3244         case NmiSource::pciBusError:
3245             bmcSourceSignal = nmi::NMISource::BMCSourceSignal::PciBusError;
3246             break;
3247         case NmiSource::pch:
3248             bmcSourceSignal = nmi::NMISource::BMCSourceSignal::PCH;
3249             break;
3250         case NmiSource::chipset:
3251             bmcSourceSignal = nmi::NMISource::BMCSourceSignal::Chipset;
3252             break;
3253         default:
3254             phosphor::logging::log<phosphor::logging::level::ERR>(
3255                 "NMI source: invalid property!");
3256             return ipmi::responseResponseError();
3257     }
3258 
3259     try
3260     {
3261         // keep NMI signal source
3262         std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
3263         std::string service =
3264             getService(*dbus, oemNmiSourceIntf, oemNmiSourceObjPath);
3265         setDbusProperty(*dbus, service, oemNmiSourceObjPath, oemNmiSourceIntf,
3266                         oemNmiBmcSourceObjPathProp,
3267                         nmi::convertForMessage(bmcSourceSignal));
3268         // set Enabled property to inform NMI source handling
3269         // to trigger a NMI_OUT BSOD.
3270         // if it's triggered by NMI source property changed,
3271         // NMI_OUT BSOD could be missed if the same source occurs twice in a row
3272         if (bmcSourceSignal != nmi::NMISource::BMCSourceSignal::None)
3273         {
3274             setDbusProperty(*dbus, service, oemNmiSourceObjPath,
3275                             oemNmiSourceIntf, oemNmiEnabledObjPathProp,
3276                             static_cast<bool>(true));
3277         }
3278     }
3279     catch (const sdbusplus::exception_t& e)
3280     {
3281         phosphor::logging::log<phosphor::logging::level::ERR>(e.what());
3282         return ipmi::responseResponseError();
3283     }
3284 
3285     return ipmi::responseSuccess();
3286 }
3287 
3288 namespace dimmOffset
3289 {
3290 constexpr const char* dimmPower = "DimmPower";
3291 constexpr const char* staticCltt = "StaticCltt";
3292 constexpr const char* offsetPath = "/xyz/openbmc_project/Inventory/Item/Dimm";
3293 constexpr const char* offsetInterface =
3294     "xyz.openbmc_project.Inventory.Item.Dimm.Offset";
3295 constexpr const char* property = "DimmOffset";
3296 
3297 }; // namespace dimmOffset
3298 
ipmiOEMSetDimmOffset(uint8_t type,const std::vector<std::tuple<uint8_t,uint8_t>> & data)3299 ipmi::RspType<> ipmiOEMSetDimmOffset(
3300     uint8_t type, const std::vector<std::tuple<uint8_t, uint8_t>>& data)
3301 {
3302     if (type != static_cast<uint8_t>(dimmOffsetTypes::dimmPower) &&
3303         type != static_cast<uint8_t>(dimmOffsetTypes::staticCltt))
3304     {
3305         return ipmi::responseInvalidFieldRequest();
3306     }
3307 
3308     if (data.empty())
3309     {
3310         return ipmi::responseInvalidFieldRequest();
3311     }
3312     nlohmann::json json;
3313 
3314     std::ifstream jsonStream(dimmOffsetFile);
3315     if (jsonStream.good())
3316     {
3317         json = nlohmann::json::parse(jsonStream, nullptr, false);
3318         if (json.is_discarded())
3319         {
3320             json = nlohmann::json();
3321         }
3322         jsonStream.close();
3323     }
3324 
3325     std::string typeName;
3326     if (type == static_cast<uint8_t>(dimmOffsetTypes::dimmPower))
3327     {
3328         typeName = dimmOffset::dimmPower;
3329     }
3330     else
3331     {
3332         typeName = dimmOffset::staticCltt;
3333     }
3334 
3335     nlohmann::json& field = json[typeName];
3336 
3337     for (const auto& [index, value] : data)
3338     {
3339         field[index] = value;
3340     }
3341 
3342     for (nlohmann::json& val : field)
3343     {
3344         if (val == nullptr)
3345         {
3346             val = static_cast<uint8_t>(0);
3347         }
3348     }
3349 
3350     std::ofstream output(dimmOffsetFile);
3351     if (!output.good())
3352     {
3353         std::cerr << "Error writing json file\n";
3354         return ipmi::responseResponseError();
3355     }
3356 
3357     output << json.dump(4);
3358 
3359     if (type == static_cast<uint8_t>(dimmOffsetTypes::staticCltt))
3360     {
3361         std::shared_ptr<sdbusplus::asio::connection> bus = getSdBus();
3362 
3363         ipmi::DbusVariant offsets = field.get<std::vector<uint8_t>>();
3364         auto call = bus->new_method_call(
3365             settingsBusName, dimmOffset::offsetPath, PROP_INTF, "Set");
3366         call.append(dimmOffset::offsetInterface, dimmOffset::property, offsets);
3367         try
3368         {
3369             bus->call(call);
3370         }
3371         catch (const sdbusplus::exception_t& e)
3372         {
3373             phosphor::logging::log<phosphor::logging::level::ERR>(
3374                 "ipmiOEMSetDimmOffset: can't set dimm offsets!",
3375                 phosphor::logging::entry("ERR=%s", e.what()));
3376             return ipmi::responseResponseError();
3377         }
3378     }
3379 
3380     return ipmi::responseSuccess();
3381 }
3382 
ipmiOEMGetDimmOffset(uint8_t type,uint8_t index)3383 ipmi::RspType<uint8_t> ipmiOEMGetDimmOffset(uint8_t type, uint8_t index)
3384 {
3385     if (type != static_cast<uint8_t>(dimmOffsetTypes::dimmPower) &&
3386         type != static_cast<uint8_t>(dimmOffsetTypes::staticCltt))
3387     {
3388         return ipmi::responseInvalidFieldRequest();
3389     }
3390 
3391     std::ifstream jsonStream(dimmOffsetFile);
3392 
3393     auto json = nlohmann::json::parse(jsonStream, nullptr, false);
3394     if (json.is_discarded())
3395     {
3396         std::cerr << "File error in " << dimmOffsetFile << "\n";
3397         return ipmi::responseResponseError();
3398     }
3399 
3400     std::string typeName;
3401     if (type == static_cast<uint8_t>(dimmOffsetTypes::dimmPower))
3402     {
3403         typeName = dimmOffset::dimmPower;
3404     }
3405     else
3406     {
3407         typeName = dimmOffset::staticCltt;
3408     }
3409 
3410     auto it = json.find(typeName);
3411     if (it == json.end())
3412     {
3413         return ipmi::responseInvalidFieldRequest();
3414     }
3415 
3416     if (it->size() <= index)
3417     {
3418         return ipmi::responseInvalidFieldRequest();
3419     }
3420 
3421     uint8_t resp = it->at(index).get<uint8_t>();
3422     return ipmi::responseSuccess(resp);
3423 }
3424 
3425 namespace boot_options
3426 {
3427 
3428 using namespace sdbusplus::xyz::openbmc_project::Control::Boot::server;
3429 using IpmiValue = uint8_t;
3430 constexpr auto ipmiDefault = 0;
3431 
3432 std::map<IpmiValue, Source::Sources> sourceIpmiToDbus = {
3433     {0x01, Source::Sources::Network},
3434     {0x02, Source::Sources::Disk},
3435     {0x05, Source::Sources::ExternalMedia},
3436     {0x0f, Source::Sources::RemovableMedia},
3437     {ipmiDefault, Source::Sources::Default}};
3438 
3439 std::map<IpmiValue, Mode::Modes> modeIpmiToDbus = {
3440     {0x06, Mode::Modes::Setup}, {ipmiDefault, Mode::Modes::Regular}};
3441 
3442 std::map<Source::Sources, IpmiValue> sourceDbusToIpmi = {
3443     {Source::Sources::Network, 0x01},
3444     {Source::Sources::Disk, 0x02},
3445     {Source::Sources::ExternalMedia, 0x05},
3446     {Source::Sources::RemovableMedia, 0x0f},
3447     {Source::Sources::Default, ipmiDefault}};
3448 
3449 std::map<Mode::Modes, IpmiValue> modeDbusToIpmi = {
3450     {Mode::Modes::Setup, 0x06}, {Mode::Modes::Regular, ipmiDefault}};
3451 
3452 static constexpr auto bootModeIntf = "xyz.openbmc_project.Control.Boot.Mode";
3453 static constexpr auto bootSourceIntf =
3454     "xyz.openbmc_project.Control.Boot.Source";
3455 static constexpr auto enabledIntf = "xyz.openbmc_project.Object.Enable";
3456 static constexpr auto persistentObjPath =
3457     "/xyz/openbmc_project/control/host0/boot";
3458 static constexpr auto oneTimePath =
3459     "/xyz/openbmc_project/control/host0/boot/one_time";
3460 static constexpr auto bootSourceProp = "BootSource";
3461 static constexpr auto bootModeProp = "BootMode";
3462 static constexpr auto oneTimeBootEnableProp = "Enabled";
3463 static constexpr auto httpBootMode =
3464     "xyz.openbmc_project.Control.Boot.Source.Sources.Http";
3465 
3466 enum class BootOptionParameter : size_t
3467 {
3468     setInProgress = 0x0,
3469     bootFlags = 0x5,
3470 };
3471 static constexpr uint8_t setComplete = 0x0;
3472 static constexpr uint8_t setInProgress = 0x1;
3473 static uint8_t transferStatus = setComplete;
3474 static constexpr uint8_t setParmVersion = 0x01;
3475 static constexpr uint8_t setParmBootFlagsPermanent = 0x40;
3476 static constexpr uint8_t setParmBootFlagsValidOneTime = 0x80;
3477 static constexpr uint8_t setParmBootFlagsValidPermanent = 0xC0;
3478 static constexpr uint8_t httpBoot = 0xd;
3479 static constexpr uint8_t bootSourceMask = 0x3c;
3480 
3481 } // namespace boot_options
3482 
3483 ipmi::RspType<uint8_t,               // version
3484               uint8_t,               // param
3485               uint8_t,               // data0, dependent on parameter
3486               std::optional<uint8_t> // data1, dependent on parameter
3487               >
ipmiOemGetEfiBootOptions(uint8_t parameter,uint8_t set,uint8_t block)3488     ipmiOemGetEfiBootOptions(uint8_t parameter, [[maybe_unused]] uint8_t set,
3489                              [[maybe_unused]] uint8_t block)
3490 {
3491     using namespace boot_options;
3492     uint8_t bootOption = 0;
3493 
3494     if (parameter == static_cast<uint8_t>(BootOptionParameter::setInProgress))
3495     {
3496         return ipmi::responseSuccess(setParmVersion, parameter, transferStatus,
3497                                      std::nullopt);
3498     }
3499 
3500     if (parameter != static_cast<uint8_t>(BootOptionParameter::bootFlags))
3501     {
3502         phosphor::logging::log<phosphor::logging::level::ERR>(
3503             "Unsupported parameter");
3504         return ipmi::response(ccParameterNotSupported);
3505     }
3506 
3507     try
3508     {
3509         auto oneTimeEnabled = false;
3510         // read one time Enabled property
3511         std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
3512         std::string service = getService(*dbus, enabledIntf, oneTimePath);
3513         Value variant = getDbusProperty(*dbus, service, oneTimePath,
3514                                         enabledIntf, oneTimeBootEnableProp);
3515         oneTimeEnabled = std::get<bool>(variant);
3516 
3517         // get BootSource and BootMode properties
3518         // according to oneTimeEnable
3519         auto bootObjPath = oneTimePath;
3520         if (oneTimeEnabled == false)
3521         {
3522             bootObjPath = persistentObjPath;
3523         }
3524 
3525         service = getService(*dbus, bootModeIntf, bootObjPath);
3526         variant = getDbusProperty(*dbus, service, bootObjPath, bootModeIntf,
3527                                   bootModeProp);
3528 
3529         auto bootMode =
3530             Mode::convertModesFromString(std::get<std::string>(variant));
3531 
3532         service = getService(*dbus, bootSourceIntf, bootObjPath);
3533         variant = getDbusProperty(*dbus, service, bootObjPath, bootSourceIntf,
3534                                   bootSourceProp);
3535 
3536         if (std::get<std::string>(variant) == httpBootMode)
3537         {
3538             bootOption = httpBoot;
3539         }
3540         else
3541         {
3542             auto bootSource = Source::convertSourcesFromString(
3543                 std::get<std::string>(variant));
3544             bootOption = sourceDbusToIpmi.at(bootSource);
3545             if (Source::Sources::Default == bootSource)
3546             {
3547                 bootOption = modeDbusToIpmi.at(bootMode);
3548             }
3549         }
3550 
3551         uint8_t oneTime = oneTimeEnabled ? setParmBootFlagsValidOneTime
3552                                          : setParmBootFlagsValidPermanent;
3553         bootOption <<= 2; // shift for responseconstexpr
3554         return ipmi::responseSuccess(setParmVersion, parameter, oneTime,
3555                                      bootOption);
3556     }
3557     catch (const sdbusplus::exception_t& e)
3558     {
3559         phosphor::logging::log<phosphor::logging::level::ERR>(e.what());
3560         return ipmi::responseResponseError();
3561     }
3562 }
3563 
ipmiOemSetEfiBootOptions(uint8_t bootFlag,uint8_t bootParam,std::optional<uint8_t> bootOption)3564 ipmi::RspType<> ipmiOemSetEfiBootOptions(uint8_t bootFlag, uint8_t bootParam,
3565                                          std::optional<uint8_t> bootOption)
3566 {
3567     using namespace boot_options;
3568     auto oneTimeEnabled = false;
3569 
3570     if (bootFlag == 0 && bootParam == 0)
3571     {
3572         phosphor::logging::log<phosphor::logging::level::ERR>(
3573             "Unsupported parameter");
3574         return ipmi::response(ccParameterNotSupported);
3575     }
3576     if (bootFlag == static_cast<uint8_t>(BootOptionParameter::setInProgress))
3577     {
3578         if (bootOption)
3579         {
3580             return ipmi::responseReqDataLenInvalid();
3581         }
3582 
3583         if (transferStatus == setInProgress)
3584         {
3585             phosphor::logging::log<phosphor::logging::level::ERR>(
3586                 "boot option set in progress!");
3587             return ipmi::responseResponseError();
3588         }
3589 
3590         transferStatus = bootParam;
3591         return ipmi::responseSuccess();
3592     }
3593 
3594     if (bootFlag != (uint8_t)BootOptionParameter::bootFlags)
3595     {
3596         phosphor::logging::log<phosphor::logging::level::ERR>(
3597             "Unsupported parameter");
3598         return ipmi::response(ccParameterNotSupported);
3599     }
3600 
3601     if (!bootOption)
3602     {
3603         return ipmi::responseReqDataLenInvalid();
3604     }
3605 
3606     if (((bootOption.value() & bootSourceMask) >> 2) !=
3607         httpBoot) // not http boot, exit
3608     {
3609         phosphor::logging::log<phosphor::logging::level::ERR>(
3610             "wrong boot option parameter!");
3611         return ipmi::responseParmOutOfRange();
3612     }
3613 
3614     try
3615     {
3616         bool permanent = (bootParam & setParmBootFlagsPermanent) ==
3617                          setParmBootFlagsPermanent;
3618 
3619         // read one time Enabled property
3620         std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
3621         std::string service = getService(*dbus, enabledIntf, oneTimePath);
3622         Value variant = getDbusProperty(*dbus, service, oneTimePath,
3623                                         enabledIntf, oneTimeBootEnableProp);
3624         oneTimeEnabled = std::get<bool>(variant);
3625 
3626         /*
3627          * Check if the current boot setting is onetime or permanent, if the
3628          * request in the command is otherwise, then set the "Enabled"
3629          * property in one_time object path to 'True' to indicate onetime
3630          * and 'False' to indicate permanent.
3631          *
3632          * Once the onetime/permanent setting is applied, then the bootMode
3633          * and bootSource is updated for the corresponding object.
3634          */
3635         if (permanent == oneTimeEnabled)
3636         {
3637             setDbusProperty(*dbus, service, oneTimePath, enabledIntf,
3638                             oneTimeBootEnableProp, !permanent);
3639         }
3640 
3641         // set BootSource and BootMode properties
3642         // according to oneTimeEnable or persistent
3643         auto bootObjPath = oneTimePath;
3644         if (oneTimeEnabled == false)
3645         {
3646             bootObjPath = persistentObjPath;
3647         }
3648         std::string bootMode =
3649             "xyz.openbmc_project.Control.Boot.Mode.Modes.Regular";
3650         std::string bootSource = httpBootMode;
3651 
3652         service = getService(*dbus, bootModeIntf, bootObjPath);
3653         setDbusProperty(*dbus, service, bootObjPath, bootModeIntf, bootModeProp,
3654                         bootMode);
3655 
3656         service = getService(*dbus, bootSourceIntf, bootObjPath);
3657         setDbusProperty(*dbus, service, bootObjPath, bootSourceIntf,
3658                         bootSourceProp, bootSource);
3659     }
3660     catch (const sdbusplus::exception_t& e)
3661     {
3662         phosphor::logging::log<phosphor::logging::level::ERR>(e.what());
3663         return ipmi::responseResponseError();
3664     }
3665 
3666     return ipmi::responseSuccess();
3667 }
3668 
3669 using BasicVariantType = ipmi::DbusVariant;
3670 using PropertyMapType =
3671     boost::container::flat_map<std::string, BasicVariantType>;
3672 static constexpr const std::array<const char*, 1> psuPresenceTypes = {
3673     "xyz.openbmc_project.Configuration.PSUPresence"};
getPSUAddress(ipmi::Context::ptr & ctx,uint8_t & bus,std::vector<uint64_t> & addrTable)3674 int getPSUAddress(ipmi::Context::ptr& ctx, uint8_t& bus,
3675                   std::vector<uint64_t>& addrTable)
3676 {
3677     boost::system::error_code ec;
3678     GetSubTreeType subtree = ctx->bus->yield_method_call<GetSubTreeType>(
3679         ctx->yield, ec, "xyz.openbmc_project.ObjectMapper",
3680         "/xyz/openbmc_project/object_mapper",
3681         "xyz.openbmc_project.ObjectMapper", "GetSubTree",
3682         "/xyz/openbmc_project/inventory/system", 3, psuPresenceTypes);
3683     if (ec)
3684     {
3685         phosphor::logging::log<phosphor::logging::level::ERR>(
3686             "Failed to set dbus property to cold redundancy");
3687         return -1;
3688     }
3689     for (const auto& object : subtree)
3690     {
3691         std::string pathName = object.first;
3692         for (const auto& serviceIface : object.second)
3693         {
3694             std::string serviceName = serviceIface.first;
3695 
3696             ec.clear();
3697             PropertyMapType propMap =
3698                 ctx->bus->yield_method_call<PropertyMapType>(
3699                     ctx->yield, ec, serviceName, pathName,
3700                     "org.freedesktop.DBus.Properties", "GetAll",
3701                     "xyz.openbmc_project.Configuration.PSUPresence");
3702             if (ec)
3703             {
3704                 phosphor::logging::log<phosphor::logging::level::ERR>(
3705                     "Failed to set dbus property to cold redundancy");
3706                 return -1;
3707             }
3708             auto psuBus = std::get_if<uint64_t>(&propMap["Bus"]);
3709             auto psuAddress =
3710                 std::get_if<std::vector<uint64_t>>(&propMap["Address"]);
3711 
3712             if (psuBus == nullptr || psuAddress == nullptr)
3713             {
3714                 std::cerr << "error finding necessary "
3715                              "entry in configuration\n";
3716                 return -1;
3717             }
3718             bus = static_cast<uint8_t>(*psuBus);
3719             addrTable = *psuAddress;
3720             return 0;
3721         }
3722     }
3723     return -1;
3724 }
3725 
3726 static const constexpr uint8_t addrOffset = 8;
3727 static const constexpr uint8_t psuRevision = 0xd9;
3728 static const constexpr uint8_t defaultPSUBus = 7;
3729 // Second Minor, Primary Minor, Major
3730 static const constexpr size_t verLen = 3;
3731 ipmi::RspType<std::vector<uint8_t>>
ipmiOEMGetPSUVersion(ipmi::Context::ptr & ctx)3732     ipmiOEMGetPSUVersion(ipmi::Context::ptr& ctx)
3733 {
3734     uint8_t bus = defaultPSUBus;
3735     std::vector<uint64_t> addrTable;
3736     std::vector<uint8_t> result;
3737     if (getPSUAddress(ctx, bus, addrTable))
3738     {
3739         std::cerr << "Failed to get PSU bus and address\n";
3740         return ipmi::responseResponseError();
3741     }
3742 
3743     for (const auto& targetAddr : addrTable)
3744     {
3745         std::vector<uint8_t> writeData = {psuRevision};
3746         std::vector<uint8_t> readBuf(verLen);
3747         uint8_t addr = static_cast<uint8_t>(targetAddr) + addrOffset;
3748         std::string i2cBus = "/dev/i2c-" + std::to_string(bus);
3749 
3750         auto retI2C = ipmi::i2cWriteRead(i2cBus, addr, writeData, readBuf);
3751         if (retI2C != ipmi::ccSuccess)
3752         {
3753             for (size_t idx = 0; idx < verLen; idx++)
3754             {
3755                 result.emplace_back(0x00);
3756             }
3757         }
3758         else
3759         {
3760             for (const uint8_t& data : readBuf)
3761             {
3762                 result.emplace_back(data);
3763             }
3764         }
3765     }
3766 
3767     return ipmi::responseSuccess(result);
3768 }
3769 
3770 std::optional<uint8_t>
getMultiNodeInfoPresence(ipmi::Context::ptr & ctx,const std::string & name)3771     getMultiNodeInfoPresence(ipmi::Context::ptr& ctx, const std::string& name)
3772 {
3773     Value dbusValue = 0;
3774     std::string serviceName;
3775 
3776     boost::system::error_code ec =
3777         ipmi::getService(ctx, multiNodeIntf, multiNodeObjPath, serviceName);
3778 
3779     if (ec)
3780     {
3781         phosphor::logging::log<phosphor::logging::level::ERR>(
3782             "Failed to perform Multinode getService.");
3783         return std::nullopt;
3784     }
3785 
3786     ec = ipmi::getDbusProperty(ctx, serviceName, multiNodeObjPath,
3787                                multiNodeIntf, name, dbusValue);
3788     if (ec)
3789     {
3790         phosphor::logging::log<phosphor::logging::level::ERR>(
3791             "Failed to perform Multinode get property");
3792         return std::nullopt;
3793     }
3794 
3795     auto multiNodeVal = std::get_if<uint8_t>(&dbusValue);
3796     if (!multiNodeVal)
3797     {
3798         phosphor::logging::log<phosphor::logging::level::ERR>(
3799             "getMultiNodeInfoPresence: error to get multinode");
3800         return std::nullopt;
3801     }
3802     return *multiNodeVal;
3803 }
3804 
3805 /** @brief implements OEM get reading command
3806  *  @param domain ID
3807  *  @param reading Type
3808  *    - 00h = platform Power Consumption
3809  *    - 01h = inlet Air Temp
3810  *    - 02h = icc_TDC from PECI
3811  *  @param reserved, write as 0000h
3812  *
3813  *  @returns IPMI completion code plus response data
3814  *  - response
3815  *     - domain ID
3816  *     - reading Type
3817  *       - 00h = platform Power Consumption
3818  *       - 01h = inlet Air Temp
3819  *       - 02h = icc_TDC from PECI
3820  *     - reading
3821  */
3822 ipmi::RspType<uint4_t, // domain ID
3823               uint4_t, // reading Type
3824               uint16_t // reading Value
3825               >
ipmiOEMGetReading(ipmi::Context::ptr & ctx,uint4_t domainId,uint4_t readingType,uint16_t reserved)3826     ipmiOEMGetReading(ipmi::Context::ptr& ctx, uint4_t domainId,
3827                       uint4_t readingType, uint16_t reserved)
3828 {
3829     [[maybe_unused]] constexpr uint8_t platformPower = 0;
3830     constexpr uint8_t inletAirTemp = 1;
3831     constexpr uint8_t iccTdc = 2;
3832 
3833     if ((static_cast<uint8_t>(readingType) > iccTdc) || domainId || reserved)
3834     {
3835         return ipmi::responseInvalidFieldRequest();
3836     }
3837 
3838     // This command should run only from multi-node product.
3839     // For all other platforms this command will return invalid.
3840 
3841     std::optional<uint8_t> nodeInfo =
3842         getMultiNodeInfoPresence(ctx, "NodePresence");
3843     if (!nodeInfo || !*nodeInfo)
3844     {
3845         return ipmi::responseInvalidCommand();
3846     }
3847 
3848     uint16_t oemReadingValue = 0;
3849     if (static_cast<uint8_t>(readingType) == inletAirTemp)
3850     {
3851         double value = 0;
3852         boost::system::error_code ec = ipmi::getDbusProperty(
3853             ctx, "xyz.openbmc_project.HwmonTempSensor",
3854             "/xyz/openbmc_project/sensors/temperature/Inlet_BRD_Temp",
3855             "xyz.openbmc_project.Sensor.Value", "Value", value);
3856         if (ec)
3857         {
3858             phosphor::logging::log<phosphor::logging::level::ERR>(
3859                 "Failed to get BMC Get OEM temperature",
3860                 phosphor::logging::entry("EXCEPTION=%s", ec.message().c_str()));
3861             return ipmi::responseUnspecifiedError();
3862         }
3863         // Take the Inlet temperature
3864         oemReadingValue = static_cast<uint16_t>(value);
3865     }
3866     else
3867     {
3868         phosphor::logging::log<phosphor::logging::level::ERR>(
3869             "Currently Get OEM Reading support only for Inlet Air Temp");
3870         return ipmi::responseParmOutOfRange();
3871     }
3872     return ipmi::responseSuccess(domainId, readingType, oemReadingValue);
3873 }
3874 
3875 /** @brief implements the maximum size of
3876  *  bridgeable messages used between KCS and
3877  *  IPMB interfacesget security mode command.
3878  *
3879  *  @returns IPMI completion code with following data
3880  *   - KCS Buffer Size (In multiples of four bytes)
3881  *   - IPMB Buffer Size (In multiples of four bytes)
3882  **/
ipmiOEMGetBufferSize()3883 ipmi::RspType<uint8_t, uint8_t> ipmiOEMGetBufferSize()
3884 {
3885     // for now this is hard coded; really this number is dependent on
3886     // the BMC kcs driver as well as the host kcs driver....
3887     // we can't know the latter.
3888     uint8_t kcsMaxBufferSize = 63 / 4;
3889     uint8_t ipmbMaxBufferSize = 128 / 4;
3890 
3891     return ipmi::responseSuccess(kcsMaxBufferSize, ipmbMaxBufferSize);
3892 }
3893 
3894 ipmi::RspType<std::vector<uint8_t>>
ipmiOEMReadPFRMailbox(ipmi::Context::ptr & ctx,const uint8_t readRegister,const uint8_t numOfBytes,uint8_t registerIdentifier)3895     ipmiOEMReadPFRMailbox(ipmi::Context::ptr& ctx, const uint8_t readRegister,
3896                           const uint8_t numOfBytes, uint8_t registerIdentifier)
3897 {
3898     if (!ipmi::mailbox::i2cConfigLoaded)
3899     {
3900         phosphor::logging::log<phosphor::logging::level::ERR>(
3901             "Calling PFR Load Configuration Function to Get I2C Bus and Target "
3902             "Address ");
3903 
3904         ipmi::mailbox::loadPfrConfig(ctx, ipmi::mailbox::i2cConfigLoaded);
3905     }
3906 
3907     if (!numOfBytes && !readRegister)
3908     {
3909         phosphor::logging::log<phosphor::logging::level::ERR>(
3910             "OEM IPMI command: Read & write count are 0 which is invalid ");
3911         return ipmi::responseInvalidFieldRequest();
3912     }
3913 
3914     switch (registerIdentifier)
3915     {
3916         case ipmi::mailbox::registerType::fifoReadRegister:
3917         {
3918             // Check if readRegister is an FIFO read register
3919             if (ipmi::mailbox::readFifoReg.find(readRegister) ==
3920                 ipmi::mailbox::readFifoReg.end())
3921             {
3922                 phosphor::logging::log<phosphor::logging::level::ERR>(
3923                     "OEM IPMI command: Register is not a Read FIFO  ");
3924                 return ipmi::responseInvalidFieldRequest();
3925             }
3926 
3927             phosphor::logging::log<phosphor::logging::level::ERR>(
3928                 "OEM IPMI command: Register is a Read FIFO  ");
3929 
3930             ipmi::mailbox::writefifo(ipmi::mailbox::provisioningCommand,
3931                                      readRegister);
3932             ipmi::mailbox::writefifo(ipmi::mailbox::triggerCommand,
3933                                      ipmi::mailbox::flushRead);
3934 
3935             std::vector<uint8_t> writeData = {ipmi::mailbox::readFifo};
3936             std::vector<uint8_t> readBuf(1);
3937             std::vector<uint8_t> result;
3938 
3939             for (int i = 0; i < numOfBytes; i++)
3940             {
3941                 ipmi::Cc ret = ipmi::i2cWriteRead(ipmi::mailbox::i2cBus,
3942                                                   ipmi::mailbox::targetAddr,
3943                                                   writeData, readBuf);
3944                 if (ret != ipmi::ccSuccess)
3945                 {
3946                     return ipmi::response(ret);
3947                 }
3948 
3949                 else
3950                 {
3951                     for (const uint8_t& data : readBuf)
3952                     {
3953                         result.emplace_back(data);
3954                     }
3955                 }
3956             }
3957 
3958             return ipmi::responseSuccess(result);
3959         }
3960 
3961         case ipmi::mailbox::registerType::singleByteRegister:
3962         {
3963             phosphor::logging::log<phosphor::logging::level::ERR>(
3964                 "OEM IPMI command: Register is a Single Byte Register ");
3965 
3966             std::vector<uint8_t> writeData = {readRegister};
3967             std::vector<uint8_t> readBuf(numOfBytes);
3968 
3969             ipmi::Cc ret = ipmi::i2cWriteRead(ipmi::mailbox::i2cBus,
3970                                               ipmi::mailbox::targetAddr,
3971                                               writeData, readBuf);
3972             if (ret != ipmi::ccSuccess)
3973             {
3974                 return ipmi::response(ret);
3975             }
3976             return ipmi::responseSuccess(readBuf);
3977         }
3978 
3979         default:
3980         {
3981             phosphor::logging::log<phosphor::logging::level::ERR>(
3982                 "OEM IPMI command: Register identifier is not valid.It should "
3983                 "be 0 "
3984                 "for Single Byte Register and 1 for FIFO Read Register");
3985 
3986             return ipmi::responseInvalidFieldRequest();
3987         }
3988     }
3989 }
3990 
registerOEMFunctions(void)3991 static void registerOEMFunctions(void)
3992 {
3993     phosphor::logging::log<phosphor::logging::level::INFO>(
3994         "Registering OEM commands");
3995     registerHandler(prioOemBase, intel::netFnGeneral,
3996                     intel::general::cmdGetBmcVersionString, Privilege::User,
3997                     ipmiOEMGetBmcVersionString);
3998 
3999     ipmiPrintAndRegister(intel::netFnGeneral,
4000                          intel::general::cmdGetChassisIdentifier, NULL,
4001                          ipmiOEMGetChassisIdentifier,
4002                          PRIVILEGE_USER); // get chassis identifier
4003 
4004     ipmiPrintAndRegister(intel::netFnGeneral, intel::general::cmdSetSystemGUID,
4005                          NULL, ipmiOEMSetSystemGUID,
4006                          PRIVILEGE_ADMIN); // set system guid
4007 
4008     // <Disable BMC System Reset Action>
4009     registerHandler(prioOemBase, intel::netFnGeneral,
4010                     intel::general::cmdDisableBMCSystemReset, Privilege::Admin,
4011                     ipmiOEMDisableBMCSystemReset);
4012 
4013     // <Get BMC Reset Disables>
4014     registerHandler(prioOemBase, intel::netFnGeneral,
4015                     intel::general::cmdGetBMCResetDisables, Privilege::Admin,
4016                     ipmiOEMGetBMCResetDisables);
4017 
4018     ipmiPrintAndRegister(intel::netFnGeneral, intel::general::cmdSetBIOSID,
4019                          NULL, ipmiOEMSetBIOSID, PRIVILEGE_ADMIN);
4020 
4021     registerHandler(prioOemBase, intel::netFnGeneral,
4022                     intel::general::cmdGetOEMDeviceInfo, Privilege::User,
4023                     ipmiOEMGetDeviceInfo);
4024 
4025     ipmiPrintAndRegister(intel::netFnGeneral,
4026                          intel::general::cmdGetAICSlotFRUIDSlotPosRecords, NULL,
4027                          ipmiOEMGetAICFRU, PRIVILEGE_USER);
4028 
4029     registerHandler(prioOpenBmcBase, intel::netFnGeneral,
4030                     intel::general::cmdSendEmbeddedFWUpdStatus,
4031                     Privilege::Operator, ipmiOEMSendEmbeddedFwUpdStatus);
4032 
4033     registerHandler(prioOpenBmcBase, intel::netFnApp, intel::app::cmdSlotIpmb,
4034                     Privilege::Admin, ipmiOEMSlotIpmb);
4035 
4036     ipmiPrintAndRegister(intel::netFnGeneral,
4037                          intel::general::cmdSetPowerRestoreDelay, NULL,
4038                          ipmiOEMSetPowerRestoreDelay, PRIVILEGE_OPERATOR);
4039 
4040     ipmiPrintAndRegister(intel::netFnGeneral,
4041                          intel::general::cmdGetPowerRestoreDelay, NULL,
4042                          ipmiOEMGetPowerRestoreDelay, PRIVILEGE_USER);
4043 
4044     registerHandler(prioOpenBmcBase, intel::netFnGeneral,
4045                     intel::general::cmdSetOEMUser2Activation,
4046                     Privilege::Callback, ipmiOEMSetUser2Activation);
4047 
4048     registerHandler(prioOpenBmcBase, intel::netFnGeneral,
4049                     intel::general::cmdSetSpecialUserPassword,
4050                     Privilege::Callback, ipmiOEMSetSpecialUserPassword);
4051 
4052     // <Get Processor Error Config>
4053     registerHandler(prioOemBase, intel::netFnGeneral,
4054                     intel::general::cmdGetProcessorErrConfig, Privilege::User,
4055                     ipmiOEMGetProcessorErrConfig);
4056 
4057     // <Set Processor Error Config>
4058     registerHandler(prioOemBase, intel::netFnGeneral,
4059                     intel::general::cmdSetProcessorErrConfig, Privilege::Admin,
4060                     ipmiOEMSetProcessorErrConfig);
4061 
4062     ipmiPrintAndRegister(intel::netFnGeneral,
4063                          intel::general::cmdSetShutdownPolicy, NULL,
4064                          ipmiOEMSetShutdownPolicy, PRIVILEGE_ADMIN);
4065 
4066     ipmiPrintAndRegister(intel::netFnGeneral,
4067                          intel::general::cmdGetShutdownPolicy, NULL,
4068                          ipmiOEMGetShutdownPolicy, PRIVILEGE_ADMIN);
4069 
4070     registerHandler(prioOemBase, intel::netFnGeneral,
4071                     intel::general::cmdSetFanConfig, Privilege::User,
4072                     ipmiOEMSetFanConfig);
4073 
4074     registerHandler(prioOemBase, intel::netFnGeneral,
4075                     intel::general::cmdGetFanConfig, Privilege::User,
4076                     ipmiOEMGetFanConfig);
4077 
4078     registerHandler(prioOemBase, intel::netFnGeneral,
4079                     intel::general::cmdGetFanSpeedOffset, Privilege::User,
4080                     ipmiOEMGetFanSpeedOffset);
4081 
4082     registerHandler(prioOemBase, intel::netFnGeneral,
4083                     intel::general::cmdSetFanSpeedOffset, Privilege::User,
4084                     ipmiOEMSetFanSpeedOffset);
4085 
4086     registerHandler(prioOemBase, intel::netFnGeneral,
4087                     intel::general::cmdSetFscParameter, Privilege::User,
4088                     ipmiOEMSetFscParameter);
4089 
4090     registerHandler(prioOemBase, intel::netFnGeneral,
4091                     intel::general::cmdGetFscParameter, Privilege::User,
4092                     ipmiOEMGetFscParameter);
4093 
4094     registerHandler(prioOpenBmcBase, intel::netFnGeneral,
4095                     intel::general::cmdReadBaseBoardProductId, Privilege::Admin,
4096                     ipmiOEMReadBoardProductId);
4097 
4098     registerHandler(prioOemBase, intel::netFnGeneral,
4099                     intel::general::cmdGetNmiStatus, Privilege::User,
4100                     ipmiOEMGetNmiSource);
4101 
4102     registerHandler(prioOemBase, intel::netFnGeneral,
4103                     intel::general::cmdSetNmiStatus, Privilege::Operator,
4104                     ipmiOEMSetNmiSource);
4105 
4106     registerHandler(prioOemBase, intel::netFnGeneral,
4107                     intel::general::cmdGetEfiBootOptions, Privilege::User,
4108                     ipmiOemGetEfiBootOptions);
4109 
4110     registerHandler(prioOemBase, intel::netFnGeneral,
4111                     intel::general::cmdSetEfiBootOptions, Privilege::Operator,
4112                     ipmiOemSetEfiBootOptions);
4113 
4114     registerHandler(prioOemBase, intel::netFnGeneral,
4115                     intel::general::cmdGetSecurityMode, Privilege::User,
4116                     ipmiGetSecurityMode);
4117 
4118     registerHandler(prioOemBase, intel::netFnGeneral,
4119                     intel::general::cmdSetSecurityMode, Privilege::Admin,
4120                     ipmiSetSecurityMode);
4121 
4122     registerHandler(prioOemBase, intel::netFnGeneral,
4123                     intel::general::cmdGetLEDStatus, Privilege::Admin,
4124                     ipmiOEMGetLEDStatus);
4125 
4126     ipmiPrintAndRegister(ipmi::intel::netFnPlatform,
4127                          ipmi::intel::platform::cmdCfgHostSerialPortSpeed, NULL,
4128                          ipmiOEMCfgHostSerialPortSpeed, PRIVILEGE_ADMIN);
4129 
4130     registerHandler(prioOemBase, intel::netFnGeneral,
4131                     intel::general::cmdSetFaultIndication, Privilege::Operator,
4132                     ipmiOEMSetFaultIndication);
4133 
4134     registerHandler(prioOemBase, intel::netFnGeneral,
4135                     intel::general::cmdSetColdRedundancyConfig, Privilege::User,
4136                     ipmiOEMSetCRConfig);
4137 
4138     registerHandler(prioOemBase, intel::netFnGeneral,
4139                     intel::general::cmdGetColdRedundancyConfig, Privilege::User,
4140                     ipmiOEMGetCRConfig);
4141 
4142     registerHandler(prioOemBase, intel::netFnGeneral,
4143                     intel::general::cmdRestoreConfiguration, Privilege::Admin,
4144                     ipmiRestoreConfiguration);
4145 
4146     registerHandler(prioOemBase, intel::netFnGeneral,
4147                     intel::general::cmdSetDimmOffset, Privilege::Operator,
4148                     ipmiOEMSetDimmOffset);
4149 
4150     registerHandler(prioOemBase, intel::netFnGeneral,
4151                     intel::general::cmdGetDimmOffset, Privilege::Operator,
4152                     ipmiOEMGetDimmOffset);
4153 
4154     registerHandler(prioOemBase, intel::netFnGeneral,
4155                     intel::general::cmdGetPSUVersion, Privilege::User,
4156                     ipmiOEMGetPSUVersion);
4157 
4158     registerHandler(prioOemBase, intel::netFnGeneral,
4159                     intel::general::cmdGetBufferSize, Privilege::User,
4160                     ipmiOEMGetBufferSize);
4161 
4162     registerHandler(prioOemBase, intel::netFnGeneral,
4163                     intel::general::cmdOEMGetReading, Privilege::User,
4164                     ipmiOEMGetReading);
4165 
4166     registerHandler(prioOemBase, intel::netFnApp, intel::app::cmdPFRMailboxRead,
4167                     Privilege::Admin, ipmiOEMReadPFRMailbox);
4168 }
4169 
4170 } // namespace ipmi
4171