blob: 790bbf8316a16ed63f95916c3a46137e66ca11d0 [file] [log] [blame]
James Feist6714a252018-09-10 15:26:18 -07001#pragma once
Ed Tanous8a57ec02020-10-09 12:46:52 -07002#include <VariantVisitors.hpp>
Zhikui Renda98f092021-11-01 09:41:08 -07003#include <boost/algorithm/string/replace.hpp>
James Feist8086aba2020-08-25 16:00:59 -07004#include <boost/asio/steady_timer.hpp>
James Feist6714a252018-09-10 15:26:18 -07005#include <boost/container/flat_map.hpp>
James Feist38fb5982020-05-28 10:09:54 -07006#include <sdbusplus/asio/connection.hpp>
7#include <sdbusplus/asio/object_server.hpp>
8#include <sdbusplus/message/types.hpp>
9
James Feist24f02f22019-04-15 11:05:39 -070010#include <filesystem>
Patrick Venturefd6ba732019-10-31 14:27:39 -070011#include <functional>
James Feist6714a252018-09-10 15:26:18 -070012#include <iostream>
Patrick Venturefd6ba732019-10-31 14:27:39 -070013#include <memory>
Bruce Mitchell544e7dc2021-07-29 18:05:49 -050014#include <optional>
James Feist6714a252018-09-10 15:26:18 -070015#include <regex>
Zev Weiss214d9712022-08-12 12:54:31 -070016#include <span>
Patrick Venturefd6ba732019-10-31 14:27:39 -070017#include <string>
18#include <tuple>
19#include <utility>
20#include <variant>
21#include <vector>
James Feist6714a252018-09-10 15:26:18 -070022
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -070023const constexpr char* jsonStore = "/var/configuration/flattened.json";
24const constexpr char* inventoryPath = "/xyz/openbmc_project/inventory";
25const constexpr char* entityManagerName = "xyz.openbmc_project.EntityManager";
James Feist58295ad2019-05-30 15:01:41 -070026
27constexpr const char* cpuInventoryPath =
28 "/xyz/openbmc_project/inventory/system/chassis/motherboard";
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -070029const std::regex illegalDbusRegex("[^A-Za-z0-9_]");
James Feist6714a252018-09-10 15:26:18 -070030
31using BasicVariantType =
James Feist3eb82622019-02-08 13:10:22 -080032 std::variant<std::vector<std::string>, std::string, int64_t, uint64_t,
33 double, int32_t, uint32_t, int16_t, uint16_t, uint8_t, bool>;
Alex Qiu8b3f7d42020-01-06 13:54:42 -080034using SensorBaseConfigMap =
35 boost::container::flat_map<std::string, BasicVariantType>;
36using SensorBaseConfiguration = std::pair<std::string, SensorBaseConfigMap>;
37using SensorData = boost::container::flat_map<std::string, SensorBaseConfigMap>;
38using ManagedObjectType =
39 boost::container::flat_map<sdbusplus::message::object_path, SensorData>;
James Feist6714a252018-09-10 15:26:18 -070040
James Feista5e58722019-04-22 14:43:11 -070041using GetSubTreeType = std::vector<
42 std::pair<std::string,
43 std::vector<std::pair<std::string, std::vector<std::string>>>>>;
James Feistd8bd5622019-06-26 12:09:05 -070044using Association = std::tuple<std::string, std::string, std::string>;
45
Zhikui Renda98f092021-11-01 09:41:08 -070046inline std::string escapeName(const std::string& sensorName)
47{
48 return boost::replace_all_copy(sensorName, " ", "_");
49}
50
Zev Weiss88cb29d2022-05-09 03:46:15 +000051enum class PowerState
52{
53 on,
54 biosPost,
55 always
56};
57
Jason Ling100c20b2020-08-11 14:50:33 -070058std::optional<std::string> openAndRead(const std::string& hwmonFile);
59std::optional<std::string>
60 getFullHwmonFilePath(const std::string& directory,
61 const std::string& hwmonBaseName,
62 const std::set<std::string>& permitSet);
63std::set<std::string> getPermitSet(const SensorBaseConfigMap& config);
Ed Tanous8a57ec02020-10-09 12:46:52 -070064bool findFiles(const std::filesystem::path& dirPath,
Lei YU6a4e9732021-10-20 13:27:34 +080065 std::string_view matchString,
James Feistcf3bce62019-01-08 10:07:19 -080066 std::vector<std::filesystem::path>& foundPaths,
Ed Tanous8a57ec02020-10-09 12:46:52 -070067 int symlinkDepth = 1);
James Feist71d31b22019-01-02 16:57:54 -080068bool isPowerOn(void);
James Feistfc94b212019-02-06 16:14:51 -080069bool hasBiosPost(void);
Zev Weiss88cb29d2022-05-09 03:46:15 +000070void setupPowerMatchCallback(
71 const std::shared_ptr<sdbusplus::asio::connection>& conn,
72 std::function<void(PowerState type, bool state)>&& callback);
James Feist71d31b22019-01-02 16:57:54 -080073void setupPowerMatch(const std::shared_ptr<sdbusplus::asio::connection>& conn);
James Feist6714a252018-09-10 15:26:18 -070074bool getSensorConfiguration(
75 const std::string& type,
76 const std::shared_ptr<sdbusplus::asio::connection>& dbusConnection,
Ed Tanous8a57ec02020-10-09 12:46:52 -070077 ManagedObjectType& resp, bool useCache);
78
79bool getSensorConfiguration(
80 const std::string& type,
81 const std::shared_ptr<sdbusplus::asio::connection>& dbusConnection,
82 ManagedObjectType& resp);
James Feist87d713a2018-12-06 16:06:24 -080083
James Feist82bac4c2019-03-11 11:16:53 -070084void createAssociation(
85 std::shared_ptr<sdbusplus::asio::dbus_interface>& association,
86 const std::string& path);
87
James Feist87d713a2018-12-06 16:06:24 -080088// replaces limits if MinReading and MaxReading are found.
89void findLimits(std::pair<double, double>& limits,
James Feist40a72142018-12-21 10:09:53 -080090 const SensorBaseConfiguration* data);
91
Konstantin Aladyshevc7a1ae62021-04-30 08:50:43 +000092bool readingStateGood(const PowerState& powerState);
93
Zev Weiss054aad82022-08-18 01:37:34 -070094constexpr const char* configInterfacePrefix =
95 "xyz.openbmc_project.Configuration.";
96
97inline std::string configInterfaceName(const std::string& type)
98{
99 return std::string(configInterfacePrefix) + type;
100}
101
James Feista5e58722019-04-22 14:43:11 -0700102namespace mapper
103{
104constexpr const char* busName = "xyz.openbmc_project.ObjectMapper";
105constexpr const char* path = "/xyz/openbmc_project/object_mapper";
106constexpr const char* interface = "xyz.openbmc_project.ObjectMapper";
107constexpr const char* subtree = "GetSubTree";
108} // namespace mapper
109
110namespace properties
111{
112constexpr const char* interface = "org.freedesktop.DBus.Properties";
113constexpr const char* get = "Get";
James Feist49a8ccd2020-09-16 16:09:52 -0700114constexpr const char* set = "Set";
James Feista5e58722019-04-22 14:43:11 -0700115} // namespace properties
116
James Feist52497fd2019-06-07 13:01:33 -0700117namespace power
118{
119const static constexpr char* busname = "xyz.openbmc_project.State.Host";
120const static constexpr char* interface = "xyz.openbmc_project.State.Host";
121const static constexpr char* path = "/xyz/openbmc_project/state/host0";
122const static constexpr char* property = "CurrentHostState";
123} // namespace power
124namespace post
125{
126const static constexpr char* busname =
127 "xyz.openbmc_project.State.OperatingSystem";
128const static constexpr char* interface =
129 "xyz.openbmc_project.State.OperatingSystem.Status";
130const static constexpr char* path = "/xyz/openbmc_project/state/os";
131const static constexpr char* property = "OperatingSystemState";
132} // namespace post
133
James Feist2adc95c2019-09-30 14:55:28 -0700134namespace association
135{
136const static constexpr char* interface =
137 "xyz.openbmc_project.Association.Definitions";
138} // namespace association
139
James Feist40a72142018-12-21 10:09:53 -0800140template <typename T>
Zev Weissafd15042022-07-18 12:28:40 -0700141inline T loadVariant(const SensorBaseConfigMap& data, const std::string& key)
James Feist40a72142018-12-21 10:09:53 -0800142{
143 auto it = data.find(key);
144 if (it == data.end())
145 {
146 std::cerr << "Configuration missing " << key << "\n";
147 throw std::invalid_argument("Key Missing");
148 }
149 if constexpr (std::is_same_v<T, double>)
150 {
James Feist3eb82622019-02-08 13:10:22 -0800151 return std::visit(VariantToDoubleVisitor(), it->second);
James Feist40a72142018-12-21 10:09:53 -0800152 }
James Feist6ef20402019-01-07 16:45:08 -0800153 else if constexpr (std::is_unsigned_v<T>)
154 {
James Feist3eb82622019-02-08 13:10:22 -0800155 return std::visit(VariantToUnsignedIntVisitor(), it->second);
James Feist6ef20402019-01-07 16:45:08 -0800156 }
James Feist40a72142018-12-21 10:09:53 -0800157 else if constexpr (std::is_same_v<T, std::string>)
158 {
James Feist3eb82622019-02-08 13:10:22 -0800159 return std::visit(VariantToStringVisitor(), it->second);
James Feist40a72142018-12-21 10:09:53 -0800160 }
161 else
162 {
James Feist52497fd2019-06-07 13:01:33 -0700163 static_assert(!std::is_same_v<T, T>, "Type Not Implemented");
James Feist40a72142018-12-21 10:09:53 -0800164 }
165}
James Feistfc94b212019-02-06 16:14:51 -0800166
167inline void setReadState(const std::string& str, PowerState& val)
168{
169
170 if (str == "On")
171 {
172 val = PowerState::on;
173 }
174 else if (str == "BiosPost")
175 {
176 val = PowerState::biosPost;
177 }
178 else if (str == "Always")
179 {
180 val = PowerState::always;
181 }
182}
Cheng C Yang5580f2f2019-09-19 09:01:47 +0800183
Zev Weissa4d27682022-07-19 15:30:36 -0700184inline PowerState getPowerState(const SensorBaseConfigMap& cfg)
185{
186 PowerState state = PowerState::always;
187 auto findPowerState = cfg.find("PowerState");
188 if (findPowerState != cfg.end())
189 {
190 std::string powerState =
191 std::visit(VariantToStringVisitor(), findPowerState->second);
192 setReadState(powerState, state);
193 }
194 return state;
195}
196
Zev Weiss8569bf22022-10-11 15:37:44 -0700197inline float getPollRate(const SensorBaseConfigMap& cfg, float dflt)
198{
199 float pollRate = dflt;
200 auto findPollRate = cfg.find("PollRate");
201 if (findPollRate != cfg.end())
202 {
203 pollRate = std::visit(VariantToFloatVisitor(), findPollRate->second);
204 if (!std::isfinite(pollRate) || pollRate <= 0.0F)
205 {
206 pollRate = dflt; // poll time invalid, fall back to default
207 }
208 }
209 return pollRate;
210}
211
Ed Tanous8a57ec02020-10-09 12:46:52 -0700212inline void setLed(const std::shared_ptr<sdbusplus::asio::connection>& conn,
James Feist49a8ccd2020-09-16 16:09:52 -0700213 const std::string& name, bool on)
214{
215 conn->async_method_call(
216 [name](const boost::system::error_code ec) {
Ed Tanousbb679322022-05-16 16:10:00 -0700217 if (ec)
218 {
219 std::cerr << "Failed to set LED " << name << "\n";
220 }
James Feist49a8ccd2020-09-16 16:09:52 -0700221 },
222 "xyz.openbmc_project.LED.GroupManager",
223 "/xyz/openbmc_project/led/groups/" + name, properties::interface,
224 properties::set, "xyz.openbmc_project.Led.Group", "Asserted",
225 std::variant<bool>(on));
226}
227
Cheng C Yang5580f2f2019-09-19 09:01:47 +0800228void createInventoryAssoc(
Ed Tanous8a57ec02020-10-09 12:46:52 -0700229 const std::shared_ptr<sdbusplus::asio::connection>& conn,
230 const std::shared_ptr<sdbusplus::asio::dbus_interface>& association,
Cheng C Yang5580f2f2019-09-19 09:01:47 +0800231 const std::string& path);
James Feistc71c1192019-09-18 14:31:33 -0700232
James Feist38fb5982020-05-28 10:09:54 -0700233struct GetSensorConfiguration :
234 std::enable_shared_from_this<GetSensorConfiguration>
James Feistc71c1192019-09-18 14:31:33 -0700235{
236 GetSensorConfiguration(
237 std::shared_ptr<sdbusplus::asio::connection> connection,
238 std::function<void(ManagedObjectType& resp)>&& callbackFunc) :
Ed Tanous8a57ec02020-10-09 12:46:52 -0700239 dbusConnection(std::move(connection)),
James Feistc71c1192019-09-18 14:31:33 -0700240 callback(std::move(callbackFunc))
James Feist38fb5982020-05-28 10:09:54 -0700241 {}
James Feistf27a55c2020-08-04 14:27:30 -0700242
243 void getPath(const std::string& path, const std::string& interface,
244 const std::string& owner, size_t retries = 5)
James Feistc71c1192019-09-18 14:31:33 -0700245 {
James Feistf27a55c2020-08-04 14:27:30 -0700246 if (retries > 5)
247 {
248 retries = 5;
249 }
250 std::shared_ptr<GetSensorConfiguration> self = shared_from_this();
251
252 self->dbusConnection->async_method_call(
Zev Weissafd15042022-07-18 12:28:40 -0700253 [self, path, interface, owner, retries](
254 const boost::system::error_code ec, SensorBaseConfigMap& data) {
Ed Tanousbb679322022-05-16 16:10:00 -0700255 if (ec)
256 {
257 std::cerr << "Error getting " << path << ": retries left"
258 << retries - 1 << "\n";
Ed Tanous2049bd22022-07-09 07:20:26 -0700259 if (retries == 0U)
James Feistf27a55c2020-08-04 14:27:30 -0700260 {
James Feistf27a55c2020-08-04 14:27:30 -0700261 return;
262 }
Ed Tanousbb679322022-05-16 16:10:00 -0700263 auto timer = std::make_shared<boost::asio::steady_timer>(
264 self->dbusConnection->get_io_context());
265 timer->expires_after(std::chrono::seconds(10));
266 timer->async_wait([self, timer, path, interface, owner,
267 retries](boost::system::error_code ec) {
268 if (ec)
269 {
270 std::cerr << "Timer error!\n";
271 return;
272 }
273 self->getPath(path, interface, owner, retries - 1);
274 });
275 return;
276 }
James Feistf27a55c2020-08-04 14:27:30 -0700277
Ed Tanousbb679322022-05-16 16:10:00 -0700278 self->respData[path][interface] = std::move(data);
James Feistf27a55c2020-08-04 14:27:30 -0700279 },
280 owner, path, "org.freedesktop.DBus.Properties", "GetAll",
281 interface);
282 }
283
Zev Weiss054aad82022-08-18 01:37:34 -0700284 void getConfiguration(const std::vector<std::string>& types,
James Feistf27a55c2020-08-04 14:27:30 -0700285 size_t retries = 0)
286 {
287 if (retries > 5)
288 {
289 retries = 5;
290 }
291
Zev Weiss054aad82022-08-18 01:37:34 -0700292 std::vector<std::string> interfaces(types.size());
293 for (const auto& type : types)
294 {
295 interfaces.push_back(configInterfaceName(type));
296 }
297
James Feistc71c1192019-09-18 14:31:33 -0700298 std::shared_ptr<GetSensorConfiguration> self = shared_from_this();
299 dbusConnection->async_method_call(
James Feistf27a55c2020-08-04 14:27:30 -0700300 [self, interfaces, retries](const boost::system::error_code ec,
301 const GetSubTreeType& ret) {
Ed Tanousbb679322022-05-16 16:10:00 -0700302 if (ec)
303 {
304 std::cerr << "Error calling mapper\n";
Ed Tanous2049bd22022-07-09 07:20:26 -0700305 if (retries == 0U)
James Feistc71c1192019-09-18 14:31:33 -0700306 {
James Feistc71c1192019-09-18 14:31:33 -0700307 return;
308 }
Ed Tanousbb679322022-05-16 16:10:00 -0700309 auto timer = std::make_shared<boost::asio::steady_timer>(
310 self->dbusConnection->get_io_context());
311 timer->expires_after(std::chrono::seconds(10));
312 timer->async_wait([self, timer, interfaces,
313 retries](boost::system::error_code ec) {
314 if (ec)
James Feistc71c1192019-09-18 14:31:33 -0700315 {
Ed Tanousbb679322022-05-16 16:10:00 -0700316 std::cerr << "Timer error!\n";
James Feistc71c1192019-09-18 14:31:33 -0700317 return;
318 }
Ed Tanousbb679322022-05-16 16:10:00 -0700319 self->getConfiguration(interfaces, retries - 1);
320 });
James Feistc71c1192019-09-18 14:31:33 -0700321
Ed Tanousbb679322022-05-16 16:10:00 -0700322 return;
323 }
324 for (const auto& [path, objDict] : ret)
325 {
326 if (objDict.empty())
327 {
328 return;
James Feistc71c1192019-09-18 14:31:33 -0700329 }
Ed Tanousbb679322022-05-16 16:10:00 -0700330 const std::string& owner = objDict.begin()->first;
331
332 for (const std::string& interface : objDict.begin()->second)
333 {
334 // anything that starts with a requested configuration
335 // is good
336 if (std::find_if(interfaces.begin(), interfaces.end(),
337 [interface](const std::string& possible) {
Zev Weiss6c106d62022-08-17 20:50:00 -0700338 return interface.starts_with(possible);
Ed Tanousbb679322022-05-16 16:10:00 -0700339 }) == interfaces.end())
340 {
341 continue;
342 }
343 self->getPath(path, interface, owner);
344 }
345 }
James Feistc71c1192019-09-18 14:31:33 -0700346 },
347 mapper::busName, mapper::path, mapper::interface, mapper::subtree,
348 "/", 0, interfaces);
349 }
350
351 ~GetSensorConfiguration()
352 {
353 callback(respData);
354 }
355
356 std::shared_ptr<sdbusplus::asio::connection> dbusConnection;
357 std::function<void(ManagedObjectType& resp)> callback;
358 ManagedObjectType respData;
359};
Zbigniew Kurzynski63f38662020-06-09 13:02:11 +0200360
361// The common scheme for sysfs files naming is: <type><number>_<item>.
362// This function returns optionally these 3 elements as a tuple.
363std::optional<std::tuple<std::string, std::string, std::string>>
364 splitFileName(const std::filesystem::path& filePath);
365std::optional<double> readFile(const std::string& thresholdFile,
James Feist8086aba2020-08-25 16:00:59 -0700366 const double& scaleFactor);
Bruce Lee1263c3d2021-06-04 15:16:33 +0800367void setupManufacturingModeMatch(sdbusplus::asio::connection& conn);
368bool getManufacturingMode();
Zev Weiss214d9712022-08-12 12:54:31 -0700369std::vector<std::unique_ptr<sdbusplus::bus::match_t>>
370 setupPropertiesChangedMatches(
371 sdbusplus::asio::connection& bus, std::span<const char* const> types,
372 const std::function<void(sdbusplus::message_t&)>& handler);