| #include <iostream> |
| #include <string> |
| #include <algorithm> |
| #include <phosphor-logging/log.hpp> |
| #include <xyz/openbmc_project/Led/Physical/server.hpp> |
| #include "manager.hpp" |
| namespace phosphor |
| { |
| namespace led |
| { |
| |
| // Assert -or- De-assert |
| bool Manager::setGroupState(const std::string& path, bool assert, |
| group& ledsAssert, group& ledsDeAssert, |
| group& ledsUpdate) |
| { |
| if (assert) |
| { |
| assertedGroups.insert(&ledMap.at(path)); |
| } |
| else |
| { |
| auto search = assertedGroups.find(&ledMap.at(path)); |
| if (search != assertedGroups.end()) |
| { |
| assertedGroups.erase(&ledMap.at(path)); |
| } |
| } |
| |
| // This will contain the union of what's already in the asserted group |
| group desiredState {}; |
| for(const auto& grp : assertedGroups) |
| { |
| desiredState.insert(grp->cbegin(), grp->cend()); |
| } |
| |
| // Has the LEDs that are either to be turned off -or- want a new assertion |
| group transient {}; |
| std::set_difference(currentState.begin(), currentState.end(), |
| desiredState.begin(), desiredState.end(), |
| std::inserter(transient, transient.begin())); |
| if(transient.size()) |
| { |
| // We really do not want the Manager to know how a particular LED |
| // transitions from State-A --> State-B and all this must be handled by |
| // the physical LED controller implementation. |
| // So in this case, Manager really does not want to turn off the |
| // LEDs and then turning it back on and let the physical LED controller |
| // handle that. |
| |
| // If we previously had a FRU in ON state , and then if there was a |
| // request to make it blink, the end state would now be blink. |
| // If we either turn off blink / fault, then we need to go back to its |
| // previous state. |
| std::set_intersection(desiredState.begin(), desiredState.end(), |
| transient.begin(), transient.end(), |
| std::inserter(ledsUpdate, ledsUpdate.begin()), |
| ledComp); |
| |
| // These LEDs are only to be De-Asserted. |
| std::set_difference(transient.begin(), transient.end(), |
| ledsUpdate.begin(), ledsUpdate.end(), |
| std::inserter(ledsDeAssert, ledsDeAssert.begin()), |
| ledComp); |
| |
| } |
| |
| // Turn on these |
| std::set_difference(desiredState.begin(), desiredState.end(), |
| currentState.begin(), currentState.end(), |
| std::inserter(ledsAssert, ledsAssert.begin())); |
| |
| |
| // Done.. Save the latest and greatest. |
| currentState = std::move(desiredState); |
| |
| // If we survive, then set the state accordingly. |
| return assert; |
| } |
| |
| /** @brief Run through the map and apply action on the LEDs */ |
| void Manager::driveLEDs(group& ledsAssert, group& ledsDeAssert, |
| group& ledsUpdate) |
| { |
| // For now, physical LED is driven by xyz.openbmc_project.Led.Controller |
| // at /xyz/openbmc_project/led/physical. However, its possible that in the |
| // future, the physical LEDs are driven by different dbus services. |
| // when that happens, service name needs to be obtained everytime a |
| // particular LED would be targeted as opposed to getting one now and then |
| // using it for all |
| |
| // This order of LED operation is important. |
| if (ledsUpdate.size()) |
| { |
| std::cout << "Updating LED states between (On <--> Blink)" |
| << std::endl; |
| for (const auto& it: ledsUpdate) |
| { |
| std::string objPath = std::string(PHY_LED_PATH) + it.name; |
| drivePhysicalLED(objPath, it.action, it.dutyOn); |
| } |
| } |
| |
| if (ledsDeAssert.size()) |
| { |
| std::cout << "De-Asserting LEDs" << std::endl; |
| for (const auto& it: ledsDeAssert) |
| { |
| std::string objPath = std::string(PHY_LED_PATH) + it.name; |
| drivePhysicalLED(objPath, Layout::Action::Off, it.dutyOn); |
| } |
| } |
| |
| if(ledsAssert.size()) |
| { |
| std::cout << "Asserting LEDs" << std::endl; |
| for (const auto& it: ledsAssert) |
| { |
| std::string objPath = std::string(PHY_LED_PATH) + it.name; |
| drivePhysicalLED(objPath, it.action, it.dutyOn); |
| } |
| } |
| return; |
| } |
| |
| // Calls into driving physical LED post choosing the action |
| void Manager::drivePhysicalLED(const std::string& objPath, |
| Layout::Action action, |
| uint8_t dutyOn) |
| { |
| auto service = getServiceName(objPath, PHY_LED_IFACE); |
| if (!service.empty()) |
| { |
| // If Blink, set its property |
| if (action == Layout::Action::Blink) |
| { |
| drivePhysicalLED(service, objPath, "DutyOn", dutyOn); |
| } |
| drivePhysicalLED(service, objPath, "State", |
| getPhysicalAction(action)); |
| } |
| } |
| |
| /** @brief Returns action string based on enum */ |
| std::string Manager::getPhysicalAction(Layout::Action action) |
| { |
| namespace server = sdbusplus::xyz::openbmc_project::Led::server; |
| |
| // TODO: openbmc/phosphor-led-manager#5 |
| // Somehow need to use the generated Action enum than giving one |
| // in ledlayout. |
| if(action == Layout::Action::On) |
| { |
| return server::convertForMessage(server::Physical::Action::On); |
| } |
| else if(action == Layout::Action::Blink) |
| { |
| return server::convertForMessage(server::Physical::Action::Blink); |
| } |
| else |
| { |
| return server::convertForMessage(server::Physical::Action::Off); |
| } |
| } |
| |
| /** Given the LED dbus path and interface, returns the service name */ |
| std::string Manager::getServiceName(const std::string& objPath, |
| const std::string& interface) const |
| { |
| using namespace phosphor::logging; |
| |
| // Mapper dbus constructs |
| constexpr auto MAPPER_BUSNAME = "xyz.openbmc_project.ObjectMapper"; |
| constexpr auto MAPPER_OBJ_PATH = "/xyz/openbmc_project/object_mapper"; |
| constexpr auto MAPPER_IFACE = "xyz.openbmc_project.ObjectMapper"; |
| |
| // Make a mapper call |
| auto mapperCall = bus.new_method_call(MAPPER_BUSNAME, MAPPER_OBJ_PATH, |
| MAPPER_IFACE, "GetObject"); |
| // Cook rest of the things. |
| mapperCall.append(objPath); |
| mapperCall.append(std::vector<std::string>({interface})); |
| |
| auto reply = bus.call(mapperCall); |
| if (reply.is_method_error()) |
| { |
| // Its okay if we do not see a corresponding physical LED. |
| log<level::INFO>("Error looking up Physical LED service", |
| entry("PATH=%s",objPath.c_str())); |
| return ""; |
| } |
| |
| // Response by mapper in the case of success |
| std::map<std::string, std::vector<std::string>> serviceNames; |
| |
| // This is the service name for the passed in objpath |
| reply.read(serviceNames); |
| if (serviceNames.empty()) |
| { |
| log<level::INFO>("Physical LED lookup did not return any service", |
| entry("PATH=%s",objPath.c_str())); |
| return ""; |
| } |
| |
| return serviceNames.begin()->first; |
| } |
| |
| } // namespace led |
| } // namespace phosphor |