14b916f13SBrad Bishop #pragma once
24b916f13SBrad Bishop
3fccdc39fSBrad Bishop #include "callback.hpp"
44b916f13SBrad Bishop #include "data_types.hpp"
54b916f13SBrad Bishop #include "propertywatch.hpp"
64b916f13SBrad Bishop
73d6d3182SPatrick Venture #include <sdbusplus/bus/match.hpp>
83d6d3182SPatrick Venture #include <sdbusplus/message.hpp>
93fe976ccSGeorge Liu
10ae4c95c6SAndrew Geissler #include <string>
113d6d3182SPatrick Venture #include <vector>
123d6d3182SPatrick Venture
134b916f13SBrad Bishop namespace phosphor
144b916f13SBrad Bishop {
154b916f13SBrad Bishop namespace dbus
164b916f13SBrad Bishop {
174b916f13SBrad Bishop namespace monitoring
184b916f13SBrad Bishop {
194b916f13SBrad Bishop
204b916f13SBrad Bishop using MappedPropertyIndex =
214b916f13SBrad Bishop RefKeyMap<const std::string,
22d1eac88dSBrad Bishop RefKeyMap<const std::string, RefVector<const std::string>>>;
234b916f13SBrad Bishop
244b916f13SBrad Bishop MappedPropertyIndex convert(const PropertyIndex& index);
254b916f13SBrad Bishop
264b916f13SBrad Bishop template <typename DBusInterfaceType>
start()274b916f13SBrad Bishop void PropertyWatch<DBusInterfaceType>::start()
284b916f13SBrad Bishop {
294b916f13SBrad Bishop if (alreadyRan)
304b916f13SBrad Bishop {
314b916f13SBrad Bishop return;
324b916f13SBrad Bishop }
334b916f13SBrad Bishop
344b916f13SBrad Bishop // The index has a flat layout which is not optimal here. Nest
354b916f13SBrad Bishop // properties in a map of interface names in a map of object paths.
364b916f13SBrad Bishop auto mapped = convert(index);
374b916f13SBrad Bishop
384b916f13SBrad Bishop for (const auto& m : mapped)
394b916f13SBrad Bishop {
404b916f13SBrad Bishop const auto& path = m.first.get();
414b916f13SBrad Bishop const auto& interfaces = m.second;
424b916f13SBrad Bishop
434b916f13SBrad Bishop // Watch for new interfaces on this path.
444b916f13SBrad Bishop DBusInterfaceType::addMatch(
454b916f13SBrad Bishop sdbusplus::bus::match::rules::interfacesAdded(path),
464b916f13SBrad Bishop [this](auto& msg)
474b916f13SBrad Bishop // *INDENT-OFF*
48d1eac88dSBrad Bishop { this->interfacesAdded(msg); });
494b916f13SBrad Bishop // *INDENT-ON*
504b916f13SBrad Bishop
514b916f13SBrad Bishop // Do a query to populate the cache. Start with a mapper query.
524b916f13SBrad Bishop // The specific services are queried below.
53cca7a7c8SBrad Bishop auto getObjectFromMapper = [](const auto& path) {
544b916f13SBrad Bishop const std::vector<std::string> queryInterfaces; // all interfaces
55cca7a7c8SBrad Bishop try
56cca7a7c8SBrad Bishop {
57cca7a7c8SBrad Bishop return DBusInterfaceType::template callMethodAndRead<GetObject>(
58d1eac88dSBrad Bishop MAPPER_BUSNAME, MAPPER_PATH, MAPPER_INTERFACE, "GetObject",
59d1eac88dSBrad Bishop path, queryInterfaces);
60cca7a7c8SBrad Bishop }
61*413a4857SPatrick Williams catch (const sdbusplus::exception_t&)
62cca7a7c8SBrad Bishop {
63cca7a7c8SBrad Bishop // Paths in the configuration may not exist yet. Prime those
64cca7a7c8SBrad Bishop // later, when/if InterfacesAdded occurs.
65cca7a7c8SBrad Bishop return GetObject();
66cca7a7c8SBrad Bishop }
67cca7a7c8SBrad Bishop };
68cca7a7c8SBrad Bishop auto mapperResp = getObjectFromMapper(path);
694b916f13SBrad Bishop
704b916f13SBrad Bishop for (const auto& i : interfaces)
714b916f13SBrad Bishop {
724b916f13SBrad Bishop const auto& interface = i.first.get();
734b916f13SBrad Bishop
744b916f13SBrad Bishop // Watch for property changes on this interface.
754b916f13SBrad Bishop DBusInterfaceType::addMatch(
76d1eac88dSBrad Bishop sdbusplus::bus::match::rules::propertiesChanged(path,
77d1eac88dSBrad Bishop interface),
784b916f13SBrad Bishop [this](auto& msg)
794b916f13SBrad Bishop // *INDENT-OFF*
804b916f13SBrad Bishop {
814b916f13SBrad Bishop std::string interface;
824b916f13SBrad Bishop msg.read(interface);
834b916f13SBrad Bishop auto path = msg.get_path();
844b916f13SBrad Bishop this->propertiesChanged(msg, path, interface);
854b916f13SBrad Bishop });
864b916f13SBrad Bishop // *INDENT-ON*
874b916f13SBrad Bishop
884b916f13SBrad Bishop // The mapper response is a busname:[interfaces] map. Look for
894b916f13SBrad Bishop // each interface in the index and if found, query the service and
904b916f13SBrad Bishop // populate the cache entries for the interface.
914b916f13SBrad Bishop for (const auto& mr : mapperResp)
924b916f13SBrad Bishop {
934b916f13SBrad Bishop const auto& busName = mr.first;
944b916f13SBrad Bishop const auto& mapperInterfaces = mr.second;
95d1eac88dSBrad Bishop if (mapperInterfaces.end() ==
96d1eac88dSBrad Bishop std::find(mapperInterfaces.begin(), mapperInterfaces.end(),
974b916f13SBrad Bishop interface))
984b916f13SBrad Bishop {
994b916f13SBrad Bishop // This interface isn't being watched.
1004b916f13SBrad Bishop continue;
1014b916f13SBrad Bishop }
1024b916f13SBrad Bishop
1034b916f13SBrad Bishop // Delegate type specific property updates to subclasses.
104cca7a7c8SBrad Bishop try
105cca7a7c8SBrad Bishop {
1064b916f13SBrad Bishop updateProperties(busName, path, interface);
1074b916f13SBrad Bishop }
108*413a4857SPatrick Williams catch (const sdbusplus::exception_t&)
109cca7a7c8SBrad Bishop {
110cca7a7c8SBrad Bishop // If for some reason the path has gone away since
111cca7a7c8SBrad Bishop // the mapper lookup we'll simply try again if/when
112cca7a7c8SBrad Bishop // InterfacesAdded occurs the next time it shows up.
113cca7a7c8SBrad Bishop }
114cca7a7c8SBrad Bishop }
1154b916f13SBrad Bishop }
1164b916f13SBrad Bishop }
117c1283ae8SBrad Bishop
118c1283ae8SBrad Bishop alreadyRan = true;
1194b916f13SBrad Bishop }
1204b916f13SBrad Bishop
121ce4fbe11SBrad Bishop template <typename DBusInterfaceType>
callback(Context ctx)122a45e086dSRatan Gupta void PropertyWatch<DBusInterfaceType>::callback(Context ctx)
123ce4fbe11SBrad Bishop {
12498d6462aSLei YU // Ignore callback if ignoreStartCallback is true and it's the START
12598d6462aSLei YU // callback
12698d6462aSLei YU if (ctx == Context::START && ignoreStartCallback)
12798d6462aSLei YU {
12898d6462aSLei YU return;
12998d6462aSLei YU }
130ce4fbe11SBrad Bishop // Invoke callback if present.
131ce4fbe11SBrad Bishop if (this->alreadyRan && this->cb)
132ce4fbe11SBrad Bishop {
133a45e086dSRatan Gupta (*this->cb)(ctx);
134ce4fbe11SBrad Bishop }
135ce4fbe11SBrad Bishop }
136ce4fbe11SBrad Bishop
1374b916f13SBrad Bishop template <typename T, typename DBusInterfaceType>
updateProperties(const std::string & busName,const std::string & path,const std::string & interface)1384b916f13SBrad Bishop void PropertyWatchOfType<T, DBusInterfaceType>::updateProperties(
139d1eac88dSBrad Bishop const std::string& busName, const std::string& path,
1404b916f13SBrad Bishop const std::string& interface)
1414b916f13SBrad Bishop {
1424b916f13SBrad Bishop auto properties =
1434b916f13SBrad Bishop DBusInterfaceType::template callMethodAndRead<PropertiesChanged<T>>(
144d1eac88dSBrad Bishop busName.c_str(), path.c_str(), "org.freedesktop.DBus.Properties",
145d1eac88dSBrad Bishop "GetAll", interface);
1464b916f13SBrad Bishop propertiesChanged(path, interface, properties);
1474b916f13SBrad Bishop }
1484b916f13SBrad Bishop
1494b916f13SBrad Bishop template <typename T, typename DBusInterfaceType>
propertiesChanged(const std::string & path,const std::string & interface,const PropertiesChanged<T> & properties)1504b916f13SBrad Bishop void PropertyWatchOfType<T, DBusInterfaceType>::propertiesChanged(
151d1eac88dSBrad Bishop const std::string& path, const std::string& interface,
1524b916f13SBrad Bishop const PropertiesChanged<T>& properties)
1534b916f13SBrad Bishop {
1544b916f13SBrad Bishop // Update the cache for any watched properties.
1554b916f13SBrad Bishop for (const auto& p : properties)
1564b916f13SBrad Bishop {
1574b916f13SBrad Bishop auto key = std::make_tuple(path, interface, p.first);
1584b916f13SBrad Bishop auto item = this->index.find(key);
1594b916f13SBrad Bishop if (item == this->index.end())
1604b916f13SBrad Bishop {
1614b916f13SBrad Bishop // This property isn't being watched.
1624b916f13SBrad Bishop continue;
1634b916f13SBrad Bishop }
1644b916f13SBrad Bishop
165ae786ef6SMatthew Barth // Run property value thru filter operations
166ae786ef6SMatthew Barth auto isFiltered = false;
167ae786ef6SMatthew Barth const auto& storage = std::get<storageIndex>(item->second);
16834ef1e52SPatrick Williams auto value = std::get<T>(p.second);
169efe0158cSMatthew Barth if (filterOps)
170ae786ef6SMatthew Barth {
17126dc0bcbSPatrick Williams std::any anyValue = value;
172efe0158cSMatthew Barth if ((*filterOps)(anyValue))
173ae786ef6SMatthew Barth {
174ae786ef6SMatthew Barth // Property value filtered, clear it from storage so
175ae786ef6SMatthew Barth // callback functions do not use it
176ae786ef6SMatthew Barth isFiltered = true;
17726dc0bcbSPatrick Williams std::get<valueIndex>(storage.get()).reset();
178ae786ef6SMatthew Barth }
179ae786ef6SMatthew Barth }
180ae786ef6SMatthew Barth if (!isFiltered)
181ae786ef6SMatthew Barth {
182ae786ef6SMatthew Barth // Property value not filtered out, update
183ae786ef6SMatthew Barth std::get<valueIndex>(storage.get()) = value;
184fccdc39fSBrad Bishop // Invoke callback if present.
185a45e086dSRatan Gupta this->callback(Context::SIGNAL);
1864b916f13SBrad Bishop }
1874b916f13SBrad Bishop }
188ae786ef6SMatthew Barth }
1894b916f13SBrad Bishop
1904b916f13SBrad Bishop template <typename T, typename DBusInterfaceType>
propertiesChanged(sdbusplus::message_t & msg,const std::string & path,const std::string & interface)1914b916f13SBrad Bishop void PropertyWatchOfType<T, DBusInterfaceType>::propertiesChanged(
192*413a4857SPatrick Williams sdbusplus::message_t& msg, const std::string& path,
1934b916f13SBrad Bishop const std::string& interface)
1944b916f13SBrad Bishop {
1954b916f13SBrad Bishop PropertiesChanged<T> properties;
1964b916f13SBrad Bishop msg.read(properties);
1974b916f13SBrad Bishop propertiesChanged(path, interface, properties);
1984b916f13SBrad Bishop }
1994b916f13SBrad Bishop
2004b916f13SBrad Bishop template <typename T, typename DBusInterfaceType>
interfacesAdded(const std::string & path,const InterfacesAdded<T> & interfaces)2014b916f13SBrad Bishop void PropertyWatchOfType<T, DBusInterfaceType>::interfacesAdded(
202d1eac88dSBrad Bishop const std::string& path, const InterfacesAdded<T>& interfaces)
2034b916f13SBrad Bishop {
2044b916f13SBrad Bishop for (const auto& i : interfaces)
2054b916f13SBrad Bishop {
2064b916f13SBrad Bishop propertiesChanged(path, i.first, i.second);
2074b916f13SBrad Bishop }
2084b916f13SBrad Bishop }
2094b916f13SBrad Bishop
2104b916f13SBrad Bishop template <typename T, typename DBusInterfaceType>
interfacesAdded(sdbusplus::message_t & msg)2114b916f13SBrad Bishop void PropertyWatchOfType<T, DBusInterfaceType>::interfacesAdded(
212*413a4857SPatrick Williams sdbusplus::message_t& msg)
2134b916f13SBrad Bishop {
2144b916f13SBrad Bishop sdbusplus::message::object_path path;
2154b916f13SBrad Bishop InterfacesAdded<T> interfaces;
2164b916f13SBrad Bishop msg.read(path, interfaces);
2174b916f13SBrad Bishop interfacesAdded(path, interfaces);
2184b916f13SBrad Bishop }
2194b916f13SBrad Bishop
2204b916f13SBrad Bishop } // namespace monitoring
2214b916f13SBrad Bishop } // namespace dbus
2224b916f13SBrad Bishop } // namespace phosphor
223