| /* |
| // 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_t> powerMatch = nullptr; |
| static std::unique_ptr<sdbusplus::bus::match_t> 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_t getManagedObjects = |
| dbusConnection->new_method_call( |
| entityManagerName, "/", "org.freedesktop.DBus.ObjectManager", |
| "GetManagedObjects"); |
| bool err = false; |
| try |
| { |
| sdbusplus::message_t reply = |
| dbusConnection->call(getManagedObjects); |
| reply.read(managedObj); |
| } |
| catch (const sdbusplus::exception_t& 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, std::string_view matchString, |
| std::vector<fs::path>& foundPaths, int symlinkDepth) |
| { |
| std::error_code ec; |
| if (!fs::exists(dirPath, ec)) |
| { |
| return false; |
| } |
| |
| std::vector<std::regex> matchPieces; |
| |
| size_t pos = 0; |
| std::string token; |
| // Generate the regex expressions list from the match we were given |
| while ((pos = matchString.find('/')) != std::string::npos) |
| { |
| token = matchString.substr(0, pos); |
| matchPieces.emplace_back(token); |
| matchString.remove_prefix(pos + 1); |
| } |
| matchPieces.emplace_back(std::string{matchString}); |
| |
| // Check if the match string contains directories, and skip the match of |
| // subdirectory if not |
| if (matchPieces.size() <= 1) |
| { |
| std::regex search(std::string{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; |
| } |
| |
| // The match string contains directories, verify each level of sub |
| // directories |
| for (auto p = fs::recursive_directory_iterator( |
| dirPath, fs::directory_options::follow_directory_symlink); |
| p != fs::recursive_directory_iterator(); ++p) |
| { |
| std::vector<std::regex>::iterator matchPiece = matchPieces.begin(); |
| fs::path::iterator pathIt = p->path().begin(); |
| for (const fs::path& dir : dirPath) |
| { |
| if (dir.empty()) |
| { |
| // When the path ends with '/', it gets am empty path |
| // skip such case. |
| break; |
| } |
| pathIt++; |
| } |
| |
| while (pathIt != p->path().end()) |
| { |
| // Found a path deeper than match. |
| if (matchPiece == matchPieces.end()) |
| { |
| p.disable_recursion_pending(); |
| break; |
| } |
| std::smatch match; |
| std::string component = pathIt->string(); |
| std::regex regexPiece(*matchPiece); |
| if (!std::regex_match(component, match, regexPiece)) |
| { |
| // path prefix doesn't match, no need to iterate further |
| p.disable_recursion_pending(); |
| break; |
| } |
| matchPiece++; |
| pathIt++; |
| } |
| |
| if (!is_directory(*p)) |
| { |
| if (matchPiece == matchPieces.end()) |
| { |
| 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 != 0U) |
| { |
| 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 != 0U) |
| { |
| 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; |
| } |
| const auto& value = std::get<std::string>(state); |
| biosHasPost = (value != "Inactive") && |
| (value != "xyz.openbmc_project.State.OperatingSystem." |
| "Status.OSStatus.Inactive"); |
| }, |
| post::busname, post::path, properties::interface, properties::get, |
| post::interface, post::property); |
| } |
| |
| void setupPowerMatchCallback( |
| const std::shared_ptr<sdbusplus::asio::connection>& conn, |
| std::function<void(PowerState type, bool state)>&& hostStatusCallback) |
| { |
| 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_t>( |
| static_cast<sdbusplus::bus_t&>(*conn), |
| "type='signal',interface='" + std::string(properties::interface) + |
| "',path='" + std::string(power::path) + "',arg0='" + |
| std::string(power::interface) + "'", |
| [hostStatusCallback](sdbusplus::message_t& 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; |
| hostStatusCallback(PowerState::on, powerStatusOn); |
| return; |
| } |
| // on comes too quickly |
| timer.expires_after(std::chrono::seconds(10)); |
| timer.async_wait( |
| [hostStatusCallback](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; |
| hostStatusCallback(PowerState::on, powerStatusOn); |
| }); |
| } |
| }); |
| |
| postMatch = std::make_unique<sdbusplus::bus::match_t>( |
| static_cast<sdbusplus::bus_t&>(*conn), |
| "type='signal',interface='" + std::string(properties::interface) + |
| "',path='" + std::string(post::path) + "',arg0='" + |
| std::string(post::interface) + "'", |
| [hostStatusCallback](sdbusplus::message_t& 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()) |
| { |
| auto& value = std::get<std::string>(findState->second); |
| biosHasPost = (value != "Inactive") && |
| (value != "xyz.openbmc_project.State.OperatingSystem." |
| "Status.OSStatus.Inactive"); |
| hostStatusCallback(PowerState::biosPost, biosHasPost); |
| } |
| }); |
| |
| getPowerStatus(conn); |
| getPostStatus(conn); |
| } |
| |
| void setupPowerMatch(const std::shared_ptr<sdbusplus::asio::connection>& conn) |
| { |
| setupPowerMatchCallback(conn, [](PowerState, bool) {}); |
| } |
| |
| // replaces limits if MinReading and MaxReading are found. |
| void findLimits(std::pair<double, double>& limits, |
| const SensorBaseConfiguration* data) |
| { |
| if (data == nullptr) |
| { |
| 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) |
| { |
| fs::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) |
| { |
| fs::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 fs::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; |
| } |
| |
| static void handleSpecialModeChange(const std::string& manufacturingModeStatus) |
| { |
| manufacturingMode = false; |
| if (manufacturingModeStatus == "xyz.openbmc_project.Control.Security." |
| "SpecialMode.Modes.Manufacturing") |
| { |
| manufacturingMode = true; |
| } |
| if (validateUnsecureFeature == 1) |
| { |
| if (manufacturingModeStatus == "xyz.openbmc_project.Control.Security." |
| "SpecialMode.Modes.ValidationUnsecure") |
| { |
| manufacturingMode = true; |
| } |
| } |
| } |
| |
| void setupManufacturingModeMatch(sdbusplus::asio::connection& conn) |
| { |
| namespace rules = sdbusplus::bus::match::rules; |
| static constexpr const char* specialModeInterface = |
| "xyz.openbmc_project.Security.SpecialMode"; |
| |
| const std::string filterSpecialModeIntfAdd = |
| rules::interfacesAdded() + |
| rules::argNpath(0, "/xyz/openbmc_project/security/special_mode"); |
| static std::unique_ptr<sdbusplus::bus::match_t> specialModeIntfMatch = |
| std::make_unique<sdbusplus::bus::match_t>(conn, |
| filterSpecialModeIntfAdd, |
| [](sdbusplus::message_t& m) { |
| sdbusplus::message::object_path path; |
| using PropertyMap = |
| boost::container::flat_map<std::string, std::variant<std::string>>; |
| boost::container::flat_map<std::string, PropertyMap> interfaceAdded; |
| m.read(path, interfaceAdded); |
| auto intfItr = interfaceAdded.find(specialModeInterface); |
| if (intfItr == interfaceAdded.end()) |
| { |
| return; |
| } |
| PropertyMap& propertyList = intfItr->second; |
| auto itr = propertyList.find("SpecialMode"); |
| if (itr == propertyList.end()) |
| { |
| std::cerr << "error getting SpecialMode property " |
| << "\n"; |
| return; |
| } |
| auto* manufacturingModeStatus = std::get_if<std::string>(&itr->second); |
| handleSpecialModeChange(*manufacturingModeStatus); |
| }); |
| |
| const std::string filterSpecialModeChange = |
| rules::type::signal() + rules::member("PropertiesChanged") + |
| rules::interface("org.freedesktop.DBus.Properties") + |
| rules::argN(0, specialModeInterface); |
| static std::unique_ptr<sdbusplus::bus::match_t> specialModeChangeMatch = |
| std::make_unique<sdbusplus::bus::match_t>(conn, filterSpecialModeChange, |
| [](sdbusplus::message_t& m) { |
| std::string interfaceName; |
| boost::container::flat_map<std::string, std::variant<std::string>> |
| propertiesChanged; |
| |
| m.read(interfaceName, propertiesChanged); |
| auto itr = propertiesChanged.find("SpecialMode"); |
| if (itr == propertiesChanged.end()) |
| { |
| return; |
| } |
| auto* manufacturingModeStatus = std::get_if<std::string>(&itr->second); |
| handleSpecialModeChange(*manufacturingModeStatus); |
| }); |
| |
| conn.async_method_call( |
| [](const boost::system::error_code ec, |
| const std::variant<std::string>& getManufactMode) { |
| if (ec) |
| { |
| std::cerr << "error getting SpecialMode status " << ec.message() |
| << "\n"; |
| return; |
| } |
| const auto* manufacturingModeStatus = |
| std::get_if<std::string>(&getManufactMode); |
| handleSpecialModeChange(*manufacturingModeStatus); |
| }, |
| "xyz.openbmc_project.SpecialMode", |
| "/xyz/openbmc_project/security/special_mode", |
| "org.freedesktop.DBus.Properties", "Get", specialModeInterface, |
| "SpecialMode"); |
| } |
| |
| bool getManufacturingMode() |
| { |
| return manufacturingMode; |
| } |
| |
| std::vector<std::unique_ptr<sdbusplus::bus::match_t>> |
| setupPropertiesChangedMatches( |
| sdbusplus::asio::connection& bus, std::span<const char* const> types, |
| const std::function<void(sdbusplus::message_t&)>& handler) |
| { |
| std::vector<std::unique_ptr<sdbusplus::bus::match_t>> matches; |
| for (const char* type : types) |
| { |
| auto match = std::make_unique<sdbusplus::bus::match_t>( |
| static_cast<sdbusplus::bus_t&>(bus), |
| "type='signal',member='PropertiesChanged',path_namespace='" + |
| std::string(inventoryPath) + "',arg0namespace='" + type + "'", |
| handler); |
| matches.emplace_back(std::move(match)); |
| } |
| return matches; |
| } |