blob: 88a8390cbf6950747f074b5ad3ccdf6bc83dc7c1 [file] [log] [blame]
/*
// Copyright (c) 2017 Intel 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 <Utils.hpp>
#include <boost/algorithm/string/predicate.hpp>
#include <filesystem>
#include <fstream>
#include <regex>
#include <sdbusplus/asio/connection.hpp>
#include <sdbusplus/asio/object_server.hpp>
#include <sdbusplus/bus/match.hpp>
namespace fs = std::filesystem;
const static constexpr char* powerInterfaceName =
"xyz.openbmc_project.Chassis.Control.Power";
const static constexpr char* powerObjectName =
"/xyz/openbmc_project/Chassis/Control/Power0";
static bool powerStatusOn = false;
static bool biosHasPost = false;
static std::unique_ptr<sdbusplus::bus::match::match> powerMatch = nullptr;
bool getSensorConfiguration(
const std::string& type,
const std::shared_ptr<sdbusplus::asio::connection>& dbusConnection,
ManagedObjectType& resp, bool useCache)
{
static ManagedObjectType managedObj;
if (!useCache)
{
managedObj.clear();
sdbusplus::message::message getManagedObjects =
dbusConnection->new_method_call(
entityManagerName, "/", "org.freedesktop.DBus.ObjectManager",
"GetManagedObjects");
bool err = false;
try
{
sdbusplus::message::message reply =
dbusConnection->call(getManagedObjects);
reply.read(managedObj);
}
catch (const sdbusplus::exception::exception&)
{
err = true;
}
if (err)
{
std::cerr << "Error communicating to entity manager\n";
return false;
}
}
for (const auto& pathPair : managedObj)
{
std::vector<boost::container::flat_map<std::string, BasicVariantType>>
sensorData;
bool correctType = false;
for (const auto& entry : pathPair.second)
{
if (boost::starts_with(entry.first, type))
{
correctType = true;
break;
}
}
if (correctType)
{
resp.emplace(pathPair);
}
}
return true;
}
bool findFiles(const fs::path dirPath, const std::string& matchString,
std::vector<fs::path>& foundPaths, unsigned int symlinkDepth)
{
if (!fs::exists(dirPath))
return false;
std::regex search(matchString);
std::smatch match;
for (auto& p : fs::recursive_directory_iterator(dirPath))
{
std::string path = p.path().string();
if (!is_directory(p))
{
if (std::regex_search(path, match, search))
foundPaths.emplace_back(p.path());
}
else if (is_symlink(p) && symlinkDepth)
{
findFiles(p.path(), matchString, foundPaths, symlinkDepth - 1);
}
}
return true;
}
bool isPowerOn(void)
{
if (!powerMatch)
{
throw std::runtime_error("Power Match Not Created");
}
return powerStatusOn;
}
bool hasBiosPost(void)
{
if (!powerMatch)
{
throw std::runtime_error("Power Match Not Created");
}
return biosHasPost;
}
void setupPowerMatch(const std::shared_ptr<sdbusplus::asio::connection>& conn)
{
// create a match for powergood changes, first time do a method call to
// cache the correct value
std::function<void(sdbusplus::message::message & message)> eventHandler =
[](sdbusplus::message::message& message) {
std::string objectName;
boost::container::flat_map<std::string, std::variant<int32_t, bool>>
values;
message.read(objectName, values);
auto findPgood = values.find("pgood");
if (findPgood != values.end())
{
powerStatusOn = std::get<int32_t>(findPgood->second);
}
auto findPostComplete = values.find("post_complete");
if (findPostComplete != values.end())
{
biosHasPost = std::get<bool>(findPostComplete->second);
}
};
powerMatch = std::make_unique<sdbusplus::bus::match::match>(
static_cast<sdbusplus::bus::bus&>(*conn),
"type='signal',interface='org.freedesktop.DBus.Properties',path_"
"namespace='/xyz/openbmc_project/Chassis/Control/"
"Power0',arg0='xyz.openbmc_project.Chassis.Control.Power'",
eventHandler);
conn->async_method_call(
[](boost::system::error_code ec, const std::variant<int32_t>& pgood) {
if (ec)
{
// we commonly come up before power control, we'll capture the
// property change later
return;
}
powerStatusOn = std::get<int32_t>(pgood);
},
powerInterfaceName, powerObjectName, "org.freedesktop.DBus.Properties",
"Get", powerInterfaceName, "pgood");
conn->async_method_call(
[](boost::system::error_code ec,
const std::variant<int32_t>& postComplete) {
if (ec)
{
// we commonly come up before power control, we'll capture the
// property change later
return;
}
biosHasPost = std::get<int32_t>(postComplete);
},
powerInterfaceName, powerObjectName, "org.freedesktop.DBus.Properties",
"Get", powerInterfaceName, "post_complete");
}
// replaces limits if MinReading and MaxReading are found.
void findLimits(std::pair<double, double>& limits,
const SensorBaseConfiguration* data)
{
if (!data)
{
return;
}
auto maxFind = data->second.find("MaxReading");
auto minFind = data->second.find("MinReading");
if (minFind != data->second.end())
{
limits.first = std::visit(VariantToDoubleVisitor(), minFind->second);
}
if (maxFind != data->second.end())
{
limits.second = std::visit(VariantToDoubleVisitor(), maxFind->second);
}
}
void createAssociation(
std::shared_ptr<sdbusplus::asio::dbus_interface>& association,
const std::string& path)
{
if (association)
{
std::filesystem::path p(path);
using Association = std::tuple<std::string, std::string, std::string>;
std::vector<Association> associations;
associations.push_back(
Association("inventory", "sensors", p.parent_path().string()));
association->register_property("associations", associations);
association->initialize();
}
}