blob: 0fd422da3d1cdd0d71d9980acb707ccc3dbfe52c [file] [log] [blame]
/*
// Copyright (c) 2018-2019 Intel Corporation
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
*/
#include "power_control.hpp"
#include <sys/sysinfo.h>
#include <systemd/sd-journal.h>
#include <boost/asio/io_context.hpp>
#include <boost/asio/posix/stream_descriptor.hpp>
#include <boost/asio/steady_timer.hpp>
#include <boost/container/flat_map.hpp>
#include <boost/container/flat_set.hpp>
#include <gpiod.hpp>
#include <nlohmann/json.hpp>
#include <phosphor-logging/lg2.hpp>
#include <sdbusplus/asio/object_server.hpp>
#include <filesystem>
#include <fstream>
#include <optional>
#include <regex>
#include <string_view>
namespace power_control
{
static boost::asio::io_context io;
std::shared_ptr<sdbusplus::asio::connection> conn;
PersistentState appState;
PowerRestoreController powerRestore(io);
static std::string node = "0";
static const std::string appName = "power-control";
enum class DbusConfigType
{
name = 1,
path,
interface,
property
};
// Mandatory config parameters for dbus inputs
boost::container::flat_map<DbusConfigType, std::string> dbusParams = {
{DbusConfigType::name, "DbusName"},
{DbusConfigType::path, "Path"},
{DbusConfigType::interface, "Interface"},
{DbusConfigType::property, "Property"}};
enum class ConfigType
{
GPIO = 1,
DBUS
};
struct ConfigData
{
std::string name;
std::string lineName;
std::string dbusName;
std::string path;
std::string interface;
std::optional<std::regex> matchRegex;
bool polarity;
ConfigType type;
};
static ConfigData powerOutConfig;
static ConfigData powerOkConfig;
static ConfigData resetOutConfig;
static ConfigData nmiOutConfig;
static ConfigData sioPwrGoodConfig;
static ConfigData sioOnControlConfig;
static ConfigData sioS5Config;
static ConfigData postCompleteConfig;
static ConfigData powerButtonConfig;
static ConfigData resetButtonConfig;
static ConfigData idButtonConfig;
static ConfigData nmiButtonConfig;
static ConfigData slotPowerConfig;
static ConfigData hpmStbyEnConfig;
// map for storing list of gpio parameters whose config are to be read from x86
// power control json config
boost::container::flat_map<std::string, ConfigData*> powerSignalMap = {
{"PowerOut", &powerOutConfig},
{"PowerOk", &powerOkConfig},
{"ResetOut", &resetOutConfig},
{"NMIOut", &nmiOutConfig},
{"SioPowerGood", &sioPwrGoodConfig},
{"SioOnControl", &sioOnControlConfig},
{"SIOS5", &sioS5Config},
{"PostComplete", &postCompleteConfig},
{"PowerButton", &powerButtonConfig},
{"ResetButton", &resetButtonConfig},
{"IdButton", &idButtonConfig},
{"NMIButton", &nmiButtonConfig},
{"SlotPower", &slotPowerConfig},
{"HpmStbyEn", &hpmStbyEnConfig}};
static std::string hostDbusName = "xyz.openbmc_project.State.Host";
static std::string chassisDbusName = "xyz.openbmc_project.State.Chassis";
static std::string osDbusName = "xyz.openbmc_project.State.OperatingSystem";
static std::string buttonDbusName = "xyz.openbmc_project.Chassis.Buttons";
static std::string nmiDbusName = "xyz.openbmc_project.Control.Host.NMI";
static std::string rstCauseDbusName =
"xyz.openbmc_project.Control.Host.RestartCause";
static std::shared_ptr<sdbusplus::asio::dbus_interface> hostIface;
static std::shared_ptr<sdbusplus::asio::dbus_interface> chassisIface;
#ifdef CHASSIS_SYSTEM_RESET
static std::shared_ptr<sdbusplus::asio::dbus_interface> chassisSysIface;
static std::shared_ptr<sdbusplus::asio::dbus_interface> chassisSlotIface;
#endif
static std::shared_ptr<sdbusplus::asio::dbus_interface> powerButtonIface;
static std::shared_ptr<sdbusplus::asio::dbus_interface> resetButtonIface;
static std::shared_ptr<sdbusplus::asio::dbus_interface> nmiButtonIface;
static std::shared_ptr<sdbusplus::asio::dbus_interface> osIface;
static std::shared_ptr<sdbusplus::asio::dbus_interface> idButtonIface;
static std::shared_ptr<sdbusplus::asio::dbus_interface> nmiOutIface;
static std::shared_ptr<sdbusplus::asio::dbus_interface> restartCauseIface;
static gpiod::line powerButtonMask;
static gpiod::line resetButtonMask;
static bool nmiButtonMasked = false;
#if IGNORE_SOFT_RESETS_DURING_POST
static bool ignoreNextSoftReset = false;
#endif
// This map contains all timer values that are to be read from json config
boost::container::flat_map<std::string, int> TimerMap = {
{"PowerPulseMs", 200},
{"ForceOffPulseMs", 15000},
{"ResetPulseMs", 500},
{"PowerCycleMs", 5000},
{"SioPowerGoodWatchdogMs", 1000},
{"PsPowerOKWatchdogMs", 8000},
{"GracefulPowerOffS", (5 * 60)},
{"WarmResetCheckMs", 500},
{"PowerOffSaveMs", 7000},
{"SlotPowerCycleMs", 200},
{"DbusGetPropertyRetry", 1000}};
static bool nmiEnabled = true;
static bool nmiWhenPoweredOff = true;
static bool sioEnabled = true;
// Timers
// Time holding GPIOs asserted
static boost::asio::steady_timer gpioAssertTimer(io);
// Time between off and on during a power cycle
static boost::asio::steady_timer powerCycleTimer(io);
// Time OS gracefully powering off
static boost::asio::steady_timer gracefulPowerOffTimer(io);
// Time the warm reset check
static boost::asio::steady_timer warmResetCheckTimer(io);
// Time power supply power OK assertion on power-on
static boost::asio::steady_timer psPowerOKWatchdogTimer(io);
// Time SIO power good assertion on power-on
static boost::asio::steady_timer sioPowerGoodWatchdogTimer(io);
// Time power-off state save for power loss tracking
static boost::asio::steady_timer powerStateSaveTimer(io);
// POH timer
static boost::asio::steady_timer pohCounterTimer(io);
// Time when to allow restart cause updates
static boost::asio::steady_timer restartCauseTimer(io);
static boost::asio::steady_timer slotPowerCycleTimer(io);
// Map containing timers used for D-Bus get-property retries
static boost::container::flat_map<std::string, boost::asio::steady_timer>
dBusRetryTimers;
// GPIO Lines and Event Descriptors
static gpiod::line psPowerOKLine;
static boost::asio::posix::stream_descriptor psPowerOKEvent(io);
static gpiod::line sioPowerGoodLine;
static boost::asio::posix::stream_descriptor sioPowerGoodEvent(io);
static gpiod::line sioOnControlLine;
static boost::asio::posix::stream_descriptor sioOnControlEvent(io);
static gpiod::line sioS5Line;
static boost::asio::posix::stream_descriptor sioS5Event(io);
static gpiod::line powerButtonLine;
static boost::asio::posix::stream_descriptor powerButtonEvent(io);
static gpiod::line resetButtonLine;
static boost::asio::posix::stream_descriptor resetButtonEvent(io);
static gpiod::line nmiButtonLine;
static boost::asio::posix::stream_descriptor nmiButtonEvent(io);
static gpiod::line idButtonLine;
static boost::asio::posix::stream_descriptor idButtonEvent(io);
static gpiod::line postCompleteLine;
static boost::asio::posix::stream_descriptor postCompleteEvent(io);
static gpiod::line nmiOutLine;
static gpiod::line slotPowerLine;
static constexpr uint8_t beepPowerFail = 8;
static void beep(const uint8_t& beepPriority)
{
lg2::info("Beep with priority: {BEEP_PRIORITY}", "BEEP_PRIORITY",
beepPriority);
conn->async_method_call(
[](boost::system::error_code ec) {
if (ec)
{
lg2::error(
"beep returned error with async_method_call (ec = {ERROR_MSG})",
"ERROR_MSG", ec.message());
return;
}
},
"xyz.openbmc_project.BeepCode", "/xyz/openbmc_project/BeepCode",
"xyz.openbmc_project.BeepCode", "Beep", uint8_t(beepPriority));
}
enum class OperatingSystemStateStage
{
Inactive,
Standby,
};
static OperatingSystemStateStage operatingSystemState;
static constexpr std::string_view
getOperatingSystemStateStage(const OperatingSystemStateStage stage)
{
switch (stage)
{
case OperatingSystemStateStage::Inactive:
return "xyz.openbmc_project.State.OperatingSystem.Status.OSStatus.Inactive";
break;
case OperatingSystemStateStage::Standby:
return "xyz.openbmc_project.State.OperatingSystem.Status.OSStatus.Standby";
break;
default:
return "xyz.openbmc_project.State.OperatingSystem.Status.OSStatus.Inactive";
break;
}
};
static void setOperatingSystemState(const OperatingSystemStateStage stage)
{
operatingSystemState = stage;
#if IGNORE_SOFT_RESETS_DURING_POST
// If POST complete has asserted set ignoreNextSoftReset to false to avoid
// masking soft resets after POST
if (operatingSystemState == OperatingSystemStateStage::Standby)
{
ignoreNextSoftReset = false;
}
#endif
osIface->set_property("OperatingSystemState",
std::string(getOperatingSystemStateStage(stage)));
lg2::info("Moving os state to {STATE} stage", "STATE",
getOperatingSystemStateStage(stage));
}
enum class PowerState
{
on,
waitForPSPowerOK,
waitForSIOPowerGood,
off,
transitionToOff,
gracefulTransitionToOff,
cycleOff,
transitionToCycleOff,
gracefulTransitionToCycleOff,
checkForWarmReset,
};
static PowerState powerState;
static std::string getPowerStateName(PowerState state)
{
switch (state)
{
case PowerState::on:
return "On";
break;
case PowerState::waitForPSPowerOK:
return "Wait for Power Supply Power OK";
break;
case PowerState::waitForSIOPowerGood:
return "Wait for SIO Power Good";
break;
case PowerState::off:
return "Off";
break;
case PowerState::transitionToOff:
return "Transition to Off";
break;
case PowerState::gracefulTransitionToOff:
return "Graceful Transition to Off";
break;
case PowerState::cycleOff:
return "Power Cycle Off";
break;
case PowerState::transitionToCycleOff:
return "Transition to Power Cycle Off";
break;
case PowerState::gracefulTransitionToCycleOff:
return "Graceful Transition to Power Cycle Off";
break;
case PowerState::checkForWarmReset:
return "Check for Warm Reset";
break;
default:
return "unknown state: " + std::to_string(static_cast<int>(state));
break;
}
}
static void logStateTransition(const PowerState state)
{
lg2::info("Host{HOST}: Moving to \"{STATE}\" state", "HOST", node, "STATE",
getPowerStateName(state));
}
enum class Event
{
psPowerOKAssert,
psPowerOKDeAssert,
sioPowerGoodAssert,
sioPowerGoodDeAssert,
sioS5Assert,
sioS5DeAssert,
pltRstAssert,
pltRstDeAssert,
postCompleteAssert,
postCompleteDeAssert,
powerButtonPressed,
resetButtonPressed,
powerCycleTimerExpired,
psPowerOKWatchdogTimerExpired,
sioPowerGoodWatchdogTimerExpired,
gracefulPowerOffTimerExpired,
powerOnRequest,
powerOffRequest,
powerCycleRequest,
resetRequest,
gracefulPowerOffRequest,
gracefulPowerCycleRequest,
warmResetDetected,
};
static std::string getEventName(Event event)
{
switch (event)
{
case Event::psPowerOKAssert:
return "power supply power OK assert";
break;
case Event::psPowerOKDeAssert:
return "power supply power OK de-assert";
break;
case Event::sioPowerGoodAssert:
return "SIO power good assert";
break;
case Event::sioPowerGoodDeAssert:
return "SIO power good de-assert";
break;
case Event::sioS5Assert:
return "SIO S5 assert";
break;
case Event::sioS5DeAssert:
return "SIO S5 de-assert";
break;
case Event::pltRstAssert:
return "PLT_RST assert";
break;
case Event::pltRstDeAssert:
return "PLT_RST de-assert";
break;
case Event::postCompleteAssert:
return "POST Complete assert";
break;
case Event::postCompleteDeAssert:
return "POST Complete de-assert";
break;
case Event::powerButtonPressed:
return "power button pressed";
break;
case Event::resetButtonPressed:
return "reset button pressed";
break;
case Event::powerCycleTimerExpired:
return "power cycle timer expired";
break;
case Event::psPowerOKWatchdogTimerExpired:
return "power supply power OK watchdog timer expired";
break;
case Event::sioPowerGoodWatchdogTimerExpired:
return "SIO power good watchdog timer expired";
break;
case Event::gracefulPowerOffTimerExpired:
return "graceful power-off timer expired";
break;
case Event::powerOnRequest:
return "power-on request";
break;
case Event::powerOffRequest:
return "power-off request";
break;
case Event::powerCycleRequest:
return "power-cycle request";
break;
case Event::resetRequest:
return "reset request";
break;
case Event::gracefulPowerOffRequest:
return "graceful power-off request";
break;
case Event::gracefulPowerCycleRequest:
return "graceful power-cycle request";
break;
case Event::warmResetDetected:
return "warm reset detected";
break;
default:
return "unknown event: " + std::to_string(static_cast<int>(event));
break;
}
}
static void logEvent(const std::string_view stateHandler, const Event event)
{
lg2::info("{STATE_HANDLER}: {EVENT} event received", "STATE_HANDLER",
stateHandler, "EVENT", getEventName(event));
}
// Power state handlers
static void powerStateOn(const Event event);
static void powerStateWaitForPSPowerOK(const Event event);
static void powerStateWaitForSIOPowerGood(const Event event);
static void powerStateOff(const Event event);
static void powerStateTransitionToOff(const Event event);
static void powerStateGracefulTransitionToOff(const Event event);
static void powerStateCycleOff(const Event event);
static void powerStateTransitionToCycleOff(const Event event);
static void powerStateGracefulTransitionToCycleOff(const Event event);
static void powerStateCheckForWarmReset(const Event event);
static std::function<void(const Event)> getPowerStateHandler(PowerState state)
{
switch (state)
{
case PowerState::on:
return powerStateOn;
break;
case PowerState::waitForPSPowerOK:
return powerStateWaitForPSPowerOK;
break;
case PowerState::waitForSIOPowerGood:
return powerStateWaitForSIOPowerGood;
break;
case PowerState::off:
return powerStateOff;
break;
case PowerState::transitionToOff:
return powerStateTransitionToOff;
break;
case PowerState::gracefulTransitionToOff:
return powerStateGracefulTransitionToOff;
break;
case PowerState::cycleOff:
return powerStateCycleOff;
break;
case PowerState::transitionToCycleOff:
return powerStateTransitionToCycleOff;
break;
case PowerState::gracefulTransitionToCycleOff:
return powerStateGracefulTransitionToCycleOff;
break;
case PowerState::checkForWarmReset:
return powerStateCheckForWarmReset;
break;
default:
return nullptr;
break;
}
};
static void sendPowerControlEvent(const Event event)
{
std::function<void(const Event)> handler = getPowerStateHandler(powerState);
if (handler == nullptr)
{
lg2::error("Failed to find handler for power state: {STATE}", "STATE",
static_cast<int>(powerState));
return;
}
handler(event);
}
static uint64_t getCurrentTimeMs()
{
struct timespec time = {};
if (clock_gettime(CLOCK_REALTIME, &time) < 0)
{
return 0;
}
uint64_t currentTimeMs = static_cast<uint64_t>(time.tv_sec) * 1000;
currentTimeMs += static_cast<uint64_t>(time.tv_nsec) / 1000 / 1000;
return currentTimeMs;
}
static constexpr std::string_view getHostState(const PowerState state)
{
switch (state)
{
case PowerState::on:
case PowerState::gracefulTransitionToOff:
case PowerState::gracefulTransitionToCycleOff:
return "xyz.openbmc_project.State.Host.HostState.Running";
break;
case PowerState::waitForPSPowerOK:
case PowerState::waitForSIOPowerGood:
case PowerState::off:
case PowerState::transitionToOff:
case PowerState::transitionToCycleOff:
case PowerState::cycleOff:
case PowerState::checkForWarmReset:
return "xyz.openbmc_project.State.Host.HostState.Off";
break;
default:
return "";
break;
}
};
static constexpr std::string_view getChassisState(const PowerState state)
{
switch (state)
{
case PowerState::on:
case PowerState::transitionToOff:
case PowerState::gracefulTransitionToOff:
case PowerState::transitionToCycleOff:
case PowerState::gracefulTransitionToCycleOff:
case PowerState::checkForWarmReset:
return "xyz.openbmc_project.State.Chassis.PowerState.On";
break;
case PowerState::waitForPSPowerOK:
case PowerState::waitForSIOPowerGood:
case PowerState::off:
case PowerState::cycleOff:
return "xyz.openbmc_project.State.Chassis.PowerState.Off";
break;
default:
return "";
break;
}
};
#ifdef CHASSIS_SYSTEM_RESET
enum class SlotPowerState
{
on,
off,
};
static SlotPowerState slotPowerState;
static constexpr std::string_view getSlotState(const SlotPowerState state)
{
switch (state)
{
case SlotPowerState::on:
return "xyz.openbmc_project.State.Chassis.PowerState.On";
break;
case SlotPowerState::off:
return "xyz.openbmc_project.State.Chassis.PowerState.Off";
break;
default:
return "";
break;
}
};
static void setSlotPowerState(const SlotPowerState state)
{
slotPowerState = state;
chassisSlotIface->set_property("CurrentPowerState",
std::string(getSlotState(slotPowerState)));
chassisSlotIface->set_property("LastStateChangeTime", getCurrentTimeMs());
}
#endif
static void savePowerState(const PowerState state)
{
powerStateSaveTimer.expires_after(
std::chrono::milliseconds(TimerMap["PowerOffSaveMs"]));
powerStateSaveTimer.async_wait([state](const boost::system::error_code ec) {
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error("Power-state save async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
return;
}
appState.set(PersistentState::Params::PowerState,
std::string{getChassisState(state)});
});
}
static void setPowerState(const PowerState state)
{
powerState = state;
logStateTransition(state);
hostIface->set_property("CurrentHostState",
std::string(getHostState(powerState)));
chassisIface->set_property("CurrentPowerState",
std::string(getChassisState(powerState)));
chassisIface->set_property("LastStateChangeTime", getCurrentTimeMs());
// Save the power state for the restore policy
savePowerState(state);
}
enum class RestartCause
{
command,
resetButton,
powerButton,
watchdog,
powerPolicyOn,
powerPolicyRestore,
softReset,
};
static boost::container::flat_set<RestartCause> causeSet;
static std::string getRestartCause(RestartCause cause)
{
switch (cause)
{
case RestartCause::command:
return "xyz.openbmc_project.State.Host.RestartCause.IpmiCommand";
break;
case RestartCause::resetButton:
return "xyz.openbmc_project.State.Host.RestartCause.ResetButton";
break;
case RestartCause::powerButton:
return "xyz.openbmc_project.State.Host.RestartCause.PowerButton";
break;
case RestartCause::watchdog:
return "xyz.openbmc_project.State.Host.RestartCause.WatchdogTimer";
break;
case RestartCause::powerPolicyOn:
return "xyz.openbmc_project.State.Host.RestartCause.PowerPolicyAlwaysOn";
break;
case RestartCause::powerPolicyRestore:
return "xyz.openbmc_project.State.Host.RestartCause.PowerPolicyPreviousState";
break;
case RestartCause::softReset:
return "xyz.openbmc_project.State.Host.RestartCause.SoftReset";
break;
default:
return "xyz.openbmc_project.State.Host.RestartCause.Unknown";
break;
}
}
static void addRestartCause(const RestartCause cause)
{
// Add this to the set of causes for this restart
causeSet.insert(cause);
}
static void clearRestartCause()
{
// Clear the set for the next restart
causeSet.clear();
}
static void setRestartCauseProperty(const std::string& cause)
{
lg2::info("RestartCause set to {RESTART_CAUSE}", "RESTART_CAUSE", cause);
restartCauseIface->set_property("RestartCause", cause);
}
#ifdef USE_ACBOOT
static void resetACBootProperty()
{
if ((causeSet.contains(RestartCause::command)) ||
(causeSet.contains(RestartCause::softReset)))
{
conn->async_method_call(
[](boost::system::error_code ec) {
if (ec)
{
lg2::error("failed to reset ACBoot property");
}
},
"xyz.openbmc_project.Settings",
"/xyz/openbmc_project/control/host0/ac_boot",
"org.freedesktop.DBus.Properties", "Set",
"xyz.openbmc_project.Common.ACBoot", "ACBoot",
std::variant<std::string>{"False"});
}
}
#endif // USE_ACBOOT
static void setRestartCause()
{
// Determine the actual restart cause based on the set of causes
std::string restartCause =
"xyz.openbmc_project.State.Host.RestartCause.Unknown";
if (causeSet.contains(RestartCause::watchdog))
{
restartCause = getRestartCause(RestartCause::watchdog);
}
else if (causeSet.contains(RestartCause::command))
{
restartCause = getRestartCause(RestartCause::command);
}
else if (causeSet.contains(RestartCause::resetButton))
{
restartCause = getRestartCause(RestartCause::resetButton);
}
else if (causeSet.contains(RestartCause::powerButton))
{
restartCause = getRestartCause(RestartCause::powerButton);
}
else if (causeSet.contains(RestartCause::powerPolicyOn))
{
restartCause = getRestartCause(RestartCause::powerPolicyOn);
}
else if (causeSet.contains(RestartCause::powerPolicyRestore))
{
restartCause = getRestartCause(RestartCause::powerPolicyRestore);
}
else if (causeSet.contains(RestartCause::softReset))
{
#if IGNORE_SOFT_RESETS_DURING_POST
if (ignoreNextSoftReset)
{
ignoreNextSoftReset = false;
return;
}
#endif
restartCause = getRestartCause(RestartCause::softReset);
}
setRestartCauseProperty(restartCause);
}
static void systemPowerGoodFailedLog()
{
sd_journal_send(
"MESSAGE=PowerControl: system power good failed to assert (VR failure)",
"PRIORITY=%i", LOG_INFO, "REDFISH_MESSAGE_ID=%s",
"OpenBMC.0.1.SystemPowerGoodFailed", "REDFISH_MESSAGE_ARGS=%d",
TimerMap["SioPowerGoodWatchdogMs"], NULL);
}
static void psPowerOKFailedLog()
{
sd_journal_send(
"MESSAGE=PowerControl: power supply power good failed to assert",
"PRIORITY=%i", LOG_INFO, "REDFISH_MESSAGE_ID=%s",
"OpenBMC.0.1.PowerSupplyPowerGoodFailed", "REDFISH_MESSAGE_ARGS=%d",
TimerMap["PsPowerOKWatchdogMs"], NULL);
}
static void powerRestorePolicyLog()
{
sd_journal_send("MESSAGE=PowerControl: power restore policy applied",
"PRIORITY=%i", LOG_INFO, "REDFISH_MESSAGE_ID=%s",
"OpenBMC.0.1.PowerRestorePolicyApplied", NULL);
}
static void powerButtonPressLog()
{
sd_journal_send("MESSAGE=PowerControl: power button pressed", "PRIORITY=%i",
LOG_INFO, "REDFISH_MESSAGE_ID=%s",
"OpenBMC.0.1.PowerButtonPressed", NULL);
}
static void resetButtonPressLog()
{
sd_journal_send("MESSAGE=PowerControl: reset button pressed", "PRIORITY=%i",
LOG_INFO, "REDFISH_MESSAGE_ID=%s",
"OpenBMC.0.1.ResetButtonPressed", NULL);
}
static void nmiButtonPressLog()
{
sd_journal_send("MESSAGE=PowerControl: NMI button pressed", "PRIORITY=%i",
LOG_INFO, "REDFISH_MESSAGE_ID=%s",
"OpenBMC.0.1.NMIButtonPressed", NULL);
}
static void nmiDiagIntLog()
{
sd_journal_send("MESSAGE=PowerControl: NMI Diagnostic Interrupt",
"PRIORITY=%i", LOG_INFO, "REDFISH_MESSAGE_ID=%s",
"OpenBMC.0.1.NMIDiagnosticInterrupt", NULL);
}
PersistentState::PersistentState()
{
// create the power control directory if it doesn't exist
std::error_code ec;
if (!(std::filesystem::create_directories(powerControlDir, ec)))
{
if (ec.value() != 0)
{
lg2::error("failed to create {DIR_NAME}: {ERROR_MSG}", "DIR_NAME",
powerControlDir.string(), "ERROR_MSG", ec.message());
throw std::runtime_error("Failed to create state directory");
}
}
// read saved state, it's ok, if the file doesn't exists
std::ifstream appStateStream(powerControlDir / stateFile);
if (!appStateStream.is_open())
{
lg2::info("Cannot open state file \'{PATH}\'", "PATH",
std::string(powerControlDir / stateFile));
stateData = nlohmann::json({});
return;
}
try
{
appStateStream >> stateData;
if (stateData.is_discarded())
{
lg2::info("Cannot parse state file \'{PATH}\'", "PATH",
std::string(powerControlDir / stateFile));
stateData = nlohmann::json({});
return;
}
}
catch (const std::exception& ex)
{
lg2::info("Cannot read state file \'{PATH}\'", "PATH",
std::string(powerControlDir / stateFile));
stateData = nlohmann::json({});
return;
}
}
PersistentState::~PersistentState()
{
saveState();
}
const std::string PersistentState::get(Params parameter)
{
auto val = stateData.find(getName(parameter));
if (val != stateData.end())
{
return val->get<std::string>();
}
return getDefault(parameter);
}
void PersistentState::set(Params parameter, const std::string& value)
{
stateData[getName(parameter)] = value;
saveState();
}
const std::string PersistentState::getName(const Params parameter)
{
switch (parameter)
{
case Params::PowerState:
return "PowerState";
}
return "";
}
const std::string PersistentState::getDefault(const Params parameter)
{
switch (parameter)
{
case Params::PowerState:
return "xyz.openbmc_project.State.Chassis.PowerState.Off";
}
return "";
}
void PersistentState::saveState()
{
std::ofstream appStateStream(powerControlDir / stateFile, std::ios::trunc);
if (!appStateStream.is_open())
{
lg2::error("Cannot write state file \'{PATH}\'", "PATH",
std::string(powerControlDir / stateFile));
return;
}
appStateStream << stateData.dump(indentationSize);
}
static constexpr const char* setingsService = "xyz.openbmc_project.Settings";
static constexpr const char* powerRestorePolicyIface =
"xyz.openbmc_project.Control.Power.RestorePolicy";
#ifdef USE_ACBOOT
static constexpr const char* powerACBootObject =
"/xyz/openbmc_project/control/host0/ac_boot";
static constexpr const char* powerACBootIface =
"xyz.openbmc_project.Common.ACBoot";
#endif // USE_ACBOOT
namespace match_rules = sdbusplus::bus::match::rules;
static int powerRestoreConfigHandler(sd_bus_message* m, void* context,
sd_bus_error*)
{
if (context == nullptr || m == nullptr)
{
throw std::runtime_error("Invalid match");
}
sdbusplus::message_t message(m);
PowerRestoreController* powerRestore =
static_cast<PowerRestoreController*>(context);
if (std::string(message.get_member()) == "InterfacesAdded")
{
sdbusplus::message::object_path path;
boost::container::flat_map<std::string, dbusPropertiesList> data;
message.read(path, data);
for (auto& [iface, properties] : data)
{
if ((iface == powerRestorePolicyIface)
#ifdef USE_ACBOOT
|| (iface == powerACBootIface)
#endif // USE_ACBOOT
)
{
powerRestore->setProperties(properties);
}
}
}
else if (std::string(message.get_member()) == "PropertiesChanged")
{
std::string interfaceName;
dbusPropertiesList propertiesChanged;
message.read(interfaceName, propertiesChanged);
powerRestore->setProperties(propertiesChanged);
}
return 1;
}
void PowerRestoreController::run()
{
std::string powerRestorePolicyObject =
"/xyz/openbmc_project/control/host" + node + "/power_restore_policy";
powerRestorePolicyLog();
// this list only needs to be created once
if (matches.empty())
{
matches.emplace_back(
*conn,
match_rules::interfacesAdded() +
match_rules::argNpath(0, powerRestorePolicyObject) +
match_rules::sender(setingsService),
powerRestoreConfigHandler, this);
#ifdef USE_ACBOOT
matches.emplace_back(*conn,
match_rules::interfacesAdded() +
match_rules::argNpath(0, powerACBootObject) +
match_rules::sender(setingsService),
powerRestoreConfigHandler, this);
matches.emplace_back(*conn,
match_rules::propertiesChanged(powerACBootObject,
powerACBootIface) +
match_rules::sender(setingsService),
powerRestoreConfigHandler, this);
#endif // USE_ACBOOT
}
// Check if it's already on DBus
conn->async_method_call(
[this](boost::system::error_code ec,
const dbusPropertiesList properties) {
if (ec)
{
return;
}
setProperties(properties);
},
setingsService, powerRestorePolicyObject,
"org.freedesktop.DBus.Properties", "GetAll", powerRestorePolicyIface);
#ifdef USE_ACBOOT
// Check if it's already on DBus
conn->async_method_call(
[this](boost::system::error_code ec,
const dbusPropertiesList properties) {
if (ec)
{
return;
}
setProperties(properties);
},
setingsService, powerACBootObject, "org.freedesktop.DBus.Properties",
"GetAll", powerACBootIface);
#endif
}
void PowerRestoreController::setProperties(const dbusPropertiesList& props)
{
for (auto& [property, propValue] : props)
{
if (property == "PowerRestorePolicy")
{
const std::string* value = std::get_if<std::string>(&propValue);
if (value == nullptr)
{
lg2::error("Unable to read Power Restore Policy");
continue;
}
powerRestorePolicy = *value;
}
else if (property == "PowerRestoreDelay")
{
const uint64_t* value = std::get_if<uint64_t>(&propValue);
if (value == nullptr)
{
lg2::error("Unable to read Power Restore Delay");
continue;
}
powerRestoreDelay = *value / 1000000; // usec to sec
}
#ifdef USE_ACBOOT
else if (property == "ACBoot")
{
const std::string* value = std::get_if<std::string>(&propValue);
if (value == nullptr)
{
lg2::error("Unable to read AC Boot status");
continue;
}
acBoot = *value;
}
#endif // USE_ACBOOT
}
invokeIfReady();
}
void PowerRestoreController::invokeIfReady()
{
if ((powerRestorePolicy.empty()) || (powerRestoreDelay < 0))
{
return;
}
#ifdef USE_ACBOOT
if (acBoot.empty() || acBoot == "Unknown")
{
return;
}
#endif
matches.clear();
if (!timerFired)
{
// Calculate the delay from now to meet the requested delay
// Subtract the approximate uboot time
static constexpr const int ubootSeconds = 20;
int delay = powerRestoreDelay - ubootSeconds;
// Subtract the time since boot
struct sysinfo info = {};
if (sysinfo(&info) == 0)
{
delay -= info.uptime;
}
if (delay > 0)
{
powerRestoreTimer.expires_after(std::chrono::seconds(delay));
lg2::info("Power Restore delay of {DELAY} seconds started", "DELAY",
delay);
powerRestoreTimer.async_wait([this](const boost::system::error_code
ec) {
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec == boost::asio::error::operation_aborted)
{
return;
}
lg2::error(
"power restore policy async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
else
{
lg2::info("Power Restore delay timer expired");
}
invoke();
});
timerFired = true;
}
else
{
invoke();
}
}
}
void PowerRestoreController::invoke()
{
// we want to run Power Restore only once
if (policyInvoked)
{
return;
}
policyInvoked = true;
lg2::info("Invoking Power Restore Policy {POLICY}", "POLICY",
powerRestorePolicy);
if (powerRestorePolicy ==
"xyz.openbmc_project.Control.Power.RestorePolicy.Policy.AlwaysOn")
{
sendPowerControlEvent(Event::powerOnRequest);
setRestartCauseProperty(getRestartCause(RestartCause::powerPolicyOn));
}
else if (powerRestorePolicy ==
"xyz.openbmc_project.Control.Power.RestorePolicy.Policy.Restore")
{
if (wasPowerDropped())
{
lg2::info("Power was dropped, restoring Host On state");
sendPowerControlEvent(Event::powerOnRequest);
setRestartCauseProperty(
getRestartCause(RestartCause::powerPolicyRestore));
}
else
{
lg2::info("No power drop, restoring Host Off state");
}
}
// We're done with the previous power state for the restore policy, so store
// the current state
savePowerState(powerState);
}
bool PowerRestoreController::wasPowerDropped()
{
std::string state = appState.get(PersistentState::Params::PowerState);
return state == "xyz.openbmc_project.State.Chassis.PowerState.On";
}
static void waitForGPIOEvent(
const std::string& name, const std::function<void(bool)>& eventHandler,
gpiod::line& line, boost::asio::posix::stream_descriptor& event)
{
event.async_wait(
boost::asio::posix::stream_descriptor::wait_read,
[&name, eventHandler, &line,
&event](const boost::system::error_code ec) {
if (ec)
{
lg2::error("{GPIO_NAME} fd handler error: {ERROR_MSG}",
"GPIO_NAME", name, "ERROR_MSG", ec.message());
// TODO: throw here to force power-control to
// restart?
return;
}
gpiod::line_event line_event = line.event_read();
eventHandler(line_event.event_type ==
gpiod::line_event::RISING_EDGE);
waitForGPIOEvent(name, eventHandler, line, event);
});
}
static bool requestGPIOEvents(
const std::string& name, const std::function<void(bool)>& handler,
gpiod::line& gpioLine,
boost::asio::posix::stream_descriptor& gpioEventDescriptor)
{
// Find the GPIO line
gpioLine = gpiod::find_line(name);
if (!gpioLine)
{
lg2::error("Failed to find the {GPIO_NAME} line", "GPIO_NAME", name);
return false;
}
try
{
gpioLine.request({appName, gpiod::line_request::EVENT_BOTH_EDGES, {}});
}
catch (const std::exception& e)
{
lg2::error("Failed to request events for {GPIO_NAME}: {ERROR}",
"GPIO_NAME", name, "ERROR", e);
return false;
}
int gpioLineFd = gpioLine.event_get_fd();
if (gpioLineFd < 0)
{
lg2::error("Failed to get {GPIO_NAME} fd", "GPIO_NAME", name);
return false;
}
gpioEventDescriptor.assign(gpioLineFd);
waitForGPIOEvent(name, handler, gpioLine, gpioEventDescriptor);
return true;
}
static bool setGPIOOutput(const std::string& name, const int value,
gpiod::line& gpioLine)
{
// Find the GPIO line
gpioLine = gpiod::find_line(name);
if (!gpioLine)
{
lg2::error("Failed to find the {GPIO_NAME} line", "GPIO_NAME", name);
return false;
}
// Request GPIO output to specified value
try
{
gpioLine.request({appName, gpiod::line_request::DIRECTION_OUTPUT, {}},
value);
}
catch (const std::exception& e)
{
lg2::error("Failed to request {GPIO_NAME} output: {ERROR}", "GPIO_NAME",
name, "ERROR", e);
return false;
}
lg2::info("{GPIO_NAME} set to {GPIO_VALUE}", "GPIO_NAME", name,
"GPIO_VALUE", value);
return true;
}
static int setMaskedGPIOOutputForMs(gpiod::line& maskedGPIOLine,
const std::string& name, const int value,
const int durationMs)
{
// Set the masked GPIO line to the specified value
maskedGPIOLine.set_value(value);
lg2::info("{GPIO_NAME} set to {GPIO_VALUE}", "GPIO_NAME", name,
"GPIO_VALUE", value);
gpioAssertTimer.expires_after(std::chrono::milliseconds(durationMs));
gpioAssertTimer.async_wait(
[maskedGPIOLine, value, name](const boost::system::error_code ec) {
// Set the masked GPIO line back to the opposite value
maskedGPIOLine.set_value(!value);
lg2::info("{GPIO_NAME} released", "GPIO_NAME", name);
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error("{GPIO_NAME} async_wait failed: {ERROR_MSG}",
"GPIO_NAME", name, "ERROR_MSG", ec.message());
}
}
});
return 0;
}
static int setGPIOOutputForMs(const ConfigData& config, const int value,
const int durationMs)
{
// If the requested GPIO is masked, use the mask line to set the output
if (powerButtonMask && config.lineName == powerOutConfig.lineName)
{
return setMaskedGPIOOutputForMs(powerButtonMask, config.lineName, value,
durationMs);
}
if (resetButtonMask && config.lineName == resetOutConfig.lineName)
{
return setMaskedGPIOOutputForMs(resetButtonMask, config.lineName, value,
durationMs);
}
// No mask set, so request and set the GPIO normally
gpiod::line gpioLine;
if (!setGPIOOutput(config.lineName, value, gpioLine))
{
return -1;
}
const std::string name = config.lineName;
gpioAssertTimer.expires_after(std::chrono::milliseconds(durationMs));
gpioAssertTimer.async_wait(
[gpioLine, value, name](const boost::system::error_code ec) {
// Set the GPIO line back to the opposite value
gpioLine.set_value(!value);
lg2::info("{GPIO_NAME} released", "GPIO_NAME", name);
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error("{GPIO_NAME} async_wait failed: {ERROR_MSG}",
"GPIO_NAME", name, "ERROR_MSG", ec.message());
}
}
});
return 0;
}
static int assertGPIOForMs(const ConfigData& config, const int durationMs)
{
return setGPIOOutputForMs(config, config.polarity, durationMs);
}
static void powerOn()
{
assertGPIOForMs(powerOutConfig, TimerMap["PowerPulseMs"]);
}
#ifdef CHASSIS_SYSTEM_RESET
static int slotPowerOn()
{
if (power_control::slotPowerState != power_control::SlotPowerState::on)
{
slotPowerLine.set_value(1);
if (slotPowerLine.get_value() > 0)
{
setSlotPowerState(SlotPowerState::on);
lg2::info("Slot Power is switched On\n");
}
else
{
return -1;
}
}
else
{
lg2::info("Slot Power is already in 'On' state\n");
return -1;
}
return 0;
}
static int slotPowerOff()
{
if (power_control::slotPowerState != power_control::SlotPowerState::off)
{
slotPowerLine.set_value(0);
if (!(slotPowerLine.get_value() > 0))
{
setSlotPowerState(SlotPowerState::off);
setPowerState(PowerState::off);
lg2::info("Slot Power is switched Off\n");
}
else
{
return -1;
}
}
else
{
lg2::info("Slot Power is already in 'Off' state\n");
return -1;
}
return 0;
}
static void slotPowerCycle()
{
lg2::info("Slot Power Cycle started\n");
slotPowerOff();
slotPowerCycleTimer.expires_after(
std::chrono::milliseconds(TimerMap["SlotPowerCycleMs"]));
slotPowerCycleTimer.async_wait([](const boost::system::error_code ec) {
if (ec)
{
if (ec != boost::asio::error::operation_aborted)
{
lg2::error(
"Slot Power cycle timer async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
lg2::info("Slot Power cycle timer canceled\n");
return;
}
lg2::info("Slot Power cycle timer completed\n");
slotPowerOn();
lg2::info("Slot Power Cycle Completed\n");
});
}
#endif
static void gracefulPowerOff()
{
assertGPIOForMs(powerOutConfig, TimerMap["PowerPulseMs"]);
}
static void forcePowerOff()
{
if (assertGPIOForMs(powerOutConfig, TimerMap["ForceOffPulseMs"]) < 0)
{
return;
}
// If the force off timer expires, then the power-button override failed
gpioAssertTimer.async_wait([](const boost::system::error_code ec) {
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error("Force power off async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
return;
}
lg2::error("Power-button override failed. Not sure what to do now.");
});
}
static void reset()
{
assertGPIOForMs(resetOutConfig, TimerMap["ResetPulseMs"]);
}
static void gracefulPowerOffTimerStart()
{
lg2::info("Graceful power-off timer started");
gracefulPowerOffTimer.expires_after(
std::chrono::seconds(TimerMap["GracefulPowerOffS"]));
gracefulPowerOffTimer.async_wait([](const boost::system::error_code ec) {
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error("Graceful power-off async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
lg2::info("Graceful power-off timer canceled");
return;
}
lg2::info("Graceful power-off timer completed");
sendPowerControlEvent(Event::gracefulPowerOffTimerExpired);
});
}
static void powerCycleTimerStart()
{
lg2::info("Power-cycle timer started");
powerCycleTimer.expires_after(
std::chrono::milliseconds(TimerMap["PowerCycleMs"]));
powerCycleTimer.async_wait([](const boost::system::error_code ec) {
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error("Power-cycle async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
lg2::info("Power-cycle timer canceled");
return;
}
lg2::info("Power-cycle timer completed");
sendPowerControlEvent(Event::powerCycleTimerExpired);
});
}
static void psPowerOKWatchdogTimerStart()
{
lg2::info("power supply power OK watchdog timer started");
psPowerOKWatchdogTimer.expires_after(
std::chrono::milliseconds(TimerMap["PsPowerOKWatchdogMs"]));
psPowerOKWatchdogTimer.async_wait([](const boost::system::error_code ec) {
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error(
"power supply power OK watchdog async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
lg2::info("power supply power OK watchdog timer canceled");
return;
}
lg2::info("power supply power OK watchdog timer expired");
sendPowerControlEvent(Event::psPowerOKWatchdogTimerExpired);
});
}
static void warmResetCheckTimerStart()
{
lg2::info("Warm reset check timer started");
warmResetCheckTimer.expires_after(
std::chrono::milliseconds(TimerMap["WarmResetCheckMs"]));
warmResetCheckTimer.async_wait([](const boost::system::error_code ec) {
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error("Warm reset check async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
lg2::info("Warm reset check timer canceled");
return;
}
lg2::info("Warm reset check timer completed");
sendPowerControlEvent(Event::warmResetDetected);
});
}
static void pohCounterTimerStart()
{
lg2::info("POH timer started");
// Set the time-out as 1 hour, to align with POH command in ipmid
pohCounterTimer.expires_after(std::chrono::hours(1));
pohCounterTimer.async_wait([](const boost::system::error_code& ec) {
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error("POH timer async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
lg2::info("POH timer canceled");
return;
}
if (getHostState(powerState) !=
"xyz.openbmc_project.State.Host.HostState.Running")
{
return;
}
conn->async_method_call(
[](boost::system::error_code ec,
const std::variant<uint32_t>& pohCounterProperty) {
if (ec)
{
lg2::error("error getting poh counter");
return;
}
const uint32_t* pohCounter =
std::get_if<uint32_t>(&pohCounterProperty);
if (pohCounter == nullptr)
{
lg2::error("unable to read poh counter");
return;
}
conn->async_method_call(
[](boost::system::error_code ec) {
if (ec)
{
lg2::error("failed to set poh counter");
}
},
"xyz.openbmc_project.Settings",
"/xyz/openbmc_project/state/chassis0",
"org.freedesktop.DBus.Properties", "Set",
"xyz.openbmc_project.State.PowerOnHours", "POHCounter",
std::variant<uint32_t>(*pohCounter + 1));
},
"xyz.openbmc_project.Settings",
"/xyz/openbmc_project/state/chassis0",
"org.freedesktop.DBus.Properties", "Get",
"xyz.openbmc_project.State.PowerOnHours", "POHCounter");
pohCounterTimerStart();
});
}
static void currentHostStateMonitor()
{
if (getHostState(powerState) ==
"xyz.openbmc_project.State.Host.HostState.Running")
{
pohCounterTimerStart();
// Clear the restart cause set for the next restart
clearRestartCause();
}
else
{
pohCounterTimer.cancel();
// Set the restart cause set for this restart
setRestartCause();
}
static auto match = sdbusplus::bus::match_t(
*conn,
"type='signal',member='PropertiesChanged', "
"interface='org.freedesktop.DBus.Properties', "
"arg0='xyz.openbmc_project.State.Host'",
[](sdbusplus::message_t& message) {
std::string intfName;
std::map<std::string, std::variant<std::string>> properties;
try
{
message.read(intfName, properties);
}
catch (const std::exception& e)
{
lg2::error("Unable to read host state: {ERROR}", "ERROR", e);
return;
}
if (properties.empty())
{
lg2::error("ERROR: Empty PropertiesChanged signal received");
return;
}
// We only want to check for CurrentHostState
if (properties.begin()->first != "CurrentHostState")
{
return;
}
std::string* currentHostState =
std::get_if<std::string>(&(properties.begin()->second));
if (currentHostState == nullptr)
{
lg2::error("{PROPERTY} property invalid", "PROPERTY",
properties.begin()->first);
return;
}
if (*currentHostState ==
"xyz.openbmc_project.State.Host.HostState.Running")
{
pohCounterTimerStart();
// Clear the restart cause set for the next restart
clearRestartCause();
sd_journal_send("MESSAGE=Host system DC power is on",
"PRIORITY=%i", LOG_INFO,
"REDFISH_MESSAGE_ID=%s",
"OpenBMC.0.1.DCPowerOn", NULL);
}
else
{
pohCounterTimer.cancel();
// POST_COMPLETE GPIO event is not working in some platforms
// when power state is changed to OFF. This resulted in
// 'OperatingSystemState' to stay at 'Standby', even though
// system is OFF. Set 'OperatingSystemState' to 'Inactive'
// if HostState is trurned to OFF.
setOperatingSystemState(OperatingSystemStateStage::Inactive);
// Set the restart cause set for this restart
setRestartCause();
#ifdef USE_ACBOOT
resetACBootProperty();
#endif // USE_ACBOOT
sd_journal_send("MESSAGE=Host system DC power is off",
"PRIORITY=%i", LOG_INFO,
"REDFISH_MESSAGE_ID=%s",
"OpenBMC.0.1.DCPowerOff", NULL);
}
});
}
static void sioPowerGoodWatchdogTimerStart()
{
lg2::info("SIO power good watchdog timer started");
sioPowerGoodWatchdogTimer.expires_after(
std::chrono::milliseconds(TimerMap["SioPowerGoodWatchdogMs"]));
sioPowerGoodWatchdogTimer.async_wait([](const boost::system::error_code
ec) {
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error(
"SIO power good watchdog async_wait failed: {ERROR_MSG}",
"ERROR_MSG", ec.message());
}
lg2::info("SIO power good watchdog timer canceled");
return;
}
lg2::info("SIO power good watchdog timer completed");
sendPowerControlEvent(Event::sioPowerGoodWatchdogTimerExpired);
});
}
static void powerStateOn(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::psPowerOKDeAssert:
setPowerState(PowerState::off);
// DC power is unexpectedly lost, beep
beep(beepPowerFail);
break;
case Event::sioS5Assert:
setPowerState(PowerState::transitionToOff);
#if IGNORE_SOFT_RESETS_DURING_POST
// Only recognize soft resets once host gets past POST COMPLETE
if (operatingSystemState != OperatingSystemStateStage::Standby)
{
ignoreNextSoftReset = true;
}
#endif
addRestartCause(RestartCause::softReset);
break;
#if USE_PLT_RST
case Event::pltRstAssert:
#else
case Event::postCompleteDeAssert:
#endif
setPowerState(PowerState::checkForWarmReset);
#if IGNORE_SOFT_RESETS_DURING_POST
// Only recognize soft resets once host gets past POST COMPLETE
if (operatingSystemState != OperatingSystemStateStage::Standby)
{
ignoreNextSoftReset = true;
}
#endif
addRestartCause(RestartCause::softReset);
warmResetCheckTimerStart();
break;
case Event::powerButtonPressed:
setPowerState(PowerState::gracefulTransitionToOff);
gracefulPowerOffTimerStart();
break;
case Event::powerOffRequest:
setPowerState(PowerState::transitionToOff);
forcePowerOff();
break;
case Event::gracefulPowerOffRequest:
setPowerState(PowerState::gracefulTransitionToOff);
gracefulPowerOffTimerStart();
gracefulPowerOff();
break;
case Event::powerCycleRequest:
setPowerState(PowerState::transitionToCycleOff);
forcePowerOff();
break;
case Event::gracefulPowerCycleRequest:
setPowerState(PowerState::gracefulTransitionToCycleOff);
gracefulPowerOffTimerStart();
gracefulPowerOff();
break;
case Event::resetButtonPressed:
setPowerState(PowerState::checkForWarmReset);
warmResetCheckTimerStart();
break;
case Event::resetRequest:
reset();
break;
default:
lg2::info("No action taken.");
break;
}
}
static void powerStateWaitForPSPowerOK(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::psPowerOKAssert:
{
// Cancel any GPIO assertions held during the transition
gpioAssertTimer.cancel();
psPowerOKWatchdogTimer.cancel();
if (sioEnabled == true)
{
sioPowerGoodWatchdogTimerStart();
setPowerState(PowerState::waitForSIOPowerGood);
}
else
{
setPowerState(PowerState::on);
}
break;
}
case Event::psPowerOKWatchdogTimerExpired:
setPowerState(PowerState::off);
psPowerOKFailedLog();
break;
case Event::sioPowerGoodAssert:
psPowerOKWatchdogTimer.cancel();
setPowerState(PowerState::on);
break;
default:
lg2::info("No action taken.");
break;
}
}
static void powerStateWaitForSIOPowerGood(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::sioPowerGoodAssert:
sioPowerGoodWatchdogTimer.cancel();
setPowerState(PowerState::on);
break;
case Event::sioPowerGoodWatchdogTimerExpired:
setPowerState(PowerState::off);
systemPowerGoodFailedLog();
break;
default:
lg2::info("No action taken.");
break;
}
}
static void powerStateOff(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::psPowerOKAssert:
{
if (sioEnabled == true)
{
sioPowerGoodWatchdogTimerStart();
setPowerState(PowerState::waitForSIOPowerGood);
}
else
{
setPowerState(PowerState::on);
}
break;
}
case Event::sioS5DeAssert:
psPowerOKWatchdogTimerStart();
setPowerState(PowerState::waitForPSPowerOK);
break;
case Event::sioPowerGoodAssert:
setPowerState(PowerState::on);
break;
case Event::powerButtonPressed:
psPowerOKWatchdogTimerStart();
setPowerState(PowerState::waitForPSPowerOK);
break;
case Event::powerOnRequest:
psPowerOKWatchdogTimerStart();
setPowerState(PowerState::waitForPSPowerOK);
powerOn();
break;
default:
lg2::info("No action taken.");
break;
}
}
static void powerStateTransitionToOff(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::psPowerOKDeAssert:
// Cancel any GPIO assertions held during the transition
gpioAssertTimer.cancel();
setPowerState(PowerState::off);
break;
default:
lg2::info("No action taken.");
break;
}
}
static void powerStateGracefulTransitionToOff(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::psPowerOKDeAssert:
gracefulPowerOffTimer.cancel();
setPowerState(PowerState::off);
break;
case Event::gracefulPowerOffTimerExpired:
setPowerState(PowerState::on);
break;
case Event::powerOffRequest:
gracefulPowerOffTimer.cancel();
setPowerState(PowerState::transitionToOff);
forcePowerOff();
break;
case Event::powerCycleRequest:
gracefulPowerOffTimer.cancel();
setPowerState(PowerState::transitionToCycleOff);
forcePowerOff();
break;
case Event::resetRequest:
gracefulPowerOffTimer.cancel();
setPowerState(PowerState::on);
reset();
break;
default:
lg2::info("No action taken.");
break;
}
}
static void powerStateCycleOff(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::psPowerOKAssert:
{
powerCycleTimer.cancel();
if (sioEnabled == true)
{
sioPowerGoodWatchdogTimerStart();
setPowerState(PowerState::waitForSIOPowerGood);
}
else
{
setPowerState(PowerState::on);
}
break;
}
case Event::sioS5DeAssert:
powerCycleTimer.cancel();
psPowerOKWatchdogTimerStart();
setPowerState(PowerState::waitForPSPowerOK);
break;
case Event::powerButtonPressed:
powerCycleTimer.cancel();
psPowerOKWatchdogTimerStart();
setPowerState(PowerState::waitForPSPowerOK);
break;
case Event::powerCycleTimerExpired:
psPowerOKWatchdogTimerStart();
setPowerState(PowerState::waitForPSPowerOK);
powerOn();
break;
default:
lg2::info("No action taken.");
break;
}
}
static void powerStateTransitionToCycleOff(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::psPowerOKDeAssert:
// Cancel any GPIO assertions held during the transition
gpioAssertTimer.cancel();
setPowerState(PowerState::cycleOff);
powerCycleTimerStart();
break;
default:
lg2::info("No action taken.");
break;
}
}
static void powerStateGracefulTransitionToCycleOff(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::psPowerOKDeAssert:
gracefulPowerOffTimer.cancel();
setPowerState(PowerState::cycleOff);
powerCycleTimerStart();
break;
case Event::gracefulPowerOffTimerExpired:
setPowerState(PowerState::on);
break;
case Event::powerOffRequest:
gracefulPowerOffTimer.cancel();
setPowerState(PowerState::transitionToOff);
forcePowerOff();
break;
case Event::powerCycleRequest:
gracefulPowerOffTimer.cancel();
setPowerState(PowerState::transitionToCycleOff);
forcePowerOff();
break;
case Event::resetRequest:
gracefulPowerOffTimer.cancel();
setPowerState(PowerState::on);
reset();
break;
default:
lg2::info("No action taken.");
break;
}
}
static void powerStateCheckForWarmReset(const Event event)
{
logEvent(__FUNCTION__, event);
switch (event)
{
case Event::sioS5Assert:
warmResetCheckTimer.cancel();
setPowerState(PowerState::transitionToOff);
break;
case Event::warmResetDetected:
setPowerState(PowerState::on);
break;
case Event::psPowerOKDeAssert:
warmResetCheckTimer.cancel();
setPowerState(PowerState::off);
// DC power is unexpectedly lost, beep
beep(beepPowerFail);
break;
default:
lg2::info("No action taken.");
break;
}
}
static void psPowerOKHandler(bool state)
{
Event powerControlEvent = (state == powerOkConfig.polarity)
? Event::psPowerOKAssert
: Event::psPowerOKDeAssert;
sendPowerControlEvent(powerControlEvent);
}
static void sioPowerGoodHandler(bool state)
{
Event powerControlEvent = (state == sioPwrGoodConfig.polarity)
? Event::sioPowerGoodAssert
: Event::sioPowerGoodDeAssert;
sendPowerControlEvent(powerControlEvent);
}
static void sioOnControlHandler(bool state)
{
lg2::info("SIO_ONCONTROL value changed: {VALUE}", "VALUE",
static_cast<int>(state));
}
static void sioS5Handler(bool state)
{
Event powerControlEvent = (state == sioS5Config.polarity)
? Event::sioS5Assert
: Event::sioS5DeAssert;
sendPowerControlEvent(powerControlEvent);
}
static void powerButtonHandler(bool state)
{
bool asserted = state == powerButtonConfig.polarity;
powerButtonIface->set_property("ButtonPressed", asserted);
if (asserted)
{
powerButtonPressLog();
if (!powerButtonMask)
{
sendPowerControlEvent(Event::powerButtonPressed);
addRestartCause(RestartCause::powerButton);
}
else
{
lg2::info("power button press masked");
}
}
#if USE_BUTTON_PASSTHROUGH
gpiod::line gpioLine;
bool outputState =
asserted ? powerOutConfig.polarity : (!powerOutConfig.polarity);
if (!setGPIOOutput(powerOutConfig.lineName, outputState, gpioLine))
{
lg2::error("{GPIO_NAME} power button passthrough failed", "GPIO_NAME",
powerOutConfig.lineName);
}
#endif
}
static void resetButtonHandler(bool state)
{
bool asserted = state == resetButtonConfig.polarity;
resetButtonIface->set_property("ButtonPressed", asserted);
if (asserted)
{
resetButtonPressLog();
if (!resetButtonMask)
{
sendPowerControlEvent(Event::resetButtonPressed);
addRestartCause(RestartCause::resetButton);
}
else
{
lg2::info("reset button press masked");
}
}
#if USE_BUTTON_PASSTHROUGH
gpiod::line gpioLine;
bool outputState =
asserted ? resetOutConfig.polarity : (!resetOutConfig.polarity);
if (!setGPIOOutput(resetOutConfig.lineName, outputState, gpioLine))
{
lg2::error("{GPIO_NAME} reset button passthrough failed", "GPIO_NAME",
resetOutConfig.lineName);
}
#endif
}
#ifdef CHASSIS_SYSTEM_RESET
static constexpr auto systemdBusname = "org.freedesktop.systemd1";
static constexpr auto systemdPath = "/org/freedesktop/systemd1";
static constexpr auto systemdInterface = "org.freedesktop.systemd1.Manager";
static constexpr auto systemTargetName = "chassis-system-reset.target";
void systemReset()
{
conn->async_method_call(
[](boost::system::error_code ec) {
if (ec)
{
lg2::error("Failed to call chassis system reset: {ERR}", "ERR",
ec.message());
}
},
systemdBusname, systemdPath, systemdInterface, "StartUnit",
systemTargetName, "replace");
}
#endif
static void nmiSetEnableProperty(bool value)
{
conn->async_method_call(
[](boost::system::error_code ec) {
if (ec)
{
lg2::error("failed to set NMI source");
}
},
"xyz.openbmc_project.Settings",
"/xyz/openbmc_project/Chassis/Control/NMISource",
"org.freedesktop.DBus.Properties", "Set",
"xyz.openbmc_project.Chassis.Control.NMISource", "Enabled",
std::variant<bool>{value});
}
static void nmiReset(void)
{
const static constexpr int nmiOutPulseTimeMs = 200;
lg2::info("NMI out action");
nmiOutLine.set_value(nmiOutConfig.polarity);
lg2::info("{GPIO_NAME} set to {GPIO_VALUE}", "GPIO_NAME",
nmiOutConfig.lineName, "GPIO_VALUE", nmiOutConfig.polarity);
gpioAssertTimer.expires_after(std::chrono::milliseconds(nmiOutPulseTimeMs));
gpioAssertTimer.async_wait([](const boost::system::error_code ec) {
// restore the NMI_OUT GPIO line back to the opposite value
nmiOutLine.set_value(!nmiOutConfig.polarity);
lg2::info("{GPIO_NAME} released", "GPIO_NAME", nmiOutConfig.lineName);
if (ec)
{
// operation_aborted is expected if timer is canceled before
// completion.
if (ec != boost::asio::error::operation_aborted)
{
lg2::error("{GPIO_NAME} async_wait failed: {ERROR_MSG}",
"GPIO_NAME", nmiOutConfig.lineName, "ERROR_MSG",
ec.message());
}
}
});
// log to redfish
nmiDiagIntLog();
lg2::info("NMI out action completed");
// reset Enable Property
nmiSetEnableProperty(false);
}
static void nmiSourcePropertyMonitor(void)
{
lg2::info("NMI Source Property Monitor");
static std::unique_ptr<sdbusplus::bus::match_t> nmiSourceMatch =
std::make_unique<sdbusplus::bus::match_t>(
*conn,
"type='signal',interface='org.freedesktop.DBus.Properties',"
"member='PropertiesChanged',"
"arg0namespace='xyz.openbmc_project.Chassis.Control.NMISource'",
[](sdbusplus::message_t& msg) {
std::string interfaceName;
boost::container::flat_map<std::string,
std::variant<bool, std::string>>
propertiesChanged;
std::string state;
bool value = true;
try
{
msg.read(interfaceName, propertiesChanged);
if (propertiesChanged.begin()->first == "Enabled")
{
value =
std::get<bool>(propertiesChanged.begin()->second);
lg2::info(
"NMI Enabled propertiesChanged value: {VALUE}",
"VALUE", value);
nmiEnabled = value;
if (nmiEnabled)
{
nmiReset();
}
}
}
catch (const std::exception& e)
{
lg2::error("Unable to read NMI source: {ERROR}", "ERROR",
e);
return;
}
});
}
static void setNmiSource()
{
conn->async_method_call(
[](boost::system::error_code ec) {
if (ec)
{
lg2::error("failed to set NMI source");
}
},
"xyz.openbmc_project.Settings",
"/xyz/openbmc_project/Chassis/Control/NMISource",
"org.freedesktop.DBus.Properties", "Set",
"xyz.openbmc_project.Chassis.Control.NMISource", "BMCSource",
std::variant<std::string>{
"xyz.openbmc_project.Chassis.Control.NMISource.BMCSourceSignal.FrontPanelButton"});
// set Enable Property
nmiSetEnableProperty(true);
}
static void nmiButtonHandler(bool state)
{
// Don't handle event if host not running and config doesn't force it
if (!nmiWhenPoweredOff &&
getHostState(powerState) !=
"xyz.openbmc_project.State.Host.HostState.Running")
{
return;
}
bool asserted = state == nmiButtonConfig.polarity;
nmiButtonIface->set_property("ButtonPressed", asserted);
if (asserted)
{
nmiButtonPressLog();
if (nmiButtonMasked)
{
lg2::info("NMI button press masked");
}
else
{
setNmiSource();
}
}
}
static void idButtonHandler(bool state)
{
bool asserted = state == idButtonConfig.polarity;
idButtonIface->set_property("ButtonPressed", asserted);
}
static void pltRstHandler(bool pltRst)
{
if (pltRst)
{
sendPowerControlEvent(Event::pltRstDeAssert);
}
else
{
sendPowerControlEvent(Event::pltRstAssert);
}
}
[[maybe_unused]] static void hostMiscHandler(sdbusplus::message_t& msg)
{
std::string interfaceName;
boost::container::flat_map<std::string, std::variant<bool>>
propertiesChanged;
try
{
msg.read(interfaceName, propertiesChanged);
}
catch (const std::exception& e)
{
lg2::error("Unable to read Host Misc status: {ERROR}", "ERROR", e);
return;
}
if (propertiesChanged.empty())
{
lg2::error("ERROR: Empty Host.Misc PropertiesChanged signal received");
return;
}
for (auto& [property, value] : propertiesChanged)
{
if (property == "ESpiPlatformReset")
{
bool* pltRst = std::get_if<bool>(&value);
if (pltRst == nullptr)
{
lg2::error("{PROPERTY} property invalid", "PROPERTY", property);
return;
}
pltRstHandler(*pltRst);
}
}
}
static void postCompleteHandler(bool state)
{
bool asserted = state == postCompleteConfig.polarity;
if (asserted)
{
sendPowerControlEvent(Event::postCompleteAssert);
setOperatingSystemState(OperatingSystemStateStage::Standby);
}
else
{
sendPowerControlEvent(Event::postCompleteDeAssert);
setOperatingSystemState(OperatingSystemStateStage::Inactive);
}
}
static int loadConfigValues()
{
const std::string configFilePath =
"/usr/share/x86-power-control/power-config-host" + power_control::node +
".json";
std::ifstream configFile(configFilePath.c_str());
if (!configFile.is_open())
{
lg2::error("loadConfigValues: Cannot open config path \'{PATH}\'",
"PATH", configFilePath);
return -1;
}
auto jsonData = nlohmann::json::parse(configFile, nullptr, true, true);
if (jsonData.is_discarded())
{
lg2::error("Power config readings JSON parser failure");
return -1;
}
auto gpios = jsonData["gpio_configs"];
auto timers = jsonData["timing_configs"];
ConfigData* tempGpioData;
for (nlohmann::json& gpioConfig : gpios)
{
if (!gpioConfig.contains("Name"))
{
lg2::error("The 'Name' field must be defined in Json file");
return -1;
}
// Iterate through the powersignal map to check if the gpio json config
// entry is valid
std::string gpioName = gpioConfig["Name"];
auto signalMapIter = powerSignalMap.find(gpioName);
if (signalMapIter == powerSignalMap.end())
{
lg2::error(
"{GPIO_NAME} is not a recognized power-control signal name",
"GPIO_NAME", gpioName);
return -1;
}
// assign the power signal name to the corresponding structure reference
// from map then fillup the structure with coressponding json config
// value
tempGpioData = signalMapIter->second;
tempGpioData->name = gpioName;
if (!gpioConfig.contains("Type"))
{
lg2::error("The \'Type\' field must be defined in Json file");
return -1;
}
std::string signalType = gpioConfig["Type"];
if (signalType == "GPIO")
{
tempGpioData->type = ConfigType::GPIO;
}
else if (signalType == "DBUS")
{
tempGpioData->type = ConfigType::DBUS;
}
else
{
lg2::error("{TYPE} is not a recognized power-control signal type",
"TYPE", signalType);
return -1;
}
if (tempGpioData->type == ConfigType::GPIO)
{
if (gpioConfig.contains("LineName"))
{
tempGpioData->lineName = gpioConfig["LineName"];
}
else
{
lg2::error(
"The \'LineName\' field must be defined for GPIO configuration");
return -1;
}
if (gpioConfig.contains("Polarity"))
{
std::string polarity = gpioConfig["Polarity"];
if (polarity == "ActiveLow")
{
tempGpioData->polarity = false;
}
else if (polarity == "ActiveHigh")
{
tempGpioData->polarity = true;
}
else
{
lg2::error(
"Polarity defined but not properly setup. Please only ActiveHigh or ActiveLow. Currently set to {POLARITY}",
"POLARITY", polarity);
return -1;
}
}
else
{
lg2::error("Polarity field not found for {GPIO_NAME}",
"GPIO_NAME", tempGpioData->lineName);
return -1;
}
}
else
{
// if dbus based gpio config is defined read and update the dbus
// params corresponding to the gpio config instance
for (auto& [key, dbusParamName] : dbusParams)
{
if (!gpioConfig.contains(dbusParamName))
{
lg2::error(
"The {DBUS_NAME} field must be defined for Dbus configuration ",
"DBUS_NAME", dbusParamName);
return -1;
}
}
tempGpioData->dbusName =
gpioConfig[dbusParams[DbusConfigType::name]];
tempGpioData->path = gpioConfig[dbusParams[DbusConfigType::path]];
tempGpioData->interface =
gpioConfig[dbusParams[DbusConfigType::interface]];
tempGpioData->lineName =
gpioConfig[dbusParams[DbusConfigType::property]];
// dbus-based inputs must be active-high.
tempGpioData->polarity = true;
// MatchRegex is optional
auto item = gpioConfig.find("MatchRegex");
if (item != gpioConfig.end())
{
try
{
tempGpioData->matchRegex = std::regex(*item);
}
catch (const std::regex_error& e)
{
lg2::error("Invalid MatchRegex for {NAME}: {ERR}", "NAME",
gpioName, "ERR", e.what());
return -1;
}
}
}
}
// read and store the timer values from json config to Timer Map
for (auto& [key, timerValue] : TimerMap)
{
if (timers.contains(key.c_str()))
{
timerValue = timers[key.c_str()];
}
}
// If "events_configs" key is not in json config, fallback to null
auto events = jsonData.value("event_configs",
nlohmann::json(nlohmann::json::value_t::null));
if (events.is_object())
{
nmiWhenPoweredOff = events.value("NMIWhenPoweredOff", true);
}
return 0;
}
template <typename T>
static std::optional<T>
getMessageValue(sdbusplus::message_t& msg, const std::string& name)
{
std::string event;
std::string thresholdInterface;
boost::container::flat_map<std::string, std::variant<T>> propertiesChanged;
msg.read(thresholdInterface, propertiesChanged);
if (propertiesChanged.empty())
{
return std::nullopt;
}
event = propertiesChanged.begin()->first;
if (event.empty() || event != name)
{
return std::nullopt;
}
return std::get<T>(propertiesChanged.begin()->second);
}
static bool getDbusMsgGPIOState(sdbusplus::message_t& msg,
const ConfigData& config, bool& value)
{
try
{
if (config.matchRegex.has_value())
{
std::optional<std::string> s =
getMessageValue<std::string>(msg, config.lineName);
if (!s.has_value())
{
return false;
}
std::smatch m;
value = std::regex_match(s.value(), m, config.matchRegex.value());
}
else
{
std::optional<bool> v = getMessageValue<bool>(msg, config.lineName);
if (!v.has_value())
{
return false;
}
value = v.value();
}
return true;
}
catch (const std::exception& e)
{
lg2::error(
"exception while reading dbus property \'{DBUS_NAME}\': {ERROR}",
"DBUS_NAME", config.lineName, "ERROR", e);
return false;
}
}
static sdbusplus::bus::match_t
dbusGPIOMatcher(const ConfigData& cfg, std::function<void(bool)> onMatch)
{
auto pulseEventMatcherCallback =
[&cfg, onMatch](sdbusplus::message_t& msg) {
bool value = false;
if (!getDbusMsgGPIOState(msg, cfg, value))
{
return;
}
onMatch(value);
};
return sdbusplus::bus::match_t(
static_cast<sdbusplus::bus_t&>(*conn),
"type='signal',interface='org.freedesktop.DBus.Properties',member='"
"PropertiesChanged',arg0='" +
cfg.interface + "',path='" + cfg.path + "',sender='" +
cfg.dbusName + "'",
std::move(pulseEventMatcherCallback));
}
// D-Bus property read functions
void reschedulePropertyRead(const ConfigData& configData);
int getProperty(const ConfigData& configData)
{
std::variant<bool> resp;
try
{
auto method = conn->new_method_call(
configData.dbusName.c_str(), configData.path.c_str(),
"org.freedesktop.DBus.Properties", "Get");
method.append(configData.interface.c_str(),
configData.lineName.c_str());
auto reply = conn->call(method);
if (reply.is_method_error())
{
lg2::error(
"Error reading {PROPERTY} D-Bus property on interface {INTERFACE} and path {PATH}",
"PROPERTY", configData.lineName, "INTERFACE",
configData.interface, "PATH", configData.path);
return -1;
}
reply.read(resp);
}
catch (const sdbusplus::exception_t& e)
{
lg2::error("Exception while reading {PROPERTY}: {WHAT}", "PROPERTY",
configData.lineName, "WHAT", e.what());
reschedulePropertyRead(configData);
return -1;
}
auto respValue = std::get_if<bool>(&resp);
if (!respValue)
{
lg2::error("Error: {PROPERTY} D-Bus property is not the expected type",
"PROPERTY", configData.lineName);
return -1;
}
return (*respValue);
}
void setInitialValue(const ConfigData& configData, bool initialValue)
{
if (configData.name == "PowerOk")
{
powerState = (initialValue ? PowerState::on : PowerState::off);
hostIface->set_property("CurrentHostState",
std::string(getHostState(powerState)));
}
else if (configData.name == "PowerButton")
{
powerButtonIface->set_property("ButtonPressed", !initialValue);
}
else if (configData.name == "ResetButton")
{
resetButtonIface->set_property("ButtonPressed", !initialValue);
}
else if (configData.name == "NMIButton")
{
nmiButtonIface->set_property("ButtonPressed", !initialValue);
}
else if (configData.name == "IdButton")
{
idButtonIface->set_property("ButtonPressed", !initialValue);
}
else if (configData.name == "PostComplete")
{
OperatingSystemStateStage osState =
(initialValue == postCompleteConfig.polarity
? OperatingSystemStateStage::Standby
: OperatingSystemStateStage::Inactive);
setOperatingSystemState(osState);
}
else
{
lg2::error("Unknown name {NAME}", "NAME", configData.name);
}
}
void reschedulePropertyRead(const ConfigData& configData)
{
auto item = dBusRetryTimers.find(configData.name);
if (item == dBusRetryTimers.end())
{
auto newItem = dBusRetryTimers.insert(
{configData.name, boost::asio::steady_timer(io)});
if (!newItem.second)
{
lg2::error("Failed to add new timer for {NAME}", "NAME",
configData.name);
return;
}
item = newItem.first;
}
auto& timer = item->second;
timer.expires_after(
std::chrono::milliseconds(TimerMap["DbusGetPropertyRetry"]));
timer.async_wait([&configData](const boost::system::error_code ec) {
if (ec)
{
lg2::error("Retry timer for {NAME} failed: {MSG}", "NAME",
configData.name, "MSG", ec.message());
dBusRetryTimers.erase(configData.name);
return;
}
int property = getProperty(configData);
if (property >= 0)
{
setInitialValue(configData, (property > 0));
dBusRetryTimers.erase(configData.name);
}
});
}
} // namespace power_control
int main(int argc, char* argv[])
{
using namespace power_control;
if (argc > 1)
{
node = argv[1];
}
lg2::info("Start Chassis power control service for host : {NODE}", "NODE",
node);
conn = std::make_shared<sdbusplus::asio::connection>(io);
// Load GPIO's through json config file
if (loadConfigValues() == -1)
{
lg2::error("Host{NODE}: Error in Parsing...", "NODE", node);
}
/* Currently for single host based systems additional busname is added
with "0" at the end of the name ex : xyz.openbmc_project.State.Host0.
Going forward for single hosts the old bus name without zero numbering
will be removed when all other applications adapted to the
bus name with zero numbering (xyz.openbmc_project.State.Host0). */
if (node == "0")
{
// Request all the dbus names
conn->request_name(hostDbusName.c_str());
conn->request_name(chassisDbusName.c_str());
conn->request_name(osDbusName.c_str());
conn->request_name(buttonDbusName.c_str());
conn->request_name(nmiDbusName.c_str());
conn->request_name(rstCauseDbusName.c_str());
}
hostDbusName += node;
chassisDbusName += node;
osDbusName += node;
buttonDbusName += node;
nmiDbusName += node;
rstCauseDbusName += node;
// Request all the dbus names
conn->request_name(hostDbusName.c_str());
conn->request_name(chassisDbusName.c_str());
conn->request_name(osDbusName.c_str());
conn->request_name(buttonDbusName.c_str());
conn->request_name(nmiDbusName.c_str());
conn->request_name(rstCauseDbusName.c_str());
if (sioPwrGoodConfig.lineName.empty() ||
sioOnControlConfig.lineName.empty() || sioS5Config.lineName.empty())
{
sioEnabled = false;
lg2::info("SIO control GPIOs not defined, disable SIO support.");
}
// Request PS_PWROK GPIO events
if (powerOkConfig.type == ConfigType::GPIO)
{
if (!requestGPIOEvents(powerOkConfig.lineName, psPowerOKHandler,
psPowerOKLine, psPowerOKEvent))
{
return -1;
}
}
else if (powerOkConfig.type == ConfigType::DBUS)
{
static sdbusplus::bus::match_t powerOkEventMonitor =
power_control::dbusGPIOMatcher(powerOkConfig, psPowerOKHandler);
}
else
{
lg2::error("PowerOk name should be configured from json config file");
return -1;
}
if (sioEnabled == true)
{
// Request SIO_POWER_GOOD GPIO events
if (sioPwrGoodConfig.type == ConfigType::GPIO)
{
if (!requestGPIOEvents(sioPwrGoodConfig.lineName,
sioPowerGoodHandler, sioPowerGoodLine,
sioPowerGoodEvent))
{
return -1;
}
}
else if (sioPwrGoodConfig.type == ConfigType::DBUS)
{
static sdbusplus::bus::match_t sioPwrGoodEventMonitor =
power_control::dbusGPIOMatcher(sioPwrGoodConfig,
sioPowerGoodHandler);
}
else
{
lg2::error(
"sioPwrGood name should be configured from json config file");
return -1;
}
// Request SIO_ONCONTROL GPIO events
if (sioOnControlConfig.type == ConfigType::GPIO)
{
if (!requestGPIOEvents(sioOnControlConfig.lineName,
sioOnControlHandler, sioOnControlLine,
sioOnControlEvent))
{
return -1;
}
}
else if (sioOnControlConfig.type == ConfigType::DBUS)
{
static sdbusplus::bus::match_t sioOnControlEventMonitor =
power_control::dbusGPIOMatcher(sioOnControlConfig,
sioOnControlHandler);
}
else
{
lg2::error(
"sioOnControl name should be configured from jsonconfig file\n");
return -1;
}
// Request SIO_S5 GPIO events
if (sioS5Config.type == ConfigType::GPIO)
{
if (!requestGPIOEvents(sioS5Config.lineName, sioS5Handler,
sioS5Line, sioS5Event))
{
return -1;
}
}
else if (sioS5Config.type == ConfigType::DBUS)
{
static sdbusplus::bus::match_t sioS5EventMonitor =
power_control::dbusGPIOMatcher(sioS5Config, sioS5Handler);
}
else
{
lg2::error("sioS5 name should be configured from json config file");
return -1;
}
}
// Request POWER_BUTTON GPIO events
if (powerButtonConfig.type == ConfigType::GPIO)
{
if (!requestGPIOEvents(powerButtonConfig.lineName, powerButtonHandler,
powerButtonLine, powerButtonEvent))
{
return -1;
}
}
else if (powerButtonConfig.type == ConfigType::DBUS)
{
static sdbusplus::bus::match_t powerButtonEventMonitor =
power_control::dbusGPIOMatcher(powerButtonConfig,
powerButtonHandler);
}
// Request RESET_BUTTON GPIO events
if (resetButtonConfig.type == ConfigType::GPIO)
{
if (!requestGPIOEvents(resetButtonConfig.lineName, resetButtonHandler,
resetButtonLine, resetButtonEvent))
{
return -1;
}
}
else if (resetButtonConfig.type == ConfigType::DBUS)
{
static sdbusplus::bus::match_t resetButtonEventMonitor =
power_control::dbusGPIOMatcher(resetButtonConfig,
resetButtonHandler);
}
// Request NMI_BUTTON GPIO events
if (nmiButtonConfig.type == ConfigType::GPIO)
{
if (!nmiButtonConfig.lineName.empty())
{
requestGPIOEvents(nmiButtonConfig.lineName, nmiButtonHandler,
nmiButtonLine, nmiButtonEvent);
}
}
else if (nmiButtonConfig.type == ConfigType::DBUS)
{
static sdbusplus::bus::match_t nmiButtonEventMonitor =
power_control::dbusGPIOMatcher(nmiButtonConfig, nmiButtonHandler);
}
// Request ID_BUTTON GPIO events
if (idButtonConfig.type == ConfigType::GPIO)
{
if (!idButtonConfig.lineName.empty())
{
requestGPIOEvents(idButtonConfig.lineName, idButtonHandler,
idButtonLine, idButtonEvent);
}
}
else if (idButtonConfig.type == ConfigType::DBUS)
{
static sdbusplus::bus::match_t idButtonEventMonitor =
power_control::dbusGPIOMatcher(idButtonConfig, idButtonHandler);
}
#ifdef USE_PLT_RST
sdbusplus::bus::match_t pltRstMatch(
*conn,
"type='signal',interface='org.freedesktop.DBus.Properties',member='"
"PropertiesChanged',arg0='xyz.openbmc_project.State.Host.Misc'",
hostMiscHandler);
#endif
// Request POST_COMPLETE GPIO events
if (postCompleteConfig.type == ConfigType::GPIO)
{
if (!requestGPIOEvents(postCompleteConfig.lineName, postCompleteHandler,
postCompleteLine, postCompleteEvent))
{
return -1;
}
}
else if (postCompleteConfig.type == ConfigType::DBUS)
{
static sdbusplus::bus::match_t postCompleteEventMonitor =
power_control::dbusGPIOMatcher(postCompleteConfig,
postCompleteHandler);
}
else
{
lg2::error(
"postComplete name should be configured from json config file");
return -1;
}
// initialize NMI_OUT GPIO.
if (!nmiOutConfig.lineName.empty())
{
setGPIOOutput(nmiOutConfig.lineName, !nmiOutConfig.polarity,
nmiOutLine);
}
// Initialize POWER_OUT and RESET_OUT GPIO.
gpiod::line line;
if (!powerOutConfig.lineName.empty())
{
if (!setGPIOOutput(powerOutConfig.lineName, !powerOutConfig.polarity,
line))
{
return -1;
}
}
else
{
lg2::error("powerOut name should be configured from json config file");
return -1;
}
if (!resetOutConfig.lineName.empty())
{
if (!setGPIOOutput(resetOutConfig.lineName, !resetOutConfig.polarity,
line))
{
return -1;
}
}
else
{
lg2::error("ResetOut name should be configured from json config file");
return -1;
}
// Release line
line.reset();
// Initialize the power state and operating system state
powerState = PowerState::off;
operatingSystemState = OperatingSystemStateStage::Inactive;
// Check power good
if (powerOkConfig.type == ConfigType::GPIO)
{
if (psPowerOKLine.get_value() > 0 ||
(sioEnabled &&
(sioPowerGoodLine.get_value() == sioPwrGoodConfig.polarity)))
{
powerState = PowerState::on;
}
}
else
{
if (getProperty(powerOkConfig))
{
powerState = PowerState::on;
}
}
// Check if we need to start the Power Restore policy
if (powerState != PowerState::on)
{
powerRestore.run();
}
if (nmiOutLine)
nmiSourcePropertyMonitor();
lg2::info("Initializing power state.");
logStateTransition(powerState);
// Power Control Service
sdbusplus::asio::object_server hostServer =
sdbusplus::asio::object_server(conn);
// Power Control Interface
hostIface =
hostServer.add_interface("/xyz/openbmc_project/state/host" + node,
"xyz.openbmc_project.State.Host");
// Interface for IPMI/Redfish initiated host state transitions
hostIface->register_property(
"RequestedHostTransition",
std::string("xyz.openbmc_project.State.Host.Transition.Off"),
[](const std::string& requested, std::string& resp) {
if (requested == "xyz.openbmc_project.State.Host.Transition.Off")
{
// if power button is masked, ignore this
if (!powerButtonMask)
{
sendPowerControlEvent(Event::gracefulPowerOffRequest);
addRestartCause(RestartCause::command);
}
else
{
lg2::info("Power Button Masked.");
throw std::invalid_argument("Transition Request Masked");
return 0;
}
}
else if (requested ==
"xyz.openbmc_project.State.Host.Transition.On")
{
// if power button is masked, ignore this
if (!powerButtonMask)
{
sendPowerControlEvent(Event::powerOnRequest);
addRestartCause(RestartCause::command);
}
else
{
lg2::info("Power Button Masked.");
throw std::invalid_argument("Transition Request Masked");
return 0;
}
}
else if (requested ==
"xyz.openbmc_project.State.Host.Transition.Reboot")
{
// if power button is masked, ignore this
if (!powerButtonMask)
{
sendPowerControlEvent(Event::powerCycleRequest);
addRestartCause(RestartCause::command);
}
else
{
lg2::info("Power Button Masked.");
throw std::invalid_argument("Transition Request Masked");
return 0;
}
}
else if (
requested ==
"xyz.openbmc_project.State.Host.Transition.GracefulWarmReboot")
{
// if reset button is masked, ignore this
if (!resetButtonMask)
{
sendPowerControlEvent(Event::gracefulPowerCycleRequest);
addRestartCause(RestartCause::command);
}
else
{
lg2::info("Reset Button Masked.");
throw std::invalid_argument("Transition Request Masked");
return 0;
}
}
else if (
requested ==
"xyz.openbmc_project.State.Host.Transition.ForceWarmReboot")
{
// if reset button is masked, ignore this
if (!resetButtonMask)
{
sendPowerControlEvent(Event::resetRequest);
addRestartCause(RestartCause::command);
}
else
{
lg2::info("Reset Button Masked.");
throw std::invalid_argument("Transition Request Masked");
return 0;
}
}
else
{
lg2::error("Unrecognized host state transition request.");
throw std::invalid_argument("Unrecognized Transition Request");
return 0;
}
resp = requested;
return 1;
});
hostIface->register_property("CurrentHostState",
std::string(getHostState(powerState)));
hostIface->initialize();
// Chassis Control Service
sdbusplus::asio::object_server chassisServer =
sdbusplus::asio::object_server(conn);
// Chassis Control Interface
chassisIface =
chassisServer.add_interface("/xyz/openbmc_project/state/chassis" + node,
"xyz.openbmc_project.State.Chassis");
chassisIface->register_property(
"RequestedPowerTransition",
std::string("xyz.openbmc_project.State.Chassis.Transition.Off"),
[](const std::string& requested, std::string& resp) {
if (requested == "xyz.openbmc_project.State.Chassis.Transition.Off")
{
// if power button is masked, ignore this
if (!powerButtonMask)
{
sendPowerControlEvent(Event::powerOffRequest);
addRestartCause(RestartCause::command);
}
else
{
lg2::info("Power Button Masked.");
throw std::invalid_argument("Transition Request Masked");
return 0;
}
}
else if (requested ==
"xyz.openbmc_project.State.Chassis.Transition.On")
{
// if power button is masked, ignore this
if (!powerButtonMask)
{
sendPowerControlEvent(Event::powerOnRequest);
addRestartCause(RestartCause::command);
}
else
{
lg2::info("Power Button Masked.");
throw std::invalid_argument("Transition Request Masked");
return 0;
}
}
else if (requested ==
"xyz.openbmc_project.State.Chassis.Transition.PowerCycle")
{
// if power button is masked, ignore this
if (!powerButtonMask)
{
sendPowerControlEvent(Event::powerCycleRequest);
addRestartCause(RestartCause::command);
}
else
{
lg2::info("Power Button Masked.");
throw std::invalid_argument("Transition Request Masked");
return 0;
}
}
else
{
lg2::error("Unrecognized chassis state transition request.");
throw std::invalid_argument("Unrecognized Transition Request");
return 0;
}
resp = requested;
return 1;
});
chassisIface->register_property("CurrentPowerState",
std::string(getChassisState(powerState)));
chassisIface->register_property("LastStateChangeTime", getCurrentTimeMs());
chassisIface->initialize();
#ifdef CHASSIS_SYSTEM_RESET
// Chassis System Service
sdbusplus::asio::object_server chassisSysServer =
sdbusplus::asio::object_server(conn);
// Chassis System Interface
chassisSysIface = chassisSysServer.add_interface(
"/xyz/openbmc_project/state/chassis_system0",
"xyz.openbmc_project.State.Chassis");
chassisSysIface->register_property(
"RequestedPowerTransition",
std::string("xyz.openbmc_project.State.Chassis.Transition.On"),
[](const std::string& requested, std::string& resp) {
if (requested ==
"xyz.openbmc_project.State.Chassis.Transition.PowerCycle")
{
systemReset();
addRestartCause(RestartCause::command);
}
else
{
lg2::error(
"Unrecognized chassis system state transition request.");
throw std::invalid_argument("Unrecognized Transition Request");
return 0;
}
resp = requested;
return 1;
});
chassisSysIface->register_property(
"CurrentPowerState", std::string(getChassisState(powerState)));
chassisSysIface->register_property("LastStateChangeTime",
getCurrentTimeMs());
chassisSysIface->initialize();
if (!slotPowerConfig.lineName.empty())
{
if (!setGPIOOutput(slotPowerConfig.lineName, 1, slotPowerLine))
{
return -1;
}
slotPowerState = SlotPowerState::off;
if (slotPowerLine.get_value() > 0)
{
slotPowerState = SlotPowerState::on;
}
chassisSlotIface = chassisSysServer.add_interface(
"/xyz/openbmc_project/state/chassis_system" + node,
"xyz.openbmc_project.State.Chassis");
chassisSlotIface->register_property(
"RequestedPowerTransition",
std::string("xyz.openbmc_project.State.Chassis.Transition.On"),
[](const std::string& requested, std::string& resp) {
if (requested ==
"xyz.openbmc_project.State.Chassis.Transition.On")
{
slotPowerOn();
}
else if (requested ==
"xyz.openbmc_project.State.Chassis.Transition.Off")
{
slotPowerOff();
}
else if (
requested ==
"xyz.openbmc_project.State.Chassis.Transition.PowerCycle")
{
slotPowerCycle();
}
else
{
lg2::error(
"Unrecognized chassis system state transition request.\n");
throw std::invalid_argument(
"Unrecognized Transition Request");
return 0;
}
resp = requested;
return 1;
});
chassisSlotIface->register_property(
"CurrentPowerState", std::string(getSlotState(slotPowerState)));
chassisSlotIface->register_property("LastStateChangeTime",
getCurrentTimeMs());
chassisSlotIface->initialize();
}
#endif
// Buttons Service
sdbusplus::asio::object_server buttonsServer =
sdbusplus::asio::object_server(conn);
if (!powerButtonConfig.lineName.empty())
{
// Power Button Interface
power_control::powerButtonIface = buttonsServer.add_interface(
"/xyz/openbmc_project/chassis/buttons/power",
"xyz.openbmc_project.Chassis.Buttons");
powerButtonIface->register_property(
"ButtonMasked", false, [](const bool requested, bool& current) {
if (requested)
{
if (powerButtonMask)
{
return 1;
}
if (!setGPIOOutput(powerOutConfig.lineName,
!powerOutConfig.polarity,
powerButtonMask))
{
throw std::runtime_error("Failed to request GPIO");
return 0;
}
lg2::info("Power Button Masked.");
}
else
{
if (!powerButtonMask)
{
return 1;
}
lg2::info("Power Button Un-masked");
powerButtonMask.reset();
}
// Update the mask setting
current = requested;
return 1;
});
// Check power button state
bool powerButtonPressed;
if (powerButtonConfig.type == ConfigType::GPIO)
{
powerButtonPressed = powerButtonLine.get_value() == 0;
}
else
{
powerButtonPressed = getProperty(powerButtonConfig) == 0;
}
powerButtonIface->register_property("ButtonPressed",
powerButtonPressed);
powerButtonIface->initialize();
}
if (!resetButtonConfig.lineName.empty())
{
// Reset Button Interface
resetButtonIface = buttonsServer.add_interface(
"/xyz/openbmc_project/chassis/buttons/reset",
"xyz.openbmc_project.Chassis.Buttons");
resetButtonIface->register_property(
"ButtonMasked", false, [](const bool requested, bool& current) {
if (requested)
{
if (resetButtonMask)
{
return 1;
}
if (!setGPIOOutput(resetOutConfig.lineName,
!resetOutConfig.polarity,
resetButtonMask))
{
throw std::runtime_error("Failed to request GPIO");
return 0;
}
lg2::info("Reset Button Masked.");
}
else
{
if (!resetButtonMask)
{
return 1;
}
lg2::info("Reset Button Un-masked");
resetButtonMask.reset();
}
// Update the mask setting
current = requested;
return 1;
});
// Check reset button state
bool resetButtonPressed;
if (resetButtonConfig.type == ConfigType::GPIO)
{
resetButtonPressed = resetButtonLine.get_value() == 0;
}
else
{
resetButtonPressed = getProperty(resetButtonConfig) == 0;
}
resetButtonIface->register_property("ButtonPressed",
resetButtonPressed);
resetButtonIface->initialize();
}
if (nmiButtonLine)
{
// NMI Button Interface
nmiButtonIface = buttonsServer.add_interface(
"/xyz/openbmc_project/chassis/buttons/nmi",
"xyz.openbmc_project.Chassis.Buttons");
nmiButtonIface->register_property(
"ButtonMasked", false, [](const bool requested, bool& current) {
if (nmiButtonMasked == requested)
{
// NMI button mask is already set as requested, so no change
return 1;
}
if (requested)
{
lg2::info("NMI Button Masked.");
nmiButtonMasked = true;
}
else
{
lg2::info("NMI Button Un-masked.");
nmiButtonMasked = false;
}
// Update the mask setting
current = nmiButtonMasked;
return 1;
});
// Check NMI button state
bool nmiButtonPressed;
if (nmiButtonConfig.type == ConfigType::GPIO)
{
nmiButtonPressed = nmiButtonLine.get_value() == 0;
}
else
{
nmiButtonPressed = getProperty(nmiButtonConfig) == 0;
}
nmiButtonIface->register_property("ButtonPressed", nmiButtonPressed);
nmiButtonIface->initialize();
}
if (nmiOutLine)
{
// NMI out Service
sdbusplus::asio::object_server nmiOutServer =
sdbusplus::asio::object_server(conn);
// NMI out Interface
nmiOutIface = nmiOutServer.add_interface(
"/xyz/openbmc_project/control/host" + node + "/nmi",
"xyz.openbmc_project.Control.Host.NMI");
nmiOutIface->register_method("NMI", nmiReset);
nmiOutIface->initialize();
}
if (idButtonLine)
{
// ID Button Interface
idButtonIface = buttonsServer.add_interface(
"/xyz/openbmc_project/chassis/buttons/id",
"xyz.openbmc_project.Chassis.Buttons");
// Check ID button state
bool idButtonPressed;
if (idButtonConfig.type == ConfigType::GPIO)
{
idButtonPressed = idButtonLine.get_value() == 0;
}
else
{
idButtonPressed = getProperty(idButtonConfig) == 0;
}
idButtonIface->register_property("ButtonPressed", idButtonPressed);
idButtonIface->initialize();
}
// OS State Service
sdbusplus::asio::object_server osServer =
sdbusplus::asio::object_server(conn);
// OS State Interface
osIface = osServer.add_interface(
"/xyz/openbmc_project/state/host" + node,
"xyz.openbmc_project.State.OperatingSystem.Status");
// Get the initial OS state based on POST complete
// Asserted, OS state is "Standby" (ready to boot)
// De-Asserted, OS state is "Inactive"
OperatingSystemStateStage osState;
if (postCompleteConfig.type == ConfigType::GPIO)
{
osState = postCompleteLine.get_value() == postCompleteConfig.polarity
? OperatingSystemStateStage::Standby
: OperatingSystemStateStage::Inactive;
}
else
{
osState = getProperty(postCompleteConfig) > 0
? OperatingSystemStateStage::Standby
: OperatingSystemStateStage::Inactive;
}
osIface->register_property(
"OperatingSystemState",
std::string(getOperatingSystemStateStage(osState)));
osIface->initialize();
// Restart Cause Service
sdbusplus::asio::object_server restartCauseServer =
sdbusplus::asio::object_server(conn);
// Restart Cause Interface
restartCauseIface = restartCauseServer.add_interface(
"/xyz/openbmc_project/control/host" + node + "/restart_cause",
"xyz.openbmc_project.Control.Host.RestartCause");
restartCauseIface->register_property(
"RestartCause",
std::string("xyz.openbmc_project.State.Host.RestartCause.Unknown"));
restartCauseIface->register_property(
"RequestedRestartCause",
std::string("xyz.openbmc_project.State.Host.RestartCause.Unknown"),
[](const std::string& requested, std::string& resp) {
if (requested ==
"xyz.openbmc_project.State.Host.RestartCause.WatchdogTimer")
{
addRestartCause(RestartCause::watchdog);
}
else
{
throw std::invalid_argument(
"Unrecognized RestartCause Request");
return 0;
}
lg2::info("RestartCause requested: {RESTART_CAUSE}",
"RESTART_CAUSE", requested);
resp = requested;
return 1;
});
restartCauseIface->initialize();
currentHostStateMonitor();
if (!hpmStbyEnConfig.lineName.empty())
{
// Set to indicate BMC's power control module is ready to take
// the inputs [PWR_GOOD] from the HPM FPGA
gpiod::line hpmLine;
if (!setGPIOOutput(hpmStbyEnConfig.lineName, hpmStbyEnConfig.polarity,
hpmLine))
{
return -1;
}
}
io.run();
return 0;
}