| #include "config.h" |
| |
| #include "occ_manager.hpp" |
| |
| #include "i2c_occ.hpp" |
| #include "utils.hpp" |
| |
| #include <experimental/filesystem> |
| #include <phosphor-logging/elog-errors.hpp> |
| #include <phosphor-logging/log.hpp> |
| #include <xyz/openbmc_project/Common/error.hpp> |
| |
| namespace open_power |
| { |
| namespace occ |
| { |
| |
| using namespace phosphor::logging; |
| |
| void Manager::findAndCreateObjects() |
| { |
| for (auto id = 0; id < MAX_CPUS; ++id) |
| { |
| // Create one occ per cpu |
| auto occ = std::string(OCC_NAME) + std::to_string(id); |
| createObjects(occ); |
| } |
| } |
| |
| int Manager::cpuCreated(sdbusplus::message::message& msg) |
| { |
| namespace fs = std::experimental::filesystem; |
| |
| sdbusplus::message::object_path o; |
| msg.read(o); |
| fs::path cpuPath(std::string(std::move(o))); |
| |
| auto name = cpuPath.filename().string(); |
| auto index = name.find(CPU_NAME); |
| name.replace(index, std::strlen(CPU_NAME), OCC_NAME); |
| |
| createObjects(name); |
| |
| return 0; |
| } |
| |
| void Manager::createObjects(const std::string& occ) |
| { |
| auto path = fs::path(OCC_CONTROL_ROOT) / occ; |
| |
| passThroughObjects.emplace_back( |
| std::make_unique<PassThrough>(path.c_str())); |
| |
| statusObjects.emplace_back(std::make_unique<Status>( |
| event, path.c_str(), *this, |
| std::bind(std::mem_fn(&Manager::statusCallBack), this, |
| std::placeholders::_1) |
| #ifdef PLDM |
| , |
| std::bind(std::mem_fn(&pldm::Interface::resetOCC), pldmHandle.get(), |
| std::placeholders::_1) |
| #endif |
| )); |
| |
| // Create the power cap monitor object for master occ (0) |
| if (!pcap) |
| { |
| pcap = std::make_unique<open_power::occ::powercap::PowerCap>( |
| *statusObjects.front()); |
| } |
| |
| #ifdef POWER10 |
| // Create the power mode monitor object for master occ (0) |
| if (!pmode) |
| { |
| pmode = std::make_unique<open_power::occ::powermode::PowerMode>( |
| *statusObjects.front()); |
| } |
| #endif |
| } |
| |
| void Manager::statusCallBack(bool status) |
| { |
| using InternalFailure = |
| sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure; |
| |
| // At this time, it won't happen but keeping it |
| // here just in case something changes in the future |
| if ((activeCount == 0) && (!status)) |
| { |
| log<level::ERR>("Invalid update on OCCActive"); |
| elog<InternalFailure>(); |
| } |
| |
| activeCount += status ? 1 : -1; |
| |
| // Only start presence detection if all the OCCs are bound |
| if (activeCount == statusObjects.size()) |
| { |
| for (auto& obj : statusObjects) |
| { |
| obj->addPresenceWatchMaster(); |
| } |
| } |
| |
| if ((!_pollTimer->isEnabled()) && (activeCount > 0)) |
| { |
| log<level::INFO>(fmt::format("Manager::statusCallBack(): {} OCCs will " |
| "be polled every {} seconds", |
| activeCount, pollInterval) |
| .c_str()); |
| |
| // Send poll and start OCC poll timer |
| pollerTimerExpired(); |
| } |
| else if ((_pollTimer->isEnabled()) && (activeCount == 0)) |
| { |
| // Stop OCC poll timer |
| log<level::INFO>("Manager::statusCallBack(): OCCs are not running, " |
| "stopping poll timer"); |
| _pollTimer->setEnabled(false); |
| } |
| } |
| |
| #ifdef I2C_OCC |
| void Manager::initStatusObjects() |
| { |
| // Make sure we have a valid path string |
| static_assert(sizeof(DEV_PATH) != 0); |
| |
| auto deviceNames = i2c_occ::getOccHwmonDevices(DEV_PATH); |
| auto occMasterName = deviceNames.front(); |
| for (auto& name : deviceNames) |
| { |
| i2c_occ::i2cToDbus(name); |
| name = std::string(OCC_NAME) + '_' + name; |
| auto path = fs::path(OCC_CONTROL_ROOT) / name; |
| statusObjects.emplace_back( |
| std::make_unique<Status>(event, path.c_str(), *this)); |
| } |
| // The first device is master occ |
| pcap = std::make_unique<open_power::occ::powercap::PowerCap>( |
| *statusObjects.front(), occMasterName); |
| #ifdef POWER10 |
| pmode = std::make_unique<open_power::occ::powermode::PowerMode>( |
| *statusObjects.front()); |
| #endif |
| } |
| #endif |
| |
| #ifdef PLDM |
| bool Manager::updateOCCActive(instanceID instance, bool status) |
| { |
| return (statusObjects[instance])->occActive(status); |
| } |
| #endif |
| |
| void Manager::pollerTimerExpired() |
| { |
| if (activeCount == 0) |
| { |
| // No OCCs running, so poll timer will not be restarted |
| log<level::INFO>("Manager::pollerTimerExpire(): No OCCs running, poll " |
| "timer not restarted"); |
| } |
| |
| if (!_pollTimer) |
| { |
| log<level::ERR>( |
| "Manager::pollerTimerExpired() ERROR: Timer not defined"); |
| return; |
| } |
| |
| for (auto& obj : statusObjects) |
| { |
| // Read sysfs to force kernel to poll OCC |
| obj->readOccState(); |
| } |
| |
| // Restart OCC poll timer |
| _pollTimer->restartOnce(std::chrono::seconds(pollInterval)); |
| } |
| |
| } // namespace occ |
| } // namespace open_power |