#include "config.h" #include "occ_manager.hpp" #include "i2c_occ.hpp" #include "utils.hpp" #include #include #include #include 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(path.c_str())); statusObjects.emplace_back(std::make_unique( 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( *statusObjects.front()); } } 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("Invalid update on OCCActive"); elog(); } 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(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("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(event, path.c_str(), *this)); } // The first device is master occ pcap = std::make_unique( *statusObjects.front(), occMasterName); } #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("Manager::pollerTimerExpire(): No OCCs running, poll " "timer not restarted"); } if (!_pollTimer) { log( "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