| #include "powermode.hpp" |
| |
| #include <fcntl.h> |
| #include <sys/ioctl.h> |
| |
| #include <format> |
| |
| #ifdef POWERVM_CHECK |
| #include <com/ibm/Host/Target/server.hpp> |
| #endif |
| #include <org/open_power/OCC/Device/error.hpp> |
| #include <phosphor-logging/elog-errors.hpp> |
| #include <phosphor-logging/log.hpp> |
| #include <xyz/openbmc_project/Common/error.hpp> |
| #include <xyz/openbmc_project/Control/Power/Mode/server.hpp> |
| |
| #include <cassert> |
| #include <fstream> |
| #include <regex> |
| |
| namespace open_power |
| { |
| namespace occ |
| { |
| namespace powermode |
| { |
| |
| using namespace phosphor::logging; |
| using namespace std::literals::string_literals; |
| using namespace sdbusplus::org::open_power::OCC::Device::Error; |
| |
| using Mode = sdbusplus::xyz::openbmc_project::Control::Power::server::Mode; |
| |
| using NotAllowed = sdbusplus::xyz::openbmc_project::Common::Error::NotAllowed; |
| |
| // List of all Power Modes that are currently supported (and in Redfish) |
| #define VALID_POWER_MODE_SETTING(mode) \ |
| ((mode == SysPwrMode::STATIC) || (mode == SysPwrMode::POWER_SAVING) || \ |
| (mode == SysPwrMode::BALANCED_PERF) || (mode == SysPwrMode::MAX_PERF) || \ |
| (mode == SysPwrMode::EFF_FAVOR_POWER) || \ |
| (mode == SysPwrMode::EFF_FAVOR_PERF)) |
| // List of OEM Power Modes that are currently supported |
| #define VALID_OEM_POWER_MODE_SETTING(mode) \ |
| ((mode == SysPwrMode::SFP) || (mode == SysPwrMode::FFO) || \ |
| (mode == SysPwrMode::MAX_FREQ) || \ |
| (mode == SysPwrMode::NON_DETERMINISTIC)) |
| // List of all Power Modes that disable IPS |
| #define IS_ECO_MODE(mode) \ |
| ((mode == SysPwrMode::EFF_FAVOR_POWER) || \ |
| (mode == SysPwrMode::EFF_FAVOR_PERF)) |
| |
| // Constructor |
| PowerMode::PowerMode(const Manager& managerRef, const char* modePath, |
| const char* ipsPath, EventPtr& event) : |
| ModeInterface(utils::getBus(), modePath, |
| ModeInterface::action::emit_no_signals), |
| manager(managerRef), |
| ipsMatch(utils::getBus(), |
| sdbusplus::bus::match::rules::propertiesChanged(PIPS_PATH, |
| PIPS_INTERFACE), |
| [this](auto& msg) { this->ipsChanged(msg); }), |
| defaultsUpdateMatch( |
| utils::getBus(), |
| sdbusplus::bus::match::rules::propertiesChangedNamespace( |
| "/xyz/openbmc_project/inventory", PMODE_DEFAULT_INTERFACE), |
| [this](auto& msg) { this->defaultsReady(msg); }), |
| masterOccSet(false), masterActive(false), ipsObjectPath(ipsPath), |
| event(event) |
| { |
| // Get supported power modes from entity manager |
| if (false == getSupportedModes()) |
| { |
| // Did not find them so use default customer modes |
| using Mode = |
| sdbusplus::xyz::openbmc_project::Control::Power::server::Mode; |
| // Update power modes that will be allowed by the Redfish interface |
| ModeInterface::allowedPowerModes( |
| {Mode::PowerMode::Static, Mode::PowerMode::MaximumPerformance, |
| Mode::PowerMode::PowerSaving}); |
| } |
| |
| SysPwrMode currentMode; |
| uint16_t oemModeData = 0; |
| // Read the persisted power mode |
| if (getMode(currentMode, oemModeData)) |
| { |
| // Validate persisted mode is supported |
| if (isValidMode(currentMode)) |
| { |
| // Update power mode on DBus and create IPS object if allowed |
| updateDbusMode(currentMode); |
| } |
| else |
| { |
| log<level::ERR>( |
| std::format( |
| "PowerMode: Persisted power mode ({}/{}) is not valid. Reading system default mode", |
| currentMode, oemModeData) |
| .c_str()); |
| persistedData.invalidateMode(); |
| // Read default power mode |
| initPersistentData(); |
| } |
| } |
| }; |
| |
| void PowerMode::createIpsObject() |
| { |
| if (!ipsObject) |
| { |
| log<level::INFO>("createIpsObject: Creating IPS object"); |
| ipsObject = |
| std::make_unique<IpsInterface>(utils::getBus(), ipsObjectPath); |
| |
| uint8_t enterUtil, exitUtil; |
| uint16_t enterTime, exitTime; |
| bool ipsEnabled; |
| // Read the persisted Idle Power Saver parametres |
| if (getIPSParms(ipsEnabled, enterUtil, enterTime, exitUtil, exitTime)) |
| { |
| // Update Idle Power Saver parameters on DBus |
| updateDbusIPS(ipsEnabled, enterUtil, enterTime, exitUtil, exitTime); |
| } |
| |
| // Starts watching for IPS state changes. |
| addIpsWatch(true); |
| |
| needToSendIpsData = true; |
| } |
| } |
| |
| void PowerMode::removeIpsObject() |
| { |
| if (ipsObject) |
| { |
| // Stop watching for IPS state changes. |
| removeIpsWatch(); |
| |
| log<level::INFO>("removeIpsObject: Deleting IPS object"); |
| ipsObject.reset(nullptr); |
| } |
| needToSendIpsData = false; |
| } |
| |
| // Set the Master OCC |
| void PowerMode::setMasterOcc(const std::string& masterOccPath) |
| { |
| if (masterOccSet) |
| { |
| if (masterOccPath != path) |
| { |
| log<level::ERR>( |
| std::format( |
| "PowerMode::setMasterOcc: Master changed (was OCC{}, {})", |
| occInstance, masterOccPath) |
| .c_str()); |
| if (occCmd) |
| { |
| occCmd.reset(); |
| } |
| } |
| } |
| path = masterOccPath; |
| occInstance = path.back() - '0'; |
| log<level::DEBUG>(std::format("PowerMode::setMasterOcc(OCC{}, {})", |
| occInstance, path.c_str()) |
| .c_str()); |
| if (!occCmd) |
| { |
| occCmd = std::make_unique<open_power::occ::OccCommand>( |
| occInstance, path.c_str()); |
| } |
| masterOccSet = true; |
| }; |
| |
| // Set the state of power mode lock. Writing persistent data via dbus method. |
| bool PowerMode::powerModeLock() |
| { |
| log<level::INFO>("PowerMode::powerModeLock: locking mode change"); |
| persistedData.updateModeLock(true); // write persistent data |
| return true; |
| } |
| |
| // Get the state of power mode. Reading persistent data via dbus method. |
| bool PowerMode::powerModeLockStatus() |
| { |
| bool status = persistedData.getModeLock(); // read persistent data |
| log<level::INFO>(std::format("PowerMode::powerModeLockStatus: {}", |
| status ? "locked" : "unlocked") |
| .c_str()); |
| return status; |
| } |
| |
| // Called from OCC PassThrough interface (via CE login / BMC command line) |
| bool PowerMode::setMode(const SysPwrMode newMode, const uint16_t oemModeData) |
| { |
| if (persistedData.getModeLock()) |
| { |
| log<level::INFO>("PowerMode::setMode: mode change blocked"); |
| return false; |
| } |
| |
| if (updateDbusMode(newMode) == false) |
| { |
| // Unsupported mode |
| return false; |
| } |
| |
| // Save mode |
| persistedData.updateMode(newMode, oemModeData); |
| |
| // Send mode change to OCC |
| if (sendModeChange() != CmdStatus::SUCCESS) |
| { |
| // Mode change failed |
| return false; |
| } |
| |
| return true; |
| } |
| |
| // Convert PowerMode value to occ-control internal SysPwrMode |
| // Returns SysPwrMode::NO_CHANGE if mode not valid |
| SysPwrMode getInternalMode(const Mode::PowerMode& mode) |
| { |
| if (mode == Mode::PowerMode::MaximumPerformance) |
| { |
| return SysPwrMode::MAX_PERF; |
| } |
| else if (mode == Mode::PowerMode::PowerSaving) |
| { |
| return SysPwrMode::POWER_SAVING; |
| } |
| else if (mode == Mode::PowerMode::Static) |
| { |
| return SysPwrMode::STATIC; |
| } |
| else if (mode == Mode::PowerMode::EfficiencyFavorPower) |
| { |
| return SysPwrMode::EFF_FAVOR_POWER; |
| } |
| else if (mode == Mode::PowerMode::EfficiencyFavorPerformance) |
| { |
| return SysPwrMode::EFF_FAVOR_PERF; |
| } |
| else if (mode == Mode::PowerMode::BalancedPerformance) |
| { |
| return SysPwrMode::BALANCED_PERF; |
| } |
| |
| log<level::WARNING>("getInternalMode: Invalid PowerMode specified"); |
| return SysPwrMode::NO_CHANGE; |
| } |
| |
| // Convert PowerMode string to OCC SysPwrMode |
| // Returns NO_CHANGE if OEM or unsupported mode |
| SysPwrMode convertStringToMode(const std::string& i_modeString) |
| { |
| SysPwrMode newMode = SysPwrMode::NO_CHANGE; |
| try |
| { |
| Mode::PowerMode newPMode = |
| Mode::convertPowerModeFromString(i_modeString); |
| newMode = getInternalMode(newPMode); |
| } |
| catch (const std::exception& e) |
| { |
| // Strip off prefix to to search OEM modes not part of Redfish |
| auto prefix = PMODE_INTERFACE + ".PowerMode."s; |
| std::string shortMode = i_modeString; |
| std::string::size_type index = i_modeString.find(prefix); |
| if (index != std::string::npos) |
| { |
| shortMode.erase(0, prefix.length()); |
| } |
| if (shortMode == "FFO") |
| { |
| newMode = SysPwrMode::FFO; |
| } |
| else if (shortMode == "SFP") |
| { |
| newMode = SysPwrMode::SFP; |
| } |
| else if (shortMode == "MaxFrequency") |
| { |
| newMode = SysPwrMode::MAX_FREQ; |
| } |
| else if (shortMode == "NonDeterministic") |
| { |
| newMode = SysPwrMode::NON_DETERMINISTIC; |
| } |
| else |
| { |
| log<level::ERR>( |
| std::format("convertStringToMode: Invalid Power Mode: {} ({})", |
| shortMode, e.what()) |
| .c_str()); |
| } |
| } |
| return newMode; |
| } |
| |
| // Check if Hypervisor target is PowerVM |
| bool isPowerVM() |
| { |
| bool powerVmTarget = true; |
| #ifdef POWERVM_CHECK |
| namespace Hyper = sdbusplus::com::ibm::Host::server; |
| constexpr auto HYPE_PATH = "/com/ibm/host0/hypervisor"; |
| constexpr auto HYPE_INTERFACE = "com.ibm.Host.Target"; |
| constexpr auto HYPE_PROP = "Target"; |
| |
| // This will throw exception on failure |
| auto& bus = utils::getBus(); |
| auto service = utils::getService(HYPE_PATH, HYPE_INTERFACE); |
| auto method = bus.new_method_call(service.c_str(), HYPE_PATH, |
| "org.freedesktop.DBus.Properties", "Get"); |
| method.append(HYPE_INTERFACE, HYPE_PROP); |
| auto reply = bus.call(method); |
| |
| std::variant<std::string> hyperEntryValue; |
| reply.read(hyperEntryValue); |
| auto propVal = std::get<std::string>(hyperEntryValue); |
| if (Hyper::Target::convertHypervisorFromString(propVal) == |
| Hyper::Target::Hypervisor::PowerVM) |
| { |
| powerVmTarget = true; |
| } |
| |
| log<level::DEBUG>( |
| std::format("isPowerVM returning {}", powerVmTarget).c_str()); |
| #endif |
| |
| return powerVmTarget; |
| } |
| |
| // Initialize persistent data and return true if successful |
| bool PowerMode::initPersistentData() |
| { |
| if (!persistedData.modeAvailable()) |
| { |
| // Read the default mode |
| SysPwrMode currentMode; |
| if (!getDefaultMode(currentMode)) |
| { |
| // Unable to read defaults |
| return false; |
| } |
| log<level::INFO>( |
| std::format("PowerMode::initPersistentData: Using default mode: {}", |
| currentMode) |
| .c_str()); |
| |
| // Save default mode as current mode |
| persistedData.updateMode(currentMode, 0); |
| |
| // Write default mode to DBus and create IPS object if allowed |
| updateDbusMode(currentMode); |
| } |
| |
| if (!persistedData.ipsAvailable()) |
| { |
| // Read the default IPS parameters, write persistent file and update |
| // DBus |
| return useDefaultIPSParms(); |
| } |
| return true; |
| } |
| |
| // Get the requested power mode and return true if successful |
| bool PowerMode::getMode(SysPwrMode& currentMode, uint16_t& oemModeData) |
| { |
| currentMode = SysPwrMode::NO_CHANGE; |
| oemModeData = 0; |
| |
| if (!persistedData.getMode(currentMode, oemModeData)) |
| { |
| // Persistent data not initialized, read defaults and update DBus |
| if (!initPersistentData()) |
| { |
| // Unable to read defaults from entity manager yet |
| return false; |
| } |
| return persistedData.getMode(currentMode, oemModeData); |
| } |
| |
| return true; |
| } |
| |
| // Set the power mode on DBus and create IPS object if allowed/needed |
| bool PowerMode::updateDbusMode(const SysPwrMode newMode) |
| { |
| if (!isValidMode(newMode)) |
| { |
| log<level::ERR>( |
| std::format( |
| "PowerMode::updateDbusMode - Requested power mode not supported: {}", |
| newMode) |
| .c_str()); |
| return false; |
| } |
| |
| ModeInterface::PowerMode dBusMode = Mode::PowerMode::OEM; |
| if (customerModeList.contains(newMode)) |
| { |
| // Convert mode for DBus |
| switch (newMode) |
| { |
| case SysPwrMode::STATIC: |
| dBusMode = Mode::PowerMode::Static; |
| break; |
| case SysPwrMode::POWER_SAVING: |
| dBusMode = Mode::PowerMode::PowerSaving; |
| break; |
| case SysPwrMode::MAX_PERF: |
| dBusMode = Mode::PowerMode::MaximumPerformance; |
| break; |
| case SysPwrMode::EFF_FAVOR_POWER: |
| dBusMode = Mode::PowerMode::EfficiencyFavorPower; |
| break; |
| case SysPwrMode::EFF_FAVOR_PERF: |
| dBusMode = Mode::PowerMode::EfficiencyFavorPerformance; |
| break; |
| case SysPwrMode::BALANCED_PERF: |
| dBusMode = Mode::PowerMode::BalancedPerformance; |
| break; |
| default: |
| break; |
| } |
| } |
| // else return OEM mode |
| |
| // Determine if IPS is allowed and create/remove as needed |
| if (IS_ECO_MODE(newMode)) |
| { |
| removeIpsObject(); |
| } |
| else |
| { |
| createIpsObject(); |
| } |
| |
| ModeInterface::powerMode(dBusMode); |
| |
| return true; |
| } |
| |
| // Send mode change request to the master OCC |
| CmdStatus PowerMode::sendModeChange() |
| { |
| CmdStatus status; |
| |
| SysPwrMode newMode; |
| uint16_t oemModeData = 0; |
| getMode(newMode, oemModeData); |
| |
| if (isValidMode(newMode)) |
| { |
| if (IS_ECO_MODE(newMode)) |
| { |
| // Customer no longer able to enable IPS |
| removeIpsObject(); |
| } |
| else |
| { |
| // Customer now able to enable IPS |
| if (!ipsObject) |
| { |
| createIpsObject(); |
| } |
| } |
| |
| if (!masterActive || !masterOccSet) |
| { |
| // Nothing to do until OCC goes active |
| log<level::DEBUG>( |
| "PowerMode::sendModeChange: OCC master not active"); |
| return CmdStatus::SUCCESS; |
| } |
| |
| if (!isPowerVM()) |
| { |
| // Mode change is only supported on PowerVM systems |
| log<level::DEBUG>( |
| "PowerMode::sendModeChange: MODE CHANGE does not get sent on non-PowerVM systems"); |
| return CmdStatus::SUCCESS; |
| } |
| |
| std::vector<std::uint8_t> cmd, rsp; |
| cmd.reserve(9); |
| cmd.push_back(uint8_t(CmdType::SET_MODE_AND_STATE)); |
| cmd.push_back(0x00); // Data Length (2 bytes) |
| cmd.push_back(0x06); |
| cmd.push_back(0x30); // Data (Version) |
| cmd.push_back(uint8_t(OccState::NO_CHANGE)); |
| cmd.push_back(uint8_t(newMode)); |
| cmd.push_back(oemModeData >> 8); // Mode Data (Freq Point) |
| cmd.push_back(oemModeData & 0xFF); // |
| cmd.push_back(0x00); // reserved |
| log<level::INFO>( |
| std::format( |
| "PowerMode::sendModeChange: SET_MODE({},{}) command to OCC{} ({} bytes)", |
| newMode, oemModeData, occInstance, cmd.size()) |
| .c_str()); |
| status = occCmd->send(cmd, rsp); |
| if (status == CmdStatus::SUCCESS) |
| { |
| if (rsp.size() == 5) |
| { |
| if (RspStatus::SUCCESS == RspStatus(rsp[2])) |
| { |
| if (needToSendIpsData) |
| { |
| // Successful mode change and IPS is now allowed, so |
| // send IPS config |
| sendIpsData(); |
| } |
| } |
| else |
| { |
| log<level::ERR>( |
| std::format( |
| "PowerMode::sendModeChange: SET MODE failed with status 0x{:02X}", |
| rsp[2]) |
| .c_str()); |
| dump_hex(rsp); |
| status = CmdStatus::FAILURE; |
| } |
| } |
| else |
| { |
| log<level::ERR>( |
| "PowerMode::sendModeChange: INVALID SET MODE response"); |
| dump_hex(rsp); |
| status = CmdStatus::FAILURE; |
| } |
| } |
| else |
| { |
| log<level::ERR>( |
| std::format( |
| "PowerMode::sendModeChange: SET_MODE FAILED with status={}", |
| status) |
| .c_str()); |
| } |
| } |
| else |
| { |
| log<level::ERR>( |
| std::format( |
| "PowerMode::sendModeChange: Unable to set power mode to {}", |
| newMode) |
| .c_str()); |
| status = CmdStatus::FAILURE; |
| } |
| |
| return status; |
| } |
| |
| // Handle IPS changed event (from GUI/Redfish) |
| void PowerMode::ipsChanged(sdbusplus::message_t& msg) |
| { |
| bool parmsChanged = false; |
| std::string interface; |
| std::map<std::string, std::variant<bool, uint8_t, uint64_t>> |
| ipsProperties{}; |
| msg.read(interface, ipsProperties); |
| |
| // Read persisted values |
| bool ipsEnabled; |
| uint8_t enterUtil, exitUtil; |
| uint16_t enterTime, exitTime; |
| getIPSParms(ipsEnabled, enterUtil, enterTime, exitUtil, exitTime); |
| |
| if (!ipsObject) |
| { |
| log<level::WARNING>( |
| "ipsChanged: Idle Power Saver can not be modified in an ECO power mode"); |
| return; |
| } |
| |
| // Check for any changed data |
| auto ipsEntry = ipsProperties.find(IPS_ENABLED_PROP); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| ipsEnabled = std::get<bool>(ipsEntry->second); |
| log<level::INFO>( |
| std::format("Idle Power Saver change: Enabled={}", ipsEnabled) |
| .c_str()); |
| parmsChanged = true; |
| } |
| ipsEntry = ipsProperties.find(IPS_ENTER_UTIL); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| enterUtil = std::get<uint8_t>(ipsEntry->second); |
| log<level::INFO>( |
| std::format("Idle Power Saver change: Enter Util={}%", enterUtil) |
| .c_str()); |
| parmsChanged = true; |
| } |
| ipsEntry = ipsProperties.find(IPS_ENTER_TIME); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| std::chrono::milliseconds ms(std::get<uint64_t>(ipsEntry->second)); |
| enterTime = |
| std::chrono::duration_cast<std::chrono::seconds>(ms).count(); |
| log<level::INFO>( |
| std::format("Idle Power Saver change: Enter Time={}sec", enterTime) |
| .c_str()); |
| parmsChanged = true; |
| } |
| ipsEntry = ipsProperties.find(IPS_EXIT_UTIL); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| exitUtil = std::get<uint8_t>(ipsEntry->second); |
| log<level::INFO>( |
| std::format("Idle Power Saver change: Exit Util={}%", exitUtil) |
| .c_str()); |
| parmsChanged = true; |
| } |
| ipsEntry = ipsProperties.find(IPS_EXIT_TIME); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| std::chrono::milliseconds ms(std::get<uint64_t>(ipsEntry->second)); |
| exitTime = std::chrono::duration_cast<std::chrono::seconds>(ms).count(); |
| log<level::INFO>( |
| std::format("Idle Power Saver change: Exit Time={}sec", exitTime) |
| .c_str()); |
| parmsChanged = true; |
| } |
| |
| if (parmsChanged) |
| { |
| if (exitUtil == 0) |
| { |
| // Setting the exitUtil to 0 will force restoring the default IPS |
| // parmeters (0 is not valid exit utilization) |
| log<level::INFO>( |
| "Idle Power Saver Exit Utilization is 0%. Restoring default parameters"); |
| // Read the default IPS parameters, write persistent file and update |
| // DBus |
| useDefaultIPSParms(); |
| } |
| else |
| { |
| // Update persistant data with new DBus values |
| persistedData.updateIPS(ipsEnabled, enterUtil, enterTime, exitUtil, |
| exitTime); |
| } |
| |
| // Trigger IPS data to get sent to the OCC |
| sendIpsData(); |
| } |
| |
| return; |
| } |
| |
| /** @brief Get the Idle Power Saver properties from persisted data |
| * @return true if IPS parameters were read |
| */ |
| bool PowerMode::getIPSParms(bool& ipsEnabled, uint8_t& enterUtil, |
| uint16_t& enterTime, uint8_t& exitUtil, |
| uint16_t& exitTime) |
| { |
| // Defaults: |
| ipsEnabled = true; // Enabled |
| enterUtil = 8; // Enter Utilization (8%) |
| enterTime = 240; // Enter Delay Time (240s) |
| exitUtil = 12; // Exit Utilization (12%) |
| exitTime = 10; // Exit Delay Time (10s) |
| |
| if (!persistedData.getIPS(ipsEnabled, enterUtil, enterTime, exitUtil, |
| exitTime)) |
| { |
| // Persistent data not initialized, read defaults and update DBus |
| if (!initPersistentData()) |
| { |
| // Unable to read defaults from entity manager yet |
| return false; |
| } |
| |
| persistedData.getIPS(ipsEnabled, enterUtil, enterTime, exitUtil, |
| exitTime); |
| } |
| |
| if (enterUtil > exitUtil) |
| { |
| log<level::ERR>( |
| std::format( |
| "ERROR: Idle Power Saver Enter Utilization ({}%) is > Exit Utilization ({}%) - using Exit for both", |
| enterUtil, exitUtil) |
| .c_str()); |
| enterUtil = exitUtil; |
| } |
| |
| return true; |
| } |
| |
| // Set the Idle Power Saver data on DBus |
| bool PowerMode::updateDbusIPS(const bool enabled, const uint8_t enterUtil, |
| const uint16_t enterTime, const uint8_t exitUtil, |
| const uint16_t exitTime) |
| { |
| if (ipsObject) |
| { |
| // true = skip update signal |
| ipsObject->setPropertyByName(IPS_ENABLED_PROP, enabled, true); |
| ipsObject->setPropertyByName(IPS_ENTER_UTIL, enterUtil, true); |
| // Convert time from seconds to ms |
| uint64_t msTime = enterTime * 1000; |
| ipsObject->setPropertyByName(IPS_ENTER_TIME, msTime, true); |
| ipsObject->setPropertyByName(IPS_EXIT_UTIL, exitUtil, true); |
| msTime = exitTime * 1000; |
| ipsObject->setPropertyByName(IPS_EXIT_TIME, msTime, true); |
| } |
| else |
| { |
| log<level::WARNING>("updateDbusIPS: No IPS object was found"); |
| } |
| |
| return true; |
| } |
| |
| // Send Idle Power Saver config data to the master OCC |
| CmdStatus PowerMode::sendIpsData() |
| { |
| if (!masterActive || !masterOccSet) |
| { |
| // Nothing to do |
| return CmdStatus::SUCCESS; |
| } |
| |
| if (!isPowerVM()) |
| { |
| // Idle Power Saver data is only supported on PowerVM systems |
| log<level::DEBUG>( |
| "PowerMode::sendIpsData: SET_CFG_DATA[IPS] does not get sent on non-PowerVM systems"); |
| return CmdStatus::SUCCESS; |
| } |
| |
| if (!ipsObject) |
| { |
| // Idle Power Saver data is not available in Eco Modes |
| log<level::INFO>( |
| "PowerMode::sendIpsData: Skipping IPS data due to being in an ECO Power Mode"); |
| return CmdStatus::SUCCESS; |
| } |
| |
| bool ipsEnabled; |
| uint8_t enterUtil, exitUtil; |
| uint16_t enterTime, exitTime; |
| getIPSParms(ipsEnabled, enterUtil, enterTime, exitUtil, exitTime); |
| |
| log<level::INFO>( |
| std::format( |
| "Idle Power Saver Parameters: enabled:{}, enter:{}%/{}s, exit:{}%/{}s", |
| ipsEnabled, enterUtil, enterTime, exitUtil, exitTime) |
| .c_str()); |
| |
| std::vector<std::uint8_t> cmd, rsp; |
| cmd.reserve(12); |
| cmd.push_back(uint8_t(CmdType::SET_CONFIG_DATA)); |
| cmd.push_back(0x00); // Data Length (2 bytes) |
| cmd.push_back(0x09); // |
| cmd.push_back(0x11); // Config Format: IPS Settings |
| cmd.push_back(0x00); // Version |
| cmd.push_back(ipsEnabled ? 1 : 0); // IPS Enable |
| cmd.push_back(enterTime >> 8); // Enter Delay Time |
| cmd.push_back(enterTime & 0xFF); // |
| cmd.push_back(enterUtil); // Enter Utilization |
| cmd.push_back(exitTime >> 8); // Exit Delay Time |
| cmd.push_back(exitTime & 0xFF); // |
| cmd.push_back(exitUtil); // Exit Utilization |
| log<level::INFO>(std::format("PowerMode::sendIpsData: SET_CFG_DATA[IPS] " |
| "command to OCC{} ({} bytes)", |
| occInstance, cmd.size()) |
| .c_str()); |
| CmdStatus status = occCmd->send(cmd, rsp); |
| if (status == CmdStatus::SUCCESS) |
| { |
| if (rsp.size() == 5) |
| { |
| if (RspStatus::SUCCESS == RspStatus(rsp[2])) |
| { |
| needToSendIpsData = false; |
| } |
| else |
| { |
| log<level::ERR>( |
| std::format( |
| "PowerMode::sendIpsData: SET_CFG_DATA[IPS] failed with status 0x{:02X}", |
| rsp[2]) |
| .c_str()); |
| dump_hex(rsp); |
| status = CmdStatus::FAILURE; |
| } |
| } |
| else |
| { |
| log<level::ERR>( |
| "PowerMode::sendIpsData: INVALID SET_CFG_DATA[IPS] response"); |
| dump_hex(rsp); |
| status = CmdStatus::FAILURE; |
| } |
| } |
| else |
| { |
| log<level::ERR>( |
| std::format( |
| "PowerMode::sendIpsData: SET_CFG_DATA[IPS] with status={}", |
| status) |
| .c_str()); |
| } |
| |
| return status; |
| } |
| |
| // Print the current values |
| void OccPersistData::print() |
| { |
| if (modeData.modeInitialized) |
| { |
| log<level::INFO>( |
| std::format( |
| "OccPersistData: Mode: 0x{:02X}, OEM Mode Data: {} (0x{:04X} Locked{})", |
| modeData.mode, modeData.oemModeData, modeData.oemModeData, |
| modeData.modeLocked) |
| .c_str()); |
| } |
| if (modeData.ipsInitialized) |
| { |
| log<level::INFO>( |
| std::format( |
| "OccPersistData: IPS enabled:{}, enter:{}%/{}s, exit:{}%/{}s", |
| modeData.ipsEnabled, modeData.ipsEnterUtil, |
| modeData.ipsEnterTime, modeData.ipsExitUtil, |
| modeData.ipsExitTime) |
| .c_str()); |
| } |
| } |
| |
| // Saves the OEM mode data in the filesystem using cereal. |
| void OccPersistData::save() |
| { |
| std::filesystem::path opath = |
| std::filesystem::path{OCC_CONTROL_PERSIST_PATH} / powerModeFilename; |
| |
| if (!std::filesystem::exists(opath.parent_path())) |
| { |
| std::filesystem::create_directory(opath.parent_path()); |
| } |
| |
| log<level::DEBUG>( |
| std::format( |
| "OccPersistData::save: Writing Power Mode persisted data to {}", |
| opath.c_str()) |
| .c_str()); |
| // print(); |
| |
| std::ofstream stream{opath.c_str()}; |
| cereal::JSONOutputArchive oarchive{stream}; |
| |
| oarchive(modeData); |
| } |
| |
| // Loads the OEM mode data in the filesystem using cereal. |
| void OccPersistData::load() |
| { |
| std::filesystem::path ipath = |
| std::filesystem::path{OCC_CONTROL_PERSIST_PATH} / powerModeFilename; |
| |
| if (!std::filesystem::exists(ipath)) |
| { |
| modeData.modeInitialized = false; |
| modeData.ipsInitialized = false; |
| return; |
| } |
| |
| log<level::DEBUG>( |
| std::format( |
| "OccPersistData::load: Reading Power Mode persisted data from {}", |
| ipath.c_str()) |
| .c_str()); |
| try |
| { |
| std::ifstream stream{ipath.c_str()}; |
| cereal::JSONInputArchive iarchive(stream); |
| iarchive(modeData); |
| } |
| catch (const std::exception& e) |
| { |
| auto error = errno; |
| log<level::ERR>( |
| std::format("OccPersistData::load: failed to read {}, errno={}", |
| ipath.c_str(), error) |
| .c_str()); |
| modeData.modeInitialized = false; |
| modeData.ipsInitialized = false; |
| } |
| |
| // print(); |
| } |
| |
| // Called when PowerModeProperties defaults are available on DBus |
| void PowerMode::defaultsReady(sdbusplus::message_t& msg) |
| { |
| std::map<std::string, std::variant<std::string>> properties{}; |
| std::string interface; |
| msg.read(interface, properties); |
| |
| if (persistedData.modeAvailable()) |
| { |
| // Validate persisted mode is supported |
| SysPwrMode pMode = SysPwrMode::NO_CHANGE; |
| uint16_t oemModeData = 0; |
| persistedData.getMode(pMode, oemModeData); |
| if (!isValidMode(pMode)) |
| { |
| log<level::ERR>( |
| std::format( |
| "defaultsReady: Persisted power mode ({}/{}) is not valid. Reading system default mode", |
| pMode, oemModeData) |
| .c_str()); |
| persistedData.invalidateMode(); |
| } |
| } |
| |
| // If persistent data exists, then don't need to read defaults |
| if ((!persistedData.modeAvailable()) || (!persistedData.ipsAvailable())) |
| { |
| log<level::INFO>( |
| std::format( |
| "Default PowerModeProperties are now available (persistent modeAvail={}, ipsAvail={})", |
| persistedData.modeAvailable() ? 'y' : 'n', |
| persistedData.ipsAvailable() ? 'y' : 'n') |
| .c_str()); |
| |
| // Read default power mode defaults and update DBus |
| initPersistentData(); |
| } |
| } |
| |
| // Get the default power mode from DBus and return true if success |
| bool PowerMode::getDefaultMode(SysPwrMode& defaultMode) |
| { |
| try |
| { |
| auto& bus = utils::getBus(); |
| std::string path = "/"; |
| std::string service = |
| utils::getServiceUsingSubTree(PMODE_DEFAULT_INTERFACE, path); |
| auto method = |
| bus.new_method_call(service.c_str(), path.c_str(), |
| "org.freedesktop.DBus.Properties", "Get"); |
| method.append(PMODE_DEFAULT_INTERFACE, "PowerMode"); |
| auto reply = bus.call(method); |
| |
| std::variant<std::string> stateEntryValue; |
| reply.read(stateEntryValue); |
| auto propVal = std::get<std::string>(stateEntryValue); |
| |
| const std::string fullModeString = |
| PMODE_INTERFACE + ".PowerMode."s + propVal; |
| defaultMode = powermode::convertStringToMode(fullModeString); |
| if (!VALID_POWER_MODE_SETTING(defaultMode)) |
| { |
| log<level::ERR>(std::format("PowerMode::getDefaultMode: Invalid " |
| "default power mode found: {}", |
| defaultMode) |
| .c_str()); |
| // If default was read but not valid, use Max Performance |
| defaultMode = SysPwrMode::MAX_PERF; |
| return true; |
| } |
| } |
| catch (const sdbusplus::exception_t& e) |
| { |
| log<level::ERR>( |
| std::format("Unable to read Default Power Mode: {}", e.what()) |
| .c_str()); |
| return false; |
| } |
| |
| return true; |
| } |
| |
| /* Get the default Idle Power Saver properties and return true if successful */ |
| bool PowerMode::getDefaultIPSParms(bool& ipsEnabled, uint8_t& enterUtil, |
| uint16_t& enterTime, uint8_t& exitUtil, |
| uint16_t& exitTime) |
| { |
| // Defaults: |
| ipsEnabled = true; // Enabled |
| enterUtil = 8; // Enter Utilization (8%) |
| enterTime = 240; // Enter Delay Time (240s) |
| exitUtil = 12; // Exit Utilization (12%) |
| exitTime = 10; // Exit Delay Time (10s) |
| |
| std::map<std::string, std::variant<bool, uint8_t, uint16_t, uint64_t>> |
| ipsProperties{}; |
| |
| // Get all IPS properties from DBus |
| try |
| { |
| auto& bus = utils::getBus(); |
| std::string path = "/"; |
| std::string service = |
| utils::getServiceUsingSubTree(PMODE_DEFAULT_INTERFACE, path); |
| auto method = |
| bus.new_method_call(service.c_str(), path.c_str(), |
| "org.freedesktop.DBus.Properties", "GetAll"); |
| method.append(PMODE_DEFAULT_INTERFACE); |
| auto reply = bus.call(method); |
| reply.read(ipsProperties); |
| } |
| catch (const sdbusplus::exception_t& e) |
| { |
| log<level::ERR>( |
| std::format( |
| "Unable to read Default Idle Power Saver parameters so it will be disabled: {}", |
| e.what()) |
| .c_str()); |
| return false; |
| } |
| |
| auto ipsEntry = ipsProperties.find("IdlePowerSaverEnabled"); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| ipsEnabled = std::get<bool>(ipsEntry->second); |
| } |
| else |
| { |
| log<level::ERR>( |
| "PowerMode::getDefaultIPSParms could not find property: IdlePowerSaverEnabled"); |
| } |
| |
| ipsEntry = ipsProperties.find("EnterUtilizationPercent"); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| enterUtil = std::get<uint64_t>(ipsEntry->second); |
| } |
| else |
| { |
| log<level::ERR>( |
| "PowerMode::getDefaultIPSParms could not find property: EnterUtilizationPercent"); |
| } |
| |
| ipsEntry = ipsProperties.find("EnterUtilizationDwellTime"); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| enterTime = std::get<uint64_t>(ipsEntry->second); |
| } |
| else |
| { |
| log<level::ERR>( |
| "PowerMode::getDefaultIPSParms could not find property: EnterUtilizationDwellTime"); |
| } |
| |
| ipsEntry = ipsProperties.find("ExitUtilizationPercent"); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| exitUtil = std::get<uint64_t>(ipsEntry->second); |
| } |
| else |
| { |
| log<level::ERR>( |
| "PowerMode::getDefaultIPSParms could not find property: ExitUtilizationPercent"); |
| } |
| |
| ipsEntry = ipsProperties.find("ExitUtilizationDwellTime"); |
| if (ipsEntry != ipsProperties.end()) |
| { |
| exitTime = std::get<uint64_t>(ipsEntry->second); |
| } |
| else |
| { |
| log<level::ERR>( |
| "PowerMode::getDefaultIPSParms could not find property: ExitUtilizationDwellTime"); |
| } |
| |
| if (enterUtil > exitUtil) |
| { |
| log<level::ERR>( |
| std::format( |
| "ERROR: Default Idle Power Saver Enter Utilization ({}%) is > Exit Utilization ({}%) - using Exit for both", |
| enterUtil, exitUtil) |
| .c_str()); |
| enterUtil = exitUtil; |
| } |
| |
| return true; |
| } |
| |
| /* Read default IPS parameters, save them to the persistent file and update |
| DBus. Return true if successful */ |
| bool PowerMode::useDefaultIPSParms() |
| { |
| // Read the default IPS parameters |
| bool ipsEnabled; |
| uint8_t enterUtil, exitUtil; |
| uint16_t enterTime, exitTime; |
| if (!getDefaultIPSParms(ipsEnabled, enterUtil, enterTime, exitUtil, |
| exitTime)) |
| { |
| // Unable to read defaults |
| return false; |
| } |
| log<level::INFO>( |
| std::format( |
| "PowerMode::useDefaultIPSParms: Using default IPS parms: Enabled: {}, EnterUtil: {}%, EnterTime: {}s, ExitUtil: {}%, ExitTime: {}s", |
| ipsEnabled, enterUtil, enterTime, exitUtil, exitTime) |
| .c_str()); |
| |
| // Save IPS parms to the persistent file |
| persistedData.updateIPS(ipsEnabled, enterUtil, enterTime, exitUtil, |
| exitTime); |
| |
| // Write IPS parms to DBus |
| return updateDbusIPS(ipsEnabled, enterUtil, enterTime, exitUtil, exitTime); |
| } |
| |
| // Starts to watch for IPS active state changes. |
| bool PowerMode::openIpsFile() |
| { |
| bool rc = true; |
| fd = open(ipsStatusFile.c_str(), O_RDONLY | O_NONBLOCK); |
| const int open_errno = errno; |
| if (fd < 0) |
| { |
| log<level::ERR>( |
| std::format("openIpsFile Error({})={} : File={}", open_errno, |
| strerror(open_errno), ipsStatusFile.c_str()) |
| .c_str()); |
| |
| close(fd); |
| |
| using namespace sdbusplus::org::open_power::OCC::Device::Error; |
| report<OpenFailure>( |
| phosphor::logging::org::open_power::OCC::Device::OpenFailure:: |
| CALLOUT_ERRNO(open_errno), |
| phosphor::logging::org::open_power::OCC::Device::OpenFailure:: |
| CALLOUT_DEVICE_PATH(ipsStatusFile.c_str())); |
| |
| // We are no longer watching the error |
| if (ipsObject) |
| { |
| ipsObject->active(false); |
| } |
| |
| watching = false; |
| rc = false; |
| // NOTE: this will leave the system not reporting IPS active state to |
| // Fan Controls, Until an APP reload, or IPL and we will attempt again. |
| } |
| return rc; |
| } |
| |
| // Starts to watch for IPS active state changes. |
| void PowerMode::addIpsWatch(bool poll) |
| { |
| // open file and register callback on file if we are not currently watching, |
| // and if poll=true, and if we are the master. |
| if ((!watching) && poll) |
| { |
| // Open the file |
| if (openIpsFile()) |
| { |
| // register the callback handler which sets 'watching' |
| registerIpsStatusCallBack(); |
| } |
| } |
| } |
| |
| // Stops watching for IPS active state changes. |
| void PowerMode::removeIpsWatch() |
| { |
| // NOTE: we want to remove event, close file, and IPS active false no |
| // matter what the 'watching' flags is set to. |
| |
| // We are no longer watching the error |
| if (ipsObject) |
| { |
| ipsObject->active(false); |
| } |
| |
| watching = false; |
| |
| // Close file |
| close(fd); |
| |
| // clears sourcePtr in the event source. |
| eventSource.reset(); |
| } |
| |
| // Attaches the FD to event loop and registers the callback handler |
| void PowerMode::registerIpsStatusCallBack() |
| { |
| decltype(eventSource.get()) sourcePtr = nullptr; |
| |
| auto r = sd_event_add_io(event.get(), &sourcePtr, fd, EPOLLPRI | EPOLLERR, |
| ipsStatusCallBack, this); |
| if (r < 0) |
| { |
| log<level::ERR>(std::format("sd_event_add_io: Error({})={} : File={}", |
| r, strerror(-r), ipsStatusFile.c_str()) |
| .c_str()); |
| |
| using InternalFailure = |
| sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure; |
| report<InternalFailure>(); |
| |
| removeIpsWatch(); |
| // NOTE: this will leave the system not reporting IPS active state to |
| // Fan Controls, Until an APP reload, or IPL and we will attempt again. |
| } |
| else |
| { |
| // puts sourcePtr in the event source. |
| eventSource.reset(sourcePtr); |
| // Set we are watching the error |
| watching = true; |
| } |
| } |
| |
| // Static function to redirect to non static analyze event function to be |
| // able to read file and push onto dBus. |
| int PowerMode::ipsStatusCallBack(sd_event_source* /*es*/, int /*fd*/, |
| uint32_t /*revents*/, void* userData) |
| { |
| auto pmode = static_cast<PowerMode*>(userData); |
| pmode->analyzeIpsEvent(); |
| return 0; |
| } |
| |
| // Function to Read SysFs file change on IPS state and push on dBus. |
| void PowerMode::analyzeIpsEvent() |
| { |
| // Need to seek to START, else the poll returns immediately telling |
| // there is data to be read. if not done this floods the system. |
| auto r = lseek(fd, 0, SEEK_SET); |
| const int open_errno = errno; |
| if (r < 0) |
| { |
| // NOTE: upon file access error we can not just re-open file, we have to |
| // remove and add to watch. |
| removeIpsWatch(); |
| addIpsWatch(true); |
| } |
| |
| // if we are 'watching' that is the file seek, or the re-open passed.. we |
| // can read the data |
| if (watching) |
| { |
| // This file gets created when polling OCCs. A value or length of 0 is |
| // deemed success. That means we would disable IPS active on dbus. |
| char data; |
| bool ipsState = false; |
| const auto len = read(fd, &data, sizeof(data)); |
| const int readErrno = errno; |
| if (len <= 0) |
| { |
| removeIpsWatch(); |
| |
| log<level::ERR>( |
| std::format("IPS state Read Error({})={} : File={} : len={}", |
| readErrno, strerror(readErrno), |
| ipsStatusFile.c_str(), len) |
| .c_str()); |
| |
| report<ReadFailure>( |
| phosphor::logging::org::open_power::OCC::Device::ReadFailure:: |
| CALLOUT_ERRNO(readErrno), |
| phosphor::logging::org::open_power::OCC::Device::ReadFailure:: |
| CALLOUT_DEVICE_PATH(ipsStatusFile.c_str())); |
| |
| // NOTE: this will leave the system not reporting IPS active state |
| // to Fan Controls, Until an APP reload, or IPL and we will attempt |
| // again. |
| } |
| else |
| { |
| // Data returned in ASCII. |
| // convert to integer. atoi() |
| // from OCC_P10_FW_Interfaces spec |
| // Bit 6: IPS active 1 indicates enabled. |
| // Bit 7: IPS enabled. 1 indicates enabled. |
| // mask off bit 6 --> & 0x02 |
| // Shift left one bit and store as bool. >> 1 |
| ipsState = static_cast<bool>(((atoi(&data)) & 0x2) >> 1); |
| } |
| |
| // This will only set IPS active dbus if different than current. |
| if (ipsObject) |
| { |
| ipsObject->active(ipsState); |
| } |
| } |
| else |
| { |
| removeIpsWatch(); |
| |
| // If the Retry did not get to "watching = true" we already have an |
| // error log, just post trace. |
| log<level::ERR>( |
| std::format("Retry on File seek Error({})={} : File={}", open_errno, |
| strerror(open_errno), ipsStatusFile.c_str()) |
| .c_str()); |
| |
| // NOTE: this will leave the system not reporting IPS active state to |
| // Fan Controls, Until an APP reload, or IPL and we will attempt again. |
| } |
| |
| return; |
| } |
| |
| // overrides read/write to powerMode dbus property. |
| Mode::PowerMode PowerMode::powerMode(Mode::PowerMode requestedMode) |
| { |
| if (persistedData.getModeLock()) |
| { |
| log<level::INFO>("PowerMode::powerMode: mode property change blocked"); |
| elog<NotAllowed>(xyz::openbmc_project::Common::NotAllowed::REASON( |
| "mode change not allowed due to lock")); |
| } |
| else |
| { |
| // Verify requested mode is allowed |
| |
| // Convert PowerMode to internal SysPwrMode |
| SysPwrMode newMode = getInternalMode(requestedMode); |
| if (newMode != SysPwrMode::NO_CHANGE) |
| { |
| // Validate it is an allowed customer mode |
| if (customerModeList.contains(newMode)) |
| { |
| // Update persisted data with new mode |
| persistedData.updateMode(newMode, 0); |
| |
| log<level::INFO>( |
| std::format("DBus PowerMode Changed: {}", |
| convertPowerModeToString(requestedMode)) |
| .c_str()); |
| |
| // Send mode change to OCC |
| sendModeChange(); |
| } |
| else |
| { |
| // Not Allowed |
| log<level::ERR>( |
| std::format( |
| "PowerMode change not allowed. {} is not in AllowedPowerModes", |
| convertPowerModeToString(requestedMode)) |
| .c_str()); |
| elog<NotAllowed>( |
| xyz::openbmc_project::Common::NotAllowed::REASON( |
| "PowerMode value not allowed")); |
| } |
| } |
| else |
| { |
| // Value is not valid |
| using InvalidArgument = |
| sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument; |
| using Argument = xyz::openbmc_project::Common::InvalidArgument; |
| log<level::ERR>( |
| std::format( |
| "PowerMode not valid. {} is not in AllowedPowerModes", |
| convertPowerModeToString(requestedMode)) |
| .c_str()); |
| elog<InvalidArgument>(Argument::ARGUMENT_NAME("PowerMode"), |
| Argument::ARGUMENT_VALUE("INVALID MODE")); |
| } |
| } |
| |
| // All elog<> calls will cause trap (so code will not make it here) |
| |
| return Mode::powerMode(requestedMode); |
| } |
| |
| /* Set dbus property to SAFE mode(true) or clear(false) only if different |
| */ |
| void PowerMode::updateDbusSafeMode(const bool safeModeReq) |
| { |
| log<level::DEBUG>( |
| std::format("PowerMode:updateDbusSafeMode: Update dbus state ({})", |
| safeModeReq) |
| .c_str()); |
| |
| // Note; this function checks and only updates if different. |
| Mode::safeMode(safeModeReq); |
| } |
| |
| // Get the supported power modes from DBus and return true if success |
| bool PowerMode::getSupportedModes() |
| { |
| bool foundCustomerMode = false; |
| using ModePropertyVariants = |
| std::variant<bool, uint8_t, uint16_t, std::vector<std::string>>; |
| std::map<std::string, ModePropertyVariants> powerModeProperties{}; |
| |
| // Get all power mode properties from DBus |
| try |
| { |
| auto& bus = utils::getBus(); |
| std::string path = "/"; |
| std::string service = |
| utils::getServiceUsingSubTree(PMODE_DEFAULT_INTERFACE, path); |
| auto method = |
| bus.new_method_call(service.c_str(), path.c_str(), |
| "org.freedesktop.DBus.Properties", "GetAll"); |
| method.append(PMODE_DEFAULT_INTERFACE); |
| auto reply = bus.call(method); |
| reply.read(powerModeProperties); |
| } |
| catch (const sdbusplus::exception_t& e) |
| { |
| log<level::ERR>( |
| std::format("Unable to read PowerModeProperties: {}", e.what()) |
| .c_str()); |
| return false; |
| } |
| |
| // Determine if system suports EcoModes |
| auto ecoSupport = powerModeProperties.find("EcoModeSupport"); |
| if (ecoSupport != powerModeProperties.end()) |
| { |
| ecoModeSupport = std::get<bool>(ecoSupport->second); |
| log<level::INFO>(std::format("getSupportedModes(): ecoModeSupport: {}", |
| ecoModeSupport) |
| .c_str()); |
| } |
| |
| // Determine what customer modes are supported |
| using PMode = sdbusplus::xyz::openbmc_project::Control::Power::server::Mode; |
| std::set<PMode::PowerMode> modesToAllow; |
| auto custList = powerModeProperties.find("CustomerModes"); |
| if (custList != powerModeProperties.end()) |
| { |
| auto modeList = std::get<std::vector<std::string>>(custList->second); |
| for (auto mode : modeList) |
| { |
| // Ensure mode is valid |
| const std::string fullModeString = |
| PMODE_INTERFACE + ".PowerMode."s + mode; |
| log<level::INFO>( |
| std::format("getSupportedModes(): {}", mode).c_str()); |
| SysPwrMode modeValue = |
| powermode::convertStringToMode(fullModeString); |
| if (VALID_POWER_MODE_SETTING(modeValue)) |
| { |
| if (!foundCustomerMode) |
| { |
| // Start with empty list |
| customerModeList.clear(); |
| foundCustomerMode = true; |
| } |
| // Add mode to list |
| std::optional<PMode::PowerMode> cMode = |
| PMode::convertStringToPowerMode(fullModeString); |
| if (cMode) |
| modesToAllow.insert(cMode.value()); |
| customerModeList.insert(modeValue); |
| } |
| else |
| { |
| log<level::ERR>( |
| std::format( |
| "getSupportedModes(): Ignoring unsupported customer mode {}", |
| mode) |
| .c_str()); |
| } |
| } |
| } |
| if (foundCustomerMode) |
| { |
| ModeInterface::allowedPowerModes(modesToAllow); |
| } |
| |
| // Determine what OEM modes are supported |
| auto oemList = powerModeProperties.find("OemModes"); |
| if (oemList != powerModeProperties.end()) |
| { |
| bool foundValidMode = false; |
| auto OmodeList = std::get<std::vector<std::string>>(oemList->second); |
| for (auto mode : OmodeList) |
| { |
| // Ensure mode is valid |
| const std::string fullModeString = |
| PMODE_INTERFACE + ".PowerMode."s + mode; |
| SysPwrMode modeValue = |
| powermode::convertStringToMode(fullModeString); |
| if (VALID_POWER_MODE_SETTING(modeValue) || |
| VALID_OEM_POWER_MODE_SETTING(modeValue)) |
| { |
| if (!foundValidMode) |
| { |
| // Start with empty list |
| oemModeList.clear(); |
| foundValidMode = true; |
| } |
| // Add mode to list |
| oemModeList.insert(modeValue); |
| } |
| else |
| { |
| log<level::ERR>( |
| std::format( |
| "getSupportedModes(): Ignoring unsupported OEM mode {}", |
| mode) |
| .c_str()); |
| } |
| } |
| } |
| |
| return foundCustomerMode; |
| } |
| |
| bool PowerMode::isValidMode(const SysPwrMode mode) |
| { |
| if (customerModeList.contains(mode) || oemModeList.contains(mode)) |
| { |
| return true; |
| } |
| return false; |
| } |
| |
| } // namespace powermode |
| |
| } // namespace occ |
| |
| } // namespace open_power |