blob: 7fd2b106e34b6d43ec45112dc53245d5cbabe8aa [file] [log] [blame]
Andrew Geisslera80a3af2019-02-04 14:01:49 -06001#include "associations.hpp"
Andrew Geissler3b025e62019-02-01 10:33:54 -06002#include "processing.hpp"
Matt Spinlerdd945862018-09-07 12:41:05 -05003#include "src/argument.hpp"
Matt Spinler35396c12019-04-05 11:46:57 -05004#include "types.hpp"
Matt Spinlerdd945862018-09-07 12:41:05 -05005
Ed Tanous60520632018-06-11 17:46:52 -07006#include <tinyxml2.h>
7
8#include <atomic>
9#include <boost/algorithm/string/predicate.hpp>
Ed Tanous21c60592020-08-17 23:43:46 -070010#include <boost/asio/io_context.hpp>
11#include <boost/asio/signal_set.hpp>
Ed Tanous60520632018-06-11 17:46:52 -070012#include <boost/container/flat_map.hpp>
Ed Tanous60520632018-06-11 17:46:52 -070013#include <chrono>
14#include <iomanip>
15#include <iostream>
16#include <sdbusplus/asio/connection.hpp>
17#include <sdbusplus/asio/object_server.hpp>
Konstantin Aladyshevb15df6b2022-01-11 14:50:55 +030018#include <xyz/openbmc_project/Common/error.hpp>
Ed Tanous60520632018-06-11 17:46:52 -070019
Matt Spinlere2359fb2019-04-05 14:11:33 -050020AssociationMaps associationMaps;
Matt Spinler937a2322019-01-23 13:54:22 -060021
Andrew Geissler82815da2019-02-04 12:19:41 -060022static WhiteBlackList service_whitelist;
23static WhiteBlackList service_blacklist;
Matt Spinlerdd945862018-09-07 12:41:05 -050024
Ed Tanous60520632018-06-11 17:46:52 -070025void update_owners(sdbusplus::asio::connection* conn,
26 boost::container::flat_map<std::string, std::string>& owners,
27 const std::string& new_object)
28{
29 if (boost::starts_with(new_object, ":"))
30 {
31 return;
32 }
33 conn->async_method_call(
34 [&, new_object](const boost::system::error_code ec,
35 const std::string& nameOwner) {
36 if (ec)
37 {
38 std::cerr << "Error getting owner of " << new_object << " : "
39 << ec << "\n";
40 return;
41 }
42 owners[nameOwner] = new_object;
43 },
44 "org.freedesktop.DBus", "/", "org.freedesktop.DBus", "GetNameOwner",
45 new_object);
46}
47
48void send_introspection_complete_signal(sdbusplus::asio::connection* system_bus,
49 const std::string& process_name)
50{
51 // TODO(ed) This signal doesn't get exposed properly in the
52 // introspect right now. Find out how to register signals in
53 // sdbusplus
54 sdbusplus::message::message m = system_bus->new_signal(
Brad Bishopa02cd542021-10-12 19:12:42 -040055 "/xyz/openbmc_project/object_mapper",
56 "xyz.openbmc_project.ObjectMapper.Private", "IntrospectionComplete");
Ed Tanous60520632018-06-11 17:46:52 -070057 m.append(process_name);
58 m.signal_send();
59}
60
61struct InProgressIntrospect
62{
63 InProgressIntrospect(
Ed Tanous21c60592020-08-17 23:43:46 -070064 sdbusplus::asio::connection* system_bus, boost::asio::io_context& io,
Matt Spinler11401e22019-04-08 13:13:25 -050065 const std::string& process_name, AssociationMaps& am
Matt Spinleraecabe82018-09-19 13:25:42 -050066#ifdef DEBUG
67 ,
Ed Tanous60520632018-06-11 17:46:52 -070068 std::shared_ptr<std::chrono::time_point<std::chrono::steady_clock>>
Matt Spinleraecabe82018-09-19 13:25:42 -050069 global_start_time
70#endif
71 ) :
Ed Tanous60520632018-06-11 17:46:52 -070072 system_bus(system_bus),
Matt Spinler11401e22019-04-08 13:13:25 -050073 io(io), process_name(process_name), assocMaps(am)
Matt Spinleraecabe82018-09-19 13:25:42 -050074#ifdef DEBUG
75 ,
Ed Tanous60520632018-06-11 17:46:52 -070076 global_start_time(global_start_time),
77 process_start_time(std::chrono::steady_clock::now())
Matt Spinleraecabe82018-09-19 13:25:42 -050078#endif
Ed Tanous60520632018-06-11 17:46:52 -070079 {
80 }
81 ~InProgressIntrospect()
82 {
83 send_introspection_complete_signal(system_bus, process_name);
Matt Spinleraecabe82018-09-19 13:25:42 -050084
85#ifdef DEBUG
Ed Tanous60520632018-06-11 17:46:52 -070086 std::chrono::duration<float> diff =
87 std::chrono::steady_clock::now() - process_start_time;
88 std::cout << std::setw(50) << process_name << " scan took "
89 << diff.count() << " seconds\n";
90
91 // If we're the last outstanding caller globally, calculate the
92 // time it took
93 if (global_start_time != nullptr && global_start_time.use_count() == 1)
94 {
95 diff = std::chrono::steady_clock::now() - *global_start_time;
96 std::cout << "Total scan took " << diff.count()
97 << " seconds to complete\n";
98 }
Matt Spinleraecabe82018-09-19 13:25:42 -050099#endif
Ed Tanous60520632018-06-11 17:46:52 -0700100 }
101 sdbusplus::asio::connection* system_bus;
Ed Tanous21c60592020-08-17 23:43:46 -0700102 boost::asio::io_context& io;
Ed Tanous60520632018-06-11 17:46:52 -0700103 std::string process_name;
Matt Spinler11401e22019-04-08 13:13:25 -0500104 AssociationMaps& assocMaps;
Matt Spinleraecabe82018-09-19 13:25:42 -0500105#ifdef DEBUG
Ed Tanous60520632018-06-11 17:46:52 -0700106 std::shared_ptr<std::chrono::time_point<std::chrono::steady_clock>>
107 global_start_time;
108 std::chrono::time_point<std::chrono::steady_clock> process_start_time;
Matt Spinleraecabe82018-09-19 13:25:42 -0500109#endif
Ed Tanous60520632018-06-11 17:46:52 -0700110};
111
Ed Tanous60520632018-06-11 17:46:52 -0700112void do_associations(sdbusplus::asio::connection* system_bus,
Matt Spinlere0b0e3a2019-04-08 10:39:23 -0500113 interface_map_type& interfaceMap,
Ed Tanous60520632018-06-11 17:46:52 -0700114 sdbusplus::asio::object_server& objectServer,
Adrian Ambrożewiczbb40bd32021-02-12 13:36:26 +0100115 const std::string& processName, const std::string& path,
116 int timeoutRetries = 0)
Ed Tanous60520632018-06-11 17:46:52 -0700117{
Adrian Ambrożewiczbb40bd32021-02-12 13:36:26 +0100118 constexpr int maxTimeoutRetries = 3;
Ed Tanous60520632018-06-11 17:46:52 -0700119 system_bus->async_method_call(
Adrian Ambrożewiczbb40bd32021-02-12 13:36:26 +0100120 [&objectServer, path, processName, &interfaceMap, system_bus,
121 timeoutRetries](
Matt Spinler937a2322019-01-23 13:54:22 -0600122 const boost::system::error_code ec,
Patrick Williams2bb2d6b2020-05-13 17:59:02 -0500123 const std::variant<std::vector<Association>>& variantAssociations) {
Ed Tanous60520632018-06-11 17:46:52 -0700124 if (ec)
125 {
Adrian Ambrożewiczbb40bd32021-02-12 13:36:26 +0100126 if (ec.value() == boost::system::errc::timed_out &&
127 timeoutRetries < maxTimeoutRetries)
128 {
129 do_associations(system_bus, interfaceMap, objectServer,
130 processName, path, timeoutRetries + 1);
131 return;
132 }
Ed Tanous60520632018-06-11 17:46:52 -0700133 std::cerr << "Error getting associations from " << path << "\n";
134 }
135 std::vector<Association> associations =
Patrick Williamsb05bc122020-05-13 12:21:00 -0500136 std::get<std::vector<Association>>(variantAssociations);
Andrew Geissler4511b332019-02-21 15:40:40 -0600137 associationChanged(objectServer, associations, path, processName,
Matt Spinlere0b0e3a2019-04-08 10:39:23 -0500138 interfaceMap, associationMaps);
Ed Tanous60520632018-06-11 17:46:52 -0700139 },
140 processName, path, "org.freedesktop.DBus.Properties", "Get",
John Wangd0cf9422019-09-17 16:01:34 +0800141 assocDefsInterface, assocDefsProperty);
Ed Tanous60520632018-06-11 17:46:52 -0700142}
143
144void do_introspect(sdbusplus::asio::connection* system_bus,
145 std::shared_ptr<InProgressIntrospect> transaction,
146 interface_map_type& interface_map,
147 sdbusplus::asio::object_server& objectServer,
Vernon Maueryc52be0d2020-01-14 11:14:25 -0800148 std::string path, int timeoutRetries = 0)
Ed Tanous60520632018-06-11 17:46:52 -0700149{
Vernon Maueryc52be0d2020-01-14 11:14:25 -0800150 constexpr int maxTimeoutRetries = 3;
Ed Tanous60520632018-06-11 17:46:52 -0700151 system_bus->async_method_call(
Vernon Maueryc52be0d2020-01-14 11:14:25 -0800152 [&interface_map, &objectServer, transaction, path, system_bus,
153 timeoutRetries](const boost::system::error_code ec,
154 const std::string& introspect_xml) {
Ed Tanous60520632018-06-11 17:46:52 -0700155 if (ec)
156 {
Vernon Maueryc52be0d2020-01-14 11:14:25 -0800157 if (ec.value() == boost::system::errc::timed_out &&
158 timeoutRetries < maxTimeoutRetries)
159 {
160 do_introspect(system_bus, transaction, interface_map,
161 objectServer, path, timeoutRetries + 1);
162 return;
163 }
Ed Tanous60520632018-06-11 17:46:52 -0700164 std::cerr << "Introspect call failed with error: " << ec << ", "
165 << ec.message()
166 << " on process: " << transaction->process_name
167 << " path: " << path << "\n";
168 return;
169 }
170
171 tinyxml2::XMLDocument doc;
172
173 tinyxml2::XMLError e = doc.Parse(introspect_xml.c_str());
174 if (e != tinyxml2::XMLError::XML_SUCCESS)
175 {
176 std::cerr << "XML parsing failed\n";
177 return;
178 }
179
180 tinyxml2::XMLNode* pRoot = doc.FirstChildElement("node");
181 if (pRoot == nullptr)
182 {
183 std::cerr << "XML document did not contain any data\n";
184 return;
185 }
186 auto& thisPathMap = interface_map[path];
Ed Tanous60520632018-06-11 17:46:52 -0700187 tinyxml2::XMLElement* pElement =
188 pRoot->FirstChildElement("interface");
189 while (pElement != nullptr)
190 {
191 const char* iface_name = pElement->Attribute("name");
192 if (iface_name == nullptr)
193 {
194 continue;
195 }
196
Ed Tanousd4dd96a2018-11-12 11:37:44 -0800197 thisPathMap[transaction->process_name].emplace(iface_name);
198
John Wangd0cf9422019-09-17 16:01:34 +0800199 if (std::strcmp(iface_name, assocDefsInterface) == 0)
Ed Tanous60520632018-06-11 17:46:52 -0700200 {
Matt Spinlere0b0e3a2019-04-08 10:39:23 -0500201 do_associations(system_bus, interface_map, objectServer,
John Wangd0cf9422019-09-17 16:01:34 +0800202 transaction->process_name, path);
Ed Tanous60520632018-06-11 17:46:52 -0700203 }
Ed Tanous60520632018-06-11 17:46:52 -0700204
205 pElement = pElement->NextSiblingElement("interface");
206 }
207
Matt Spinler11401e22019-04-08 13:13:25 -0500208 // Check if this new path has a pending association that can
209 // now be completed.
210 checkIfPendingAssociation(path, interface_map,
211 transaction->assocMaps, objectServer);
212
Ed Tanous50232cd2018-11-12 11:34:43 -0800213 pElement = pRoot->FirstChildElement("node");
214 while (pElement != nullptr)
Ed Tanous60520632018-06-11 17:46:52 -0700215 {
Ed Tanous50232cd2018-11-12 11:34:43 -0800216 const char* child_path = pElement->Attribute("name");
217 if (child_path != nullptr)
Ed Tanous60520632018-06-11 17:46:52 -0700218 {
Ed Tanous50232cd2018-11-12 11:34:43 -0800219 std::string parent_path(path);
220 if (parent_path == "/")
Ed Tanous60520632018-06-11 17:46:52 -0700221 {
Ed Tanous50232cd2018-11-12 11:34:43 -0800222 parent_path.clear();
Ed Tanous60520632018-06-11 17:46:52 -0700223 }
Ed Tanous50232cd2018-11-12 11:34:43 -0800224
225 do_introspect(system_bus, transaction, interface_map,
226 objectServer, parent_path + "/" + child_path);
Ed Tanous60520632018-06-11 17:46:52 -0700227 }
Ed Tanous50232cd2018-11-12 11:34:43 -0800228 pElement = pElement->NextSiblingElement("node");
Ed Tanous60520632018-06-11 17:46:52 -0700229 }
230 },
231 transaction->process_name, path, "org.freedesktop.DBus.Introspectable",
232 "Introspect");
233}
234
Ed Tanous60520632018-06-11 17:46:52 -0700235void start_new_introspect(
Ed Tanous21c60592020-08-17 23:43:46 -0700236 sdbusplus::asio::connection* system_bus, boost::asio::io_context& io,
Ed Tanous60520632018-06-11 17:46:52 -0700237 interface_map_type& interface_map, const std::string& process_name,
Matt Spinler11401e22019-04-08 13:13:25 -0500238 AssociationMaps& assocMaps,
Matt Spinleraecabe82018-09-19 13:25:42 -0500239#ifdef DEBUG
Ed Tanous60520632018-06-11 17:46:52 -0700240 std::shared_ptr<std::chrono::time_point<std::chrono::steady_clock>>
241 global_start_time,
Matt Spinleraecabe82018-09-19 13:25:42 -0500242#endif
Ed Tanous60520632018-06-11 17:46:52 -0700243 sdbusplus::asio::object_server& objectServer)
244{
Andrew Geissler82815da2019-02-04 12:19:41 -0600245 if (needToIntrospect(process_name, service_whitelist, service_blacklist))
Ed Tanous60520632018-06-11 17:46:52 -0700246 {
Ed Tanous60520632018-06-11 17:46:52 -0700247 std::shared_ptr<InProgressIntrospect> transaction =
Matt Spinler11401e22019-04-08 13:13:25 -0500248 std::make_shared<InProgressIntrospect>(system_bus, io, process_name,
249 assocMaps
Matt Spinleraecabe82018-09-19 13:25:42 -0500250#ifdef DEBUG
251 ,
252 global_start_time
253#endif
254 );
Ed Tanous60520632018-06-11 17:46:52 -0700255
256 do_introspect(system_bus, transaction, interface_map, objectServer,
257 "/");
258 }
259}
260
261// TODO(ed) replace with std::set_intersection once c++17 is available
262template <class InputIt1, class InputIt2>
263bool intersect(InputIt1 first1, InputIt1 last1, InputIt2 first2, InputIt2 last2)
264{
265 while (first1 != last1 && first2 != last2)
266 {
267 if (*first1 < *first2)
268 {
269 ++first1;
270 continue;
271 }
272 if (*first2 < *first1)
273 {
274 ++first2;
275 continue;
276 }
277 return true;
278 }
279 return false;
280}
281
282void doListNames(
Ed Tanous21c60592020-08-17 23:43:46 -0700283 boost::asio::io_context& io, interface_map_type& interface_map,
Ed Tanous60520632018-06-11 17:46:52 -0700284 sdbusplus::asio::connection* system_bus,
285 boost::container::flat_map<std::string, std::string>& name_owners,
Matt Spinler11401e22019-04-08 13:13:25 -0500286 AssociationMaps& assocMaps, sdbusplus::asio::object_server& objectServer)
Ed Tanous60520632018-06-11 17:46:52 -0700287{
288 system_bus->async_method_call(
Matt Spinler11401e22019-04-08 13:13:25 -0500289 [&io, &interface_map, &name_owners, &objectServer, system_bus,
290 &assocMaps](const boost::system::error_code ec,
Ed Tanous60520632018-06-11 17:46:52 -0700291 std::vector<std::string> process_names) {
292 if (ec)
293 {
294 std::cerr << "Error getting names: " << ec << "\n";
295 std::exit(EXIT_FAILURE);
296 return;
297 }
Ed Tanous60520632018-06-11 17:46:52 -0700298 // Try to make startup consistent
299 std::sort(process_names.begin(), process_names.end());
Matt Spinleraecabe82018-09-19 13:25:42 -0500300#ifdef DEBUG
Ed Tanous60520632018-06-11 17:46:52 -0700301 std::shared_ptr<std::chrono::time_point<std::chrono::steady_clock>>
302 global_start_time = std::make_shared<
303 std::chrono::time_point<std::chrono::steady_clock>>(
304 std::chrono::steady_clock::now());
Matt Spinleraecabe82018-09-19 13:25:42 -0500305#endif
Ed Tanous60520632018-06-11 17:46:52 -0700306 for (const std::string& process_name : process_names)
307 {
Andrew Geissler82815da2019-02-04 12:19:41 -0600308 if (needToIntrospect(process_name, service_whitelist,
309 service_blacklist))
Ed Tanous60520632018-06-11 17:46:52 -0700310 {
311 start_new_introspect(system_bus, io, interface_map,
Matt Spinler11401e22019-04-08 13:13:25 -0500312 process_name, assocMaps,
Matt Spinleraecabe82018-09-19 13:25:42 -0500313#ifdef DEBUG
314 global_start_time,
315#endif
Ed Tanous60520632018-06-11 17:46:52 -0700316 objectServer);
317 update_owners(system_bus, name_owners, process_name);
318 }
319 }
320 },
321 "org.freedesktop.DBus", "/org/freedesktop/DBus", "org.freedesktop.DBus",
322 "ListNames");
323}
324
Matt Spinlerdd945862018-09-07 12:41:05 -0500325void splitArgs(const std::string& stringArgs,
326 boost::container::flat_set<std::string>& listArgs)
327{
328 std::istringstream args;
329 std::string arg;
330
331 args.str(stringArgs);
332
333 while (!args.eof())
334 {
335 args >> arg;
336 if (!arg.empty())
337 {
338 listArgs.insert(arg);
339 }
340 }
341}
342
Matt Spinler47c09752018-11-29 14:54:13 -0600343void addObjectMapResult(
344 std::vector<interface_map_type::value_type>& objectMap,
Matt Spinler9f0958e2018-09-11 08:26:10 -0500345 const std::string& objectPath,
346 const std::pair<std::string, boost::container::flat_set<std::string>>&
347 interfaceMap)
348{
349 // Adds an object path/service name/interface list entry to
Matt Spinler47c09752018-11-29 14:54:13 -0600350 // the results of GetSubTree and GetAncestors.
Matt Spinler9f0958e2018-09-11 08:26:10 -0500351 // If an entry for the object path already exists, just add the
352 // service name and interfaces to that entry, otherwise create
353 // a new entry.
354 auto entry = std::find_if(
Matt Spinler47c09752018-11-29 14:54:13 -0600355 objectMap.begin(), objectMap.end(),
Matt Spinler9f0958e2018-09-11 08:26:10 -0500356 [&objectPath](const auto& i) { return objectPath == i.first; });
357
Matt Spinler47c09752018-11-29 14:54:13 -0600358 if (entry != objectMap.end())
Matt Spinler9f0958e2018-09-11 08:26:10 -0500359 {
360 entry->second.emplace(interfaceMap);
361 }
362 else
363 {
364 interface_map_type::value_type object;
365 object.first = objectPath;
366 object.second.emplace(interfaceMap);
Matt Spinler47c09752018-11-29 14:54:13 -0600367 objectMap.push_back(object);
Matt Spinler9f0958e2018-09-11 08:26:10 -0500368 }
369}
370
Matt Spinlera82779f2019-01-09 12:39:42 -0600371// Remove parents of the passed in path that:
372// 1) Only have the 3 default interfaces on them
373// - Means D-Bus created these, not application code,
374// with the Properties, Introspectable, and Peer ifaces
375// 2) Have no other child for this owner
376void removeUnneededParents(const std::string& objectPath,
377 const std::string& owner,
378 interface_map_type& interface_map)
379{
380 auto parent = objectPath;
381
382 while (true)
383 {
384 auto pos = parent.find_last_of('/');
385 if ((pos == std::string::npos) || (pos == 0))
386 {
387 break;
388 }
389 parent = parent.substr(0, pos);
390
391 auto parent_it = interface_map.find(parent);
392 if (parent_it == interface_map.end())
393 {
394 break;
395 }
396
397 auto ifaces_it = parent_it->second.find(owner);
398 if (ifaces_it == parent_it->second.end())
399 {
400 break;
401 }
402
403 if (ifaces_it->second.size() != 3)
404 {
405 break;
406 }
407
408 auto child_path = parent + '/';
409
410 // Remove this parent if there isn't a remaining child on this owner
411 auto child = std::find_if(
412 interface_map.begin(), interface_map.end(),
413 [&owner, &child_path](const auto& entry) {
414 return boost::starts_with(entry.first, child_path) &&
415 (entry.second.find(owner) != entry.second.end());
416 });
417
418 if (child == interface_map.end())
419 {
420 parent_it->second.erase(ifaces_it);
421 if (parent_it->second.empty())
422 {
423 interface_map.erase(parent_it);
424 }
425 }
426 else
427 {
428 break;
429 }
430 }
431}
432
Ed Tanous0a13c762021-09-28 13:29:25 -0700433std::vector<interface_map_type::value_type>
434 getAncestors(const interface_map_type& interface_map, std::string req_path,
435 std::vector<std::string>& interfaces)
436{
437 // Interfaces need to be sorted for intersect to function
438 std::sort(interfaces.begin(), interfaces.end());
439
440 if (boost::ends_with(req_path, "/"))
441 {
442 req_path.pop_back();
443 }
444 if (req_path.size() && interface_map.find(req_path) == interface_map.end())
445 {
Konstantin Aladyshevb15df6b2022-01-11 14:50:55 +0300446 throw sdbusplus::xyz::openbmc_project::Common::Error::
447 ResourceNotFound();
Ed Tanous0a13c762021-09-28 13:29:25 -0700448 }
449
450 std::vector<interface_map_type::value_type> ret;
451 for (auto& object_path : interface_map)
452 {
453 auto& this_path = object_path.first;
454 if (boost::starts_with(req_path, this_path) && (req_path != this_path))
455 {
456 if (interfaces.empty())
457 {
458 ret.emplace_back(object_path);
459 }
460 else
461 {
462 for (auto& interface_map : object_path.second)
463 {
464
465 if (intersect(interfaces.begin(), interfaces.end(),
466 interface_map.second.begin(),
467 interface_map.second.end()))
468 {
469 addObjectMapResult(ret, this_path, interface_map);
470 }
471 }
472 }
473 }
474 }
475
476 return ret;
477}
478
479boost::container::flat_map<std::string, boost::container::flat_set<std::string>>
480 getObject(const interface_map_type& interface_map, const std::string& path,
481 std::vector<std::string>& interfaces)
482{
483 boost::container::flat_map<std::string,
484 boost::container::flat_set<std::string>>
485 results;
486
487 // Interfaces need to be sorted for intersect to function
488 std::sort(interfaces.begin(), interfaces.end());
489 auto path_ref = interface_map.find(path);
490 if (path_ref == interface_map.end())
491 {
Konstantin Aladyshevb15df6b2022-01-11 14:50:55 +0300492 throw sdbusplus::xyz::openbmc_project::Common::Error::
493 ResourceNotFound();
Ed Tanous0a13c762021-09-28 13:29:25 -0700494 }
495 if (interfaces.empty())
496 {
497 return path_ref->second;
498 }
499 for (auto& interface_map : path_ref->second)
500 {
501 if (intersect(interfaces.begin(), interfaces.end(),
502 interface_map.second.begin(), interface_map.second.end()))
503 {
504 results.emplace(interface_map.first, interface_map.second);
505 }
506 }
507
508 if (results.empty())
509 {
Konstantin Aladyshevb15df6b2022-01-11 14:50:55 +0300510 throw sdbusplus::xyz::openbmc_project::Common::Error::
511 ResourceNotFound();
Ed Tanous0a13c762021-09-28 13:29:25 -0700512 }
513
514 return results;
515}
516
517std::vector<interface_map_type::value_type>
518 getSubTree(const interface_map_type& interface_map, std::string req_path,
519 int32_t depth, std::vector<std::string>& interfaces)
520{
521 if (depth <= 0)
522 {
523 depth = std::numeric_limits<int32_t>::max();
524 }
525 // Interfaces need to be sorted for intersect to function
526 std::sort(interfaces.begin(), interfaces.end());
527 std::vector<interface_map_type::value_type> ret;
528
529 if (boost::ends_with(req_path, "/"))
530 {
531 req_path.pop_back();
532 }
533 if (req_path.size() && interface_map.find(req_path) == interface_map.end())
534 {
Konstantin Aladyshevb15df6b2022-01-11 14:50:55 +0300535 throw sdbusplus::xyz::openbmc_project::Common::Error::
536 ResourceNotFound();
Ed Tanous0a13c762021-09-28 13:29:25 -0700537 }
538
539 for (auto& object_path : interface_map)
540 {
541 auto& this_path = object_path.first;
542
543 if (this_path == req_path)
544 {
545 continue;
546 }
547
548 if (boost::starts_with(this_path, req_path))
549 {
550 // count the number of slashes past the search term
551 int32_t this_depth = std::count(this_path.begin() + req_path.size(),
552 this_path.end(), '/');
553 if (this_depth <= depth)
554 {
555 for (auto& interface_map : object_path.second)
556 {
557 if (intersect(interfaces.begin(), interfaces.end(),
558 interface_map.second.begin(),
559 interface_map.second.end()) ||
560 interfaces.empty())
561 {
562 addObjectMapResult(ret, this_path, interface_map);
563 }
564 }
565 }
566 }
567 }
568
569 return ret;
570}
571
572std::vector<std::string>
573 getSubTreePaths(const interface_map_type& interface_map,
574 std::string req_path, int32_t depth,
575 std::vector<std::string>& interfaces)
576{
577 if (depth <= 0)
578 {
579 depth = std::numeric_limits<int32_t>::max();
580 }
581 // Interfaces need to be sorted for intersect to function
582 std::sort(interfaces.begin(), interfaces.end());
583 std::vector<std::string> ret;
584
585 if (boost::ends_with(req_path, "/"))
586 {
587 req_path.pop_back();
588 }
589 if (req_path.size() && interface_map.find(req_path) == interface_map.end())
590 {
Konstantin Aladyshevb15df6b2022-01-11 14:50:55 +0300591 throw sdbusplus::xyz::openbmc_project::Common::Error::
592 ResourceNotFound();
Ed Tanous0a13c762021-09-28 13:29:25 -0700593 }
594
595 for (auto& object_path : interface_map)
596 {
597 auto& this_path = object_path.first;
598
599 if (this_path == req_path)
600 {
601 continue;
602 }
603
604 if (boost::starts_with(this_path, req_path))
605 {
606 // count the number of slashes past the search term
607 int this_depth = std::count(this_path.begin() + req_path.size(),
608 this_path.end(), '/');
609 if (this_depth <= depth)
610 {
611 bool add = interfaces.empty();
612 for (auto& interface_map : object_path.second)
613 {
614 if (intersect(interfaces.begin(), interfaces.end(),
615 interface_map.second.begin(),
616 interface_map.second.end()))
617 {
618 add = true;
619 break;
620 }
621 }
622 if (add)
623 {
624 // TODO(ed) this is a copy
625 ret.emplace_back(this_path);
626 }
627 }
628 }
629 }
630
631 return ret;
632}
633
Ed Tanous60520632018-06-11 17:46:52 -0700634int main(int argc, char** argv)
635{
Matt Spinlerdd945862018-09-07 12:41:05 -0500636 auto options = ArgumentParser(argc, argv);
Ed Tanous21c60592020-08-17 23:43:46 -0700637 boost::asio::io_context io;
Ed Tanous60520632018-06-11 17:46:52 -0700638 std::shared_ptr<sdbusplus::asio::connection> system_bus =
639 std::make_shared<sdbusplus::asio::connection>(io);
640
Matt Spinlerdd945862018-09-07 12:41:05 -0500641 splitArgs(options["service-namespaces"], service_whitelist);
Matt Spinlerdd945862018-09-07 12:41:05 -0500642 splitArgs(options["service-blacklists"], service_blacklist);
643
Ed Tanousd4dd96a2018-11-12 11:37:44 -0800644 // TODO(Ed) Remove this once all service files are updated to not use this.
645 // For now, simply squash the input, and ignore it.
646 boost::container::flat_set<std::string> iface_whitelist;
647 splitArgs(options["interface-namespaces"], iface_whitelist);
648
Ed Tanous60520632018-06-11 17:46:52 -0700649 sdbusplus::asio::object_server server(system_bus);
650
651 // Construct a signal set registered for process termination.
652 boost::asio::signal_set signals(io, SIGINT, SIGTERM);
Brad Bishop2d41d6a2021-08-03 08:14:45 -0400653 signals.async_wait(
654 [&io](const boost::system::error_code&, int) { io.stop(); });
Ed Tanous60520632018-06-11 17:46:52 -0700655
656 interface_map_type interface_map;
657 boost::container::flat_map<std::string, std::string> name_owners;
658
659 std::function<void(sdbusplus::message::message & message)>
660 nameChangeHandler = [&interface_map, &io, &name_owners, &server,
661 system_bus](sdbusplus::message::message& message) {
Andrew Geissler20679262019-02-11 20:20:40 -0600662 std::string name; // well-known
663 std::string old_owner; // unique-name
664 std::string new_owner; // unique-name
Ed Tanous60520632018-06-11 17:46:52 -0700665
666 message.read(name, old_owner, new_owner);
667
668 if (!old_owner.empty())
669 {
Andrew Geissler20679262019-02-11 20:20:40 -0600670 processNameChangeDelete(name_owners, name, old_owner,
Matt Spinlere2359fb2019-04-05 14:11:33 -0500671 interface_map, associationMaps, server);
Ed Tanous60520632018-06-11 17:46:52 -0700672 }
673
674 if (!new_owner.empty())
675 {
Matt Spinleraecabe82018-09-19 13:25:42 -0500676#ifdef DEBUG
Ed Tanous60520632018-06-11 17:46:52 -0700677 auto transaction = std::make_shared<
678 std::chrono::time_point<std::chrono::steady_clock>>(
679 std::chrono::steady_clock::now());
Matt Spinleraecabe82018-09-19 13:25:42 -0500680#endif
Ed Tanous60520632018-06-11 17:46:52 -0700681 // New daemon added
Andrew Geissler82815da2019-02-04 12:19:41 -0600682 if (needToIntrospect(name, service_whitelist,
683 service_blacklist))
Ed Tanous60520632018-06-11 17:46:52 -0700684 {
685 name_owners[new_owner] = name;
686 start_new_introspect(system_bus.get(), io, interface_map,
Matt Spinler11401e22019-04-08 13:13:25 -0500687 name, associationMaps,
Matt Spinleraecabe82018-09-19 13:25:42 -0500688#ifdef DEBUG
689 transaction,
690#endif
691 server);
Ed Tanous60520632018-06-11 17:46:52 -0700692 }
693 }
694 };
695
696 sdbusplus::bus::match::match nameOwnerChanged(
697 static_cast<sdbusplus::bus::bus&>(*system_bus),
698 sdbusplus::bus::match::rules::nameOwnerChanged(), nameChangeHandler);
699
700 std::function<void(sdbusplus::message::message & message)>
701 interfacesAddedHandler = [&interface_map, &name_owners, &server](
702 sdbusplus::message::message& message) {
703 sdbusplus::message::object_path obj_path;
Andrew Geissler70461892019-02-27 09:57:37 -0600704 InterfacesAdded interfaces_added;
Ed Tanous60520632018-06-11 17:46:52 -0700705 message.read(obj_path, interfaces_added);
706 std::string well_known;
Andrew Geissler3b025e62019-02-01 10:33:54 -0600707 if (!getWellKnown(name_owners, message.get_sender(), well_known))
Ed Tanous60520632018-06-11 17:46:52 -0700708 {
709 return; // only introspect well-known
710 }
Andrew Geissler82815da2019-02-04 12:19:41 -0600711 if (needToIntrospect(well_known, service_whitelist,
712 service_blacklist))
Ed Tanous60520632018-06-11 17:46:52 -0700713 {
Andrew Geissler70461892019-02-27 09:57:37 -0600714 processInterfaceAdded(interface_map, obj_path, interfaces_added,
Matt Spinlere2359fb2019-04-05 14:11:33 -0500715 well_known, associationMaps, server);
Ed Tanous60520632018-06-11 17:46:52 -0700716 }
717 };
718
719 sdbusplus::bus::match::match interfacesAdded(
720 static_cast<sdbusplus::bus::bus&>(*system_bus),
721 sdbusplus::bus::match::rules::interfacesAdded(),
722 interfacesAddedHandler);
723
724 std::function<void(sdbusplus::message::message & message)>
725 interfacesRemovedHandler = [&interface_map, &name_owners, &server](
726 sdbusplus::message::message& message) {
727 sdbusplus::message::object_path obj_path;
728 std::vector<std::string> interfaces_removed;
729 message.read(obj_path, interfaces_removed);
730 auto connection_map = interface_map.find(obj_path.str);
731 if (connection_map == interface_map.end())
732 {
733 return;
734 }
735
736 std::string sender;
Andrew Geissler3b025e62019-02-01 10:33:54 -0600737 if (!getWellKnown(name_owners, message.get_sender(), sender))
Ed Tanous60520632018-06-11 17:46:52 -0700738 {
739 return;
740 }
741 for (const std::string& interface : interfaces_removed)
742 {
743 auto interface_set = connection_map->second.find(sender);
744 if (interface_set == connection_map->second.end())
745 {
746 continue;
747 }
748
John Wangd0cf9422019-09-17 16:01:34 +0800749 if (interface == assocDefsInterface)
Ed Tanous60520632018-06-11 17:46:52 -0700750 {
Andrew Geissler67e5a422019-02-04 13:00:59 -0600751 removeAssociation(obj_path.str, sender, server,
Matt Spinlere2359fb2019-04-05 14:11:33 -0500752 associationMaps);
Ed Tanous60520632018-06-11 17:46:52 -0700753 }
754
755 interface_set->second.erase(interface);
Matt Spinler9c3d2852019-04-08 15:57:19 -0500756
Ed Tanous60520632018-06-11 17:46:52 -0700757 if (interface_set->second.empty())
758 {
Matt Spinler9c3d2852019-04-08 15:57:19 -0500759 // If this was the last interface on this connection,
760 // erase the connection
Ed Tanous60520632018-06-11 17:46:52 -0700761 connection_map->second.erase(interface_set);
Matt Spinler9c3d2852019-04-08 15:57:19 -0500762
763 // Instead of checking if every single path is the endpoint
764 // of an association that needs to be moved to pending,
765 // only check when the only remaining owner of this path is
766 // ourself, which would be because we still own the
767 // association path.
768 if ((connection_map->second.size() == 1) &&
769 (connection_map->second.begin()->first ==
Brad Bishopa02cd542021-10-12 19:12:42 -0400770 "xyz.openbmc_project.ObjectMapper"))
Matt Spinler9c3d2852019-04-08 15:57:19 -0500771 {
772 // Remove the 2 association D-Bus paths and move the
773 // association to pending.
774 moveAssociationToPending(obj_path.str, associationMaps,
775 server);
776 }
Ed Tanous60520632018-06-11 17:46:52 -0700777 }
778 }
779 // If this was the last connection on this object path,
780 // erase the object path
781 if (connection_map->second.empty())
782 {
783 interface_map.erase(connection_map);
784 }
Matt Spinlera82779f2019-01-09 12:39:42 -0600785
786 removeUnneededParents(obj_path.str, sender, interface_map);
Ed Tanous60520632018-06-11 17:46:52 -0700787 };
788
789 sdbusplus::bus::match::match interfacesRemoved(
790 static_cast<sdbusplus::bus::bus&>(*system_bus),
791 sdbusplus::bus::match::rules::interfacesRemoved(),
792 interfacesRemovedHandler);
793
794 std::function<void(sdbusplus::message::message & message)>
Matt Spinlere0b0e3a2019-04-08 10:39:23 -0500795 associationChangedHandler = [&server, &name_owners, &interface_map](
Matt Spinler8f876a52019-04-15 13:22:50 -0500796 sdbusplus::message::message& message) {
797 std::string objectName;
Patrick Williams2bb2d6b2020-05-13 17:59:02 -0500798 boost::container::flat_map<std::string,
799 std::variant<std::vector<Association>>>
Matt Spinler8f876a52019-04-15 13:22:50 -0500800 values;
801 message.read(objectName, values);
John Wangd0cf9422019-09-17 16:01:34 +0800802 auto prop = values.find(assocDefsProperty);
Matt Spinler8f876a52019-04-15 13:22:50 -0500803 if (prop != values.end())
804 {
805 std::vector<Association> associations =
Patrick Williamsb05bc122020-05-13 12:21:00 -0500806 std::get<std::vector<Association>>(prop->second);
Matt Spinler8f876a52019-04-15 13:22:50 -0500807
808 std::string well_known;
809 if (!getWellKnown(name_owners, message.get_sender(),
810 well_known))
811 {
812 return;
Ed Tanous60520632018-06-11 17:46:52 -0700813 }
Matt Spinler8f876a52019-04-15 13:22:50 -0500814 associationChanged(server, associations, message.get_path(),
Matt Spinlere0b0e3a2019-04-08 10:39:23 -0500815 well_known, interface_map, associationMaps);
Matt Spinler8f876a52019-04-15 13:22:50 -0500816 }
817 };
818 sdbusplus::bus::match::match assocChangedMatch(
Ed Tanous60520632018-06-11 17:46:52 -0700819 static_cast<sdbusplus::bus::bus&>(*system_bus),
820 sdbusplus::bus::match::rules::interface(
821 "org.freedesktop.DBus.Properties") +
822 sdbusplus::bus::match::rules::member("PropertiesChanged") +
Matt Spinler8f876a52019-04-15 13:22:50 -0500823 sdbusplus::bus::match::rules::argN(0, assocDefsInterface),
824 associationChangedHandler);
825
Ed Tanous60520632018-06-11 17:46:52 -0700826 std::shared_ptr<sdbusplus::asio::dbus_interface> iface =
Brad Bishopa02cd542021-10-12 19:12:42 -0400827 server.add_interface("/xyz/openbmc_project/object_mapper",
828 "xyz.openbmc_project.ObjectMapper");
Ed Tanous60520632018-06-11 17:46:52 -0700829
830 iface->register_method(
Matt Spinler6a39e8c2018-11-13 11:11:36 -0600831 "GetAncestors", [&interface_map](std::string& req_path,
Ed Tanous60520632018-06-11 17:46:52 -0700832 std::vector<std::string>& interfaces) {
Ed Tanous0a13c762021-09-28 13:29:25 -0700833 return getAncestors(interface_map, req_path, interfaces);
Ed Tanous60520632018-06-11 17:46:52 -0700834 });
835
836 iface->register_method(
837 "GetObject", [&interface_map](const std::string& path,
838 std::vector<std::string>& interfaces) {
Ed Tanous0a13c762021-09-28 13:29:25 -0700839 return getObject(interface_map, path, interfaces);
Ed Tanous60520632018-06-11 17:46:52 -0700840 });
841
842 iface->register_method(
Matt Spinler153494e2018-11-07 16:35:40 -0600843 "GetSubTree", [&interface_map](std::string& req_path, int32_t depth,
844 std::vector<std::string>& interfaces) {
Ed Tanous0a13c762021-09-28 13:29:25 -0700845 return getSubTree(interface_map, req_path, depth, interfaces);
Ed Tanous60520632018-06-11 17:46:52 -0700846 });
847
848 iface->register_method(
849 "GetSubTreePaths",
Matt Spinler153494e2018-11-07 16:35:40 -0600850 [&interface_map](std::string& req_path, int32_t depth,
Ed Tanous60520632018-06-11 17:46:52 -0700851 std::vector<std::string>& interfaces) {
Ed Tanous0a13c762021-09-28 13:29:25 -0700852 return getSubTreePaths(interface_map, req_path, depth, interfaces);
Ed Tanous60520632018-06-11 17:46:52 -0700853 });
854
855 iface->initialize();
856
857 io.post([&]() {
Matt Spinler11401e22019-04-08 13:13:25 -0500858 doListNames(io, interface_map, system_bus.get(), name_owners,
859 associationMaps, server);
Ed Tanous60520632018-06-11 17:46:52 -0700860 });
861
Brad Bishopa02cd542021-10-12 19:12:42 -0400862 system_bus->request_name("xyz.openbmc_project.ObjectMapper");
Vishwanatha Subbanna64354ef2020-08-21 03:35:26 -0500863
Ed Tanous60520632018-06-11 17:46:52 -0700864 io.run();
865}