blob: 2c4e16d82074ffc90a125d140564894b8a8599b1 [file] [log] [blame]
George Liu616a0712021-02-18 10:50:24 +08001#include "utils.hpp"
George Liudef5f5a2020-04-10 11:23:52 +08002
George Liue9fb5c62021-07-01 14:05:32 +08003#include <phosphor-logging/lg2.hpp>
George Liu616a0712021-02-18 10:50:24 +08004#include <sdbusplus/exception.hpp>
5#include <sdeventplus/event.hpp>
George Liudef5f5a2020-04-10 11:23:52 +08006
7#include <filesystem>
8#include <fstream>
George Liudef5f5a2020-04-10 11:23:52 +08009
10namespace fs = std::filesystem;
11
George Liu616a0712021-02-18 10:50:24 +080012namespace phosphor
George Liudef5f5a2020-04-10 11:23:52 +080013{
George Liu616a0712021-02-18 10:50:24 +080014namespace led
15{
George Liudef5f5a2020-04-10 11:23:52 +080016
George Liu616a0712021-02-18 10:50:24 +080017static constexpr auto confFileName = "led-group-config.json";
18static constexpr auto confOverridePath = "/etc/phosphor-led-manager";
19static constexpr auto confBasePath = "/usr/share/phosphor-led-manager";
20static constexpr auto confCompatibleInterface =
21 "xyz.openbmc_project.Configuration.IBMCompatibleSystem";
22static constexpr auto confCompatibleProperty = "Names";
23
24class JsonConfig
25{
26 public:
27 /**
28 * @brief Constructor
29 *
30 * Looks for the JSON config file. If it can't find one, then it
31 * will watch entity-manager for the IBMCompatibleSystem interface
32 * to show up.
33 *
34 * @param[in] bus - The D-Bus object
35 * @param[in] event - sd event handler
36 */
Patrick Williams3e073ba2022-07-22 19:26:52 -050037 JsonConfig(sdbusplus::bus_t& bus, sdeventplus::Event& event) : event(event)
George Liudef5f5a2020-04-10 11:23:52 +080038 {
Patrick Williamsf178c0f2021-11-19 12:23:09 -060039 match = std::make_unique<sdbusplus::bus::match_t>(
George Liu616a0712021-02-18 10:50:24 +080040 bus,
41 sdbusplus::bus::match::rules::interfacesAdded() +
42 sdbusplus::bus::match::rules::sender(
43 "xyz.openbmc_project.EntityManager"),
44 std::bind(&JsonConfig::ifacesAddedCallback, this,
45 std::placeholders::_1));
46 getFilePath();
47
48 if (!confFile.empty())
49 {
50 match.reset();
51 }
George Liudef5f5a2020-04-10 11:23:52 +080052 }
53
George Liu616a0712021-02-18 10:50:24 +080054 /**
55 * @brief Get the configuration file
56 *
57 * @return filesystem path
58 */
59 inline const fs::path& getConfFile() const
George Liudef5f5a2020-04-10 11:23:52 +080060 {
George Liu616a0712021-02-18 10:50:24 +080061 return confFile;
George Liudef5f5a2020-04-10 11:23:52 +080062 }
George Liu616a0712021-02-18 10:50:24 +080063
64 private:
65 /** @brief Check the file path exists
66 *
67 * @param[in] names - Vector of the confCompatible Property
68 *
69 * @return - True or False
70 */
71 bool filePathExists(const std::vector<std::string>& names)
George Liudef5f5a2020-04-10 11:23:52 +080072 {
George Liu616a0712021-02-18 10:50:24 +080073 auto it =
74 std::find_if(names.begin(), names.end(), [this](const auto& name) {
75 auto tempConfFile =
76 fs::path{confBasePath} / name / confFileName;
77 if (fs::exists(tempConfFile))
78 {
79 confFile = tempConfFile;
80 return true;
81 }
82 return false;
83 });
84 return it == names.end() ? false : true;
85 }
86
87 /**
88 * @brief The interfacesAdded callback function that looks for
89 * the IBMCompatibleSystem interface. If it finds it,
90 * it uses the Names property in the interface to find
91 * the JSON config file to use.
92 *
93 * @param[in] msg - The D-Bus message contents
94 */
Patrick Williams3e073ba2022-07-22 19:26:52 -050095 void ifacesAddedCallback(sdbusplus::message_t& msg)
George Liu616a0712021-02-18 10:50:24 +080096 {
97 sdbusplus::message::object_path path;
Patrick Williamsf2044032022-03-17 05:12:30 -050098 std::unordered_map<
99 std::string,
100 std::unordered_map<std::string,
101 std::variant<std::vector<std::string>>>>
George Liu616a0712021-02-18 10:50:24 +0800102 interfaces;
103
104 msg.read(path, interfaces);
105
George Liu7f53a032021-05-04 11:18:21 +0800106 if (!interfaces.contains(confCompatibleInterface))
George Liu616a0712021-02-18 10:50:24 +0800107 {
108 return;
109 }
110
111 // Get the "Name" property value of the
112 // "xyz.openbmc_project.Configuration.IBMCompatibleSystem" interface
113 const auto& properties = interfaces.at(confCompatibleInterface);
114
George Liu7f53a032021-05-04 11:18:21 +0800115 if (!properties.contains(confCompatibleProperty))
George Liu616a0712021-02-18 10:50:24 +0800116 {
117 return;
118 }
119 auto names = std::get<std::vector<std::string>>(
120 properties.at(confCompatibleProperty));
121
122 if (filePathExists(names))
123 {
124 match.reset();
125
126 // This results in event.loop() exiting in getSystemLedMap
127 event.exit(0);
128 }
129 }
130
131 /**
132 * Get the json configuration file. The first location found to contain the
133 * json config file from the following locations in order.
134 * confOverridePath: /etc/phosphor-led-manager/led-group-config.json
135 * confBasePath: /usr/shard/phosphor-led-manager/led-group-config.json
136 * the name property of the confCompatibleInterface:
137 * /usr/shard/phosphor-led-manager/${Name}/led-group-config.json
138 *
139 * @brief Get the configuration file to be used
140 *
141 * @return
142 */
George Liu4b062012020-10-13 15:26:58 +0800143 void getFilePath()
George Liu616a0712021-02-18 10:50:24 +0800144 {
145 // Check override location
146 confFile = fs::path{confOverridePath} / confFileName;
147 if (fs::exists(confFile))
148 {
149 return;
150 }
151
152 // If the default file is there, use it
153 confFile = fs::path{confBasePath} / confFileName;
154 if (fs::exists(confFile))
155 {
156 return;
157 }
158 confFile.clear();
159
160 try
161 {
162 // Get all objects implementing the compatible interface
163 auto objects =
164 dBusHandler.getSubTreePaths("/", confCompatibleInterface);
165 for (const auto& path : objects)
166 {
167 try
168 {
169 // Retrieve json config compatible relative path locations
170 auto value = dBusHandler.getProperty(
171 path, confCompatibleInterface, confCompatibleProperty);
172
173 auto confCompatValues =
174 std::get<std::vector<std::string>>(value);
175
176 // Look for a config file at each name relative to the base
177 // path and use the first one found
178 if (filePathExists(confCompatValues))
179 {
180 // Use the first config file found at a listed location
181 break;
182 }
183 confFile.clear();
184 }
Patrick Williams3e073ba2022-07-22 19:26:52 -0500185 catch (const sdbusplus::exception_t& e)
George Liu616a0712021-02-18 10:50:24 +0800186 {
187 // Property unavailable on object.
George Liue9fb5c62021-07-01 14:05:32 +0800188 lg2::error(
189 "Failed to get Names property, ERROR = {ERROR}, INTERFACES = {INTERFACES}, PATH = {PATH}",
190 "ERROR", e, "INTERFACE", confCompatibleInterface,
191 "PATH", path);
George Liudef5f5a2020-04-10 11:23:52 +0800192
George Liu616a0712021-02-18 10:50:24 +0800193 confFile.clear();
194 }
195 }
196 }
Patrick Williams3e073ba2022-07-22 19:26:52 -0500197 catch (const sdbusplus::exception_t& e)
George Liu616a0712021-02-18 10:50:24 +0800198 {
George Liue9fb5c62021-07-01 14:05:32 +0800199 lg2::error(
200 "Failed to call the SubTreePaths method, ERROR = {ERROR}, INTERFACE = {INTERFACE}",
201 "ERROR", e, "INTERFACE", confCompatibleInterface);
George Liu616a0712021-02-18 10:50:24 +0800202 }
George Liu17fcb4d2020-06-30 18:12:53 +0800203 return;
204 }
205
George Liu616a0712021-02-18 10:50:24 +0800206 private:
207 /**
208 * @brief sd event handler.
209 */
210 sdeventplus::Event& event;
George Liu17fcb4d2020-06-30 18:12:53 +0800211
George Liu616a0712021-02-18 10:50:24 +0800212 /**
213 * @brief The JSON config file
214 */
215 fs::path confFile;
George Liu17fcb4d2020-06-30 18:12:53 +0800216
George Liu616a0712021-02-18 10:50:24 +0800217 /**
218 * @brief The interfacesAdded match that is used to wait
219 * for the IBMCompatibleSystem interface to show up.
220 */
Patrick Williamsf178c0f2021-11-19 12:23:09 -0600221 std::unique_ptr<sdbusplus::bus::match_t> match;
George Liudef5f5a2020-04-10 11:23:52 +0800222
George Liu616a0712021-02-18 10:50:24 +0800223 /** DBusHandler class handles the D-Bus operations */
224 utils::DBusHandler dBusHandler;
225};
Patrick Williams7217c032022-03-16 16:26:09 -0500226
227/** Blocking call to find the JSON Config from DBus. */
228auto getJsonConfig()
229{
230 // Get a new Dbus
231 auto bus = sdbusplus::bus::new_bus();
232
233 // Get a new event loop
234 auto event = sdeventplus::Event::get_new();
235
236 // Attach the bus to sd_event to service user requests
237 bus.attach_event(event.get(), SD_EVENT_PRIORITY_IMPORTANT);
238 phosphor::led::JsonConfig jsonConfig(bus, event);
239
240 // The event loop will be terminated from inside of a function in JsonConfig
241 // after finding the configuration file
242 if (jsonConfig.getConfFile().empty())
243 {
244 event.loop();
245 }
246
247 // Detach the bus from its sd_event event loop object
248 bus.detach_event();
249
250 return jsonConfig.getConfFile();
251}
252
George Liu616a0712021-02-18 10:50:24 +0800253} // namespace led
254} // namespace phosphor