1 #include "config.h" 2 3 #include "dcmihandler.hpp" 4 5 #include "user_channel/channel_layer.hpp" 6 7 #include <bitset> 8 #include <cmath> 9 #include <fstream> 10 #include <ipmid/api.hpp> 11 #include <ipmid/utils.hpp> 12 #include <nlohmann/json.hpp> 13 #include <phosphor-logging/elog-errors.hpp> 14 #include <phosphor-logging/log.hpp> 15 #include <sdbusplus/bus.hpp> 16 #include <sdbusplus/message/types.hpp> 17 #include <xyz/openbmc_project/Common/error.hpp> 18 19 using namespace phosphor::logging; 20 using InternalFailure = 21 sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure; 22 23 void register_netfn_dcmi_functions() __attribute__((constructor)); 24 25 constexpr auto PCAP_PATH = "/xyz/openbmc_project/control/host0/power_cap"; 26 constexpr auto PCAP_INTERFACE = "xyz.openbmc_project.Control.Power.Cap"; 27 28 constexpr auto POWER_CAP_PROP = "PowerCap"; 29 constexpr auto POWER_CAP_ENABLE_PROP = "PowerCapEnable"; 30 31 constexpr auto DCMI_PARAMETER_REVISION = 2; 32 constexpr auto DCMI_SPEC_MAJOR_VERSION = 1; 33 constexpr auto DCMI_SPEC_MINOR_VERSION = 5; 34 constexpr auto DCMI_CONFIG_PARAMETER_REVISION = 1; 35 constexpr auto DCMI_RAND_BACK_OFF_MASK = 0x80; 36 constexpr auto DCMI_OPTION_60_43_MASK = 0x02; 37 constexpr auto DCMI_OPTION_12_MASK = 0x01; 38 constexpr auto DCMI_ACTIVATE_DHCP_MASK = 0x01; 39 constexpr auto DCMI_ACTIVATE_DHCP_REPLY = 0x00; 40 constexpr auto DCMI_SET_CONF_PARAM_REQ_PACKET_MAX_SIZE = 0x05; 41 constexpr auto DCMI_SET_CONF_PARAM_REQ_PACKET_MIN_SIZE = 0x04; 42 constexpr auto DHCP_TIMING1 = 0x04; // 4 sec 43 constexpr auto DHCP_TIMING2_UPPER = 0x00; // 2 min 44 constexpr auto DHCP_TIMING2_LOWER = 0x78; 45 constexpr auto DHCP_TIMING3_UPPER = 0x00; // 64 sec 46 constexpr auto DHCP_TIMING3_LOWER = 0x40; 47 // When DHCP Option 12 is enabled the string "SendHostName=true" will be 48 // added into n/w configuration file and the parameter 49 // SendHostNameEnabled will set to true. 50 constexpr auto DHCP_OPT12_ENABLED = "SendHostNameEnabled"; 51 52 constexpr auto SENSOR_VALUE_INTF = "xyz.openbmc_project.Sensor.Value"; 53 constexpr auto SENSOR_VALUE_PROP = "Value"; 54 constexpr auto SENSOR_SCALE_PROP = "Scale"; 55 56 using namespace phosphor::logging; 57 namespace variant_ns = sdbusplus::message::variant_ns; 58 59 namespace dcmi 60 { 61 62 // Refer Table 6-14, DCMI Entity ID Extension, DCMI v1.5 spec 63 static const std::map<uint8_t, std::string> entityIdToName{ 64 {0x40, "inlet"}, {0x37, "inlet"}, {0x41, "cpu"}, 65 {0x03, "cpu"}, {0x42, "baseboard"}, {0x07, "baseboard"}}; 66 67 bool isDCMIPowerMgmtSupported() 68 { 69 auto data = parseJSONConfig(gDCMICapabilitiesConfig); 70 71 return (gDCMIPowerMgmtSupported == data.value(gDCMIPowerMgmtCapability, 0)); 72 } 73 74 uint32_t getPcap(sdbusplus::bus::bus& bus) 75 { 76 auto settingService = ipmi::getService(bus, PCAP_INTERFACE, PCAP_PATH); 77 78 auto method = bus.new_method_call(settingService.c_str(), PCAP_PATH, 79 "org.freedesktop.DBus.Properties", "Get"); 80 81 method.append(PCAP_INTERFACE, POWER_CAP_PROP); 82 auto reply = bus.call(method); 83 84 if (reply.is_method_error()) 85 { 86 log<level::ERR>("Error in getPcap prop"); 87 elog<InternalFailure>(); 88 } 89 sdbusplus::message::variant<uint32_t> pcap; 90 reply.read(pcap); 91 92 return variant_ns::get<uint32_t>(pcap); 93 } 94 95 bool getPcapEnabled(sdbusplus::bus::bus& bus) 96 { 97 auto settingService = ipmi::getService(bus, PCAP_INTERFACE, PCAP_PATH); 98 99 auto method = bus.new_method_call(settingService.c_str(), PCAP_PATH, 100 "org.freedesktop.DBus.Properties", "Get"); 101 102 method.append(PCAP_INTERFACE, POWER_CAP_ENABLE_PROP); 103 auto reply = bus.call(method); 104 105 if (reply.is_method_error()) 106 { 107 log<level::ERR>("Error in getPcapEnabled prop"); 108 elog<InternalFailure>(); 109 } 110 sdbusplus::message::variant<bool> pcapEnabled; 111 reply.read(pcapEnabled); 112 113 return variant_ns::get<bool>(pcapEnabled); 114 } 115 116 void setPcap(sdbusplus::bus::bus& bus, const uint32_t powerCap) 117 { 118 auto service = ipmi::getService(bus, PCAP_INTERFACE, PCAP_PATH); 119 120 auto method = bus.new_method_call(service.c_str(), PCAP_PATH, 121 "org.freedesktop.DBus.Properties", "Set"); 122 123 method.append(PCAP_INTERFACE, POWER_CAP_PROP); 124 method.append(sdbusplus::message::variant<uint32_t>(powerCap)); 125 126 auto reply = bus.call(method); 127 128 if (reply.is_method_error()) 129 { 130 log<level::ERR>("Error in setPcap property"); 131 elog<InternalFailure>(); 132 } 133 } 134 135 void setPcapEnable(sdbusplus::bus::bus& bus, bool enabled) 136 { 137 auto service = ipmi::getService(bus, PCAP_INTERFACE, PCAP_PATH); 138 139 auto method = bus.new_method_call(service.c_str(), PCAP_PATH, 140 "org.freedesktop.DBus.Properties", "Set"); 141 142 method.append(PCAP_INTERFACE, POWER_CAP_ENABLE_PROP); 143 method.append(sdbusplus::message::variant<bool>(enabled)); 144 145 auto reply = bus.call(method); 146 147 if (reply.is_method_error()) 148 { 149 log<level::ERR>("Error in setPcapEnabled property"); 150 elog<InternalFailure>(); 151 } 152 } 153 154 void readAssetTagObjectTree(dcmi::assettag::ObjectTree& objectTree) 155 { 156 static constexpr auto mapperBusName = "xyz.openbmc_project.ObjectMapper"; 157 static constexpr auto mapperObjPath = "/xyz/openbmc_project/object_mapper"; 158 static constexpr auto mapperIface = "xyz.openbmc_project.ObjectMapper"; 159 static constexpr auto inventoryRoot = "/xyz/openbmc_project/inventory/"; 160 161 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 162 auto depth = 0; 163 164 auto mapperCall = bus.new_method_call(mapperBusName, mapperObjPath, 165 mapperIface, "GetSubTree"); 166 167 mapperCall.append(inventoryRoot); 168 mapperCall.append(depth); 169 mapperCall.append(std::vector<std::string>({dcmi::assetTagIntf})); 170 171 auto mapperReply = bus.call(mapperCall); 172 if (mapperReply.is_method_error()) 173 { 174 log<level::ERR>("Error in mapper call"); 175 elog<InternalFailure>(); 176 } 177 178 mapperReply.read(objectTree); 179 180 if (objectTree.empty()) 181 { 182 log<level::ERR>("AssetTag property is not populated"); 183 elog<InternalFailure>(); 184 } 185 } 186 187 std::string readAssetTag() 188 { 189 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 190 dcmi::assettag::ObjectTree objectTree; 191 192 // Read the object tree with the inventory root to figure out the object 193 // that has implemented the Asset tag interface. 194 readAssetTagObjectTree(objectTree); 195 196 auto method = bus.new_method_call( 197 (objectTree.begin()->second.begin()->first).c_str(), 198 (objectTree.begin()->first).c_str(), dcmi::propIntf, "Get"); 199 method.append(dcmi::assetTagIntf); 200 method.append(dcmi::assetTagProp); 201 202 auto reply = bus.call(method); 203 if (reply.is_method_error()) 204 { 205 log<level::ERR>("Error in reading asset tag"); 206 elog<InternalFailure>(); 207 } 208 209 sdbusplus::message::variant<std::string> assetTag; 210 reply.read(assetTag); 211 212 return variant_ns::get<std::string>(assetTag); 213 } 214 215 void writeAssetTag(const std::string& assetTag) 216 { 217 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 218 dcmi::assettag::ObjectTree objectTree; 219 220 // Read the object tree with the inventory root to figure out the object 221 // that has implemented the Asset tag interface. 222 readAssetTagObjectTree(objectTree); 223 224 auto method = bus.new_method_call( 225 (objectTree.begin()->second.begin()->first).c_str(), 226 (objectTree.begin()->first).c_str(), dcmi::propIntf, "Set"); 227 method.append(dcmi::assetTagIntf); 228 method.append(dcmi::assetTagProp); 229 method.append(sdbusplus::message::variant<std::string>(assetTag)); 230 231 auto reply = bus.call(method); 232 if (reply.is_method_error()) 233 { 234 log<level::ERR>("Error in writing asset tag"); 235 elog<InternalFailure>(); 236 } 237 } 238 239 std::string getHostName(void) 240 { 241 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 242 243 auto service = ipmi::getService(bus, networkConfigIntf, networkConfigObj); 244 auto value = ipmi::getDbusProperty(bus, service, networkConfigObj, 245 networkConfigIntf, hostNameProp); 246 247 return variant_ns::get<std::string>(value); 248 } 249 250 bool getDHCPEnabled() 251 { 252 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 253 254 auto ethdevice = ipmi::getChannelName(ethernetDefaultChannelNum); 255 auto ethernetObj = 256 ipmi::getDbusObject(bus, ethernetIntf, networkRoot, ethdevice); 257 auto service = ipmi::getService(bus, ethernetIntf, ethernetObj.first); 258 auto value = ipmi::getDbusProperty(bus, service, ethernetObj.first, 259 ethernetIntf, "DHCPEnabled"); 260 261 return variant_ns::get<bool>(value); 262 } 263 264 bool getDHCPOption(std::string prop) 265 { 266 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 267 268 auto service = ipmi::getService(bus, dhcpIntf, dhcpObj); 269 auto value = ipmi::getDbusProperty(bus, service, dhcpObj, dhcpIntf, prop); 270 271 return variant_ns::get<bool>(value); 272 } 273 274 void setDHCPOption(std::string prop, bool value) 275 { 276 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 277 278 auto service = ipmi::getService(bus, dhcpIntf, dhcpObj); 279 ipmi::setDbusProperty(bus, service, dhcpObj, dhcpIntf, prop, value); 280 } 281 282 Json parseJSONConfig(const std::string& configFile) 283 { 284 std::ifstream jsonFile(configFile); 285 if (!jsonFile.is_open()) 286 { 287 log<level::ERR>("Temperature readings JSON file not found"); 288 elog<InternalFailure>(); 289 } 290 291 auto data = Json::parse(jsonFile, nullptr, false); 292 if (data.is_discarded()) 293 { 294 log<level::ERR>("Temperature readings JSON parser failure"); 295 elog<InternalFailure>(); 296 } 297 298 return data; 299 } 300 301 } // namespace dcmi 302 303 ipmi_ret_t getPowerLimit(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 304 ipmi_request_t request, ipmi_response_t response, 305 ipmi_data_len_t data_len, ipmi_context_t context) 306 { 307 if (!dcmi::isDCMIPowerMgmtSupported()) 308 { 309 *data_len = 0; 310 log<level::ERR>("DCMI Power management is unsupported!"); 311 return IPMI_CC_INVALID; 312 } 313 314 auto requestData = 315 reinterpret_cast<const dcmi::GetPowerLimitRequest*>(request); 316 std::vector<uint8_t> outPayload(sizeof(dcmi::GetPowerLimitResponse)); 317 auto responseData = 318 reinterpret_cast<dcmi::GetPowerLimitResponse*>(outPayload.data()); 319 320 if (requestData->groupID != dcmi::groupExtId) 321 { 322 *data_len = 0; 323 return IPMI_CC_INVALID_FIELD_REQUEST; 324 } 325 326 sdbusplus::bus::bus sdbus{ipmid_get_sd_bus_connection()}; 327 uint32_t pcapValue = 0; 328 bool pcapEnable = false; 329 330 try 331 { 332 pcapValue = dcmi::getPcap(sdbus); 333 pcapEnable = dcmi::getPcapEnabled(sdbus); 334 } 335 catch (InternalFailure& e) 336 { 337 *data_len = 0; 338 return IPMI_CC_UNSPECIFIED_ERROR; 339 } 340 341 responseData->groupID = dcmi::groupExtId; 342 343 /* 344 * Exception action if power limit is exceeded and cannot be controlled 345 * with the correction time limit is hardcoded to Hard Power Off system 346 * and log event to SEL. 347 */ 348 constexpr auto exception = 0x01; 349 responseData->exceptionAction = exception; 350 351 responseData->powerLimit = static_cast<uint16_t>(pcapValue); 352 353 /* 354 * Correction time limit and Statistics sampling period is currently not 355 * populated. 356 */ 357 358 *data_len = outPayload.size(); 359 memcpy(response, outPayload.data(), *data_len); 360 361 if (pcapEnable) 362 { 363 return IPMI_CC_OK; 364 } 365 else 366 { 367 return IPMI_DCMI_CC_NO_ACTIVE_POWER_LIMIT; 368 } 369 } 370 371 ipmi_ret_t setPowerLimit(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 372 ipmi_request_t request, ipmi_response_t response, 373 ipmi_data_len_t data_len, ipmi_context_t context) 374 { 375 if (!dcmi::isDCMIPowerMgmtSupported()) 376 { 377 *data_len = 0; 378 log<level::ERR>("DCMI Power management is unsupported!"); 379 return IPMI_CC_INVALID; 380 } 381 382 auto requestData = 383 reinterpret_cast<const dcmi::SetPowerLimitRequest*>(request); 384 std::vector<uint8_t> outPayload(sizeof(dcmi::SetPowerLimitResponse)); 385 auto responseData = 386 reinterpret_cast<dcmi::SetPowerLimitResponse*>(outPayload.data()); 387 388 if (requestData->groupID != dcmi::groupExtId) 389 { 390 *data_len = 0; 391 return IPMI_CC_INVALID_FIELD_REQUEST; 392 } 393 394 sdbusplus::bus::bus sdbus{ipmid_get_sd_bus_connection()}; 395 396 // Only process the power limit requested in watts. 397 try 398 { 399 dcmi::setPcap(sdbus, requestData->powerLimit); 400 } 401 catch (InternalFailure& e) 402 { 403 *data_len = 0; 404 return IPMI_CC_UNSPECIFIED_ERROR; 405 } 406 407 log<level::INFO>("Set Power Cap", 408 entry("POWERCAP=%u", requestData->powerLimit)); 409 410 responseData->groupID = dcmi::groupExtId; 411 memcpy(response, outPayload.data(), outPayload.size()); 412 *data_len = outPayload.size(); 413 414 return IPMI_CC_OK; 415 } 416 417 ipmi_ret_t applyPowerLimit(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 418 ipmi_request_t request, ipmi_response_t response, 419 ipmi_data_len_t data_len, ipmi_context_t context) 420 { 421 if (!dcmi::isDCMIPowerMgmtSupported()) 422 { 423 *data_len = 0; 424 log<level::ERR>("DCMI Power management is unsupported!"); 425 return IPMI_CC_INVALID; 426 } 427 428 auto requestData = 429 reinterpret_cast<const dcmi::ApplyPowerLimitRequest*>(request); 430 std::vector<uint8_t> outPayload(sizeof(dcmi::ApplyPowerLimitResponse)); 431 auto responseData = 432 reinterpret_cast<dcmi::ApplyPowerLimitResponse*>(outPayload.data()); 433 434 if (requestData->groupID != dcmi::groupExtId) 435 { 436 *data_len = 0; 437 return IPMI_CC_INVALID_FIELD_REQUEST; 438 } 439 440 sdbusplus::bus::bus sdbus{ipmid_get_sd_bus_connection()}; 441 442 try 443 { 444 dcmi::setPcapEnable(sdbus, 445 static_cast<bool>(requestData->powerLimitAction)); 446 } 447 catch (InternalFailure& e) 448 { 449 *data_len = 0; 450 return IPMI_CC_UNSPECIFIED_ERROR; 451 } 452 453 log<level::INFO>("Set Power Cap Enable", 454 entry("POWERCAPENABLE=%u", requestData->powerLimitAction)); 455 456 responseData->groupID = dcmi::groupExtId; 457 memcpy(response, outPayload.data(), outPayload.size()); 458 *data_len = outPayload.size(); 459 460 return IPMI_CC_OK; 461 } 462 463 ipmi_ret_t getAssetTag(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 464 ipmi_request_t request, ipmi_response_t response, 465 ipmi_data_len_t data_len, ipmi_context_t context) 466 { 467 auto requestData = 468 reinterpret_cast<const dcmi::GetAssetTagRequest*>(request); 469 std::vector<uint8_t> outPayload(sizeof(dcmi::GetAssetTagResponse)); 470 auto responseData = 471 reinterpret_cast<dcmi::GetAssetTagResponse*>(outPayload.data()); 472 473 if (requestData->groupID != dcmi::groupExtId) 474 { 475 *data_len = 0; 476 return IPMI_CC_INVALID_FIELD_REQUEST; 477 } 478 479 // Verify offset to read and number of bytes to read are not exceeding the 480 // range. 481 if ((requestData->offset > dcmi::assetTagMaxOffset) || 482 (requestData->bytes > dcmi::maxBytes) || 483 ((requestData->offset + requestData->bytes) > dcmi::assetTagMaxSize)) 484 { 485 *data_len = 0; 486 return IPMI_CC_PARM_OUT_OF_RANGE; 487 } 488 489 std::string assetTag; 490 491 try 492 { 493 assetTag = dcmi::readAssetTag(); 494 } 495 catch (InternalFailure& e) 496 { 497 *data_len = 0; 498 return IPMI_CC_UNSPECIFIED_ERROR; 499 } 500 501 responseData->groupID = dcmi::groupExtId; 502 503 // Return if the asset tag is not populated. 504 if (!assetTag.size()) 505 { 506 responseData->tagLength = 0; 507 memcpy(response, outPayload.data(), outPayload.size()); 508 *data_len = outPayload.size(); 509 return IPMI_CC_OK; 510 } 511 512 // If the asset tag is longer than 63 bytes, restrict it to 63 bytes to suit 513 // Get Asset Tag command. 514 if (assetTag.size() > dcmi::assetTagMaxSize) 515 { 516 assetTag.resize(dcmi::assetTagMaxSize); 517 } 518 519 // If the requested offset is beyond the asset tag size. 520 if (requestData->offset >= assetTag.size()) 521 { 522 *data_len = 0; 523 return IPMI_CC_PARM_OUT_OF_RANGE; 524 } 525 526 auto returnData = assetTag.substr(requestData->offset, requestData->bytes); 527 528 responseData->tagLength = assetTag.size(); 529 530 memcpy(response, outPayload.data(), outPayload.size()); 531 memcpy(static_cast<uint8_t*>(response) + outPayload.size(), 532 returnData.data(), returnData.size()); 533 *data_len = outPayload.size() + returnData.size(); 534 535 return IPMI_CC_OK; 536 } 537 538 ipmi_ret_t setAssetTag(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 539 ipmi_request_t request, ipmi_response_t response, 540 ipmi_data_len_t data_len, ipmi_context_t context) 541 { 542 auto requestData = 543 reinterpret_cast<const dcmi::SetAssetTagRequest*>(request); 544 std::vector<uint8_t> outPayload(sizeof(dcmi::SetAssetTagResponse)); 545 auto responseData = 546 reinterpret_cast<dcmi::SetAssetTagResponse*>(outPayload.data()); 547 548 if (requestData->groupID != dcmi::groupExtId) 549 { 550 *data_len = 0; 551 return IPMI_CC_INVALID_FIELD_REQUEST; 552 } 553 554 // Verify offset to read and number of bytes to read are not exceeding the 555 // range. 556 if ((requestData->offset > dcmi::assetTagMaxOffset) || 557 (requestData->bytes > dcmi::maxBytes) || 558 ((requestData->offset + requestData->bytes) > dcmi::assetTagMaxSize)) 559 { 560 *data_len = 0; 561 return IPMI_CC_PARM_OUT_OF_RANGE; 562 } 563 564 std::string assetTag; 565 566 try 567 { 568 assetTag = dcmi::readAssetTag(); 569 570 if (requestData->offset > assetTag.size()) 571 { 572 *data_len = 0; 573 return IPMI_CC_PARM_OUT_OF_RANGE; 574 } 575 576 assetTag.replace(requestData->offset, 577 assetTag.size() - requestData->offset, 578 static_cast<const char*>(request) + 579 sizeof(dcmi::SetAssetTagRequest), 580 requestData->bytes); 581 582 dcmi::writeAssetTag(assetTag); 583 584 responseData->groupID = dcmi::groupExtId; 585 responseData->tagLength = assetTag.size(); 586 memcpy(response, outPayload.data(), outPayload.size()); 587 *data_len = outPayload.size(); 588 589 return IPMI_CC_OK; 590 } 591 catch (InternalFailure& e) 592 { 593 *data_len = 0; 594 return IPMI_CC_UNSPECIFIED_ERROR; 595 } 596 } 597 598 ipmi_ret_t getMgmntCtrlIdStr(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 599 ipmi_request_t request, ipmi_response_t response, 600 ipmi_data_len_t data_len, ipmi_context_t context) 601 { 602 auto requestData = 603 reinterpret_cast<const dcmi::GetMgmntCtrlIdStrRequest*>(request); 604 auto responseData = 605 reinterpret_cast<dcmi::GetMgmntCtrlIdStrResponse*>(response); 606 std::string hostName; 607 608 *data_len = 0; 609 610 if (requestData->groupID != dcmi::groupExtId || 611 requestData->bytes > dcmi::maxBytes || 612 requestData->offset + requestData->bytes > dcmi::maxCtrlIdStrLen) 613 { 614 return IPMI_CC_INVALID_FIELD_REQUEST; 615 } 616 617 try 618 { 619 hostName = dcmi::getHostName(); 620 } 621 catch (InternalFailure& e) 622 { 623 return IPMI_CC_UNSPECIFIED_ERROR; 624 } 625 626 if (requestData->offset > hostName.length()) 627 { 628 return IPMI_CC_PARM_OUT_OF_RANGE; 629 } 630 auto responseStr = hostName.substr(requestData->offset, requestData->bytes); 631 auto responseStrLen = std::min(static_cast<std::size_t>(requestData->bytes), 632 responseStr.length() + 1); 633 responseData->groupID = dcmi::groupExtId; 634 responseData->strLen = hostName.length(); 635 std::copy(begin(responseStr), end(responseStr), responseData->data); 636 637 *data_len = sizeof(*responseData) + responseStrLen; 638 return IPMI_CC_OK; 639 } 640 641 ipmi_ret_t setMgmntCtrlIdStr(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 642 ipmi_request_t request, ipmi_response_t response, 643 ipmi_data_len_t data_len, ipmi_context_t context) 644 { 645 static std::array<char, dcmi::maxCtrlIdStrLen + 1> newCtrlIdStr; 646 647 auto requestData = 648 reinterpret_cast<const dcmi::SetMgmntCtrlIdStrRequest*>(request); 649 auto responseData = 650 reinterpret_cast<dcmi::SetMgmntCtrlIdStrResponse*>(response); 651 652 *data_len = 0; 653 654 if (requestData->groupID != dcmi::groupExtId || 655 requestData->bytes > dcmi::maxBytes || 656 requestData->offset + requestData->bytes > dcmi::maxCtrlIdStrLen + 1 || 657 (requestData->offset + requestData->bytes == 658 dcmi::maxCtrlIdStrLen + 1 && 659 requestData->data[requestData->bytes - 1] != '\0')) 660 { 661 return IPMI_CC_INVALID_FIELD_REQUEST; 662 } 663 664 try 665 { 666 /* if there is no old value and offset is not 0 */ 667 if (newCtrlIdStr[0] == '\0' && requestData->offset != 0) 668 { 669 /* read old ctrlIdStr */ 670 auto hostName = dcmi::getHostName(); 671 hostName.resize(dcmi::maxCtrlIdStrLen); 672 std::copy(begin(hostName), end(hostName), begin(newCtrlIdStr)); 673 newCtrlIdStr[hostName.length()] = '\0'; 674 } 675 676 /* replace part of string and mark byte after the last as \0 */ 677 auto restStrIter = 678 std::copy_n(requestData->data, requestData->bytes, 679 begin(newCtrlIdStr) + requestData->offset); 680 /* if the last written byte is not 64th - add '\0' */ 681 if (requestData->offset + requestData->bytes <= dcmi::maxCtrlIdStrLen) 682 { 683 *restStrIter = '\0'; 684 } 685 686 /* if input data contains '\0' whole string is sent - update hostname */ 687 auto it = std::find(requestData->data, 688 requestData->data + requestData->bytes, '\0'); 689 if (it != requestData->data + requestData->bytes) 690 { 691 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 692 ipmi::setDbusProperty(bus, dcmi::networkServiceName, 693 dcmi::networkConfigObj, 694 dcmi::networkConfigIntf, dcmi::hostNameProp, 695 std::string(newCtrlIdStr.data())); 696 } 697 } 698 catch (InternalFailure& e) 699 { 700 *data_len = 0; 701 return IPMI_CC_UNSPECIFIED_ERROR; 702 } 703 704 responseData->groupID = dcmi::groupExtId; 705 responseData->offset = requestData->offset + requestData->bytes; 706 *data_len = sizeof(*responseData); 707 return IPMI_CC_OK; 708 } 709 710 // List of the capabilities under each parameter 711 dcmi::DCMICaps dcmiCaps = { 712 // Supported DCMI Capabilities 713 {dcmi::DCMICapParameters::SUPPORTED_DCMI_CAPS, 714 {3, 715 {{"PowerManagement", 2, 0, 1}, 716 {"OOBSecondaryLan", 3, 2, 1}, 717 {"SerialTMODE", 3, 1, 1}, 718 {"InBandSystemInterfaceChannel", 3, 0, 1}}}}, 719 // Mandatory Platform Attributes 720 {dcmi::DCMICapParameters::MANDATORY_PLAT_ATTRIBUTES, 721 {5, 722 {{"SELAutoRollOver", 1, 15, 1}, 723 {"FlushEntireSELUponRollOver", 1, 14, 1}, 724 {"RecordLevelSELFlushUponRollOver", 1, 13, 1}, 725 {"NumberOfSELEntries", 1, 0, 12}, 726 {"TempMonitoringSamplingFreq", 5, 0, 8}}}}, 727 // Optional Platform Attributes 728 {dcmi::DCMICapParameters::OPTIONAL_PLAT_ATTRIBUTES, 729 {2, 730 {{"PowerMgmtDeviceSlaveAddress", 1, 1, 7}, 731 {"BMCChannelNumber", 2, 4, 4}, 732 {"DeviceRivision", 2, 0, 4}}}}, 733 // Manageability Access Attributes 734 {dcmi::DCMICapParameters::MANAGEABILITY_ACCESS_ATTRIBUTES, 735 {3, 736 {{"MandatoryPrimaryLanOOBSupport", 1, 0, 8}, 737 {"OptionalSecondaryLanOOBSupport", 2, 0, 8}, 738 {"OptionalSerialOOBMTMODECapability", 3, 0, 8}}}}}; 739 740 ipmi_ret_t getDCMICapabilities(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 741 ipmi_request_t request, ipmi_response_t response, 742 ipmi_data_len_t data_len, ipmi_context_t context) 743 { 744 745 std::ifstream dcmiCapFile(dcmi::gDCMICapabilitiesConfig); 746 if (!dcmiCapFile.is_open()) 747 { 748 log<level::ERR>("DCMI Capabilities file not found"); 749 return IPMI_CC_UNSPECIFIED_ERROR; 750 } 751 752 auto data = nlohmann::json::parse(dcmiCapFile, nullptr, false); 753 if (data.is_discarded()) 754 { 755 log<level::ERR>("DCMI Capabilities JSON parser failure"); 756 return IPMI_CC_UNSPECIFIED_ERROR; 757 } 758 759 auto requestData = 760 reinterpret_cast<const dcmi::GetDCMICapRequest*>(request); 761 762 // get list of capabilities in a parameter 763 auto caps = 764 dcmiCaps.find(static_cast<dcmi::DCMICapParameters>(requestData->param)); 765 if (caps == dcmiCaps.end()) 766 { 767 log<level::ERR>("Invalid input parameter"); 768 return IPMI_CC_INVALID_FIELD_REQUEST; 769 } 770 771 if (requestData->groupID != dcmi::groupExtId) 772 { 773 *data_len = 0; 774 return IPMI_CC_INVALID_FIELD_REQUEST; 775 } 776 777 auto responseData = reinterpret_cast<dcmi::GetDCMICapResponse*>(response); 778 779 // For each capabilities in a parameter fill the data from 780 // the json file based on the capability name. 781 for (auto cap : caps->second.capList) 782 { 783 // If the data is beyond first byte boundary, insert in a 784 // 16bit pattern for example number of SEL entries are represented 785 // in 12bits. 786 if ((cap.length + cap.position) > dcmi::gByteBitSize) 787 { 788 uint16_t val = data.value(cap.name.c_str(), 0); 789 // According to DCMI spec v1.5, max number of SEL entries is 790 // 4096, but bit 12b of DCMI capabilities Mandatory Platform 791 // Attributes field is reserved and therefore we can use only 792 // the provided 12 bits with maximum value of 4095. 793 // We're playing safe here by applying the mask 794 // to ensure that provided value will fit into 12 bits. 795 if (cap.length > dcmi::gByteBitSize) 796 { 797 val &= dcmi::gMaxSELEntriesMask; 798 } 799 val <<= cap.position; 800 responseData->data[cap.bytePosition - 1] |= 801 static_cast<uint8_t>(val); 802 responseData->data[cap.bytePosition] |= val >> dcmi::gByteBitSize; 803 } 804 else 805 { 806 responseData->data[cap.bytePosition - 1] |= 807 data.value(cap.name.c_str(), 0) << cap.position; 808 } 809 } 810 811 responseData->groupID = dcmi::groupExtId; 812 responseData->major = DCMI_SPEC_MAJOR_VERSION; 813 responseData->minor = DCMI_SPEC_MINOR_VERSION; 814 responseData->paramRevision = DCMI_PARAMETER_REVISION; 815 *data_len = sizeof(*responseData) + caps->second.size; 816 817 return IPMI_CC_OK; 818 } 819 820 namespace dcmi 821 { 822 namespace temp_readings 823 { 824 825 Temperature readTemp(const std::string& dbusService, 826 const std::string& dbusPath) 827 { 828 // Read the temperature value from d-bus object. Need some conversion. 829 // As per the interface xyz.openbmc_project.Sensor.Value, the temperature 830 // is an double and in degrees C. It needs to be scaled by using the 831 // formula Value * 10^Scale. The ipmi spec has the temperature as a uint8_t, 832 // with a separate single bit for the sign. 833 834 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 835 auto result = ipmi::getAllDbusProperties( 836 bus, dbusService, dbusPath, "xyz.openbmc_project.Sensor.Value"); 837 auto temperature = sdbusplus::message::variant_ns::visit( 838 ipmi::VariantToDoubleVisitor(), result.at("Value")); 839 double absTemp = std::abs(temperature); 840 841 auto findFactor = result.find("Scale"); 842 double factor = 0.0; 843 if (findFactor != result.end()) 844 { 845 factor = sdbusplus::message::variant_ns::visit( 846 ipmi::VariantToDoubleVisitor(), findFactor->second); 847 } 848 double scale = std::pow(10, factor); 849 850 auto tempDegrees = absTemp * scale; 851 // Max absolute temp as per ipmi spec is 128. 852 if (tempDegrees > maxTemp) 853 { 854 tempDegrees = maxTemp; 855 } 856 857 return std::make_tuple(static_cast<uint8_t>(tempDegrees), 858 (temperature < 0)); 859 } 860 861 std::tuple<Response, NumInstances> read(const std::string& type, 862 uint8_t instance) 863 { 864 Response response{}; 865 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 866 867 if (!instance) 868 { 869 log<level::ERR>("Expected non-zero instance"); 870 elog<InternalFailure>(); 871 } 872 873 auto data = parseJSONConfig(gDCMISensorsConfig); 874 static const std::vector<Json> empty{}; 875 std::vector<Json> readings = data.value(type, empty); 876 size_t numInstances = readings.size(); 877 for (const auto& j : readings) 878 { 879 uint8_t instanceNum = j.value("instance", 0); 880 // Not the instance we're interested in 881 if (instanceNum != instance) 882 { 883 continue; 884 } 885 886 std::string path = j.value("dbus", ""); 887 std::string service; 888 try 889 { 890 service = 891 ipmi::getService(bus, "xyz.openbmc_project.Sensor.Value", path); 892 } 893 catch (std::exception& e) 894 { 895 log<level::DEBUG>(e.what()); 896 return std::make_tuple(response, numInstances); 897 } 898 899 response.instance = instance; 900 uint8_t temp{}; 901 bool sign{}; 902 std::tie(temp, sign) = readTemp(service, path); 903 response.temperature = temp; 904 response.sign = sign; 905 906 // Found the instance we're interested in 907 break; 908 } 909 910 if (numInstances > maxInstances) 911 { 912 numInstances = maxInstances; 913 } 914 return std::make_tuple(response, numInstances); 915 } 916 917 std::tuple<ResponseList, NumInstances> readAll(const std::string& type, 918 uint8_t instanceStart) 919 { 920 ResponseList response{}; 921 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 922 923 size_t numInstances = 0; 924 auto data = parseJSONConfig(gDCMISensorsConfig); 925 static const std::vector<Json> empty{}; 926 std::vector<Json> readings = data.value(type, empty); 927 numInstances = readings.size(); 928 for (const auto& j : readings) 929 { 930 try 931 { 932 // Max of 8 response data sets 933 if (response.size() == maxDataSets) 934 { 935 break; 936 } 937 938 uint8_t instanceNum = j.value("instance", 0); 939 // Not in the instance range we're interested in 940 if (instanceNum < instanceStart) 941 { 942 continue; 943 } 944 945 std::string path = j.value("dbus", ""); 946 auto service = 947 ipmi::getService(bus, "xyz.openbmc_project.Sensor.Value", path); 948 949 Response r{}; 950 r.instance = instanceNum; 951 uint8_t temp{}; 952 bool sign{}; 953 std::tie(temp, sign) = readTemp(service, path); 954 r.temperature = temp; 955 r.sign = sign; 956 response.push_back(r); 957 } 958 catch (std::exception& e) 959 { 960 log<level::DEBUG>(e.what()); 961 continue; 962 } 963 } 964 965 if (numInstances > maxInstances) 966 { 967 numInstances = maxInstances; 968 } 969 return std::make_tuple(response, numInstances); 970 } 971 972 } // namespace temp_readings 973 } // namespace dcmi 974 975 ipmi_ret_t getTempReadings(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 976 ipmi_request_t request, ipmi_response_t response, 977 ipmi_data_len_t data_len, ipmi_context_t context) 978 { 979 auto requestData = 980 reinterpret_cast<const dcmi::GetTempReadingsRequest*>(request); 981 auto responseData = 982 reinterpret_cast<dcmi::GetTempReadingsResponseHdr*>(response); 983 984 if (*data_len != sizeof(dcmi::GetTempReadingsRequest)) 985 { 986 log<level::ERR>("Malformed request data", 987 entry("DATA_SIZE=%d", *data_len)); 988 return IPMI_CC_REQ_DATA_LEN_INVALID; 989 } 990 *data_len = 0; 991 992 auto it = dcmi::entityIdToName.find(requestData->entityId); 993 if (it == dcmi::entityIdToName.end()) 994 { 995 log<level::ERR>("Unknown Entity ID", 996 entry("ENTITY_ID=%d", requestData->entityId)); 997 return IPMI_CC_INVALID_FIELD_REQUEST; 998 } 999 1000 if (requestData->groupID != dcmi::groupExtId) 1001 { 1002 log<level::ERR>("Invalid Group ID", 1003 entry("GROUP_ID=%d", requestData->groupID)); 1004 return IPMI_CC_INVALID_FIELD_REQUEST; 1005 } 1006 1007 if (requestData->sensorType != dcmi::temperatureSensorType) 1008 { 1009 log<level::ERR>("Invalid sensor type", 1010 entry("SENSOR_TYPE=%d", requestData->sensorType)); 1011 return IPMI_CC_INVALID_FIELD_REQUEST; 1012 } 1013 1014 dcmi::temp_readings::ResponseList temps{}; 1015 try 1016 { 1017 if (!requestData->entityInstance) 1018 { 1019 // Read all instances 1020 std::tie(temps, responseData->numInstances) = 1021 dcmi::temp_readings::readAll(it->second, 1022 requestData->instanceStart); 1023 } 1024 else 1025 { 1026 // Read one instance 1027 temps.resize(1); 1028 std::tie(temps[0], responseData->numInstances) = 1029 dcmi::temp_readings::read(it->second, 1030 requestData->entityInstance); 1031 } 1032 responseData->numDataSets = temps.size(); 1033 } 1034 catch (InternalFailure& e) 1035 { 1036 return IPMI_CC_UNSPECIFIED_ERROR; 1037 } 1038 1039 responseData->groupID = dcmi::groupExtId; 1040 size_t payloadSize = temps.size() * sizeof(dcmi::temp_readings::Response); 1041 if (!temps.empty()) 1042 { 1043 memcpy(responseData + 1, // copy payload right after the response header 1044 temps.data(), payloadSize); 1045 } 1046 *data_len = sizeof(dcmi::GetTempReadingsResponseHdr) + payloadSize; 1047 1048 return IPMI_CC_OK; 1049 } 1050 1051 int64_t getPowerReading(sdbusplus::bus::bus& bus) 1052 { 1053 std::ifstream sensorFile(POWER_READING_SENSOR); 1054 std::string objectPath; 1055 if (!sensorFile.is_open()) 1056 { 1057 log<level::ERR>("Power reading configuration file not found", 1058 entry("POWER_SENSOR_FILE=%s", POWER_READING_SENSOR)); 1059 elog<InternalFailure>(); 1060 } 1061 1062 auto data = nlohmann::json::parse(sensorFile, nullptr, false); 1063 if (data.is_discarded()) 1064 { 1065 log<level::ERR>("Error in parsing configuration file", 1066 entry("POWER_SENSOR_FILE=%s", POWER_READING_SENSOR)); 1067 elog<InternalFailure>(); 1068 } 1069 1070 objectPath = data.value("path", ""); 1071 if (objectPath.empty()) 1072 { 1073 log<level::ERR>("Power sensor D-Bus object path is empty", 1074 entry("POWER_SENSOR_FILE=%s", POWER_READING_SENSOR)); 1075 elog<InternalFailure>(); 1076 } 1077 1078 // Return default value if failed to read from D-Bus object 1079 int64_t power = 0; 1080 try 1081 { 1082 auto service = ipmi::getService(bus, SENSOR_VALUE_INTF, objectPath); 1083 1084 // Read the sensor value and scale properties 1085 auto properties = ipmi::getAllDbusProperties(bus, service, objectPath, 1086 SENSOR_VALUE_INTF); 1087 auto value = variant_ns::get<int64_t>(properties[SENSOR_VALUE_PROP]); 1088 auto scale = variant_ns::get<int64_t>(properties[SENSOR_SCALE_PROP]); 1089 1090 // Power reading needs to be scaled with the Scale value using the 1091 // formula Value * 10^Scale. 1092 power = value * std::pow(10, scale); 1093 } 1094 catch (std::exception& e) 1095 { 1096 log<level::INFO>("Failure to read power value from D-Bus object", 1097 entry("OBJECT_PATH=%s", objectPath.c_str()), 1098 entry("INTERFACE=%s", SENSOR_VALUE_INTF)); 1099 } 1100 return power; 1101 } 1102 1103 ipmi_ret_t setDCMIConfParams(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 1104 ipmi_request_t request, ipmi_response_t response, 1105 ipmi_data_len_t data_len, ipmi_context_t context) 1106 { 1107 auto requestData = 1108 reinterpret_cast<const dcmi::SetConfParamsRequest*>(request); 1109 auto responseData = 1110 reinterpret_cast<dcmi::SetConfParamsResponse*>(response); 1111 1112 if (requestData->groupID != dcmi::groupExtId || 1113 *data_len < DCMI_SET_CONF_PARAM_REQ_PACKET_MIN_SIZE || 1114 *data_len > DCMI_SET_CONF_PARAM_REQ_PACKET_MAX_SIZE) 1115 { 1116 log<level::ERR>("Invalid Group ID or Invalid Requested Packet size", 1117 entry("GROUP_ID=%d", requestData->groupID), 1118 entry("PACKET SIZE=%d", *data_len)); 1119 return IPMI_CC_INVALID_FIELD_REQUEST; 1120 } 1121 1122 *data_len = 0; 1123 1124 try 1125 { 1126 // Take action based on the Parameter Selector 1127 switch ( 1128 static_cast<dcmi::DCMIConfigParameters>(requestData->paramSelect)) 1129 { 1130 case dcmi::DCMIConfigParameters::ActivateDHCP: 1131 1132 if ((requestData->data[0] & DCMI_ACTIVATE_DHCP_MASK) && 1133 dcmi::getDHCPEnabled()) 1134 { 1135 // When these conditions are met we have to trigger DHCP 1136 // protocol restart using the latest parameter settings, but 1137 // as per n/w manager design, each time when we update n/w 1138 // parameters, n/w service is restarted. So we no need to 1139 // take any action in this case. 1140 } 1141 break; 1142 1143 case dcmi::DCMIConfigParameters::DiscoveryConfig: 1144 1145 if (requestData->data[0] & DCMI_OPTION_12_MASK) 1146 { 1147 dcmi::setDHCPOption(DHCP_OPT12_ENABLED, true); 1148 } 1149 else 1150 { 1151 dcmi::setDHCPOption(DHCP_OPT12_ENABLED, false); 1152 } 1153 1154 // Systemd-networkd doesn't support Random Back off 1155 if (requestData->data[0] & DCMI_RAND_BACK_OFF_MASK) 1156 { 1157 return IPMI_CC_INVALID; 1158 } 1159 break; 1160 // Systemd-networkd doesn't allow to configure DHCP timigs 1161 case dcmi::DCMIConfigParameters::DHCPTiming1: 1162 case dcmi::DCMIConfigParameters::DHCPTiming2: 1163 case dcmi::DCMIConfigParameters::DHCPTiming3: 1164 default: 1165 return IPMI_CC_INVALID; 1166 } 1167 } 1168 catch (std::exception& e) 1169 { 1170 log<level::ERR>(e.what()); 1171 return IPMI_CC_UNSPECIFIED_ERROR; 1172 } 1173 responseData->groupID = dcmi::groupExtId; 1174 *data_len = sizeof(dcmi::SetConfParamsResponse); 1175 1176 return IPMI_CC_OK; 1177 } 1178 1179 ipmi_ret_t getDCMIConfParams(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 1180 ipmi_request_t request, ipmi_response_t response, 1181 ipmi_data_len_t data_len, ipmi_context_t context) 1182 { 1183 1184 auto requestData = 1185 reinterpret_cast<const dcmi::GetConfParamsRequest*>(request); 1186 auto responseData = 1187 reinterpret_cast<dcmi::GetConfParamsResponse*>(response); 1188 1189 responseData->data[0] = 0x00; 1190 1191 if (requestData->groupID != dcmi::groupExtId || 1192 *data_len != sizeof(dcmi::GetConfParamsRequest)) 1193 { 1194 log<level::ERR>("Invalid Group ID or Invalid Requested Packet size", 1195 entry("GROUP_ID=%d", requestData->groupID), 1196 entry("PACKET SIZE=%d", *data_len)); 1197 return IPMI_CC_INVALID_FIELD_REQUEST; 1198 } 1199 1200 *data_len = 0; 1201 1202 try 1203 { 1204 // Take action based on the Parameter Selector 1205 switch ( 1206 static_cast<dcmi::DCMIConfigParameters>(requestData->paramSelect)) 1207 { 1208 case dcmi::DCMIConfigParameters::ActivateDHCP: 1209 responseData->data[0] = DCMI_ACTIVATE_DHCP_REPLY; 1210 *data_len = sizeof(dcmi::GetConfParamsResponse) + 1; 1211 break; 1212 case dcmi::DCMIConfigParameters::DiscoveryConfig: 1213 if (dcmi::getDHCPOption(DHCP_OPT12_ENABLED)) 1214 { 1215 responseData->data[0] |= DCMI_OPTION_12_MASK; 1216 } 1217 *data_len = sizeof(dcmi::GetConfParamsResponse) + 1; 1218 break; 1219 // Get below values from Systemd-networkd source code 1220 case dcmi::DCMIConfigParameters::DHCPTiming1: 1221 responseData->data[0] = DHCP_TIMING1; 1222 *data_len = sizeof(dcmi::GetConfParamsResponse) + 1; 1223 break; 1224 case dcmi::DCMIConfigParameters::DHCPTiming2: 1225 responseData->data[0] = DHCP_TIMING2_LOWER; 1226 responseData->data[1] = DHCP_TIMING2_UPPER; 1227 *data_len = sizeof(dcmi::GetConfParamsResponse) + 2; 1228 break; 1229 case dcmi::DCMIConfigParameters::DHCPTiming3: 1230 responseData->data[0] = DHCP_TIMING3_LOWER; 1231 responseData->data[1] = DHCP_TIMING3_UPPER; 1232 *data_len = sizeof(dcmi::GetConfParamsResponse) + 2; 1233 break; 1234 default: 1235 *data_len = 0; 1236 return IPMI_CC_INVALID; 1237 } 1238 } 1239 catch (std::exception& e) 1240 { 1241 log<level::ERR>(e.what()); 1242 return IPMI_CC_UNSPECIFIED_ERROR; 1243 } 1244 1245 responseData->groupID = dcmi::groupExtId; 1246 responseData->major = DCMI_SPEC_MAJOR_VERSION; 1247 responseData->minor = DCMI_SPEC_MINOR_VERSION; 1248 responseData->paramRevision = DCMI_CONFIG_PARAMETER_REVISION; 1249 1250 return IPMI_CC_OK; 1251 } 1252 1253 ipmi_ret_t getPowerReading(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 1254 ipmi_request_t request, ipmi_response_t response, 1255 ipmi_data_len_t data_len, ipmi_context_t context) 1256 { 1257 if (!dcmi::isDCMIPowerMgmtSupported()) 1258 { 1259 *data_len = 0; 1260 log<level::ERR>("DCMI Power management is unsupported!"); 1261 return IPMI_CC_INVALID; 1262 } 1263 1264 ipmi_ret_t rc = IPMI_CC_OK; 1265 auto requestData = 1266 reinterpret_cast<const dcmi::GetPowerReadingRequest*>(request); 1267 auto responseData = 1268 reinterpret_cast<dcmi::GetPowerReadingResponse*>(response); 1269 1270 if (requestData->groupID != dcmi::groupExtId) 1271 { 1272 *data_len = 0; 1273 return IPMI_CC_INVALID_FIELD_REQUEST; 1274 } 1275 1276 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 1277 int64_t power = 0; 1278 try 1279 { 1280 power = getPowerReading(bus); 1281 } 1282 catch (InternalFailure& e) 1283 { 1284 log<level::ERR>("Error in reading power sensor value", 1285 entry("INTERFACE=%s", SENSOR_VALUE_INTF), 1286 entry("PROPERTY=%s", SENSOR_VALUE_PROP)); 1287 return IPMI_CC_UNSPECIFIED_ERROR; 1288 } 1289 responseData->groupID = dcmi::groupExtId; 1290 1291 // TODO: openbmc/openbmc#2819 1292 // Minimum, Maximum, Average power, TimeFrame, TimeStamp, 1293 // PowerReadingState readings need to be populated 1294 // after Telemetry changes. 1295 uint16_t totalPower = static_cast<uint16_t>(power); 1296 responseData->currentPower = totalPower; 1297 responseData->minimumPower = totalPower; 1298 responseData->maximumPower = totalPower; 1299 responseData->averagePower = totalPower; 1300 1301 *data_len = sizeof(*responseData); 1302 return rc; 1303 } 1304 1305 namespace dcmi 1306 { 1307 namespace sensor_info 1308 { 1309 1310 Response createFromJson(const Json& config) 1311 { 1312 Response response{}; 1313 uint16_t recordId = config.value("record_id", 0); 1314 response.recordIdLsb = recordId & 0xFF; 1315 response.recordIdMsb = (recordId >> 8) & 0xFF; 1316 return response; 1317 } 1318 1319 std::tuple<Response, NumInstances> read(const std::string& type, 1320 uint8_t instance, const Json& config) 1321 { 1322 Response response{}; 1323 1324 if (!instance) 1325 { 1326 log<level::ERR>("Expected non-zero instance"); 1327 elog<InternalFailure>(); 1328 } 1329 1330 static const std::vector<Json> empty{}; 1331 std::vector<Json> readings = config.value(type, empty); 1332 size_t numInstances = readings.size(); 1333 for (const auto& reading : readings) 1334 { 1335 uint8_t instanceNum = reading.value("instance", 0); 1336 // Not the instance we're interested in 1337 if (instanceNum != instance) 1338 { 1339 continue; 1340 } 1341 1342 response = createFromJson(reading); 1343 1344 // Found the instance we're interested in 1345 break; 1346 } 1347 1348 if (numInstances > maxInstances) 1349 { 1350 log<level::DEBUG>("Trimming IPMI num instances", 1351 entry("NUM_INSTANCES=%d", numInstances)); 1352 numInstances = maxInstances; 1353 } 1354 return std::make_tuple(response, numInstances); 1355 } 1356 1357 std::tuple<ResponseList, NumInstances> 1358 readAll(const std::string& type, uint8_t instanceStart, const Json& config) 1359 { 1360 ResponseList responses{}; 1361 1362 size_t numInstances = 0; 1363 static const std::vector<Json> empty{}; 1364 std::vector<Json> readings = config.value(type, empty); 1365 numInstances = readings.size(); 1366 for (const auto& reading : readings) 1367 { 1368 try 1369 { 1370 // Max of 8 records 1371 if (responses.size() == maxRecords) 1372 { 1373 break; 1374 } 1375 1376 uint8_t instanceNum = reading.value("instance", 0); 1377 // Not in the instance range we're interested in 1378 if (instanceNum < instanceStart) 1379 { 1380 continue; 1381 } 1382 1383 Response response = createFromJson(reading); 1384 responses.push_back(response); 1385 } 1386 catch (std::exception& e) 1387 { 1388 log<level::DEBUG>(e.what()); 1389 continue; 1390 } 1391 } 1392 1393 if (numInstances > maxInstances) 1394 { 1395 log<level::DEBUG>("Trimming IPMI num instances", 1396 entry("NUM_INSTANCES=%d", numInstances)); 1397 numInstances = maxInstances; 1398 } 1399 return std::make_tuple(responses, numInstances); 1400 } 1401 1402 } // namespace sensor_info 1403 } // namespace dcmi 1404 1405 ipmi_ret_t getSensorInfo(ipmi_netfn_t netfn, ipmi_cmd_t cmd, 1406 ipmi_request_t request, ipmi_response_t response, 1407 ipmi_data_len_t data_len, ipmi_context_t context) 1408 { 1409 auto requestData = 1410 reinterpret_cast<const dcmi::GetSensorInfoRequest*>(request); 1411 auto responseData = 1412 reinterpret_cast<dcmi::GetSensorInfoResponseHdr*>(response); 1413 1414 if (*data_len != sizeof(dcmi::GetSensorInfoRequest)) 1415 { 1416 log<level::ERR>("Malformed request data", 1417 entry("DATA_SIZE=%d", *data_len)); 1418 return IPMI_CC_REQ_DATA_LEN_INVALID; 1419 } 1420 *data_len = 0; 1421 1422 auto it = dcmi::entityIdToName.find(requestData->entityId); 1423 if (it == dcmi::entityIdToName.end()) 1424 { 1425 log<level::ERR>("Unknown Entity ID", 1426 entry("ENTITY_ID=%d", requestData->entityId)); 1427 return IPMI_CC_INVALID_FIELD_REQUEST; 1428 } 1429 1430 if (requestData->groupID != dcmi::groupExtId) 1431 { 1432 log<level::ERR>("Invalid Group ID", 1433 entry("GROUP_ID=%d", requestData->groupID)); 1434 return IPMI_CC_INVALID_FIELD_REQUEST; 1435 } 1436 1437 if (requestData->sensorType != dcmi::temperatureSensorType) 1438 { 1439 log<level::ERR>("Invalid sensor type", 1440 entry("SENSOR_TYPE=%d", requestData->sensorType)); 1441 return IPMI_CC_INVALID_FIELD_REQUEST; 1442 } 1443 1444 dcmi::sensor_info::ResponseList sensors{}; 1445 static dcmi::Json config{}; 1446 static bool parsed = false; 1447 1448 try 1449 { 1450 if (!parsed) 1451 { 1452 config = dcmi::parseJSONConfig(dcmi::gDCMISensorsConfig); 1453 parsed = true; 1454 } 1455 1456 if (!requestData->entityInstance) 1457 { 1458 // Read all instances 1459 std::tie(sensors, responseData->numInstances) = 1460 dcmi::sensor_info::readAll(it->second, 1461 requestData->instanceStart, config); 1462 } 1463 else 1464 { 1465 // Read one instance 1466 sensors.resize(1); 1467 std::tie(sensors[0], responseData->numInstances) = 1468 dcmi::sensor_info::read(it->second, requestData->entityInstance, 1469 config); 1470 } 1471 responseData->numRecords = sensors.size(); 1472 } 1473 catch (InternalFailure& e) 1474 { 1475 return IPMI_CC_UNSPECIFIED_ERROR; 1476 } 1477 1478 responseData->groupID = dcmi::groupExtId; 1479 size_t payloadSize = sensors.size() * sizeof(dcmi::sensor_info::Response); 1480 if (!sensors.empty()) 1481 { 1482 memcpy(responseData + 1, // copy payload right after the response header 1483 sensors.data(), payloadSize); 1484 } 1485 *data_len = sizeof(dcmi::GetSensorInfoResponseHdr) + payloadSize; 1486 1487 return IPMI_CC_OK; 1488 } 1489 1490 void register_netfn_dcmi_functions() 1491 { 1492 // <Get Power Limit> 1493 1494 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::GET_POWER_LIMIT, NULL, 1495 getPowerLimit, PRIVILEGE_USER); 1496 1497 // <Set Power Limit> 1498 1499 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::SET_POWER_LIMIT, NULL, 1500 setPowerLimit, PRIVILEGE_OPERATOR); 1501 1502 // <Activate/Deactivate Power Limit> 1503 1504 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::APPLY_POWER_LIMIT, 1505 NULL, applyPowerLimit, PRIVILEGE_OPERATOR); 1506 1507 // <Get Asset Tag> 1508 1509 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::GET_ASSET_TAG, NULL, 1510 getAssetTag, PRIVILEGE_USER); 1511 1512 // <Set Asset Tag> 1513 1514 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::SET_ASSET_TAG, NULL, 1515 setAssetTag, PRIVILEGE_OPERATOR); 1516 1517 // <Get Management Controller Identifier String> 1518 1519 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::GET_MGMNT_CTRL_ID_STR, 1520 NULL, getMgmntCtrlIdStr, PRIVILEGE_USER); 1521 1522 // <Set Management Controller Identifier String> 1523 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::SET_MGMNT_CTRL_ID_STR, 1524 NULL, setMgmntCtrlIdStr, PRIVILEGE_ADMIN); 1525 1526 // <Get DCMI capabilities> 1527 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::GET_CAPABILITIES, 1528 NULL, getDCMICapabilities, PRIVILEGE_USER); 1529 1530 // <Get Temperature Readings> 1531 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::GET_TEMP_READINGS, 1532 NULL, getTempReadings, PRIVILEGE_USER); 1533 1534 // <Get Power Reading> 1535 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::GET_POWER_READING, 1536 NULL, getPowerReading, PRIVILEGE_USER); 1537 1538 // <Get Sensor Info> 1539 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::GET_SENSOR_INFO, NULL, 1540 getSensorInfo, PRIVILEGE_USER); 1541 1542 // <Get DCMI Configuration Parameters> 1543 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::GET_CONF_PARAMS, NULL, 1544 getDCMIConfParams, PRIVILEGE_USER); 1545 1546 // <Set DCMI Configuration Parameters> 1547 ipmi_register_callback(NETFUN_GRPEXT, dcmi::Commands::SET_CONF_PARAMS, NULL, 1548 setDCMIConfParams, PRIVILEGE_ADMIN); 1549 1550 return; 1551 } 1552 // 956379 1553