blob: 7b7bc81dcf1d6d593c6a236d1865814ae523bd4d [file] [log] [blame]
Potin Lai5d214202023-01-16 18:11:49 +08001#include "config.h"
2
Patrick Venture0b02be92018-08-31 11:55:55 -07003#include <arpa/inet.h>
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05304#include <fcntl.h>
Xo Wang87651332017-08-11 10:17:59 -07005#include <limits.h>
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05306#include <linux/i2c-dev.h>
7#include <linux/i2c.h>
Patrick Venture0b02be92018-08-31 11:55:55 -07008#include <mapper.h>
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05309#include <sys/ioctl.h>
10#include <sys/stat.h>
11#include <sys/types.h>
Patrick Venture0b02be92018-08-31 11:55:55 -070012#include <systemd/sd-bus.h>
Xo Wang87651332017-08-11 10:17:59 -070013#include <unistd.h>
Patrick Venture0b02be92018-08-31 11:55:55 -070014
Vernon Mauery0120b682019-03-25 13:08:54 -070015#include <app/channel.hpp>
16#include <app/watchdog.hpp>
17#include <apphandler.hpp>
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -050018#include <ipmid/api.hpp>
19#include <ipmid/sessiondef.hpp>
20#include <ipmid/sessionhelper.hpp>
21#include <ipmid/types.hpp>
22#include <ipmid/utils.hpp>
23#include <nlohmann/json.hpp>
24#include <phosphor-logging/elog-errors.hpp>
25#include <phosphor-logging/log.hpp>
26#include <sdbusplus/message/types.hpp>
27#include <sys_info_param.hpp>
28#include <xyz/openbmc_project/Common/error.hpp>
29#include <xyz/openbmc_project/Control/Power/ACPIPowerState/server.hpp>
30#include <xyz/openbmc_project/Software/Activation/server.hpp>
31#include <xyz/openbmc_project/Software/Version/server.hpp>
32#include <xyz/openbmc_project/State/BMC/server.hpp>
33
34#include <algorithm>
Patrick Venture3a5071a2018-09-12 13:27:42 -070035#include <array>
Willy Tu886d6842022-06-03 02:50:47 -070036#include <charconv>
Patrick Venture3a5071a2018-09-12 13:27:42 -070037#include <cstddef>
Vernon Mauery0120b682019-03-25 13:08:54 -070038#include <cstdint>
Vernon Mauerybdda8002019-02-26 10:18:51 -080039#include <filesystem>
Patrick Venture3a5071a2018-09-12 13:27:42 -070040#include <fstream>
41#include <memory>
Potin Lai5d214202023-01-16 18:11:49 +080042#include <regex>
Patrick Venture3a5071a2018-09-12 13:27:42 -070043#include <string>
Willy Tu886d6842022-06-03 02:50:47 -070044#include <string_view>
Patrick Venture3a5071a2018-09-12 13:27:42 -070045#include <tuple>
46#include <vector>
Ratan Guptab8e99552017-07-27 07:07:48 +053047
Patrick Venture0b02be92018-08-31 11:55:55 -070048extern sd_bus* bus;
vishwabmcba0bd5f2015-09-30 16:50:23 +053049
Alexander Amelkinba19c182018-09-04 15:49:36 +030050constexpr auto bmc_state_interface = "xyz.openbmc_project.State.BMC";
51constexpr auto bmc_state_property = "CurrentBMCState";
Marri Devender Rao5e007a52018-01-08 06:18:36 -060052
Nagaraju Goruganti744398d2018-02-20 09:52:00 -060053static constexpr auto redundancyIntf =
54 "xyz.openbmc_project.Software.RedundancyPriority";
Patrick Venture0b02be92018-08-31 11:55:55 -070055static constexpr auto versionIntf = "xyz.openbmc_project.Software.Version";
Nagaraju Goruganti744398d2018-02-20 09:52:00 -060056static constexpr auto activationIntf =
57 "xyz.openbmc_project.Software.Activation";
58static constexpr auto softwareRoot = "/xyz/openbmc_project/software";
59
Chris Austen6caf28b2015-10-13 12:40:40 -050060void register_netfn_app_functions() __attribute__((constructor));
vishwabmcba0bd5f2015-09-30 16:50:23 +053061
Ratan Guptab8e99552017-07-27 07:07:48 +053062using namespace phosphor::logging;
Willy Tu523e2d12023-09-05 11:36:48 -070063using namespace sdbusplus::error::xyz::openbmc_project::common;
64using Version = sdbusplus::server::xyz::openbmc_project::software::Version;
Nagaraju Goruganti744398d2018-02-20 09:52:00 -060065using Activation =
Willy Tu523e2d12023-09-05 11:36:48 -070066 sdbusplus::server::xyz::openbmc_project::software::Activation;
67using BMC = sdbusplus::server::xyz::openbmc_project::state::BMC;
Vernon Mauery185b9f82018-07-20 10:52:36 -070068namespace fs = std::filesystem;
Ratan Guptab8e99552017-07-27 07:07:48 +053069
Yong Libd0503a2019-08-22 17:17:17 +080070#ifdef ENABLE_I2C_WHITELIST_CHECK
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +053071typedef struct
72{
73 uint8_t busId;
74 uint8_t slaveAddr;
75 uint8_t slaveAddrMask;
76 std::vector<uint8_t> data;
77 std::vector<uint8_t> dataMask;
78} i2cMasterWRWhitelist;
79
80static std::vector<i2cMasterWRWhitelist>& getWRWhitelist()
81{
82 static std::vector<i2cMasterWRWhitelist> wrWhitelist;
83 return wrWhitelist;
84}
85
86static constexpr const char* i2cMasterWRWhitelistFile =
87 "/usr/share/ipmi-providers/master_write_read_white_list.json";
88
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +053089static constexpr const char* filtersStr = "filters";
90static constexpr const char* busIdStr = "busId";
91static constexpr const char* slaveAddrStr = "slaveAddr";
92static constexpr const char* slaveAddrMaskStr = "slaveAddrMask";
93static constexpr const char* cmdStr = "command";
94static constexpr const char* cmdMaskStr = "commandMask";
95static constexpr int base_16 = 16;
Yong Libd0503a2019-08-22 17:17:17 +080096#endif // ENABLE_I2C_WHITELIST_CHECK
Joshi-Mansi7fd91fa2021-06-11 07:18:15 +053097static constexpr uint8_t maxIPMIWriteReadSize = 255;
jayaprakash Mutyala3c5e4132020-04-27 23:00:05 +000098static constexpr uint8_t oemCmdStart = 192;
99static constexpr uint8_t oemCmdEnd = 255;
100static constexpr uint8_t invalidParamSelectorStart = 8;
101static constexpr uint8_t invalidParamSelectorEnd = 191;
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +0530102
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600103/**
104 * @brief Returns the Version info from primary s/w object
105 *
106 * Get the Version info from the active s/w object which is having high
107 * "Priority" value(a smaller number is a higher priority) and "Purpose"
108 * is "BMC" from the list of all s/w objects those are implementing
109 * RedundancyPriority interface from the given softwareRoot path.
110 *
111 * @return On success returns the Version info from primary s/w object.
112 *
113 */
Vernon Maueryea1c4012019-05-24 13:26:16 -0700114std::string getActiveSoftwareVersionInfo(ipmi::Context::ptr ctx)
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600115{
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600116 std::string revision{};
Vernon Mauery86a50822019-03-25 13:11:36 -0700117 ipmi::ObjectTree objectTree;
118 try
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600119 {
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -0500120 objectTree = ipmi::getAllDbusObjects(*ctx->bus, softwareRoot,
121 redundancyIntf);
Vernon Mauery86a50822019-03-25 13:11:36 -0700122 }
Patrick Williams5d82f472022-07-22 19:26:53 -0500123 catch (const sdbusplus::exception_t& e)
Vernon Mauery86a50822019-03-25 13:11:36 -0700124 {
125 log<level::ERR>("Failed to fetch redundancy object from dbus",
126 entry("INTERFACE=%s", redundancyIntf),
127 entry("ERRMSG=%s", e.what()));
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600128 elog<InternalFailure>();
129 }
130
131 auto objectFound = false;
132 for (auto& softObject : objectTree)
133 {
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -0500134 auto service = ipmi::getService(*ctx->bus, redundancyIntf,
135 softObject.first);
136 auto objValueTree = ipmi::getManagedObjects(*ctx->bus, service,
137 softwareRoot);
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600138
139 auto minPriority = 0xFF;
140 for (const auto& objIter : objValueTree)
141 {
142 try
143 {
144 auto& intfMap = objIter.second;
145 auto& redundancyPriorityProps = intfMap.at(redundancyIntf);
146 auto& versionProps = intfMap.at(versionIntf);
147 auto& activationProps = intfMap.at(activationIntf);
Vernon Maueryf442e112019-04-09 11:44:36 -0700148 auto priority =
149 std::get<uint8_t>(redundancyPriorityProps.at("Priority"));
William A. Kennington III4c008022018-10-12 17:18:14 -0700150 auto purpose =
Vernon Maueryf442e112019-04-09 11:44:36 -0700151 std::get<std::string>(versionProps.at("Purpose"));
152 auto activation =
153 std::get<std::string>(activationProps.at("Activation"));
William A. Kennington III4c008022018-10-12 17:18:14 -0700154 auto version =
Vernon Maueryf442e112019-04-09 11:44:36 -0700155 std::get<std::string>(versionProps.at("Version"));
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600156 if ((Version::convertVersionPurposeFromString(purpose) ==
157 Version::VersionPurpose::BMC) &&
158 (Activation::convertActivationsFromString(activation) ==
159 Activation::Activations::Active))
160 {
161 if (priority < minPriority)
162 {
163 minPriority = priority;
164 objectFound = true;
165 revision = std::move(version);
166 }
167 }
168 }
169 catch (const std::exception& e)
170 {
171 log<level::ERR>(e.what());
172 }
173 }
174 }
175
176 if (!objectFound)
177 {
178 log<level::ERR>("Could not found an BMC software Object");
179 elog<InternalFailure>();
180 }
181
182 return revision;
183}
184
Alexander Amelkinba19c182018-09-04 15:49:36 +0300185bool getCurrentBmcState()
186{
Patrick Williams5d82f472022-07-22 19:26:53 -0500187 sdbusplus::bus_t bus{ipmid_get_sd_bus_connection()};
Alexander Amelkinba19c182018-09-04 15:49:36 +0300188
189 // Get the Inventory object implementing the BMC interface
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -0500190 ipmi::DbusObjectInfo bmcObject = ipmi::getDbusObject(bus,
191 bmc_state_interface);
192 auto variant = ipmi::getDbusProperty(bus, bmcObject.second, bmcObject.first,
193 bmc_state_interface,
194 bmc_state_property);
Alexander Amelkinba19c182018-09-04 15:49:36 +0300195
Vernon Maueryf442e112019-04-09 11:44:36 -0700196 return std::holds_alternative<std::string>(variant) &&
197 BMC::convertBMCStateFromString(std::get<std::string>(variant)) ==
198 BMC::BMCState::Ready;
Alexander Amelkinba19c182018-09-04 15:49:36 +0300199}
200
Patrick Venture94930a12019-04-30 10:01:58 -0700201bool getCurrentBmcStateWithFallback(const bool fallbackAvailability)
202{
203 try
204 {
205 return getCurrentBmcState();
206 }
207 catch (...)
208 {
209 // Nothing provided the BMC interface, therefore return whatever was
210 // configured as the default.
211 return fallbackAvailability;
212 }
213}
214
Yong Li18d77262018-10-09 01:59:45 +0800215namespace acpi_state
216{
Willy Tu523e2d12023-09-05 11:36:48 -0700217using namespace sdbusplus::server::xyz::openbmc_project::control::power;
Yong Li18d77262018-10-09 01:59:45 +0800218
219const static constexpr char* acpiObjPath =
220 "/xyz/openbmc_project/control/host0/acpi_power_state";
221const static constexpr char* acpiInterface =
222 "xyz.openbmc_project.Control.Power.ACPIPowerState";
223const static constexpr char* sysACPIProp = "SysACPIStatus";
224const static constexpr char* devACPIProp = "DevACPIStatus";
225
226enum class PowerStateType : uint8_t
227{
228 sysPowerState = 0x00,
229 devPowerState = 0x01,
230};
231
232// Defined in 20.6 of ipmi doc
233enum class PowerState : uint8_t
234{
235 s0G0D0 = 0x00,
236 s1D1 = 0x01,
237 s2D2 = 0x02,
238 s3D3 = 0x03,
239 s4 = 0x04,
240 s5G2 = 0x05,
241 s4S5 = 0x06,
242 g3 = 0x07,
243 sleep = 0x08,
244 g1Sleep = 0x09,
245 override = 0x0a,
246 legacyOn = 0x20,
247 legacyOff = 0x21,
248 unknown = 0x2a,
249 noChange = 0x7f,
250};
251
252static constexpr uint8_t stateChanged = 0x80;
253
Yong Li18d77262018-10-09 01:59:45 +0800254std::map<ACPIPowerState::ACPI, PowerState> dbusToIPMI = {
255 {ACPIPowerState::ACPI::S0_G0_D0, PowerState::s0G0D0},
256 {ACPIPowerState::ACPI::S1_D1, PowerState::s1D1},
257 {ACPIPowerState::ACPI::S2_D2, PowerState::s2D2},
258 {ACPIPowerState::ACPI::S3_D3, PowerState::s3D3},
259 {ACPIPowerState::ACPI::S4, PowerState::s4},
260 {ACPIPowerState::ACPI::S5_G2, PowerState::s5G2},
261 {ACPIPowerState::ACPI::S4_S5, PowerState::s4S5},
262 {ACPIPowerState::ACPI::G3, PowerState::g3},
263 {ACPIPowerState::ACPI::SLEEP, PowerState::sleep},
264 {ACPIPowerState::ACPI::G1_SLEEP, PowerState::g1Sleep},
265 {ACPIPowerState::ACPI::OVERRIDE, PowerState::override},
266 {ACPIPowerState::ACPI::LEGACY_ON, PowerState::legacyOn},
267 {ACPIPowerState::ACPI::LEGACY_OFF, PowerState::legacyOff},
268 {ACPIPowerState::ACPI::Unknown, PowerState::unknown}};
269
270bool isValidACPIState(acpi_state::PowerStateType type, uint8_t state)
271{
272 if (type == acpi_state::PowerStateType::sysPowerState)
273 {
274 if ((state <= static_cast<uint8_t>(acpi_state::PowerState::override)) ||
275 (state == static_cast<uint8_t>(acpi_state::PowerState::legacyOn)) ||
276 (state ==
277 static_cast<uint8_t>(acpi_state::PowerState::legacyOff)) ||
278 (state == static_cast<uint8_t>(acpi_state::PowerState::unknown)) ||
279 (state == static_cast<uint8_t>(acpi_state::PowerState::noChange)))
280 {
281 return true;
282 }
283 else
284 {
285 return false;
286 }
287 }
288 else if (type == acpi_state::PowerStateType::devPowerState)
289 {
290 if ((state <= static_cast<uint8_t>(acpi_state::PowerState::s3D3)) ||
291 (state == static_cast<uint8_t>(acpi_state::PowerState::unknown)) ||
292 (state == static_cast<uint8_t>(acpi_state::PowerState::noChange)))
293 {
294 return true;
295 }
296 else
297 {
298 return false;
299 }
300 }
301 else
302 {
303 return false;
304 }
305 return false;
306}
307} // namespace acpi_state
308
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000309/** @brief implements Set ACPI Power State command
310 * @param sysAcpiState - ACPI system power state to set
311 * @param devAcpiState - ACPI device power state to set
312 *
313 * @return IPMI completion code on success
314 **/
315ipmi::RspType<> ipmiSetAcpiPowerState(uint8_t sysAcpiState,
316 uint8_t devAcpiState)
Chris Austen6caf28b2015-10-13 12:40:40 -0500317{
Yong Li18d77262018-10-09 01:59:45 +0800318 auto s = static_cast<uint8_t>(acpi_state::PowerState::unknown);
Yong Li18d77262018-10-09 01:59:45 +0800319
Patrick Williams5d82f472022-07-22 19:26:53 -0500320 sdbusplus::bus_t bus{ipmid_get_sd_bus_connection()};
Yong Li18d77262018-10-09 01:59:45 +0800321
322 auto value = acpi_state::ACPIPowerState::ACPI::Unknown;
323
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000324 if (sysAcpiState & acpi_state::stateChanged)
Yong Li18d77262018-10-09 01:59:45 +0800325 {
326 // set system power state
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000327 s = sysAcpiState & ~acpi_state::stateChanged;
Yong Li18d77262018-10-09 01:59:45 +0800328
329 if (!acpi_state::isValidACPIState(
330 acpi_state::PowerStateType::sysPowerState, s))
331 {
332 log<level::ERR>("set_acpi_power sys invalid input",
333 entry("S=%x", s));
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000334 return ipmi::responseParmOutOfRange();
Yong Li18d77262018-10-09 01:59:45 +0800335 }
336
337 // valid input
338 if (s == static_cast<uint8_t>(acpi_state::PowerState::noChange))
339 {
340 log<level::DEBUG>("No change for system power state");
341 }
342 else
343 {
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -0500344 auto found = std::find_if(acpi_state::dbusToIPMI.begin(),
345 acpi_state::dbusToIPMI.end(),
346 [&s](const auto& iter) {
347 return (static_cast<uint8_t>(iter.second) == s);
348 });
Yong Li18d77262018-10-09 01:59:45 +0800349
350 value = found->first;
351
352 try
353 {
354 auto acpiObject =
355 ipmi::getDbusObject(bus, acpi_state::acpiInterface);
356 ipmi::setDbusProperty(bus, acpiObject.second, acpiObject.first,
357 acpi_state::acpiInterface,
358 acpi_state::sysACPIProp,
359 convertForMessage(value));
360 }
361 catch (const InternalFailure& e)
362 {
363 log<level::ERR>("Failed in set ACPI system property",
364 entry("EXCEPTION=%s", e.what()));
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000365 return ipmi::responseUnspecifiedError();
Yong Li18d77262018-10-09 01:59:45 +0800366 }
367 }
368 }
369 else
370 {
371 log<level::DEBUG>("Do not change system power state");
372 }
373
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000374 if (devAcpiState & acpi_state::stateChanged)
Yong Li18d77262018-10-09 01:59:45 +0800375 {
376 // set device power state
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000377 s = devAcpiState & ~acpi_state::stateChanged;
Yong Li18d77262018-10-09 01:59:45 +0800378 if (!acpi_state::isValidACPIState(
379 acpi_state::PowerStateType::devPowerState, s))
380 {
381 log<level::ERR>("set_acpi_power dev invalid input",
382 entry("S=%x", s));
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000383 return ipmi::responseParmOutOfRange();
Yong Li18d77262018-10-09 01:59:45 +0800384 }
385
386 // valid input
387 if (s == static_cast<uint8_t>(acpi_state::PowerState::noChange))
388 {
389 log<level::DEBUG>("No change for device power state");
390 }
391 else
392 {
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -0500393 auto found = std::find_if(acpi_state::dbusToIPMI.begin(),
394 acpi_state::dbusToIPMI.end(),
395 [&s](const auto& iter) {
396 return (static_cast<uint8_t>(iter.second) == s);
397 });
Yong Li18d77262018-10-09 01:59:45 +0800398
399 value = found->first;
400
401 try
402 {
403 auto acpiObject =
404 ipmi::getDbusObject(bus, acpi_state::acpiInterface);
405 ipmi::setDbusProperty(bus, acpiObject.second, acpiObject.first,
406 acpi_state::acpiInterface,
407 acpi_state::devACPIProp,
408 convertForMessage(value));
409 }
410 catch (const InternalFailure& e)
411 {
412 log<level::ERR>("Failed in set ACPI device property",
413 entry("EXCEPTION=%s", e.what()));
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000414 return ipmi::responseUnspecifiedError();
Yong Li18d77262018-10-09 01:59:45 +0800415 }
416 }
417 }
418 else
419 {
420 log<level::DEBUG>("Do not change device power state");
421 }
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +0000422 return ipmi::responseSuccess();
Yong Li18d77262018-10-09 01:59:45 +0800423}
424
Deepak Kumar Sahu4e6d2572019-05-07 19:53:37 +0000425/**
426 * @brief implements the get ACPI power state command
427 *
428 * @return IPMI completion code plus response data on success.
429 * - ACPI system power state
430 * - ACPI device power state
431 **/
432ipmi::RspType<uint8_t, // acpiSystemPowerState
433 uint8_t // acpiDevicePowerState
434 >
435 ipmiGetAcpiPowerState()
Yong Li18d77262018-10-09 01:59:45 +0800436{
Deepak Kumar Sahu4e6d2572019-05-07 19:53:37 +0000437 uint8_t sysAcpiState;
438 uint8_t devAcpiState;
Yong Li18d77262018-10-09 01:59:45 +0800439
Patrick Williams5d82f472022-07-22 19:26:53 -0500440 sdbusplus::bus_t bus{ipmid_get_sd_bus_connection()};
Yong Li18d77262018-10-09 01:59:45 +0800441
Yong Li18d77262018-10-09 01:59:45 +0800442 try
443 {
444 auto acpiObject = ipmi::getDbusObject(bus, acpi_state::acpiInterface);
445
446 auto sysACPIVal = ipmi::getDbusProperty(
447 bus, acpiObject.second, acpiObject.first, acpi_state::acpiInterface,
448 acpi_state::sysACPIProp);
449 auto sysACPI = acpi_state::ACPIPowerState::convertACPIFromString(
Vernon Maueryf442e112019-04-09 11:44:36 -0700450 std::get<std::string>(sysACPIVal));
Deepak Kumar Sahu4e6d2572019-05-07 19:53:37 +0000451 sysAcpiState = static_cast<uint8_t>(acpi_state::dbusToIPMI.at(sysACPI));
Yong Li18d77262018-10-09 01:59:45 +0800452
453 auto devACPIVal = ipmi::getDbusProperty(
454 bus, acpiObject.second, acpiObject.first, acpi_state::acpiInterface,
455 acpi_state::devACPIProp);
456 auto devACPI = acpi_state::ACPIPowerState::convertACPIFromString(
Vernon Maueryf442e112019-04-09 11:44:36 -0700457 std::get<std::string>(devACPIVal));
Deepak Kumar Sahu4e6d2572019-05-07 19:53:37 +0000458 devAcpiState = static_cast<uint8_t>(acpi_state::dbusToIPMI.at(devACPI));
Yong Li18d77262018-10-09 01:59:45 +0800459 }
460 catch (const InternalFailure& e)
461 {
Deepak Kumar Sahu4e6d2572019-05-07 19:53:37 +0000462 return ipmi::responseUnspecifiedError();
Yong Li18d77262018-10-09 01:59:45 +0800463 }
Deepak Kumar Sahu4e6d2572019-05-07 19:53:37 +0000464
465 return ipmi::responseSuccess(sysAcpiState, devAcpiState);
Chris Austen6caf28b2015-10-13 12:40:40 -0500466}
467
Chris Austen7303bdc2016-04-17 11:50:54 -0500468typedef struct
469{
470 char major;
471 char minor;
Potin Laic7c55922023-02-16 10:33:55 +0800472 uint8_t aux[4];
Vernon Mauery86a50822019-03-25 13:11:36 -0700473} Revision;
Chris Austen7303bdc2016-04-17 11:50:54 -0500474
Potin Lai5d214202023-01-16 18:11:49 +0800475/* Use regular expression searching matched pattern X.Y, and convert it to */
476/* Major (X) and Minor (Y) version. */
477/* Example: */
478/* version = 2.14.0-dev */
479/* ^ ^ */
480/* | |---------------- Minor */
481/* |------------------ Major */
482/* */
Potin Laic7c55922023-02-16 10:33:55 +0800483/* Default regex string only tries to match Major and Minor version. */
484/* */
485/* To match more firmware version info, platforms need to define it own */
486/* regex string to match more strings, and assign correct mapping index in */
487/* matches array. */
488/* */
489/* matches[0]: matched index for major ver */
490/* matches[1]: matched index for minor ver */
491/* matches[2]: matched index for aux[0] (set 0 to skip) */
492/* matches[3]: matched index for aux[1] (set 0 to skip) */
493/* matches[4]: matched index for aux[2] (set 0 to skip) */
494/* matches[5]: matched index for aux[3] (set 0 to skip) */
495/* Example: */
496/* regex = "([\d]+).([\d]+).([\d]+)-dev-([\d]+)-g([0-9a-fA-F]{2}) */
497/* ([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})" */
498/* matches = {1,2,5,6,7,8} */
499/* version = 2.14.0-dev-750-g37a7c5ad1-dirty */
500/* ^ ^ ^ ^ ^ ^ ^ ^ */
501/* | | | | | | | | */
502/* | | | | | | | |-- Aux byte 3 (0xAD), index 8 */
503/* | | | | | | |---- Aux byte 2 (0xC5), index 7 */
504/* | | | | | |------ Aux byte 1 (0xA7), index 6 */
505/* | | | | |-------- Aux byte 0 (0x37), index 5 */
506/* | | | |------------- Not used, index 4 */
507/* | | |------------------- Not used, index 3 */
508/* | |---------------------- Minor (14), index 2 */
509/* |------------------------ Major (2), index 1 */
Potin Lai5d214202023-01-16 18:11:49 +0800510int convertVersion(std::string s, Revision& rev)
Chris Austen7303bdc2016-04-17 11:50:54 -0500511{
Potin Laic7c55922023-02-16 10:33:55 +0800512 static const std::vector<size_t> matches = {
513 MAJOR_MATCH_INDEX, MINOR_MATCH_INDEX, AUX_0_MATCH_INDEX,
514 AUX_1_MATCH_INDEX, AUX_2_MATCH_INDEX, AUX_3_MATCH_INDEX};
Potin Lai5d214202023-01-16 18:11:49 +0800515 std::regex fw_regex(FW_VER_REGEX);
516 std::smatch m;
517 Revision r = {0};
518 size_t val;
Chris Austen7303bdc2016-04-17 11:50:54 -0500519
Potin Laidcde04e2023-03-16 18:44:15 +0800520 if (std::regex_search(s, m, fw_regex))
Dinesh Chinari2b7e07d2017-11-08 15:38:50 -0600521 {
Potin Laic7c55922023-02-16 10:33:55 +0800522 if (m.size() < *std::max_element(matches.begin(), matches.end()))
523 { // max index higher than match count
Potin Laidcde04e2023-03-16 18:44:15 +0800524 return -1;
525 }
526
Potin Lai5d214202023-01-16 18:11:49 +0800527 // convert major
528 {
Potin Laic7c55922023-02-16 10:33:55 +0800529 std::string_view str = m[matches[0]].str();
Potin Lai5d214202023-01-16 18:11:49 +0800530 auto [ptr, ec]{std::from_chars(str.begin(), str.end(), val)};
531 if (ec != std::errc() || ptr != str.begin() + str.size())
532 { // failed to convert major string
Potin Laidcde04e2023-03-16 18:44:15 +0800533 return -1;
Potin Lai5d214202023-01-16 18:11:49 +0800534 }
Potin Lai058e2912023-09-07 10:16:38 +0800535
536 if (val >= 2000)
537 { // For the platforms use year as major version, it would expect to
538 // have major version between 0 - 99. If the major version is
539 // greater than or equal to 2000, it is treated as a year and
540 // converted to 0 - 99.
541 r.major = val % 100;
542 }
543 else
544 {
545 r.major = val & 0x7F;
546 }
Potin Lai5d214202023-01-16 18:11:49 +0800547 }
548
549 // convert minor
550 {
Potin Laic7c55922023-02-16 10:33:55 +0800551 std::string_view str = m[matches[1]].str();
Potin Lai5d214202023-01-16 18:11:49 +0800552 auto [ptr, ec]{std::from_chars(str.begin(), str.end(), val)};
553 if (ec != std::errc() || ptr != str.begin() + str.size())
554 { // failed to convert minor string
Potin Laidcde04e2023-03-16 18:44:15 +0800555 return -1;
Potin Lai5d214202023-01-16 18:11:49 +0800556 }
557 r.minor = val & 0xFF;
558 }
559
Potin Laic7c55922023-02-16 10:33:55 +0800560 // convert aux bytes
561 {
562 size_t i;
563 for (i = 0; i < 4; i++)
564 {
565 if (matches[i + 2] == 0)
566 {
567 continue;
568 }
569
570 std::string_view str = m[matches[i + 2]].str();
571 auto [ptr,
572 ec]{std::from_chars(str.begin(), str.end(), val, 16)};
573 if (ec != std::errc() || ptr != str.begin() + str.size())
574 { // failed to convert aux byte string
575 break;
576 }
577
578 r.aux[i] = val & 0xFF;
579 }
580
581 if (i != 4)
582 { // something wrong durign converting aux bytes
583 return -1;
584 }
585 }
586
Potin Lai5d214202023-01-16 18:11:49 +0800587 // all matched
588 rev = r;
589 return 0;
Chris Austen176c9652016-04-30 16:32:17 -0500590 }
Chris Austen7303bdc2016-04-17 11:50:54 -0500591
Potin Lai5d214202023-01-16 18:11:49 +0800592 return -1;
Chris Austen7303bdc2016-04-17 11:50:54 -0500593}
594
Vernon Maueryea1c4012019-05-24 13:26:16 -0700595/* @brief: Implement the Get Device ID IPMI command per the IPMI spec
596 * @param[in] ctx - shared_ptr to an IPMI context struct
597 *
598 * @returns IPMI completion code plus response data
599 * - Device ID (manufacturer defined)
600 * - Device revision[4 bits]; reserved[3 bits]; SDR support[1 bit]
601 * - FW revision major[7 bits] (binary encoded); available[1 bit]
602 * - FW Revision minor (BCD encoded)
603 * - IPMI version (0x02 for IPMI 2.0)
604 * - device support (bitfield of supported options)
605 * - MFG IANA ID (3 bytes)
606 * - product ID (2 bytes)
607 * - AUX info (4 bytes)
608 */
609ipmi::RspType<uint8_t, // Device ID
610 uint8_t, // Device Revision
611 uint8_t, // Firmware Revision Major
612 uint8_t, // Firmware Revision minor
613 uint8_t, // IPMI version
614 uint8_t, // Additional device support
615 uint24_t, // MFG ID
616 uint16_t, // Product ID
617 uint32_t // AUX info
618 >
Willy Tu11d68892022-01-20 10:37:34 -0800619 ipmiAppGetDeviceId([[maybe_unused]] ipmi::Context::ptr ctx)
Chris Austen6caf28b2015-10-13 12:40:40 -0500620{
Vernon Mauery86a50822019-03-25 13:11:36 -0700621 static struct
622 {
623 uint8_t id;
624 uint8_t revision;
625 uint8_t fw[2];
626 uint8_t ipmiVer;
627 uint8_t addnDevSupport;
628 uint24_t manufId;
629 uint16_t prodId;
630 uint32_t aux;
631 } devId;
David Cobbleya1adb072017-11-21 15:58:13 -0800632 static bool dev_id_initialized = false;
Patrick Venture94930a12019-04-30 10:01:58 -0700633 static bool defaultActivationSetting = true;
David Cobbleya1adb072017-11-21 15:58:13 -0800634 const char* filename = "/usr/share/ipmi-providers/dev_id.json";
Alexander Amelkinba19c182018-09-04 15:49:36 +0300635 constexpr auto ipmiDevIdStateShift = 7;
636 constexpr auto ipmiDevIdFw1Mask = ~(1 << ipmiDevIdStateShift);
Willy Tub78184e2022-10-27 22:57:38 +0000637
638#ifdef GET_DBUS_ACTIVE_SOFTWARE
639 static bool haveBMCVersion = false;
JeffLin27a62ec2021-11-24 15:40:33 +0800640 if (!haveBMCVersion || !dev_id_initialized)
David Cobbleya1adb072017-11-21 15:58:13 -0800641 {
Willy Tub78184e2022-10-27 22:57:38 +0000642 int r = -1;
643 Revision rev = {0, 0, 0, 0};
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600644 try
645 {
Vernon Maueryea1c4012019-05-24 13:26:16 -0700646 auto version = getActiveSoftwareVersionInfo(ctx);
Vernon Mauery86a50822019-03-25 13:11:36 -0700647 r = convertVersion(version, rev);
David Cobbleya1adb072017-11-21 15:58:13 -0800648 }
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600649 catch (const std::exception& e)
650 {
651 log<level::ERR>(e.what());
652 }
Nan Liee0cb902016-07-11 15:38:03 +0800653
Patrick Venture0b02be92018-08-31 11:55:55 -0700654 if (r >= 0)
655 {
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600656 // bit7 identifies if the device is available
657 // 0=normal operation
658 // 1=device firmware, SDR update,
659 // or self-initialization in progress.
Alexander Amelkinba19c182018-09-04 15:49:36 +0300660 // The availability may change in run time, so mask here
661 // and initialize later.
Vernon Mauery86a50822019-03-25 13:11:36 -0700662 devId.fw[0] = rev.major & ipmiDevIdFw1Mask;
Nagaraju Goruganti744398d2018-02-20 09:52:00 -0600663
664 rev.minor = (rev.minor > 99 ? 99 : rev.minor);
Vernon Mauery86a50822019-03-25 13:11:36 -0700665 devId.fw[1] = rev.minor % 10 + (rev.minor / 10) * 16;
Potin Laic7c55922023-02-16 10:33:55 +0800666 std::memcpy(&devId.aux, rev.aux, sizeof(rev.aux));
Brandon Kimc1f5aca2022-03-17 17:54:06 -0700667 haveBMCVersion = true;
David Cobbleya1adb072017-11-21 15:58:13 -0800668 }
JeffLin27a62ec2021-11-24 15:40:33 +0800669 }
Willy Tub78184e2022-10-27 22:57:38 +0000670#endif
JeffLin27a62ec2021-11-24 15:40:33 +0800671 if (!dev_id_initialized)
672 {
David Cobbleya1adb072017-11-21 15:58:13 -0800673 // IPMI Spec version 2.0
Vernon Mauery86a50822019-03-25 13:11:36 -0700674 devId.ipmiVer = 2;
Adriana Kobylak0e912642016-06-22 16:54:39 -0500675
Vernon Mauery86a50822019-03-25 13:11:36 -0700676 std::ifstream devIdFile(filename);
677 if (devIdFile.is_open())
David Cobbleya1adb072017-11-21 15:58:13 -0800678 {
Vernon Mauery86a50822019-03-25 13:11:36 -0700679 auto data = nlohmann::json::parse(devIdFile, nullptr, false);
David Cobbleya1adb072017-11-21 15:58:13 -0800680 if (!data.is_discarded())
681 {
Vernon Mauery86a50822019-03-25 13:11:36 -0700682 devId.id = data.value("id", 0);
683 devId.revision = data.value("revision", 0);
684 devId.addnDevSupport = data.value("addn_dev_support", 0);
685 devId.manufId = data.value("manuf_id", 0);
686 devId.prodId = data.value("prod_id", 0);
Potin Lai9a686362023-09-05 09:27:53 +0800687#ifdef GET_DBUS_ACTIVE_SOFTWARE
688 if (!(AUX_0_MATCH_INDEX || AUX_1_MATCH_INDEX ||
689 AUX_2_MATCH_INDEX || AUX_3_MATCH_INDEX))
690#endif
691 {
692 devId.aux = data.value("aux", 0);
693 }
David Cobbleya1adb072017-11-21 15:58:13 -0800694
Willy Tubfd3a172022-05-31 13:57:54 -0700695 if (data.contains("firmware_revision"))
696 {
697 const auto& firmwareRevision = data.at("firmware_revision");
698 if (firmwareRevision.contains("major"))
699 {
700 firmwareRevision.at("major").get_to(devId.fw[0]);
701 }
702 if (firmwareRevision.contains("minor"))
703 {
704 firmwareRevision.at("minor").get_to(devId.fw[1]);
705 }
706 }
707
Patrick Venture94930a12019-04-30 10:01:58 -0700708 // Set the availablitity of the BMC.
709 defaultActivationSetting = data.value("availability", true);
710
Patrick Venture0b02be92018-08-31 11:55:55 -0700711 // Don't read the file every time if successful
David Cobbleya1adb072017-11-21 15:58:13 -0800712 dev_id_initialized = true;
713 }
714 else
715 {
716 log<level::ERR>("Device ID JSON parser failure");
Vernon Maueryf2587fc2019-04-09 14:28:15 -0700717 return ipmi::responseUnspecifiedError();
David Cobbleya1adb072017-11-21 15:58:13 -0800718 }
719 }
720 else
721 {
722 log<level::ERR>("Device ID file not found");
Vernon Maueryf2587fc2019-04-09 14:28:15 -0700723 return ipmi::responseUnspecifiedError();
Chris Austen7303bdc2016-04-17 11:50:54 -0500724 }
725 }
Chris Austen6caf28b2015-10-13 12:40:40 -0500726
Alexander Amelkinba19c182018-09-04 15:49:36 +0300727 // Set availability to the actual current BMC state
Vernon Mauery86a50822019-03-25 13:11:36 -0700728 devId.fw[0] &= ipmiDevIdFw1Mask;
Patrick Venture94930a12019-04-30 10:01:58 -0700729 if (!getCurrentBmcStateWithFallback(defaultActivationSetting))
Alexander Amelkinba19c182018-09-04 15:49:36 +0300730 {
Vernon Mauery86a50822019-03-25 13:11:36 -0700731 devId.fw[0] |= (1 << ipmiDevIdStateShift);
Alexander Amelkinba19c182018-09-04 15:49:36 +0300732 }
733
Vernon Mauery86a50822019-03-25 13:11:36 -0700734 return ipmi::responseSuccess(
735 devId.id, devId.revision, devId.fw[0], devId.fw[1], devId.ipmiVer,
736 devId.addnDevSupport, devId.manufId, devId.prodId, devId.aux);
Chris Austen6caf28b2015-10-13 12:40:40 -0500737}
738
Vernon Maueryb84a5282019-03-25 13:39:03 -0700739auto ipmiAppGetSelfTestResults() -> ipmi::RspType<uint8_t, uint8_t>
Nan Li41fa24a2016-11-10 20:12:37 +0800740{
Nan Li41fa24a2016-11-10 20:12:37 +0800741 // Byte 2:
742 // 55h - No error.
Gunnar Mills8991dd62017-10-25 17:11:29 -0500743 // 56h - Self Test function not implemented in this controller.
Nan Li41fa24a2016-11-10 20:12:37 +0800744 // 57h - Corrupted or inaccesssible data or devices.
745 // 58h - Fatal hardware error.
746 // FFh - reserved.
747 // all other: Device-specific 'internal failure'.
748 // Byte 3:
749 // For byte 2 = 55h, 56h, FFh: 00h
750 // For byte 2 = 58h, all other: Device-specific
751 // For byte 2 = 57h: self-test error bitfield.
752 // Note: returning 57h does not imply that all test were run.
753 // [7] 1b = Cannot access SEL device.
754 // [6] 1b = Cannot access SDR Repository.
755 // [5] 1b = Cannot access BMC FRU device.
756 // [4] 1b = IPMB signal lines do not respond.
757 // [3] 1b = SDR Repository empty.
758 // [2] 1b = Internal Use Area of BMC FRU corrupted.
759 // [1] 1b = controller update 'boot block' firmware corrupted.
760 // [0] 1b = controller operational firmware corrupted.
Vernon Maueryb84a5282019-03-25 13:39:03 -0700761 constexpr uint8_t notImplemented = 0x56;
762 constexpr uint8_t zero = 0;
763 return ipmi::responseSuccess(notImplemented, zero);
Nan Li41fa24a2016-11-10 20:12:37 +0800764}
765
Vernon Mauery15541322019-03-25 13:33:03 -0700766static constexpr size_t uuidBinaryLength = 16;
767static std::array<uint8_t, uuidBinaryLength> rfc4122ToIpmi(std::string rfc4122)
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500768{
Willy Tu523e2d12023-09-05 11:36:48 -0700769 using Argument = xyz::openbmc_project::common::InvalidArgument;
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500770 // UUID is in RFC4122 format. Ex: 61a39523-78f2-11e5-9862-e6402cfc3223
Patrick Ventured2117022018-02-06 08:54:37 -0800771 // Per IPMI Spec 2.0 need to convert to 16 hex bytes and reverse the byte
772 // order
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500773 // Ex: 0x2332fc2c40e66298e511f2782395a361
Vernon Mauery15541322019-03-25 13:33:03 -0700774 constexpr size_t uuidHexLength = (2 * uuidBinaryLength);
775 constexpr size_t uuidRfc4122Length = (uuidHexLength + 4);
776 std::array<uint8_t, uuidBinaryLength> uuid;
777 if (rfc4122.size() == uuidRfc4122Length)
Patrick Venture0b02be92018-08-31 11:55:55 -0700778 {
Vernon Mauery15541322019-03-25 13:33:03 -0700779 rfc4122.erase(std::remove(rfc4122.begin(), rfc4122.end(), '-'),
780 rfc4122.end());
Sergey Solomineb9b8142016-08-23 09:07:28 -0500781 }
Vernon Mauery15541322019-03-25 13:33:03 -0700782 if (rfc4122.size() != uuidHexLength)
vishwa1eaea4f2016-02-26 11:57:40 -0600783 {
Vernon Mauery15541322019-03-25 13:33:03 -0700784 elog<InvalidArgument>(Argument::ARGUMENT_NAME("rfc4122"),
785 Argument::ARGUMENT_VALUE(rfc4122.c_str()));
vishwa1eaea4f2016-02-26 11:57:40 -0600786 }
Vernon Mauery15541322019-03-25 13:33:03 -0700787 for (size_t ind = 0; ind < uuidHexLength; ind += 2)
vishwa1eaea4f2016-02-26 11:57:40 -0600788 {
Vernon Mauery15541322019-03-25 13:33:03 -0700789 char v[3];
790 v[0] = rfc4122[ind];
791 v[1] = rfc4122[ind + 1];
792 v[2] = 0;
793 size_t err;
794 long b;
795 try
Emily Shafferedb8bb02018-09-27 14:50:15 -0700796 {
Vernon Mauery15541322019-03-25 13:33:03 -0700797 b = std::stoul(v, &err, 16);
Emily Shafferedb8bb02018-09-27 14:50:15 -0700798 }
Patrick Williamsa2ad2da2021-10-06 12:21:46 -0500799 catch (const std::exception& e)
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500800 {
Vernon Mauery15541322019-03-25 13:33:03 -0700801 elog<InvalidArgument>(Argument::ARGUMENT_NAME("rfc4122"),
802 Argument::ARGUMENT_VALUE(rfc4122.c_str()));
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500803 }
Vernon Mauery15541322019-03-25 13:33:03 -0700804 // check that exactly two ascii bytes were converted
805 if (err != 2)
806 {
807 elog<InvalidArgument>(Argument::ARGUMENT_NAME("rfc4122"),
808 Argument::ARGUMENT_VALUE(rfc4122.c_str()));
809 }
810 uuid[uuidBinaryLength - (ind / 2) - 1] = static_cast<uint8_t>(b);
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500811 }
Vernon Mauery15541322019-03-25 13:33:03 -0700812 return uuid;
813}
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500814
Vernon Mauery15541322019-03-25 13:33:03 -0700815auto ipmiAppGetDeviceGuid()
816 -> ipmi::RspType<std::array<uint8_t, uuidBinaryLength>>
817{
818 // return a fixed GUID based on /etc/machine-id
819 // This should match the /redfish/v1/Managers/bmc's UUID data
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500820
Vernon Mauery15541322019-03-25 13:33:03 -0700821 // machine specific application ID (for BMC ID)
822 // generated by systemd-id128 -p new as per man page
823 static constexpr sd_id128_t bmcUuidAppId = SD_ID128_MAKE(
824 e0, e1, 73, 76, 64, 61, 47, da, a5, 0c, d0, cc, 64, 12, 45, 78);
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500825
Vernon Mauery15541322019-03-25 13:33:03 -0700826 sd_id128_t bmcUuid;
827 // create the UUID from /etc/machine-id via the systemd API
828 sd_id128_get_machine_app_specific(bmcUuidAppId, &bmcUuid);
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500829
Vernon Mauery15541322019-03-25 13:33:03 -0700830 char bmcUuidCstr[SD_ID128_STRING_MAX];
831 std::string systemUuid = sd_id128_to_string(bmcUuid, bmcUuidCstr);
832
833 std::array<uint8_t, uuidBinaryLength> uuid = rfc4122ToIpmi(systemUuid);
834 return ipmi::responseSuccess(uuid);
Adriana Kobylakd100ee52015-10-20 17:02:37 -0500835}
Chris Austen6caf28b2015-10-13 12:40:40 -0500836
Vernon Mauerycc99ba42019-03-25 13:40:11 -0700837auto ipmiAppGetBtCapabilities()
838 -> ipmi::RspType<uint8_t, uint8_t, uint8_t, uint8_t, uint8_t>
vishwabmcba0bd5f2015-09-30 16:50:23 +0530839{
Adriana Kobylak88ad8152016-12-13 10:09:08 -0600840 // Per IPMI 2.0 spec, the input and output buffer size must be the max
841 // buffer size minus one byte to allocate space for the length byte.
Vernon Mauerycc99ba42019-03-25 13:40:11 -0700842 constexpr uint8_t nrOutstanding = 0x01;
843 constexpr uint8_t inputBufferSize = MAX_IPMI_BUFFER - 1;
844 constexpr uint8_t outputBufferSize = MAX_IPMI_BUFFER - 1;
845 constexpr uint8_t transactionTime = 0x0A;
846 constexpr uint8_t nrRetries = 0x01;
vishwabmcba0bd5f2015-09-30 16:50:23 +0530847
Vernon Mauerycc99ba42019-03-25 13:40:11 -0700848 return ipmi::responseSuccess(nrOutstanding, inputBufferSize,
849 outputBufferSize, transactionTime, nrRetries);
vishwabmcba0bd5f2015-09-30 16:50:23 +0530850}
851
Vernon Mauery6c44a942023-07-27 11:00:10 -0700852auto ipmiAppGetSystemGuid(ipmi::Context::ptr& ctx)
853 -> ipmi::RspType<std::array<uint8_t, 16>>
Marri Devender Rao5e007a52018-01-08 06:18:36 -0600854{
Vernon Maueryb90a5322019-03-25 13:36:55 -0700855 static constexpr auto uuidInterface = "xyz.openbmc_project.Common.UUID";
856 static constexpr auto uuidProperty = "UUID";
Marri Devender Rao5e007a52018-01-08 06:18:36 -0600857
Vernon Mauery6c44a942023-07-27 11:00:10 -0700858 // Get the Inventory object implementing BMC interface
859 ipmi::DbusObjectInfo objectInfo{};
860 boost::system::error_code ec = ipmi::getDbusObject(ctx, uuidInterface,
861 objectInfo);
862 if (ec.value())
Marri Devender Rao5e007a52018-01-08 06:18:36 -0600863 {
Vernon Mauery6c44a942023-07-27 11:00:10 -0700864 log<level::ERR>("Failed to locate System UUID object",
865 entry("INTERFACE=%s", uuidInterface),
866 entry("ERROR=%s", ec.message().c_str()));
867 return ipmi::responseUnspecifiedError();
Marri Devender Rao5e007a52018-01-08 06:18:36 -0600868 }
Vernon Mauery6c44a942023-07-27 11:00:10 -0700869
870 // Read UUID property value from bmcObject
871 // UUID is in RFC4122 format Ex: 61a39523-78f2-11e5-9862-e6402cfc3223
872 std::string rfc4122Uuid{};
873 ec = ipmi::getDbusProperty(ctx, objectInfo.second, objectInfo.first,
874 uuidInterface, uuidProperty, rfc4122Uuid);
875 if (ec.value())
Marri Devender Rao5e007a52018-01-08 06:18:36 -0600876 {
877 log<level::ERR>("Failed in reading BMC UUID property",
Vernon Maueryb90a5322019-03-25 13:36:55 -0700878 entry("INTERFACE=%s", uuidInterface),
Vernon Mauery6c44a942023-07-27 11:00:10 -0700879 entry("PROPERTY=%s", uuidProperty),
880 entry("ERROR=%s", ec.message().c_str()));
Vernon Maueryb90a5322019-03-25 13:36:55 -0700881 return ipmi::responseUnspecifiedError();
Marri Devender Rao5e007a52018-01-08 06:18:36 -0600882 }
Vernon Maueryb90a5322019-03-25 13:36:55 -0700883 std::array<uint8_t, 16> uuid;
Vernon Maueryb90a5322019-03-25 13:36:55 -0700884 try
885 {
886 // convert to IPMI format
887 uuid = rfc4122ToIpmi(rfc4122Uuid);
888 }
889 catch (const InvalidArgument& e)
890 {
891 log<level::ERR>("Failed in parsing BMC UUID property",
892 entry("INTERFACE=%s", uuidInterface),
893 entry("PROPERTY=%s", uuidProperty),
894 entry("VALUE=%s", rfc4122Uuid.c_str()));
895 return ipmi::responseUnspecifiedError();
896 }
897 return ipmi::responseSuccess(uuid);
Marri Devender Rao5e007a52018-01-08 06:18:36 -0600898}
899
Rajashekar Gade Reddye7023922019-07-10 16:54:55 +0000900/**
901 * @brief set the session state as teardown
902 *
903 * This function is to set the session state to tear down in progress if the
904 * state is active.
905 *
906 * @param[in] busp - Dbus obj
907 * @param[in] service - service name
908 * @param[in] obj - object path
909 *
910 * @return success completion code if it sets the session state to
911 * tearDownInProgress else return the corresponding error completion code.
912 **/
913uint8_t setSessionState(std::shared_ptr<sdbusplus::asio::connection>& busp,
914 const std::string& service, const std::string& obj)
915{
916 try
917 {
918 uint8_t sessionState = std::get<uint8_t>(ipmi::getDbusProperty(
919 *busp, service, obj, session::sessionIntf, "State"));
920
921 if (sessionState == static_cast<uint8_t>(session::State::active))
922 {
923 ipmi::setDbusProperty(
924 *busp, service, obj, session::sessionIntf, "State",
925 static_cast<uint8_t>(session::State::tearDownInProgress));
926 return ipmi::ccSuccess;
927 }
928 }
Patrick Williamsa2ad2da2021-10-06 12:21:46 -0500929 catch (const std::exception& e)
Rajashekar Gade Reddye7023922019-07-10 16:54:55 +0000930 {
931 log<level::ERR>("Failed in getting session state property",
932 entry("service=%s", service.c_str()),
933 entry("object path=%s", obj.c_str()),
934 entry("interface=%s", session::sessionIntf));
935 return ipmi::ccUnspecifiedError;
936 }
937
938 return ipmi::ccInvalidFieldRequest;
939}
940
941ipmi::RspType<> ipmiAppCloseSession(uint32_t reqSessionId,
942 std::optional<uint8_t> requestSessionHandle)
943{
944 auto busp = getSdBus();
945 uint8_t reqSessionHandle =
946 requestSessionHandle.value_or(session::defaultSessionHandle);
947
948 if (reqSessionId == session::sessionZero &&
949 reqSessionHandle == session::defaultSessionHandle)
950 {
951 return ipmi::response(session::ccInvalidSessionId);
952 }
953
954 if (reqSessionId == session::sessionZero &&
955 reqSessionHandle == session::invalidSessionHandle)
956 {
957 return ipmi::response(session::ccInvalidSessionHandle);
958 }
959
960 if (reqSessionId != session::sessionZero &&
961 reqSessionHandle != session::defaultSessionHandle)
962 {
963 return ipmi::response(ipmi::ccInvalidFieldRequest);
964 }
965
966 try
967 {
968 ipmi::ObjectTree objectTree = ipmi::getAllDbusObjects(
969 *busp, session::sessionManagerRootPath, session::sessionIntf);
970
971 for (auto& objectTreeItr : objectTree)
972 {
973 const std::string obj = objectTreeItr.first;
974
975 if (isSessionObjectMatched(obj, reqSessionId, reqSessionHandle))
976 {
977 auto& serviceMap = objectTreeItr.second;
978
979 // Session id and session handle are unique for each session.
980 // Session id and handler are retrived from the object path and
981 // object path will be unique for each session. Checking if
982 // multiple objects exist with same object path under multiple
983 // services.
984 if (serviceMap.size() != 1)
985 {
986 return ipmi::responseUnspecifiedError();
987 }
988
989 auto itr = serviceMap.begin();
990 const std::string service = itr->first;
991 return ipmi::response(setSessionState(busp, service, obj));
992 }
993 }
994 }
Patrick Williams5d82f472022-07-22 19:26:53 -0500995 catch (const sdbusplus::exception_t& e)
Rajashekar Gade Reddye7023922019-07-10 16:54:55 +0000996 {
997 log<level::ERR>("Failed to fetch object from dbus",
998 entry("INTERFACE=%s", session::sessionIntf),
999 entry("ERRMSG=%s", e.what()));
1000 return ipmi::responseUnspecifiedError();
1001 }
1002
1003 return ipmi::responseInvalidFieldRequest();
1004}
1005
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001006uint8_t getTotalSessionCount()
1007{
Meera-Kattac1789482021-05-18 09:53:26 +00001008 uint8_t count = 0, ch = 0;
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001009
1010 while (ch < ipmi::maxIpmiChannels &&
1011 count < session::maxNetworkInstanceSupported)
1012 {
Jayaprakash Mutyala80207e62020-07-04 16:34:15 +00001013 ipmi::ChannelInfo chInfo{};
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001014 ipmi::getChannelInfo(ch, chInfo);
1015 if (static_cast<ipmi::EChannelMediumType>(chInfo.mediumType) ==
1016 ipmi::EChannelMediumType::lan8032)
1017 {
1018 count++;
1019 }
1020 ch++;
1021 }
1022 return count * session::maxSessionCountPerChannel;
1023}
1024
1025/**
1026 * @brief get session info request data.
1027 *
1028 * This function validates the request data and retrive request session id,
1029 * session handle.
1030 *
Rajashekar Gade Reddy4d226402019-11-13 17:13:05 +05301031 * @param[in] ctx - context of current session.
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001032 * @param[in] sessionIndex - request session index
1033 * @param[in] payload - input payload
1034 * @param[in] reqSessionId - unpacked session Id will be asigned
1035 * @param[in] reqSessionHandle - unpacked session handle will be asigned
1036 *
1037 * @return success completion code if request data is valid
1038 * else return the correcponding error completion code.
1039 **/
Rajashekar Gade Reddy4d226402019-11-13 17:13:05 +05301040uint8_t getSessionInfoRequestData(const ipmi::Context::ptr ctx,
1041 const uint8_t sessionIndex,
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001042 ipmi::message::Payload& payload,
1043 uint32_t& reqSessionId,
1044 uint8_t& reqSessionHandle)
1045{
Rajashekar Gade Reddy4d226402019-11-13 17:13:05 +05301046 if ((sessionIndex > session::maxSessionCountPerChannel) &&
1047 (sessionIndex < session::searchSessionByHandle))
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001048 {
1049 return ipmi::ccInvalidFieldRequest;
1050 }
1051
1052 switch (sessionIndex)
1053 {
Rajashekar Gade Reddy4d226402019-11-13 17:13:05 +05301054 case session::searchCurrentSession:
1055
1056 ipmi::ChannelInfo chInfo;
1057 ipmi::getChannelInfo(ctx->channel, chInfo);
1058
1059 if (static_cast<ipmi::EChannelMediumType>(chInfo.mediumType) !=
1060 ipmi::EChannelMediumType::lan8032)
1061 {
1062 return ipmi::ccInvalidFieldRequest;
1063 }
1064
1065 if (!payload.fullyUnpacked())
1066 {
1067 return ipmi::ccReqDataLenInvalid;
1068 }
1069 // Check if current sessionId is 0, sessionId 0 is reserved.
1070 if (ctx->sessionId == session::sessionZero)
1071 {
1072 return session::ccInvalidSessionId;
1073 }
1074 reqSessionId = ctx->sessionId;
1075 break;
1076
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001077 case session::searchSessionByHandle:
1078
1079 if ((payload.unpack(reqSessionHandle)) ||
1080 (!payload.fullyUnpacked()))
1081 {
1082 return ipmi::ccReqDataLenInvalid;
1083 }
1084
1085 if ((reqSessionHandle == session::sessionZero) ||
1086 ((reqSessionHandle & session::multiIntfaceSessionHandleMask) >
1087 session::maxSessionCountPerChannel))
1088 {
1089 return session::ccInvalidSessionHandle;
1090 }
1091 break;
1092
1093 case session::searchSessionById:
1094
1095 if ((payload.unpack(reqSessionId)) || (!payload.fullyUnpacked()))
1096 {
1097 return ipmi::ccReqDataLenInvalid;
1098 }
1099
1100 if (reqSessionId == session::sessionZero)
1101 {
1102 return session::ccInvalidSessionId;
1103 }
1104 break;
1105
1106 default:
1107 if (!payload.fullyUnpacked())
1108 {
1109 return ipmi::ccReqDataLenInvalid;
1110 }
1111 break;
1112 }
1113 return ipmi::ccSuccess;
1114}
1115
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001116uint8_t getSessionState(ipmi::Context::ptr ctx, const std::string& service,
1117 const std::string& objPath, uint8_t& sessionState)
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001118{
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001119 boost::system::error_code ec = ipmi::getDbusProperty(
1120 ctx, service, objPath, session::sessionIntf, "State", sessionState);
1121 if (ec)
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001122 {
1123 log<level::ERR>("Failed to fetch state property ",
1124 entry("SERVICE=%s", service.c_str()),
1125 entry("OBJECTPATH=%s", objPath.c_str()),
1126 entry("INTERFACE=%s", session::sessionIntf),
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001127 entry("ERRMSG=%s", ec.message().c_str()));
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001128 return ipmi::ccUnspecifiedError;
1129 }
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001130 return ipmi::ccSuccess;
1131}
1132
1133static constexpr uint8_t macAddrLen = 6;
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001134/** Alias SessionDetails - contain the optional information about an
1135 * RMCP+ session.
1136 *
1137 * @param userID - uint6_t session user ID (0-63)
1138 * @param reserved - uint2_t reserved
1139 * @param privilege - uint4_t session privilege (0-5)
1140 * @param reserved - uint4_t reserved
1141 * @param channel - uint4_t session channel number
1142 * @param protocol - uint4_t session protocol
1143 * @param remoteIP - uint32_t remote IP address
1144 * @param macAddr - std::array<uint8_t, 6> mac address
1145 * @param port - uint16_t remote port
1146 */
1147using SessionDetails =
1148 std::tuple<uint2_t, uint6_t, uint4_t, uint4_t, uint4_t, uint4_t, uint32_t,
1149 std::array<uint8_t, macAddrLen>, uint16_t>;
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001150
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001151/** @brief get session details for a given session
1152 *
1153 * @param[in] ctx - ipmi::Context pointer for accessing D-Bus
1154 * @param[in] service - D-Bus service name to fetch details from
1155 * @param[in] objPath - D-Bus object path for session
1156 * @param[out] sessionHandle - return session handle for session
1157 * @param[out] sessionState - return session state for session
1158 * @param[out] details - return a SessionDetails tuple containing other
1159 * session info
1160 * @return - ipmi::Cc success or error code
1161 */
1162ipmi::Cc getSessionDetails(ipmi::Context::ptr ctx, const std::string& service,
1163 const std::string& objPath, uint8_t& sessionHandle,
1164 uint8_t& sessionState, SessionDetails& details)
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001165{
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001166 ipmi::PropertyMap sessionProps;
1167 boost::system::error_code ec = ipmi::getAllDbusProperties(
1168 ctx, service, objPath, session::sessionIntf, sessionProps);
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001169
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001170 if (ec)
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001171 {
1172 log<level::ERR>("Failed to fetch state property ",
1173 entry("SERVICE=%s", service.c_str()),
1174 entry("OBJECTPATH=%s", objPath.c_str()),
1175 entry("INTERFACE=%s", session::sessionIntf),
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001176 entry("ERRMSG=%s", ec.message().c_str()));
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001177 return ipmi::ccUnspecifiedError;
1178 }
1179
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001180 sessionState = ipmi::mappedVariant<uint8_t>(
1181 sessionProps, "State", static_cast<uint8_t>(session::State::inactive));
1182 if (sessionState == static_cast<uint8_t>(session::State::active))
1183 {
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -05001184 sessionHandle = ipmi::mappedVariant<uint8_t>(sessionProps,
1185 "SessionHandle", 0);
1186 std::get<0>(details) = ipmi::mappedVariant<uint8_t>(sessionProps,
1187 "UserID", 0xff);
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001188 // std::get<1>(details) = 0; // (default constructed to 0)
1189 std::get<2>(details) =
1190 ipmi::mappedVariant<uint8_t>(sessionProps, "CurrentPrivilege", 0);
1191 // std::get<3>(details) = 0; // (default constructed to 0)
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -05001192 std::get<4>(details) = ipmi::mappedVariant<uint8_t>(sessionProps,
1193 "ChannelNum", 0xff);
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001194 constexpr uint4_t rmcpPlusProtocol = 1;
1195 std::get<5>(details) = rmcpPlusProtocol;
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -05001196 std::get<6>(details) = ipmi::mappedVariant<uint32_t>(sessionProps,
1197 "RemoteIPAddr", 0);
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001198 // std::get<7>(details) = {{0}}; // default constructed to all 0
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -05001199 std::get<8>(details) = ipmi::mappedVariant<uint16_t>(sessionProps,
1200 "RemotePort", 0);
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001201 }
1202
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001203 return ipmi::ccSuccess;
1204}
1205
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001206ipmi::RspType<uint8_t, // session handle,
1207 uint8_t, // total session count
1208 uint8_t, // active session count
1209 std::optional<SessionDetails>>
Rajashekar Gade Reddy4d226402019-11-13 17:13:05 +05301210 ipmiAppGetSessionInfo(ipmi::Context::ptr ctx, uint8_t sessionIndex,
1211 ipmi::message::Payload& payload)
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001212{
1213 uint32_t reqSessionId = 0;
1214 uint8_t reqSessionHandle = session::defaultSessionHandle;
1215 // initializing state to 0xff as 0 represents state as inactive.
1216 uint8_t state = 0xFF;
1217
1218 uint8_t completionCode = getSessionInfoRequestData(
Rajashekar Gade Reddy4d226402019-11-13 17:13:05 +05301219 ctx, sessionIndex, payload, reqSessionId, reqSessionHandle);
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001220
1221 if (completionCode)
1222 {
1223 return ipmi::response(completionCode);
1224 }
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001225 ipmi::ObjectTree objectTree;
1226 boost::system::error_code ec = ipmi::getAllDbusObjects(
1227 ctx, session::sessionManagerRootPath, session::sessionIntf, objectTree);
1228 if (ec)
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001229 {
1230 log<level::ERR>("Failed to fetch object from dbus",
1231 entry("INTERFACE=%s", session::sessionIntf),
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001232 entry("ERRMSG=%s", ec.message().c_str()));
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001233 return ipmi::responseUnspecifiedError();
1234 }
1235
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001236 uint8_t totalSessionCount = getTotalSessionCount();
1237 uint8_t activeSessionCount = 0;
1238 uint8_t sessionHandle = session::defaultSessionHandle;
Johnathan Manteyca93bb12022-12-02 13:33:17 -08001239 uint8_t activeSessionHandle = 0;
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001240 std::optional<SessionDetails> maybeDetails;
1241 uint8_t index = 0;
1242 for (auto& objectTreeItr : objectTree)
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001243 {
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001244 uint32_t sessionId = 0;
1245 std::string objectPath = objectTreeItr.first;
1246
1247 if (!parseCloseSessionInputPayload(objectPath, sessionId,
1248 sessionHandle))
1249 {
1250 continue;
1251 }
1252 index++;
1253 auto& serviceMap = objectTreeItr.second;
1254 auto itr = serviceMap.begin();
1255
1256 if (serviceMap.size() != 1)
1257 {
1258 return ipmi::responseUnspecifiedError();
1259 }
1260
1261 std::string service = itr->first;
1262 uint8_t sessionState = 0;
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -05001263 completionCode = getSessionState(ctx, service, objectPath,
1264 sessionState);
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001265 if (completionCode)
1266 {
1267 return ipmi::response(completionCode);
1268 }
1269
1270 if (sessionState == static_cast<uint8_t>(session::State::active))
1271 {
1272 activeSessionCount++;
1273 }
1274
Johnathan Manteyca93bb12022-12-02 13:33:17 -08001275 if (index == sessionIndex || reqSessionId == sessionId ||
1276 reqSessionHandle == sessionHandle)
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001277 {
Johnathan Manteyca93bb12022-12-02 13:33:17 -08001278 SessionDetails details{};
1279 completionCode = getSessionDetails(ctx, service, objectPath,
1280 sessionHandle, state, details);
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001281
Johnathan Manteyca93bb12022-12-02 13:33:17 -08001282 if (completionCode)
1283 {
1284 return ipmi::response(completionCode);
1285 }
1286 activeSessionHandle = sessionHandle;
1287 maybeDetails = std::move(details);
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001288 }
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001289 }
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001290
1291 if (state == static_cast<uint8_t>(session::State::active) ||
1292 state == static_cast<uint8_t>(session::State::tearDownInProgress))
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001293 {
Johnathan Manteyca93bb12022-12-02 13:33:17 -08001294 return ipmi::responseSuccess(activeSessionHandle, totalSessionCount,
Vernon Mauerye7e8b812019-10-28 16:00:34 -07001295 activeSessionCount, maybeDetails);
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001296 }
1297
1298 return ipmi::responseInvalidFieldRequest();
1299}
1300
Xo Wangf542e8b2017-08-09 15:34:16 -07001301static std::unique_ptr<SysInfoParamStore> sysInfoParamStore;
1302
Xo Wang87651332017-08-11 10:17:59 -07001303static std::string sysInfoReadSystemName()
1304{
1305 // Use the BMC hostname as the "System Name."
1306 char hostname[HOST_NAME_MAX + 1] = {};
1307 if (gethostname(hostname, HOST_NAME_MAX) != 0)
1308 {
1309 perror("System info parameter: system name");
1310 }
1311 return hostname;
1312}
1313
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001314static constexpr uint8_t paramRevision = 0x11;
1315static constexpr size_t configParameterLength = 16;
Xo Wangf542e8b2017-08-09 15:34:16 -07001316
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001317static constexpr size_t smallChunkSize = 14;
1318static constexpr size_t fullChunkSize = 16;
Jia, chunhui449f2162019-09-11 16:51:51 +08001319static constexpr uint8_t progressMask = 0x3;
Xo Wangf542e8b2017-08-09 15:34:16 -07001320
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001321static constexpr uint8_t setComplete = 0x0;
1322static constexpr uint8_t setInProgress = 0x1;
1323static constexpr uint8_t commitWrite = 0x2;
1324static uint8_t transferStatus = setComplete;
1325
Jia, chunhui449f2162019-09-11 16:51:51 +08001326static constexpr uint8_t configDataOverhead = 2;
1327
1328// For EFI based system, 256 bytes is recommended.
1329static constexpr size_t maxBytesPerParameter = 256;
1330
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001331namespace ipmi
1332{
1333constexpr Cc ccParmNotSupported = 0x80;
Jia, chunhui449f2162019-09-11 16:51:51 +08001334constexpr Cc ccSetInProgressActive = 0x81;
1335constexpr Cc ccSystemInfoParameterSetReadOnly = 0x82;
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001336
1337static inline auto responseParmNotSupported()
1338{
1339 return response(ccParmNotSupported);
Xo Wangf542e8b2017-08-09 15:34:16 -07001340}
Jia, chunhui449f2162019-09-11 16:51:51 +08001341static inline auto responseSetInProgressActive()
1342{
1343 return response(ccSetInProgressActive);
1344}
1345static inline auto responseSystemInfoParameterSetReadOnly()
1346{
1347 return response(ccSystemInfoParameterSetReadOnly);
1348}
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001349} // namespace ipmi
Xo Wangf542e8b2017-08-09 15:34:16 -07001350
Jia, chunhui449f2162019-09-11 16:51:51 +08001351ipmi::RspType<uint8_t, // Parameter revision
1352 std::optional<uint8_t>, // data1 / setSelector / ProgressStatus
1353 std::optional<std::vector<uint8_t>>> // data2-17
jayaprakash Mutyalac2566a92020-04-23 21:18:35 +00001354 ipmiAppGetSystemInfo(uint7_t reserved, bool getRevision,
1355 uint8_t paramSelector, uint8_t setSelector,
1356 uint8_t BlockSelector)
Xo Wangf542e8b2017-08-09 15:34:16 -07001357{
Snehalatha Va5ae7722020-05-02 18:18:57 +00001358 if (reserved || (paramSelector >= invalidParamSelectorStart &&
1359 paramSelector <= invalidParamSelectorEnd))
jayaprakash Mutyalac2566a92020-04-23 21:18:35 +00001360 {
1361 return ipmi::responseInvalidFieldRequest();
1362 }
Snehalatha Va5ae7722020-05-02 18:18:57 +00001363 if ((paramSelector >= oemCmdStart) && (paramSelector <= oemCmdEnd))
1364 {
1365 return ipmi::responseParmNotSupported();
1366 }
jayaprakash Mutyalac2566a92020-04-23 21:18:35 +00001367 if (getRevision)
Xo Wangf542e8b2017-08-09 15:34:16 -07001368 {
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001369 return ipmi::responseSuccess(paramRevision, std::nullopt, std::nullopt);
Xo Wangf542e8b2017-08-09 15:34:16 -07001370 }
1371
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001372 if (paramSelector == 0)
Xo Wangf542e8b2017-08-09 15:34:16 -07001373 {
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001374 return ipmi::responseSuccess(paramRevision, transferStatus,
1375 std::nullopt);
Xo Wangf542e8b2017-08-09 15:34:16 -07001376 }
1377
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001378 if (BlockSelector != 0) // 00h if parameter does not require a block number
Xo Wangf542e8b2017-08-09 15:34:16 -07001379 {
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001380 return ipmi::responseParmNotSupported();
Xo Wangf542e8b2017-08-09 15:34:16 -07001381 }
1382
1383 if (sysInfoParamStore == nullptr)
1384 {
1385 sysInfoParamStore = std::make_unique<SysInfoParamStore>();
Xo Wang87651332017-08-11 10:17:59 -07001386 sysInfoParamStore->update(IPMI_SYSINFO_SYSTEM_NAME,
1387 sysInfoReadSystemName);
Xo Wangf542e8b2017-08-09 15:34:16 -07001388 }
1389
1390 // Parameters other than Set In Progress are assumed to be strings.
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001391 std::tuple<bool, std::string> ret =
1392 sysInfoParamStore->lookup(paramSelector);
1393 bool found = std::get<0>(ret);
Xo Wangf542e8b2017-08-09 15:34:16 -07001394 if (!found)
1395 {
Snehalatha Va5ae7722020-05-02 18:18:57 +00001396 return ipmi::responseSensorInvalid();
Xo Wangf542e8b2017-08-09 15:34:16 -07001397 }
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001398 std::string& paramString = std::get<1>(ret);
Jia, chunhui449f2162019-09-11 16:51:51 +08001399 std::vector<uint8_t> configData;
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001400 size_t count = 0;
1401 if (setSelector == 0)
Jia, chunhui449f2162019-09-11 16:51:51 +08001402 { // First chunk has only 14 bytes.
1403 configData.emplace_back(0); // encoding
1404 configData.emplace_back(paramString.length()); // string length
1405 count = std::min(paramString.length(), smallChunkSize);
1406 configData.resize(count + configDataOverhead);
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001407 std::copy_n(paramString.begin(), count,
Snehalatha Va5ae7722020-05-02 18:18:57 +00001408 configData.begin() + configDataOverhead); // 14 bytes chunk
1409
1410 // Append zero's to remaining bytes
1411 if (configData.size() < configParameterLength)
1412 {
1413 std::fill_n(std::back_inserter(configData),
1414 configParameterLength - configData.size(), 0x00);
1415 }
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001416 }
1417 else
Xo Wangf542e8b2017-08-09 15:34:16 -07001418 {
Jia, chunhui449f2162019-09-11 16:51:51 +08001419 size_t offset = (setSelector * fullChunkSize) - configDataOverhead;
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001420 if (offset >= paramString.length())
1421 {
1422 return ipmi::responseParmOutOfRange();
1423 }
Jia, chunhui449f2162019-09-11 16:51:51 +08001424 count = std::min(paramString.length() - offset, fullChunkSize);
1425 configData.resize(count);
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001426 std::copy_n(paramString.begin() + offset, count,
1427 configData.begin()); // 16 bytes chunk
Xo Wangf542e8b2017-08-09 15:34:16 -07001428 }
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001429 return ipmi::responseSuccess(paramRevision, setSelector, configData);
Xo Wangf542e8b2017-08-09 15:34:16 -07001430}
1431
Jia, chunhui449f2162019-09-11 16:51:51 +08001432ipmi::RspType<> ipmiAppSetSystemInfo(uint8_t paramSelector, uint8_t data1,
1433 std::vector<uint8_t> configData)
1434{
jayaprakash Mutyala3c5e4132020-04-27 23:00:05 +00001435 if (paramSelector >= invalidParamSelectorStart &&
1436 paramSelector <= invalidParamSelectorEnd)
1437 {
1438 return ipmi::responseInvalidFieldRequest();
1439 }
1440 if ((paramSelector >= oemCmdStart) && (paramSelector <= oemCmdEnd))
1441 {
1442 return ipmi::responseParmNotSupported();
1443 }
1444
Jia, chunhui449f2162019-09-11 16:51:51 +08001445 if (paramSelector == 0)
1446 {
1447 // attempt to set the 'set in progress' value (in parameter #0)
1448 // when not in the set complete state.
1449 if ((transferStatus != setComplete) && (data1 == setInProgress))
1450 {
1451 return ipmi::responseSetInProgressActive();
1452 }
1453 // only following 2 states are supported
1454 if (data1 > setInProgress)
1455 {
1456 phosphor::logging::log<phosphor::logging::level::ERR>(
1457 "illegal SetInProgress status");
1458 return ipmi::responseInvalidFieldRequest();
1459 }
1460
1461 transferStatus = data1 & progressMask;
1462 return ipmi::responseSuccess();
1463 }
1464
1465 if (configData.size() > configParameterLength)
1466 {
1467 return ipmi::responseInvalidFieldRequest();
1468 }
1469
jayaprakash Mutyala3c5e4132020-04-27 23:00:05 +00001470 // Append zero's to remaining bytes
1471 if (configData.size() < configParameterLength)
1472 {
1473 fill_n(back_inserter(configData),
1474 (configParameterLength - configData.size()), 0x00);
1475 }
1476
Jia, chunhui449f2162019-09-11 16:51:51 +08001477 if (!sysInfoParamStore)
1478 {
1479 sysInfoParamStore = std::make_unique<SysInfoParamStore>();
1480 sysInfoParamStore->update(IPMI_SYSINFO_SYSTEM_NAME,
1481 sysInfoReadSystemName);
1482 }
1483
1484 // lookup
1485 std::tuple<bool, std::string> ret =
1486 sysInfoParamStore->lookup(paramSelector);
1487 bool found = std::get<0>(ret);
1488 std::string& paramString = std::get<1>(ret);
1489 if (!found)
1490 {
1491 // parameter does not exist. Init new
1492 paramString = "";
1493 }
1494
1495 uint8_t setSelector = data1;
1496 size_t count = 0;
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -05001497 if (setSelector == 0) // First chunk has only 14 bytes.
Jia, chunhui449f2162019-09-11 16:51:51 +08001498 {
1499 size_t stringLen = configData.at(1); // string length
1500 // maxBytesPerParamter is 256. It will always be greater than stringLen
1501 // (unit8_t) if maxBytes changes in future, then following line is
1502 // needed.
1503 // stringLen = std::min(stringLen, maxBytesPerParameter);
1504 count = std::min(stringLen, smallChunkSize);
1505 count = std::min(count, configData.size());
1506 paramString.resize(stringLen); // reserve space
1507 std::copy_n(configData.begin() + configDataOverhead, count,
1508 paramString.begin());
1509 }
1510 else
1511 {
1512 size_t offset = (setSelector * fullChunkSize) - configDataOverhead;
1513 if (offset >= paramString.length())
1514 {
1515 return ipmi::responseParmOutOfRange();
1516 }
1517 count = std::min(paramString.length() - offset, configData.size());
1518 std::copy_n(configData.begin(), count, paramString.begin() + offset);
1519 }
1520 sysInfoParamStore->update(paramSelector, paramString);
1521 return ipmi::responseSuccess();
1522}
1523
Yong Libd0503a2019-08-22 17:17:17 +08001524#ifdef ENABLE_I2C_WHITELIST_CHECK
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301525inline std::vector<uint8_t> convertStringToData(const std::string& command)
1526{
1527 std::istringstream iss(command);
1528 std::string token;
1529 std::vector<uint8_t> dataValue;
1530 while (std::getline(iss, token, ' '))
1531 {
1532 dataValue.emplace_back(
1533 static_cast<uint8_t>(std::stoul(token, nullptr, base_16)));
1534 }
1535 return dataValue;
1536}
1537
1538static bool populateI2CMasterWRWhitelist()
1539{
1540 nlohmann::json data = nullptr;
1541 std::ifstream jsonFile(i2cMasterWRWhitelistFile);
1542
1543 if (!jsonFile.good())
1544 {
1545 log<level::WARNING>("i2c white list file not found!",
1546 entry("FILE_NAME: %s", i2cMasterWRWhitelistFile));
1547 return false;
1548 }
1549
1550 try
1551 {
1552 data = nlohmann::json::parse(jsonFile, nullptr, false);
1553 }
Patrick Williamsa2ad2da2021-10-06 12:21:46 -05001554 catch (const nlohmann::json::parse_error& e)
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301555 {
1556 log<level::ERR>("Corrupted i2c white list config file",
1557 entry("FILE_NAME: %s", i2cMasterWRWhitelistFile),
1558 entry("MSG: %s", e.what()));
1559 return false;
1560 }
1561
1562 try
1563 {
1564 // Example JSON Structure format
1565 // "filters": [
1566 // {
1567 // "Description": "Allow full read - ignore first byte write value
1568 // for 0x40 to 0x4F",
1569 // "busId": "0x01",
1570 // "slaveAddr": "0x40",
1571 // "slaveAddrMask": "0x0F",
1572 // "command": "0x00",
1573 // "commandMask": "0xFF"
1574 // },
1575 // {
1576 // "Description": "Allow full read - first byte match 0x05 and
1577 // ignore second byte",
1578 // "busId": "0x01",
1579 // "slaveAddr": "0x57",
1580 // "slaveAddrMask": "0x00",
1581 // "command": "0x05 0x00",
1582 // "commandMask": "0x00 0xFF"
1583 // },]
1584
1585 nlohmann::json filters = data[filtersStr].get<nlohmann::json>();
1586 std::vector<i2cMasterWRWhitelist>& whitelist = getWRWhitelist();
1587 for (const auto& it : filters.items())
1588 {
1589 nlohmann::json filter = it.value();
1590 if (filter.is_null())
1591 {
1592 log<level::ERR>(
1593 "Corrupted I2C master write read whitelist config file",
1594 entry("FILE_NAME: %s", i2cMasterWRWhitelistFile));
1595 return false;
1596 }
1597 const std::vector<uint8_t>& writeData =
1598 convertStringToData(filter[cmdStr].get<std::string>());
1599 const std::vector<uint8_t>& writeDataMask =
1600 convertStringToData(filter[cmdMaskStr].get<std::string>());
1601 if (writeDataMask.size() != writeData.size())
1602 {
1603 log<level::ERR>("I2C master write read whitelist filter "
1604 "mismatch for command & mask size");
1605 return false;
1606 }
1607 whitelist.push_back(
1608 {static_cast<uint8_t>(std::stoul(
1609 filter[busIdStr].get<std::string>(), nullptr, base_16)),
1610 static_cast<uint8_t>(
1611 std::stoul(filter[slaveAddrStr].get<std::string>(),
1612 nullptr, base_16)),
1613 static_cast<uint8_t>(
1614 std::stoul(filter[slaveAddrMaskStr].get<std::string>(),
1615 nullptr, base_16)),
1616 writeData, writeDataMask});
1617 }
1618 if (whitelist.size() != filters.size())
1619 {
1620 log<level::ERR>(
1621 "I2C master write read whitelist filter size mismatch");
1622 return false;
1623 }
1624 }
Patrick Williamsa2ad2da2021-10-06 12:21:46 -05001625 catch (const std::exception& e)
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301626 {
1627 log<level::ERR>("I2C master write read whitelist unexpected exception",
1628 entry("ERROR=%s", e.what()));
1629 return false;
1630 }
1631 return true;
1632}
1633
1634static inline bool isWriteDataWhitelisted(const std::vector<uint8_t>& data,
1635 const std::vector<uint8_t>& dataMask,
1636 const std::vector<uint8_t>& writeData)
1637{
1638 std::vector<uint8_t> processedDataBuf(data.size());
1639 std::vector<uint8_t> processedReqBuf(dataMask.size());
1640 std::transform(writeData.begin(), writeData.end(), dataMask.begin(),
1641 processedReqBuf.begin(), std::bit_or<uint8_t>());
1642 std::transform(data.begin(), data.end(), dataMask.begin(),
1643 processedDataBuf.begin(), std::bit_or<uint8_t>());
1644
1645 return (processedDataBuf == processedReqBuf);
1646}
1647
1648static bool isCmdWhitelisted(uint8_t busId, uint8_t slaveAddr,
1649 std::vector<uint8_t>& writeData)
1650{
1651 std::vector<i2cMasterWRWhitelist>& whiteList = getWRWhitelist();
1652 for (const auto& wlEntry : whiteList)
1653 {
1654 if ((busId == wlEntry.busId) &&
1655 ((slaveAddr | wlEntry.slaveAddrMask) ==
1656 (wlEntry.slaveAddr | wlEntry.slaveAddrMask)))
1657 {
1658 const std::vector<uint8_t>& dataMask = wlEntry.dataMask;
1659 // Skip as no-match, if requested write data is more than the
1660 // write data mask size
1661 if (writeData.size() > dataMask.size())
1662 {
1663 continue;
1664 }
1665 if (isWriteDataWhitelisted(wlEntry.data, dataMask, writeData))
1666 {
1667 return true;
1668 }
1669 }
1670 }
1671 return false;
1672}
Yong Libd0503a2019-08-22 17:17:17 +08001673#else
1674static bool populateI2CMasterWRWhitelist()
1675{
1676 log<level::INFO>(
1677 "I2C_WHITELIST_CHECK is disabled, do not populate whitelist");
1678 return true;
1679}
1680#endif // ENABLE_I2C_WHITELIST_CHECK
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301681
1682/** @brief implements master write read IPMI command which can be used for
1683 * low-level I2C/SMBus write, read or write-read access
1684 * @param isPrivateBus -to indicate private bus usage
1685 * @param busId - bus id
1686 * @param channelNum - channel number
1687 * @param reserved - skip 1 bit
1688 * @param slaveAddr - slave address
1689 * @param read count - number of bytes to be read
1690 * @param writeData - data to be written
1691 *
1692 * @returns IPMI completion code plus response data
1693 * - readData - i2c response data
1694 */
1695ipmi::RspType<std::vector<uint8_t>>
Willy Tu11d68892022-01-20 10:37:34 -08001696 ipmiMasterWriteRead([[maybe_unused]] bool isPrivateBus, uint3_t busId,
1697 [[maybe_unused]] uint4_t channelNum, bool reserved,
1698 uint7_t slaveAddr, uint8_t readCount,
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301699 std::vector<uint8_t> writeData)
1700{
Jayaprakash Mutyalac2af98b2021-09-14 09:19:11 +00001701 if (reserved)
1702 {
1703 return ipmi::responseInvalidFieldRequest();
1704 }
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301705 if (readCount > maxIPMIWriteReadSize)
1706 {
1707 log<level::ERR>("Master write read command: Read count exceeds limit");
1708 return ipmi::responseParmOutOfRange();
1709 }
1710 const size_t writeCount = writeData.size();
1711 if (!readCount && !writeCount)
1712 {
1713 log<level::ERR>("Master write read command: Read & write count are 0");
1714 return ipmi::responseInvalidFieldRequest();
1715 }
Yong Libd0503a2019-08-22 17:17:17 +08001716#ifdef ENABLE_I2C_WHITELIST_CHECK
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301717 if (!isCmdWhitelisted(static_cast<uint8_t>(busId),
1718 static_cast<uint8_t>(slaveAddr), writeData))
1719 {
1720 log<level::ERR>("Master write read request blocked!",
1721 entry("BUS=%d", static_cast<uint8_t>(busId)),
1722 entry("ADDR=0x%x", static_cast<uint8_t>(slaveAddr)));
1723 return ipmi::responseInvalidFieldRequest();
1724 }
Yong Libd0503a2019-08-22 17:17:17 +08001725#endif // ENABLE_I2C_WHITELIST_CHECK
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301726 std::vector<uint8_t> readBuf(readCount);
Patrick Williamsfbc6c9d2023-05-10 07:50:16 -05001727 std::string i2cBus = "/dev/i2c-" +
1728 std::to_string(static_cast<uint8_t>(busId));
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301729
Yong Li7dc4ac02019-08-23 17:44:32 +08001730 ipmi::Cc ret = ipmi::i2cWriteRead(i2cBus, static_cast<uint8_t>(slaveAddr),
1731 writeData, readBuf);
1732 if (ret != ipmi::ccSuccess)
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301733 {
Yong Li7dc4ac02019-08-23 17:44:32 +08001734 return ipmi::response(ret);
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301735 }
1736 return ipmi::responseSuccess(readBuf);
1737}
1738
Chris Austen6caf28b2015-10-13 12:40:40 -05001739void register_netfn_app_functions()
vishwabmcba0bd5f2015-09-30 16:50:23 +05301740{
Vernon Mauery86a50822019-03-25 13:11:36 -07001741 // <Get Device ID>
1742 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1743 ipmi::app::cmdGetDeviceId, ipmi::Privilege::User,
1744 ipmiAppGetDeviceId);
1745
Tom05732372016-09-06 17:21:23 +05301746 // <Get BT Interface Capabilities>
Vernon Mauerycc99ba42019-03-25 13:40:11 -07001747 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1748 ipmi::app::cmdGetBtIfaceCapabilities,
1749 ipmi::Privilege::User, ipmiAppGetBtCapabilities);
Chris Austen6caf28b2015-10-13 12:40:40 -05001750
Tom05732372016-09-06 17:21:23 +05301751 // <Reset Watchdog Timer>
Vernon Mauery11df4f62019-03-25 14:17:54 -07001752 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1753 ipmi::app::cmdResetWatchdogTimer,
1754 ipmi::Privilege::Operator, ipmiAppResetWatchdogTimer);
Chris Austen6caf28b2015-10-13 12:40:40 -05001755
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001756 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
AppaRao Puli5a98ea62019-11-10 21:15:02 +05301757 ipmi::app::cmdGetSessionInfo, ipmi::Privilege::User,
1758 ipmiAppGetSessionInfo);
Rajashekar Gade Reddyf71444d2019-07-25 15:12:17 +00001759
Tom05732372016-09-06 17:21:23 +05301760 // <Set Watchdog Timer>
Deepak Kumar Sahucfae9482019-05-20 14:58:58 +00001761 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1762 ipmi::app::cmdSetWatchdogTimer,
1763 ipmi::Privilege::Operator, ipmiSetWatchdogTimer);
Chris Austen6caf28b2015-10-13 12:40:40 -05001764
Rajashekar Gade Reddye7023922019-07-10 16:54:55 +00001765 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1766 ipmi::app::cmdCloseSession, ipmi::Privilege::Callback,
1767 ipmiAppCloseSession);
1768
William A. Kennington III73f44512018-02-09 15:28:46 -08001769 // <Get Watchdog Timer>
Deepak Kumar Sahucfae9482019-05-20 14:58:58 +00001770 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
AppaRao Puli5a98ea62019-11-10 21:15:02 +05301771 ipmi::app::cmdGetWatchdogTimer, ipmi::Privilege::User,
1772 ipmiGetWatchdogTimer);
William A. Kennington III73f44512018-02-09 15:28:46 -08001773
Tom05732372016-09-06 17:21:23 +05301774 // <Get Self Test Results>
Vernon Maueryb84a5282019-03-25 13:39:03 -07001775 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1776 ipmi::app::cmdGetSelfTestResults,
1777 ipmi::Privilege::User, ipmiAppGetSelfTestResults);
Nan Li41fa24a2016-11-10 20:12:37 +08001778
Tom05732372016-09-06 17:21:23 +05301779 // <Get Device GUID>
Vernon Mauery15541322019-03-25 13:33:03 -07001780 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1781 ipmi::app::cmdGetDeviceGuid, ipmi::Privilege::User,
1782 ipmiAppGetDeviceGuid);
Adriana Kobylakd100ee52015-10-20 17:02:37 -05001783
Tom05732372016-09-06 17:21:23 +05301784 // <Set ACPI Power State>
Deepak Kumar Sahu520c1312019-05-17 18:14:09 +00001785 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1786 ipmi::app::cmdSetAcpiPowerState,
1787 ipmi::Privilege::Admin, ipmiSetAcpiPowerState);
Yong Li18d77262018-10-09 01:59:45 +08001788 // <Get ACPI Power State>
Deepak Kumar Sahu4e6d2572019-05-07 19:53:37 +00001789 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1790 ipmi::app::cmdGetAcpiPowerState,
AppaRao Puli5a98ea62019-11-10 21:15:02 +05301791 ipmi::Privilege::User, ipmiGetAcpiPowerState);
Yong Li18d77262018-10-09 01:59:45 +08001792
Richard Marian Thomaiyar84bf9be2019-04-26 22:59:16 +05301793 // Note: For security reason, this command will be registered only when
1794 // there are proper I2C Master write read whitelist
1795 if (populateI2CMasterWRWhitelist())
1796 {
1797 // Note: For security reasons, registering master write read as admin
1798 // privilege command, even though IPMI 2.0 specification allows it as
1799 // operator privilege.
1800 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1801 ipmi::app::cmdMasterWriteRead,
1802 ipmi::Privilege::Admin, ipmiMasterWriteRead);
1803 }
1804
Marri Devender Rao5e007a52018-01-08 06:18:36 -06001805 // <Get System GUID Command>
Vernon Maueryb90a5322019-03-25 13:36:55 -07001806 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1807 ipmi::app::cmdGetSystemGuid, ipmi::Privilege::User,
1808 ipmiAppGetSystemGuid);
Tom Joseph7cbe2282018-03-21 21:17:33 +05301809
1810 // <Get Channel Cipher Suites Command>
Ayushi Smriti5c3b72c2019-08-30 13:47:31 +00001811 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1812 ipmi::app::cmdGetChannelCipherSuites,
Vernon Mauery79b4eea2019-11-07 09:51:39 -08001813 ipmi::Privilege::None, getChannelCipherSuites);
Vernon Maueryd2a57de2019-03-25 12:45:28 -07001814
Xo Wangf542e8b2017-08-09 15:34:16 -07001815 // <Get System Info Command>
Jia, chunhui98b43ba2019-09-05 15:54:23 +08001816 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1817 ipmi::app::cmdGetSystemInfoParameters,
1818 ipmi::Privilege::User, ipmiAppGetSystemInfo);
Jia, chunhui449f2162019-09-11 16:51:51 +08001819 // <Set System Info Command>
1820 ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnApp,
1821 ipmi::app::cmdSetSystemInfoParameters,
1822 ipmi::Privilege::Admin, ipmiAppSetSystemInfo);
vishwabmcba0bd5f2015-09-30 16:50:23 +05301823 return;
1824}