xref: /openbmc/phosphor-dbus-monitor/src/propertywatchimpl.hpp (revision eab4f8c0a047e1aaedf74d6144d83132d1b003de)
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