blob: 3582bf5774364f644878ce5810fe34e72a81f8fb [file] [log] [blame]
/**
* Copyright © 2020 IBM 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 "config.h"
#include "manager.hpp"
#include "action.hpp"
#include "event.hpp"
#include "fan.hpp"
#include "group.hpp"
#include "json_config.hpp"
#include "profile.hpp"
#include "zone.hpp"
#include <nlohmann/json.hpp>
#include <sdbusplus/bus.hpp>
#include <sdeventplus/event.hpp>
#include <sdeventplus/utility/timer.hpp>
#include <algorithm>
#include <chrono>
#include <filesystem>
#include <functional>
#include <map>
#include <memory>
#include <tuple>
#include <utility>
#include <vector>
namespace phosphor::fan::control::json
{
using json = nlohmann::json;
std::vector<std::string> Manager::_activeProfiles;
std::map<std::string,
std::map<std::string, std::pair<bool, std::vector<std::string>>>>
Manager::_servTree;
std::map<std::string,
std::map<std::string, std::map<std::string, PropertyVariantType>>>
Manager::_objects;
Manager::Manager(sdbusplus::bus::bus& bus, const sdeventplus::Event& event) :
_bus(bus), _event(event)
{
// Manager JSON config file is optional
auto confFile =
fan::JsonConfig::getConfFile(bus, confAppName, confFileName, true);
if (!confFile.empty())
{
_jsonObj = fan::JsonConfig::load(confFile);
}
// Parse and set the available profiles and which are active
setProfiles();
// Load the zone configurations
_zones = getConfig<Zone>(false, bus, bus, event, this);
// Load the fan configurations and move each fan into its zone
auto fans = getConfig<Fan>(false, bus, bus);
for (auto& fan : fans)
{
configKey fanProfile =
std::make_pair(fan.second->getZone(), fan.first.second);
auto itZone = std::find_if(
_zones.begin(), _zones.end(), [&fanProfile](const auto& zone) {
return Manager::inConfig(fanProfile, zone.first);
});
if (itZone != _zones.end())
{
if (itZone->second->getTarget() != fan.second->getTarget() &&
fan.second->getTarget() != 0)
{
// Update zone target to current target of the fan in the zone
itZone->second->setTarget(fan.second->getTarget());
}
itZone->second->addFan(std::move(fan.second));
}
}
// Load the configured groups that are copied into events where they're used
auto groups = getConfig<Group>(true, bus);
// Load any events configured
_events = getConfig<Event>(true, bus, bus, this, groups, _zones);
bus.request_name(CONTROL_BUSNAME);
}
const std::vector<std::string>& Manager::getActiveProfiles()
{
return _activeProfiles;
}
bool Manager::inConfig(const configKey& input, const configKey& comp)
{
// Config names dont match, do not include in config
if (input.first != comp.first)
{
return false;
}
// No profiles specified by input config, can be used in any config
if (input.second.empty())
{
return true;
}
else
{
// Profiles must have one match in the other's profiles(and they must be
// an active profile) to be used in the config
return std::any_of(
input.second.begin(), input.second.end(),
[&comp](const auto& lProfile) {
return std::any_of(
comp.second.begin(), comp.second.end(),
[&lProfile](const auto& rProfile) {
if (lProfile != rProfile)
{
return false;
}
auto activeProfs = getActiveProfiles();
return std::find(activeProfs.begin(), activeProfs.end(),
lProfile) != activeProfs.end();
});
});
}
}
bool Manager::hasOwner(const std::string& path, const std::string& intf)
{
auto itServ = _servTree.find(path);
if (itServ == _servTree.end())
{
// Path not found in cache, therefore owner missing
return false;
}
for (const auto& service : itServ->second)
{
auto itIntf = std::find_if(
service.second.second.begin(), service.second.second.end(),
[&intf](const auto& interface) { return intf == interface; });
if (itIntf != std::end(service.second.second))
{
// Service found, return owner state
return service.second.first;
}
}
// Interface not found in cache, therefore owner missing
return false;
}
void Manager::setOwner(const std::string& path, const std::string& serv,
const std::string& intf, bool isOwned)
{
auto itServ = _servTree.find(path);
if (itServ == _servTree.end())
{
auto intfs = {intf};
_servTree[path] = {{serv, std::make_pair(isOwned, intfs)}};
return;
}
for (auto& service : itServ->second)
{
auto itIntf = std::find_if(
service.second.second.begin(), service.second.second.end(),
[&intf](const auto& interface) { return intf == interface; });
if (itIntf != std::end(service.second.second))
{
if (service.first == serv)
{
service.second.first = isOwned;
return;
}
}
}
auto intfs = {intf};
itServ->second[serv] = std::make_pair(isOwned, intfs);
}
const std::string& Manager::findService(const std::string& path,
const std::string& intf)
{
static const std::string empty = "";
auto itServ = _servTree.find(path);
if (itServ != _servTree.end())
{
for (const auto& service : itServ->second)
{
auto itIntf = std::find_if(
service.second.second.begin(), service.second.second.end(),
[&intf](const auto& interface) { return intf == interface; });
if (itIntf != std::end(service.second.second))
{
// Service found, return service name
return service.first;
}
}
}
return empty;
}
void Manager::addServices(const std::string& intf, int32_t depth)
{
// Get all subtree objects for the given interface
auto objects = util::SDBusPlus::getSubTree(util::SDBusPlus::getBus(), "/",
intf, depth);
// Add what's returned to the cache of path->services
for (auto& itPath : objects)
{
auto pathIter = _servTree.find(itPath.first);
if (pathIter != _servTree.end())
{
// Path found in cache
for (auto& itServ : itPath.second)
{
auto servIter = pathIter->second.find(itServ.first);
if (servIter != pathIter->second.end())
{
// Service found in cache
for (auto& itIntf : itServ.second)
{
if (std::find(servIter->second.second.begin(),
servIter->second.second.end(),
itIntf) == servIter->second.second.end())
{
// Add interface to cache
servIter->second.second.emplace_back(itIntf);
}
}
}
else
{
// Service not found in cache
auto intfs = {intf};
pathIter->second[itServ.first] =
std::make_pair(true, intfs);
}
}
}
else
{
// Path not found in cache
auto intfs = {intf};
_servTree[itPath.first] = {
{itPath.second.begin()->first, std::make_pair(true, intfs)}};
}
}
}
const std::string& Manager::getService(const std::string& path,
const std::string& intf)
{
// Retrieve service from cache
const auto& serviceName = findService(path, intf);
if (serviceName.empty())
{
addServices(intf, 0);
return findService(path, intf);
}
return serviceName;
}
void Manager::addTimer(const TimerType type,
const std::chrono::microseconds interval,
std::unique_ptr<TimerPkg> pkg)
{
auto dataPtr =
std::make_unique<TimerData>(std::make_pair(type, std::move(*pkg)));
Timer timer(_event,
std::bind(&Manager::timerExpired, this, std::ref(*dataPtr)));
if (type == TimerType::repeating)
{
timer.restart(interval);
}
else if (type == TimerType::oneshot)
{
timer.restartOnce(interval);
}
else
{
throw std::invalid_argument("Invalid Timer Type");
}
_timers.emplace_back(std::move(dataPtr), std::move(timer));
}
void Manager::timerExpired(TimerData& data)
{
auto& actions =
std::get<std::vector<std::unique_ptr<ActionBase>>&>(data.second);
// Perform the actions in the timer data
std::for_each(actions.begin(), actions.end(),
[](auto& action) { action->run(); });
// Remove oneshot timers after they expired
if (data.first == TimerType::oneshot)
{
auto itTimer = std::find_if(
_timers.begin(), _timers.end(), [&data](const auto& timer) {
return (data.first == timer.first->first &&
(std::get<std::string>(data.second) ==
std::get<std::string>(timer.first->second)));
});
if (itTimer != std::end(_timers))
{
_timers.erase(itTimer);
}
}
}
void Manager::handleSignal(sdbusplus::message::message& msg,
const std::vector<SignalPkg>* pkgs)
{
for (auto& pkg : *pkgs)
{
// Handle the signal callback and only run the actions if the handler
// updated the cache for the given SignalObject
if (std::get<SignalHandler>(pkg)(msg, std::get<SignalObject>(pkg),
*this))
{
// Perform the actions in the handler package
auto& actions = std::get<SignalActions>(pkg);
std::for_each(actions.begin(), actions.end(),
[](auto& action) { action.get()->run(); });
}
}
}
unsigned int Manager::getPowerOnDelay()
{
auto powerOnDelay = 0;
// Parse optional "power_on_delay" from JSON object
if (!_jsonObj.empty() && _jsonObj.contains("power_on_delay"))
{
powerOnDelay = _jsonObj["power_on_delay"].get<unsigned int>();
}
return powerOnDelay;
}
void Manager::setProfiles()
{
// Profiles JSON config file is optional
auto confFile = fan::JsonConfig::getConfFile(_bus, confAppName,
Profile::confFileName, true);
if (!confFile.empty())
{
for (const auto& entry : fan::JsonConfig::load(confFile))
{
auto obj = std::make_unique<Profile>(entry);
_profiles.emplace(
std::make_pair(obj->getName(), obj->getProfiles()),
std::move(obj));
}
}
// Ensure all configurations use the same set of active profiles
// (In case a profile's active state changes during configuration)
for (const auto& profile : _profiles)
{
if (profile.second->isActive())
{
_activeProfiles.emplace_back(profile.first.first);
}
}
}
} // namespace phosphor::fan::control::json