blob: c9d0c1585aaad99cf4499a5b96ca353a9d660818 [file] [log] [blame]
/*
// Copyright (c) 2018 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 <Overlay.hpp>
#include <dbus/properties.hpp>
#include <nlohmann/json.hpp>
#include <fstream>
#include <future>
#include <regex>
#include <boost/algorithm/string/predicate.hpp>
#include <boost/algorithm/string/replace.hpp>
#include <boost/variant/apply_visitor.hpp>
#include <boost/lexical_cast.hpp>
#include <boost/container/flat_map.hpp>
#include <boost/container/flat_set.hpp>
#include <dbus/connection.hpp>
#include <VariantVisitors.hpp>
#include <experimental/filesystem>
constexpr const char *OUTPUT_DIR = "/var/configuration/";
constexpr const char *CONFIGURATION_DIR = "/usr/share/configurations";
constexpr const char *TEMPLATE_CHAR = "$";
constexpr const size_t PROPERTIES_CHANGED_UNTIL_FLUSH_COUNT = 20;
constexpr const size_t MAX_MAPPER_DEPTH = 99;
constexpr const size_t SLEEP_AFTER_PROPERTIES_CHANGE_SECONDS = 5;
namespace fs = std::experimental::filesystem;
struct cmp_str
{
bool operator()(const char *a, const char *b) const
{
return std::strcmp(a, b) < 0;
}
};
// underscore T for collison with dbus c api
enum class probe_type_codes
{
FALSE_T,
TRUE_T,
AND,
OR,
FOUND,
MATCH_ONE
};
const static boost::container::flat_map<const char *, probe_type_codes, cmp_str>
PROBE_TYPES{{{"FALSE", probe_type_codes::FALSE_T},
{"TRUE", probe_type_codes::TRUE_T},
{"AND", probe_type_codes::AND},
{"OR", probe_type_codes::OR},
{"FOUND", probe_type_codes::FOUND},
{"MATCH_ONE", probe_type_codes::MATCH_ONE}}};
using GetSubTreeType = std::vector<
std::pair<std::string,
std::vector<std::pair<std::string, std::vector<std::string>>>>>;
using ManagedObjectType = boost::container::flat_map<
dbus::object_path,
boost::container::flat_map<
std::string,
boost::container::flat_map<std::string, dbus::dbus_variant>>>;
boost::container::flat_map<
std::string,
std::vector<boost::container::flat_map<std::string, dbus::dbus_variant>>>
DBUS_PROBE_OBJECTS;
std::vector<std::string> PASSED_PROBES;
// todo: pass this through nicer
std::shared_ptr<dbus::connection> SYSTEM_BUS;
std::regex ILLEGAL_DBUS_REGEX("[^A-Za-z0-9_]");
void registerCallbacks(
std::vector<std::pair<std::unique_ptr<dbus::match>,
std::shared_ptr<dbus::filter>>> &dbusMatches,
nlohmann::json &systemConfiguration, dbus::DbusObjectServer &objServer);
// calls the mapper to find all exposed objects of an interface type
// and creates a vector<flat_map> that contains all the key value pairs
// getManagedObjects
bool findDbusObjects(
std::shared_ptr<dbus::connection> connection,
std::vector<boost::container::flat_map<std::string, dbus::dbus_variant>>
&interfaceDevices,
std::string interface)
{
// todo: this is only static because the mapper is unreliable as of today
static boost::container::flat_map<std::string,
boost::container::flat_set<std::string>>
connections;
// find all connections in the mapper that expose a specific type
static const dbus::endpoint mapper("xyz.openbmc_project.ObjectMapper",
"/xyz/openbmc_project/object_mapper",
"xyz.openbmc_project.ObjectMapper",
"GetSubTree");
dbus::message getMap = dbus::message::new_call(mapper);
std::vector<std::string> objects = {interface};
if (!getMap.pack("", MAX_MAPPER_DEPTH, objects))
{
std::cerr << "Pack Failed GetSensorSubtree\n";
return false;
}
GetSubTreeType interfaceSubtree;
size_t retries = 1;
bool unpackStatus = false;
// the mapper seems to hang occasionally, not responding, so we give it a
// timeout and retries
do
{
dbus::message getMapResp =
connection->send(getMap, std::chrono::seconds(2));
unpackStatus = getMapResp.unpack(interfaceSubtree);
} while (retries-- && !unpackStatus);
auto &interfaceConnections = connections[interface];
if (!unpackStatus)
{
std::cerr << "Error communicating to mapper, using cached data if "
"available\n";
if (interfaceConnections.empty())
{
return false;
}
}
if (unpackStatus)
{
interfaceConnections.clear();
for (auto &object : interfaceSubtree)
{
for (auto &connPair : object.second)
{
interfaceConnections.insert(connPair.first);
}
}
}
// iterate through the connections, adding creating individual device
// dictionaries
for (auto &conn : interfaceConnections)
{
auto managedObj =
dbus::endpoint(conn, "/", "org.freedesktop.DBus.ObjectManager",
"GetManagedObjects");
dbus::message getManagedObj = dbus::message::new_call(managedObj);
ManagedObjectType managedInterface;
retries = 1;
unpackStatus = false;
do
{
dbus::message getManagedObjResp = connection->send(getManagedObj);
unpackStatus = getManagedObjResp.unpack(managedInterface);
} while (retries-- && !unpackStatus);
if (!unpackStatus)
{
std::cerr << "error getting managed object for device " << conn
<< "\n";
continue;
}
for (auto &interfaceManagedObj : managedInterface)
{
auto ifaceObjFind = interfaceManagedObj.second.find(interface);
if (ifaceObjFind != interfaceManagedObj.second.end())
{
interfaceDevices.emplace_back(ifaceObjFind->second);
}
}
}
return true;
}
// probes interface dictionary for a key with a value that matches a regex
bool probeDbus(
const std::string &interface,
const std::map<std::string, nlohmann::json> &matches,
std::vector<boost::container::flat_map<std::string, dbus::dbus_variant>>
&devices,
bool &foundProbe)
{
auto &dbusObject = DBUS_PROBE_OBJECTS[interface];
if (dbusObject.empty())
{
if (!findDbusObjects(SYSTEM_BUS, dbusObject, interface))
{
std::cerr << "Found no dbus objects with interface "
<< interface << "\n";
foundProbe = false;
return false;
}
}
foundProbe = true;
bool foundMatch = false;
for (auto &device : dbusObject)
{
bool deviceMatches = true;
for (auto &match : matches)
{
auto deviceValue = device.find(match.first);
if (deviceValue != device.end())
{
switch (match.second.type())
{
case nlohmann::json::value_t::string:
{
std::regex search(match.second.get<std::string>());
std::smatch match;
// convert value to string respresentation
std::string probeValue = boost::apply_visitor(
[](const auto &x) {
return boost::lexical_cast<std::string>(x);
},
deviceValue->second);
if (!std::regex_search(probeValue, match, search))
{
deviceMatches = false;
break;
}
break;
}
case nlohmann::json::value_t::boolean:
case nlohmann::json::value_t::number_unsigned:
{
unsigned int probeValue = boost::apply_visitor(
VariantToUnsignedIntVisitor(), deviceValue->second);
if (probeValue != match.second.get<unsigned int>())
{
deviceMatches = false;
}
break;
}
case nlohmann::json::value_t::number_integer:
{
int probeValue = boost::apply_visitor(VariantToIntVisitor(),
deviceValue->second);
if (probeValue != match.second.get<int>())
{
deviceMatches = false;
}
break;
}
case nlohmann::json::value_t::number_float:
{
float probeValue = boost::apply_visitor(
VariantToFloatVisitor(), deviceValue->second);
if (probeValue != match.second.get<float>())
{
deviceMatches = false;
}
break;
}
}
}
else
{
deviceMatches = false;
break;
}
}
if (deviceMatches)
{
devices.emplace_back(
boost::container::flat_map<std::string, dbus::dbus_variant>(
device));
foundMatch = true;
deviceMatches = false; // for next iteration
}
}
return foundMatch;
}
// default probe entry point, iterates a list looking for specific types to
// call specific probe functions
bool probe(
const std::vector<std::string> probeCommand,
std::vector<boost::container::flat_map<std::string, dbus::dbus_variant>>
&foundDevs)
{
const static std::regex command(R"(\((.*)\))");
std::smatch match;
bool ret = false;
bool matchOne = false;
bool cur = true;
probe_type_codes lastCommand = probe_type_codes::FALSE_T;
for (auto &probe : probeCommand)
{
bool foundProbe = false;
boost::container::flat_map<const char *, probe_type_codes,
cmp_str>::const_iterator probeType;
for (probeType = PROBE_TYPES.begin(); probeType != PROBE_TYPES.end();
probeType++)
{
if (probe.find(probeType->first) != std::string::npos)
{
foundProbe = true;
break;
}
}
if (foundProbe)
{
switch (probeType->second)
{
case probe_type_codes::FALSE_T:
{
return false; // todo, actually evaluate?
break;
}
case probe_type_codes::TRUE_T:
{
return true; // todo, actually evaluate?
break;
}
case probe_type_codes::MATCH_ONE:
{
// set current value to last, this probe type shouldn't affect
// the outcome
cur = ret;
matchOne = true;
break;
}
/*case probe_type_codes::AND:
break;
case probe_type_codes::OR:
break;
// these are no-ops until the last command switch
*/
case probe_type_codes::FOUND:
{
if (!std::regex_search(probe, match, command))
{
std::cerr << "found probe sytax error " << probe << "\n";
return false;
}
std::string commandStr = *(match.begin() + 1);
boost::replace_all(commandStr, "'", "");
cur = (std::find(PASSED_PROBES.begin(), PASSED_PROBES.end(),
commandStr) != PASSED_PROBES.end());
break;
}
}
}
// look on dbus for object
else
{
if (!std::regex_search(probe, match, command))
{
std::cerr << "dbus probe sytax error " << probe << "\n";
return false;
}
std::string commandStr = *(match.begin() + 1);
// convert single ticks and single slashes into legal json
boost::replace_all(commandStr, "'", R"(")");
boost::replace_all(commandStr, R"(\)", R"(\\)");
auto json = nlohmann::json::parse(commandStr, nullptr, false);
if (json.is_discarded())
{
std::cerr << "dbus command sytax error " << commandStr << "\n";
return false;
}
// we can match any (string, variant) property. (string, string)
// does a regex
std::map<std::string, nlohmann::json> dbusProbeMap =
json.get<std::map<std::string, nlohmann::json>>();
auto findStart = probe.find("(");
if (findStart == std::string::npos)
{
return false;
}
std::string probeInterface = probe.substr(0, findStart);
cur =
probeDbus(probeInterface, dbusProbeMap, foundDevs, foundProbe);
}
// some functions like AND and OR only take affect after the
// fact
switch (lastCommand)
{
case probe_type_codes::AND:
ret = cur && ret;
break;
case probe_type_codes::OR:
ret = cur || ret;
break;
default:
ret = cur;
break;
}
lastCommand = probeType != PROBE_TYPES.end()
? probeType->second
: probe_type_codes::FALSE_T;
if (!foundProbe)
{
std::cerr << "Illegal probe type " << probe << "\n";
return false;
}
}
// probe passed, but empty device
// todo: should this be done in main?
if (ret && foundDevs.size() == 0)
{
foundDevs.emplace_back(
boost::container::flat_map<std::string, dbus::dbus_variant>());
}
if (matchOne && foundDevs.size() > 1)
{
foundDevs.erase(foundDevs.begin() + 1, foundDevs.end());
}
return ret;
}
// this function is temporary, no need to have once dbus is solified.
void writeJsonFiles(nlohmann::json &systemConfiguration)
{
std::experimental::filesystem::create_directory(OUTPUT_DIR);
std::ofstream output(std::string(OUTPUT_DIR) + "system.json");
output << systemConfiguration.dump(4);
output.close();
auto flat = nlohmann::json::array();
for (auto &pair : nlohmann::json::iterator_wrapper(systemConfiguration))
{
auto value = pair.value();
auto exposes = value.find("exposes");
if (exposes != value.end())
{
for (auto &item : *exposes)
{
flat.push_back(item);
}
}
}
output = std::ofstream(std::string(OUTPUT_DIR) + "flattened.json");
output << flat.dump(4);
output.close();
}
// adds simple json types to interface's properties
void populateInterfaceFromJson(dbus::DbusInterface *iface, nlohmann::json dict,
dbus::DbusObjectServer &objServer)
{
std::vector<std::pair<std::string, dbus::dbus_variant>> properties;
static size_t flushCount = 0;
for (auto &dictPair : nlohmann::json::iterator_wrapper(dict))
{
switch (dictPair.value().type())
{
case (nlohmann::json::value_t::boolean):
{
properties.emplace_back(std::string(dictPair.key()),
dictPair.value().get<bool>());
break;
}
case (nlohmann::json::value_t::number_integer):
{
properties.emplace_back(std::string(dictPair.key()),
dictPair.value().get<int64_t>());
break;
}
case (nlohmann::json::value_t::number_unsigned):
{
properties.emplace_back(std::string(dictPair.key()),
dictPair.value().get<uint64_t>());
break;
}
case (nlohmann::json::value_t::number_float):
{
properties.emplace_back(std::string(dictPair.key()),
dictPair.value().get<float>());
break;
}
case (nlohmann::json::value_t::string):
{
properties.emplace_back(std::string(dictPair.key()),
dictPair.value().get<std::string>());
break;
}
}
}
if (!properties.empty())
{
iface->set_properties(properties);
// flush the queue after adding an amount of properties so we don't hang
if (flushCount++ > PROPERTIES_CHANGED_UNTIL_FLUSH_COUNT)
{
objServer.flush();
flushCount = 0;
}
}
}
void postToDbus(const nlohmann::json &systemConfiguration,
dbus::DbusObjectServer &objServer)
{
for (auto &boardPair :
nlohmann::json::iterator_wrapper(systemConfiguration))
{
std::string boardKey = boardPair.key();
auto boardValues = boardPair.value();
auto findBoardType = boardValues.find("type");
std::string boardType;
if (findBoardType != boardValues.end() &&
findBoardType->type() == nlohmann::json::value_t::string)
{
boardType = findBoardType->get<std::string>();
std::regex_replace(boardType.begin(), boardType.begin(),
boardType.end(), ILLEGAL_DBUS_REGEX, "_");
}
else
{
std::cerr << "Unable to find type for " << boardKey
<< " reverting to Chassis.\n";
boardType = "Chassis";
}
std::regex_replace(boardKey.begin(), boardKey.begin(), boardKey.end(),
ILLEGAL_DBUS_REGEX, "_");
std::string boardName =
"/xyz/openbmc_project/Inventory/Item/" + boardType + "/" + boardKey;
auto boardObject = objServer.add_object(boardName);
auto boardIface = boardObject->add_interface(
"xyz.openbmc_project.Configuration." + boardType);
populateInterfaceFromJson(boardIface.get(), boardValues, objServer);
auto exposes = boardValues.find("exposes");
if (exposes == boardValues.end())
{
continue;
}
for (auto &item : *exposes)
{
auto findName = item.find("name");
if (findName == item.end())
{
std::cerr << "cannot find name in field " << item << "\n";
continue;
}
auto findStatus = item.find("status");
// if status is not found it is assumed to be status = 'okay'
if (findStatus != item.end())
{
if (*findStatus == "disabled")
{
continue;
}
}
auto findType = item.find("type");
std::string itemType;
if (findType != item.end())
{
itemType = findType->get<std::string>();
std::regex_replace(itemType.begin(), itemType.begin(),
itemType.end(), ILLEGAL_DBUS_REGEX, "_");
}
else
{
itemType = "unknown";
}
std::string itemName = findName->get<std::string>();
std::regex_replace(itemName.begin(), itemName.begin(),
itemName.end(), ILLEGAL_DBUS_REGEX, "_");
auto itemObject = objServer.add_object(boardName + "/" + itemName);
auto itemIface = itemObject->add_interface(
"xyz.openbmc_project.Configuration." + itemType);
populateInterfaceFromJson(itemIface.get(), item, objServer);
for (auto &objectPair : nlohmann::json::iterator_wrapper(item))
{
if (objectPair.value().type() ==
nlohmann::json::value_t::object)
{
auto objectIface = itemObject->add_interface(
"xyz.openbmc_project.Configuration." + itemType + "." +
objectPair.key());
populateInterfaceFromJson(objectIface.get(),
objectPair.value(), objServer);
}
else if (objectPair.value().type() ==
nlohmann::json::value_t::array)
{
size_t index = 0;
for (auto &arrayItem : objectPair.value())
{
if (arrayItem.type() != nlohmann::json::value_t::object)
{
std::cerr << "dbus format error" << arrayItem
<< "\n";
break;
}
auto objectIface = itemObject->add_interface(
"xyz.openbmc_project.Configuration." + itemType +
"." + objectPair.key() + "." +
std::to_string(index));
index++;
populateInterfaceFromJson(objectIface.get(), arrayItem,
objServer);
}
}
}
}
}
}
// finds the template character (currently set to $) and replaces the value with
// the field found in a dbus object i.e. $ADDRESS would get populated with the
// ADDRESS field from a object on dbus
void templateCharReplace(
nlohmann::json::iterator &keyPair,
const boost::container::flat_map<std::string, dbus::dbus_variant>
&foundDevice,
size_t &foundDeviceIdx)
{
if (keyPair.value().type() != nlohmann::json::value_t::string)
{
return;
}
std::string value = keyPair.value();
if (value.find(TEMPLATE_CHAR) != std::string::npos)
{
std::string templateValue = value;
templateValue.erase(0, 1); // remove template character
// special case index
if ("index" == templateValue)
{
keyPair.value() = foundDeviceIdx;
}
else
{
std::string subsitute;
for (auto &foundDevicePair : foundDevice)
{
if (boost::iequals(foundDevicePair.first, templateValue))
{
// convert value to string
// respresentation
subsitute = boost::apply_visitor(
[](const auto &x) {
return boost::lexical_cast<std::string>(x);
},
foundDevicePair.second);
break;
}
}
if (!subsitute.size())
{
std::cerr << "could not find symbol " << templateValue << "\n";
}
else
{
keyPair.value() = subsitute;
}
}
}
}
bool findJsonFiles(std::vector<nlohmann::json> &configurations)
{
// find configuration files
std::vector<fs::path> jsonPaths;
if (!find_files(fs::path(CONFIGURATION_DIR), R"(.*\.json)", jsonPaths, 0))
{
std::cerr << "Unable to find any configuration files in "
<< CONFIGURATION_DIR << "\n";
return false;
}
for (auto &jsonPath : jsonPaths)
{
std::ifstream jsonStream(jsonPath.c_str());
if (!jsonStream.good())
{
std::cerr << "unable to open " << jsonPath.string() << "\n";
continue;
}
auto data = nlohmann::json::parse(jsonStream, nullptr, false);
if (data.is_discarded())
{
std::cerr << "syntax error in " << jsonPath.string() << "\n";
continue;
}
if (data.type() == nlohmann::json::value_t::array)
{
for (auto &d : data)
{
configurations.emplace_back(d);
}
}
else
{
configurations.emplace_back(data);
}
}
}
bool rescan(nlohmann::json &systemConfiguration)
{
std::vector<nlohmann::json> configurations;
if (!findJsonFiles(configurations))
{
false;
}
// preprocess already passed configurations and missing fields
if (systemConfiguration.size())
{
for (auto it = configurations.begin(); it != configurations.end();)
{
auto findName = it->find("name");
if (findName == it->end())
{
std::cerr << "configuration missing name field " << *it << "\n";
it = configurations.erase(it);
continue;
}
else if (findName->type() != nlohmann::json::value_t::string)
{
std::cerr << "name field must be a string " << *findName
<< "\n";
it = configurations.erase(it);
continue;
}
auto findAlreadyFound =
systemConfiguration.find(findName->get<std::string>());
if (findAlreadyFound != systemConfiguration.end())
{
it = configurations.erase(it);
continue;
}
// TODO: add in tags to determine if configuration should be
// refreshed on AC / DC / Always.
it++;
}
}
// probe until no probes pass
bool probePassed = true;
while (probePassed)
{
probePassed = false;
for (auto it = configurations.begin(); it != configurations.end();)
{
bool eraseConfig = false;
auto findProbe = it->find("probe");
auto findName = it->find("name");
nlohmann::json probeCommand;
// check for poorly formatted fields, probe must be an array
if (findProbe == it->end())
{
std::cerr << "configuration file missing probe:\n " << *it
<< "\n";
eraseConfig = true;
}
else if ((*findProbe).type() != nlohmann::json::value_t::array)
{
probeCommand = nlohmann::json::array();
probeCommand.push_back(*findProbe);
}
else
{
probeCommand = *findProbe;
}
if (findName == it->end())
{
std::cerr << "configuration file missing name:\n " << *it
<< "\n";
eraseConfig = true;
}
std::vector<
boost::container::flat_map<std::string, dbus::dbus_variant>>
foundDevices;
if (!eraseConfig && probe(probeCommand, foundDevices))
{
eraseConfig = true;
probePassed = true;
std::string name = *findName;
PASSED_PROBES.push_back(name);
size_t foundDeviceIdx = 0;
for (auto &foundDevice : foundDevices)
{
for (auto keyPair = it->begin(); keyPair != it->end();
keyPair++)
{
templateCharReplace(keyPair, foundDevice,
foundDeviceIdx);
}
auto findExpose = it->find("exposes");
if (findExpose == it->end())
{
continue;
}
for (auto &expose : *findExpose)
{
for (auto keyPair = expose.begin();
keyPair != expose.end(); keyPair++)
{
// fill in template characters with devices
// found
templateCharReplace(keyPair, foundDevice,
foundDeviceIdx);
// special case bind
if (boost::starts_with(keyPair.key(), "bind_"))
{
if (keyPair.value().type() !=
nlohmann::json::value_t::string)
{
std::cerr
<< "bind_ value must be of type string "
<< keyPair.key() << "\n";
continue;
}
bool foundBind = false;
std::string bind =
keyPair.key().substr(sizeof("bind_") - 1);
for (auto &configurationPair :
nlohmann::json::iterator_wrapper(
systemConfiguration))
{
auto configListFind =
configurationPair.value().find(
"exposes");
if (configListFind ==
configurationPair.value().end() ||
configListFind->type() !=
nlohmann::json::value_t::array)
{
continue;
}
for (auto &exposedObject : *configListFind)
{
std::string foundObjectName =
(exposedObject)["name"];
if (boost::iequals(
foundObjectName,
keyPair.value()
.get<std::string>()))
{
exposedObject["status"] = "okay";
expose[bind] = exposedObject;
foundBind = true;
break;
}
}
if (foundBind)
{
break;
}
}
if (!foundBind)
{
std::cerr << "configuration file "
"dependency error, "
"could not find bind "
<< keyPair.value() << "\n";
}
}
}
}
}
systemConfiguration[name] = (*it);
foundDeviceIdx++;
}
if (eraseConfig)
{
it = configurations.erase(it);
}
else
{
it++;
}
}
}
}
void propertiesChangedCallback(
std::vector<std::pair<std::unique_ptr<dbus::match>,
std::shared_ptr<dbus::filter>>> &dbusMatches,
nlohmann::json &systemConfiguration, dbus::DbusObjectServer &objServer,
std::shared_ptr<dbus::filter> dbusFilter)
{
static std::future<void> future;
static std::atomic_bool threadRunning(false);
static std::atomic_bool pendingCallback(false);
bool notRunning = false;
if (threadRunning.compare_exchange_strong(notRunning, true))
{
future = std::async(std::launch::async, [&] {
do
{
std::this_thread::sleep_for(std::chrono::seconds(
SLEEP_AFTER_PROPERTIES_CHANGE_SECONDS));
auto oldConfiguration = systemConfiguration;
DBUS_PROBE_OBJECTS.clear();
pendingCallback = false;
rescan(systemConfiguration);
auto newConfiguration = systemConfiguration;
for (auto it = newConfiguration.begin();
it != newConfiguration.end();)
{
auto findKey = oldConfiguration.find(it.key());
if (findKey != oldConfiguration.end())
{
it = newConfiguration.erase(it);
}
else
{
it++;
}
}
registerCallbacks(dbusMatches, systemConfiguration, objServer);
// todo: for now, only add new configurations, unload to come
// later
// unloadOverlays();
loadOverlays(newConfiguration);
// this line to be removed in future
writeJsonFiles(systemConfiguration);
// only post new items to bus for now
postToDbus(newConfiguration, objServer);
} while (pendingCallback);
threadRunning = false;
});
}
else
{
pendingCallback = true;
}
if (dbusFilter != nullptr)
{
dbusFilter->async_dispatch([&, dbusFilter](boost::system::error_code ec,
dbus::message) {
if (ec)
{
std::cerr << "properties changed callback error " << ec << "\n";
}
propertiesChangedCallback(dbusMatches, systemConfiguration,
objServer, dbusFilter);
});
}
}
void registerCallbacks(
std::vector<std::pair<std::unique_ptr<dbus::match>,
std::shared_ptr<dbus::filter>>> &dbusMatches,
nlohmann::json &systemConfiguration, dbus::DbusObjectServer &objServer)
{
static boost::container::flat_set<std::string> watchedObjects;
for (const auto &objectMap : DBUS_PROBE_OBJECTS)
{
auto findObject = watchedObjects.find(objectMap.first);
if (findObject != watchedObjects.end())
{
continue;
}
// this creates a filter for properties changed for any new probe type
auto propertyChange = std::make_unique<dbus::match>(
SYSTEM_BUS,
"type='signal',member='PropertiesChanged',arg0='" +
objectMap.first + "'");
auto filter =
std::make_shared<dbus::filter>(SYSTEM_BUS, [](dbus::message &m) {
auto member = m.get_member();
return member == "PropertiesChanged";
});
filter->async_dispatch([&, filter](boost::system::error_code ec,
dbus::message) {
if (ec)
{
std::cerr << "register callbacks callback error " << ec << "\n";
}
propertiesChangedCallback(dbusMatches, systemConfiguration,
objServer, filter);
});
dbusMatches.emplace_back(std::move(propertyChange), filter);
}
}
int main(int argc, char **argv)
{
// setup connection to dbus
boost::asio::io_service io;
SYSTEM_BUS = std::make_shared<dbus::connection>(io, dbus::bus::system);
dbus::DbusObjectServer objServer(SYSTEM_BUS);
SYSTEM_BUS->request_name("xyz.openbmc_project.EntityManager");
std::vector<
std::pair<std::unique_ptr<dbus::match>, std::shared_ptr<dbus::filter>>>
dbusMatches;
nlohmann::json systemConfiguration = nlohmann::json::object();
auto iface = std::make_shared<dbus::DbusInterface>(
"xyz.openbmc_project.EntityManager", SYSTEM_BUS);
io.post([&]() {
unloadAllOverlays();
propertiesChangedCallback(dbusMatches, systemConfiguration, objServer,
nullptr);
auto object = std::make_shared<dbus::DbusObject>(
SYSTEM_BUS, "/xyz/openbmc_project/EntityManager");
objServer.register_object(object);
object->register_interface(iface);
});
// to keep reference to the match / filter objects so they don't get
// destroyed
iface->register_method("ReScan", [&]() {
propertiesChangedCallback(dbusMatches, systemConfiguration, objServer,
nullptr);
return std::tuple<>(); // this is a bug in boost-dbus, needs some sort
// of return
});
io.run();
return 0;
}