blob: 1a14911cba9ce159fb6e9b7a733dc6f484ae8975 [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 "dbus-sensor_config.h"
#include <Utils.hpp>
#include <boost/algorithm/string/predicate.hpp>
#include <boost/container/flat_map.hpp>
#include <sdbusplus/asio/connection.hpp>
#include <sdbusplus/asio/object_server.hpp>
#include <sdbusplus/bus/match.hpp>
#include <filesystem>
#include <fstream>
#include <memory>
#include <regex>
#include <stdexcept>
#include <string>
#include <utility>
#include <variant>
#include <vector>
namespace fs = std::filesystem;
static bool powerStatusOn = false;
static bool biosHasPost = false;
static bool manufacturingMode = false;
static std::unique_ptr<sdbusplus::bus::match::match> powerMatch = nullptr;
static std::unique_ptr<sdbusplus::bus::match::match> postMatch = nullptr;
/**
* return the contents of a file
* @param[in] hwmonFile - the path to the file to read
* @return the contents of the file as a string or nullopt if the file could not
* be opened.
*/
std::optional<std::string> openAndRead(const std::string& hwmonFile)
{
std::string fileVal;
std::ifstream fileStream(hwmonFile);
if (!fileStream.is_open())
{
return std::nullopt;
}
std::getline(fileStream, fileVal);
return fileVal;
}
/**
* given a hwmon temperature base name if valid return the full path else
* nullopt
* @param[in] directory - the hwmon sysfs directory
* @param[in] permitSet - a set of labels or hwmon basenames to permit. If this
* is empty then *everything* is permitted.
* @return a string to the full path of the file to create a temp sensor with or
* nullopt to indicate that no sensor should be created for this basename.
*/
std::optional<std::string>
getFullHwmonFilePath(const std::string& directory,
const std::string& hwmonBaseName,
const std::set<std::string>& permitSet)
{
std::optional<std::string> result;
std::string filename;
if (permitSet.empty())
{
result = directory + "/" + hwmonBaseName + "_input";
return result;
}
filename = directory + "/" + hwmonBaseName + "_label";
auto searchVal = openAndRead(filename);
if (!searchVal)
{
/* if the hwmon temp doesn't have a corresponding label file
* then use the hwmon temperature base name
*/
searchVal = hwmonBaseName;
}
if (permitSet.find(*searchVal) != permitSet.end())
{
result = directory + "/" + hwmonBaseName + "_input";
}
return result;
}
/**
* retrieve a set of basenames and labels to allow sensor creation for.
* @param[in] config - a map representing the configuration for a specific
* device
* @return a set of basenames and labels to allow sensor creation for. An empty
* set indicates that everything is permitted.
*/
std::set<std::string> getPermitSet(const SensorBaseConfigMap& config)
{
auto permitAttribute = config.find("Labels");
std::set<std::string> permitSet;
if (permitAttribute != config.end())
{
try
{
auto val =
std::get<std::vector<std::string>>(permitAttribute->second);
permitSet.insert(std::make_move_iterator(val.begin()),
std::make_move_iterator(val.end()));
}
catch (const std::bad_variant_access& err)
{
std::cerr << err.what()
<< ":PermitList does not contain a list, wrong "
"variant type.\n";
}
}
return permitSet;
}
bool getSensorConfiguration(
const std::string& type,
const std::shared_ptr<sdbusplus::asio::connection>& dbusConnection,
ManagedObjectType& resp)
{
return getSensorConfiguration(type, dbusConnection, resp, false);
}
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& e)
{
std::cerr << "While calling GetManagedObjects on service:"
<< entityManagerName << " exception name:" << e.name()
<< "and description:" << e.description()
<< " was thrown\n";
err = true;
}
if (err)
{
std::cerr << "Error communicating to entity manager\n";
return false;
}
}
for (const auto& pathPair : managedObj)
{
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, int symlinkDepth)
{
if (!fs::exists(dirPath))
{
return false;
}
std::regex search(matchString);
std::smatch match;
for (auto p = fs::recursive_directory_iterator(
dirPath, fs::directory_options::follow_directory_symlink);
p != fs::recursive_directory_iterator(); ++p)
{
std::string path = p->path().string();
if (!is_directory(*p))
{
if (std::regex_search(path, match, search))
{
foundPaths.emplace_back(p->path());
}
}
if (p.depth() >= symlinkDepth)
{
p.disable_recursion_pending();
}
}
return true;
}
bool isPowerOn(void)
{
if (!powerMatch)
{
throw std::runtime_error("Power Match Not Created");
}
return powerStatusOn;
}
bool hasBiosPost(void)
{
if (!postMatch)
{
throw std::runtime_error("Post Match Not Created");
}
return biosHasPost;
}
bool readingStateGood(const PowerState& powerState)
{
if (powerState == PowerState::on && !isPowerOn())
{
return false;
}
if (powerState == PowerState::biosPost && (!hasBiosPost() || !isPowerOn()))
{
return false;
}
return true;
}
static void
getPowerStatus(const std::shared_ptr<sdbusplus::asio::connection>& conn,
size_t retries = 2)
{
conn->async_method_call(
[conn, retries](boost::system::error_code ec,
const std::variant<std::string>& state) {
if (ec)
{
if (retries)
{
auto timer = std::make_shared<boost::asio::steady_timer>(
conn->get_io_context());
timer->expires_after(std::chrono::seconds(15));
timer->async_wait(
[timer, conn, retries](boost::system::error_code) {
getPowerStatus(conn, retries - 1);
});
return;
}
// we commonly come up before power control, we'll capture the
// property change later
std::cerr << "error getting power status " << ec.message()
<< "\n";
return;
}
powerStatusOn =
boost::ends_with(std::get<std::string>(state), ".Running");
},
power::busname, power::path, properties::interface, properties::get,
power::interface, power::property);
}
static void
getPostStatus(const std::shared_ptr<sdbusplus::asio::connection>& conn,
size_t retries = 2)
{
conn->async_method_call(
[conn, retries](boost::system::error_code ec,
const std::variant<std::string>& state) {
if (ec)
{
if (retries)
{
auto timer = std::make_shared<boost::asio::steady_timer>(
conn->get_io_context());
timer->expires_after(std::chrono::seconds(15));
timer->async_wait(
[timer, conn, retries](boost::system::error_code) {
getPostStatus(conn, retries - 1);
});
return;
}
// we commonly come up before power control, we'll capture the
// property change later
std::cerr << "error getting post status " << ec.message()
<< "\n";
return;
}
biosHasPost = std::get<std::string>(state) != "Inactive";
},
post::busname, post::path, properties::interface, properties::get,
post::interface, post::property);
}
void setupPowerMatch(const std::shared_ptr<sdbusplus::asio::connection>& conn)
{
static boost::asio::steady_timer timer(conn->get_io_context());
// create a match for powergood changes, first time do a method call to
// cache the correct value
if (powerMatch)
{
return;
}
powerMatch = std::make_unique<sdbusplus::bus::match::match>(
static_cast<sdbusplus::bus::bus&>(*conn),
"type='signal',interface='" + std::string(properties::interface) +
"',path='" + std::string(power::path) + "',arg0='" +
std::string(power::interface) + "'",
[](sdbusplus::message::message& message) {
std::string objectName;
boost::container::flat_map<std::string, std::variant<std::string>>
values;
message.read(objectName, values);
auto findState = values.find(power::property);
if (findState != values.end())
{
bool on = boost::ends_with(
std::get<std::string>(findState->second), ".Running");
if (!on)
{
timer.cancel();
powerStatusOn = false;
return;
}
// on comes too quickly
timer.expires_after(std::chrono::seconds(10));
timer.async_wait([](boost::system::error_code ec) {
if (ec == boost::asio::error::operation_aborted)
{
return;
}
if (ec)
{
std::cerr << "Timer error " << ec.message() << "\n";
return;
}
powerStatusOn = true;
});
}
});
postMatch = std::make_unique<sdbusplus::bus::match::match>(
static_cast<sdbusplus::bus::bus&>(*conn),
"type='signal',interface='" + std::string(properties::interface) +
"',path='" + std::string(post::path) + "',arg0='" +
std::string(post::interface) + "'",
[](sdbusplus::message::message& message) {
std::string objectName;
boost::container::flat_map<std::string, std::variant<std::string>>
values;
message.read(objectName, values);
auto findState = values.find(post::property);
if (findState != values.end())
{
biosHasPost =
std::get<std::string>(findState->second) != "Inactive";
}
});
getPowerStatus(conn);
getPostStatus(conn);
}
// 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);
std::vector<Association> associations;
associations.emplace_back("chassis", "all_sensors",
p.parent_path().string());
association->register_property("Associations", associations);
association->initialize();
}
}
void setInventoryAssociation(
const std::shared_ptr<sdbusplus::asio::dbus_interface>& association,
const std::string& path,
const std::vector<std::string>& chassisPaths = std::vector<std::string>())
{
if (association)
{
std::filesystem::path p(path);
std::vector<Association> associations;
std::string objPath(p.parent_path().string());
associations.emplace_back("inventory", "sensors", objPath);
associations.emplace_back("chassis", "all_sensors", objPath);
for (const std::string& chassisPath : chassisPaths)
{
associations.emplace_back("chassis", "all_sensors", chassisPath);
}
association->register_property("Associations", associations);
association->initialize();
}
}
void createInventoryAssoc(
const std::shared_ptr<sdbusplus::asio::connection>& conn,
const std::shared_ptr<sdbusplus::asio::dbus_interface>& association,
const std::string& path)
{
if (!association)
{
return;
}
conn->async_method_call(
[association, path](const boost::system::error_code ec,
const std::vector<std::string>& invSysObjPaths) {
if (ec)
{
// In case of error, set the default associations and
// initialize the association Interface.
setInventoryAssociation(association, path);
return;
}
setInventoryAssociation(association, path, invSysObjPaths);
},
mapper::busName, mapper::path, mapper::interface, "GetSubTreePaths",
"/xyz/openbmc_project/inventory/system", 2,
std::array<std::string, 1>{
"xyz.openbmc_project.Inventory.Item.System"});
}
std::optional<double> readFile(const std::string& thresholdFile,
const double& scaleFactor)
{
std::string line;
std::ifstream labelFile(thresholdFile);
if (labelFile.good())
{
std::getline(labelFile, line);
labelFile.close();
try
{
return std::stod(line) / scaleFactor;
}
catch (const std::invalid_argument&)
{
return std::nullopt;
}
}
return std::nullopt;
}
std::optional<std::tuple<std::string, std::string, std::string>>
splitFileName(const std::filesystem::path& filePath)
{
if (filePath.has_filename())
{
const auto fileName = filePath.filename().string();
size_t numberPos = std::strcspn(fileName.c_str(), "1234567890");
size_t itemPos = std::strcspn(fileName.c_str(), "_");
if (numberPos > 0 && itemPos > numberPos && fileName.size() > itemPos)
{
return std::make_optional(
std::make_tuple(fileName.substr(0, numberPos),
fileName.substr(numberPos, itemPos - numberPos),
fileName.substr(itemPos + 1, fileName.size())));
}
}
return std::nullopt;
}
void setupManufacturingModeMatch(sdbusplus::asio::connection& conn)
{
static std::unique_ptr<sdbusplus::bus::match::match>
setupManufacturingModeMatch =
std::make_unique<sdbusplus::bus::match::match>(
conn,
"type='signal',interface='org.freedesktop.DBus."
"Properties',member='"
"PropertiesChanged',arg0namespace='xyz.openbmc_project."
"Security.SpecialMode'",
[](sdbusplus::message::message& msg) {
std::string interfaceName;
boost::container::flat_map<std::string,
std::variant<std::string>>
propertiesChanged;
std::string manufacturingModeStatus;
msg.read(interfaceName, propertiesChanged);
if (propertiesChanged.begin() == propertiesChanged.end())
{
return;
}
manufacturingModeStatus = std::get<std::string>(
propertiesChanged.begin()->second);
manufacturingMode = false;
if (manufacturingModeStatus ==
"xyz.openbmc_project.Control.Security."
"SpecialMode.Modes.Manufacturing")
{
manufacturingMode = true;
}
if (validateUnsecureFeature == true)
{
if (manufacturingModeStatus ==
"xyz.openbmc_project.Control.Security."
"SpecialMode.Modes.ValidationUnsecure")
{
manufacturingMode = true;
}
}
});
return;
}
bool getManufacturingMode()
{
return manufacturingMode;
}