| #include "config.h" |
| |
| #include "button_handler.hpp" |
| |
| #include <phosphor-logging/lg2.hpp> |
| #include <xyz/openbmc_project/State/Chassis/server.hpp> |
| #include <xyz/openbmc_project/State/Host/server.hpp> |
| namespace phosphor |
| { |
| namespace button |
| { |
| |
| namespace sdbusRule = sdbusplus::bus::match::rules; |
| using namespace sdbusplus::xyz::openbmc_project::State::server; |
| |
| constexpr auto chassisIface = "xyz.openbmc_project.State.Chassis"; |
| constexpr auto hostIface = "xyz.openbmc_project.State.Host"; |
| constexpr auto powerButtonIface = "xyz.openbmc_project.Chassis.Buttons.Power"; |
| constexpr auto idButtonIface = "xyz.openbmc_project.Chassis.Buttons.ID"; |
| constexpr auto resetButtonIface = "xyz.openbmc_project.Chassis.Buttons.Reset"; |
| constexpr auto ledGroupIface = "xyz.openbmc_project.Led.Group"; |
| constexpr auto ledGroupBasePath = "/xyz/openbmc_project/led/groups/"; |
| constexpr auto hostSelectorIface = |
| "xyz.openbmc_project.Chassis.Buttons.HostSelector"; |
| constexpr auto debugHostSelectorIface = |
| "xyz.openbmc_project.Chassis.Buttons.DebugHostSelector"; |
| |
| constexpr auto propertyIface = "org.freedesktop.DBus.Properties"; |
| constexpr auto mapperIface = "xyz.openbmc_project.ObjectMapper"; |
| |
| constexpr auto mapperObjPath = "/xyz/openbmc_project/object_mapper"; |
| constexpr auto mapperService = "xyz.openbmc_project.ObjectMapper"; |
| constexpr auto BMC_POSITION = 0; |
| |
| Handler::Handler(sdbusplus::bus::bus& bus) : bus(bus) |
| { |
| try |
| { |
| if (!getService(POWER_DBUS_OBJECT_NAME, powerButtonIface).empty()) |
| { |
| lg2::info("Starting power button handler"); |
| powerButtonReleased = std::make_unique<sdbusplus::bus::match_t>( |
| bus, |
| sdbusRule::type::signal() + sdbusRule::member("Released") + |
| sdbusRule::path(POWER_DBUS_OBJECT_NAME) + |
| sdbusRule::interface(powerButtonIface), |
| std::bind(std::mem_fn(&Handler::powerPressed), this, |
| std::placeholders::_1)); |
| |
| powerButtonLongPressReleased = |
| std::make_unique<sdbusplus::bus::match_t>( |
| bus, |
| sdbusRule::type::signal() + |
| sdbusRule::member("PressedLong") + |
| sdbusRule::path(POWER_DBUS_OBJECT_NAME) + |
| sdbusRule::interface(powerButtonIface), |
| std::bind(std::mem_fn(&Handler::longPowerPressed), this, |
| std::placeholders::_1)); |
| } |
| } |
| catch (const sdbusplus::exception::exception& e) |
| { |
| // The button wasn't implemented |
| } |
| |
| try |
| { |
| if (!getService(ID_DBUS_OBJECT_NAME, idButtonIface).empty()) |
| { |
| lg2::info("Registering ID button handler"); |
| idButtonReleased = std::make_unique<sdbusplus::bus::match_t>( |
| bus, |
| sdbusRule::type::signal() + sdbusRule::member("Released") + |
| sdbusRule::path(ID_DBUS_OBJECT_NAME) + |
| sdbusRule::interface(idButtonIface), |
| std::bind(std::mem_fn(&Handler::idPressed), this, |
| std::placeholders::_1)); |
| } |
| } |
| catch (const sdbusplus::exception::exception& e) |
| { |
| // The button wasn't implemented |
| } |
| |
| try |
| { |
| if (!getService(RESET_DBUS_OBJECT_NAME, resetButtonIface).empty()) |
| { |
| lg2::info("Registering reset button handler"); |
| resetButtonReleased = std::make_unique<sdbusplus::bus::match_t>( |
| bus, |
| sdbusRule::type::signal() + sdbusRule::member("Released") + |
| sdbusRule::path(RESET_DBUS_OBJECT_NAME) + |
| sdbusRule::interface(resetButtonIface), |
| std::bind(std::mem_fn(&Handler::resetPressed), this, |
| std::placeholders::_1)); |
| } |
| } |
| catch (const sdbusplus::exception::exception& e) |
| { |
| // The button wasn't implemented |
| } |
| } |
| bool Handler::isMultiHost() |
| { |
| // return true in case host selector object is available |
| return (!getService(HS_DBUS_OBJECT_NAME, hostSelectorIface).empty()); |
| } |
| std::string Handler::getService(const std::string& path, |
| const std::string& interface) const |
| { |
| auto method = bus.new_method_call(mapperService, mapperObjPath, mapperIface, |
| "GetObject"); |
| method.append(path, std::vector{interface}); |
| auto result = bus.call(method); |
| |
| std::map<std::string, std::vector<std::string>> objectData; |
| result.read(objectData); |
| |
| return objectData.begin()->first; |
| } |
| size_t Handler::getHostSelectorValue() |
| { |
| auto HSService = getService(HS_DBUS_OBJECT_NAME, hostSelectorIface); |
| |
| if (HSService.empty()) |
| { |
| lg2::info("Host Selector dbus object not available"); |
| throw std::invalid_argument("Host selector dbus object not available"); |
| } |
| |
| try |
| { |
| auto method = bus.new_method_call( |
| HSService.c_str(), HS_DBUS_OBJECT_NAME, propertyIface, "Get"); |
| method.append(hostSelectorIface, "Position"); |
| auto result = bus.call(method); |
| |
| std::variant<size_t> HSPositionVariant; |
| result.read(HSPositionVariant); |
| |
| auto position = std::get<size_t>(HSPositionVariant); |
| return position; |
| } |
| catch (const sdbusplus::exception::exception& e) |
| { |
| lg2::error("Error reading Host selector Position: {ERROR}", "ERROR", e); |
| throw; |
| } |
| } |
| bool Handler::poweredOn(size_t hostNumber) const |
| { |
| auto chassisObjectName = |
| CHASSIS_STATE_OBJECT_NAME + std::to_string(hostNumber); |
| auto service = getService(chassisObjectName.c_str(), chassisIface); |
| auto method = bus.new_method_call( |
| service.c_str(), chassisObjectName.c_str(), propertyIface, "Get"); |
| method.append(chassisIface, "CurrentPowerState"); |
| auto result = bus.call(method); |
| |
| std::variant<std::string> state; |
| result.read(state); |
| |
| return Chassis::PowerState::On == |
| Chassis::convertPowerStateFromString(std::get<std::string>(state)); |
| } |
| |
| void Handler::handlePowerEvent(PowerEvent powerEventType) |
| { |
| std::string objPathName; |
| std::string dbusIfaceName; |
| std::string transitionName; |
| std::variant<Host::Transition, Chassis::Transition> transition; |
| |
| size_t hostNumber = 0; |
| auto isMultiHostSystem = isMultiHost(); |
| if (isMultiHostSystem) |
| { |
| hostNumber = getHostSelectorValue(); |
| lg2::info("Multi host system detected : {POSITION}", "POSITION", |
| hostNumber); |
| } |
| |
| std::string hostNumStr = std::to_string(hostNumber); |
| |
| // ignore power and reset button events if BMC is selected. |
| if (isMultiHostSystem && (hostNumber == BMC_POSITION) && |
| (powerEventType != PowerEvent::longPowerPressed)) |
| { |
| lg2::info( |
| "handlePowerEvent : BMC selected on multihost system. ignoring power and reset button events..."); |
| return; |
| } |
| |
| switch (powerEventType) |
| { |
| case PowerEvent::powerPressed: |
| { |
| objPathName = HOST_STATE_OBJECT_NAME + hostNumStr; |
| dbusIfaceName = hostIface; |
| transitionName = "RequestedHostTransition"; |
| |
| transition = Host::Transition::On; |
| |
| if (poweredOn(hostNumber)) |
| { |
| transition = Host::Transition::Off; |
| } |
| lg2::info("handlePowerEvent : handle power button press "); |
| |
| break; |
| } |
| case PowerEvent::longPowerPressed: |
| { |
| dbusIfaceName = chassisIface; |
| transitionName = "RequestedPowerTransition"; |
| objPathName = CHASSIS_STATE_OBJECT_NAME + hostNumStr; |
| transition = Chassis::Transition::Off; |
| |
| /* multi host system : |
| hosts (1 to N) - host shutdown |
| bmc (0) - sled cycle |
| single host system : |
| host(0) - host shutdown |
| */ |
| if (isMultiHostSystem && (hostNumber == BMC_POSITION)) |
| { |
| #if CHASSIS_SYSTEM_RESET_ENABLED |
| objPathName = CHASSISSYSTEM_STATE_OBJECT_NAME + hostNumStr; |
| transition = Chassis::Transition::PowerCycle; |
| #else |
| return; |
| #endif |
| } |
| else if (!poweredOn(hostNumber)) |
| { |
| lg2::info("Power is off so ignoring long power button press"); |
| return; |
| } |
| lg2::info("handlePowerEvent : handle long power button press"); |
| |
| break; |
| } |
| |
| case PowerEvent::resetPressed: |
| { |
| objPathName = HOST_STATE_OBJECT_NAME + hostNumStr; |
| dbusIfaceName = hostIface; |
| transitionName = "RequestedHostTransition"; |
| |
| if (!poweredOn(hostNumber)) |
| { |
| lg2::info("Power is off so ignoring reset button press"); |
| return; |
| } |
| |
| lg2::info("Handling reset button press"); |
| transition = Host::Transition::Reboot; |
| break; |
| } |
| default: |
| { |
| lg2::error("{EVENT} is invalid power event. skipping...", "EVENT", |
| static_cast<std::underlying_type_t<PowerEvent>>( |
| powerEventType)); |
| |
| return; |
| } |
| } |
| auto service = getService(objPathName.c_str(), dbusIfaceName); |
| auto method = bus.new_method_call(service.c_str(), objPathName.c_str(), |
| propertyIface, "Set"); |
| method.append(dbusIfaceName, transitionName, transition); |
| bus.call(method); |
| } |
| void Handler::powerPressed(sdbusplus::message::message& /* msg */) |
| { |
| try |
| { |
| handlePowerEvent(PowerEvent::powerPressed); |
| } |
| catch (const sdbusplus::exception::exception& e) |
| { |
| lg2::error("Failed power state change on a power button press: {ERROR}", |
| "ERROR", e); |
| } |
| } |
| void Handler::longPowerPressed(sdbusplus::message::message& /* msg */) |
| { |
| try |
| { |
| handlePowerEvent(PowerEvent::longPowerPressed); |
| } |
| catch (const sdbusplus::exception::exception& e) |
| { |
| lg2::error("Failed powering off on long power button press: {ERROR}", |
| "ERROR", e); |
| } |
| } |
| |
| void Handler::resetPressed(sdbusplus::message::message& /* msg */) |
| { |
| try |
| { |
| handlePowerEvent(PowerEvent::resetPressed); |
| } |
| catch (const sdbusplus::exception::exception& e) |
| { |
| lg2::error("Failed power state change on a reset button press: {ERROR}", |
| "ERROR", e); |
| } |
| } |
| |
| void Handler::idPressed(sdbusplus::message::message& /* msg */) |
| { |
| std::string groupPath{ledGroupBasePath}; |
| groupPath += ID_LED_GROUP; |
| |
| auto service = getService(groupPath, ledGroupIface); |
| |
| if (service.empty()) |
| { |
| lg2::info("No found {GROUP} during ID button press:", "GROUP", |
| groupPath); |
| return; |
| } |
| |
| try |
| { |
| auto method = bus.new_method_call(service.c_str(), groupPath.c_str(), |
| propertyIface, "Get"); |
| method.append(ledGroupIface, "Asserted"); |
| auto result = bus.call(method); |
| |
| std::variant<bool> state; |
| result.read(state); |
| |
| state = !std::get<bool>(state); |
| |
| lg2::info( |
| "Changing ID LED group state on ID LED press, GROUP = {GROUP}, STATE = {STATE}", |
| "GROUP", groupPath, "STATE", std::get<bool>(state)); |
| |
| method = bus.new_method_call(service.c_str(), groupPath.c_str(), |
| propertyIface, "Set"); |
| |
| method.append(ledGroupIface, "Asserted", state); |
| result = bus.call(method); |
| } |
| catch (const sdbusplus::exception::exception& e) |
| { |
| lg2::error("Error toggling ID LED group on ID button press: {ERROR}", |
| "ERROR", e); |
| } |
| } |
| } // namespace button |
| } // namespace phosphor |