blob: 5bc99cda7b33897d0a47b77a884e2316e812165a [file] [log] [blame]
Sampa Misra032bd502019-03-06 05:03:22 -06001#pragma once
2
George Liu6492f522020-06-16 10:34:05 +08003#include "libpldm/base.h"
4#include "libpldm/bios.h"
5#include "libpldm/platform.h"
Tom Joseph54922072021-06-19 02:45:46 -07006#include "libpldm/utils.h"
George Liu6492f522020-06-16 10:34:05 +08007
Sampa Misraaea5dde2020-08-31 08:33:47 -05008#include "types.hpp"
9
Sampa Misra032bd502019-03-06 05:03:22 -060010#include <stdint.h>
11#include <systemd/sd-bus.h>
Jinu Joy Thomasf666db12019-05-29 05:22:31 -050012#include <unistd.h>
Sampa Misra032bd502019-03-06 05:03:22 -060013
George Liu6492f522020-06-16 10:34:05 +080014#include <nlohmann/json.hpp>
15#include <sdbusplus/server.hpp>
16#include <xyz/openbmc_project/Logging/Entry/server.hpp>
17
Sampa Misraa2fa0702019-05-31 01:28:55 -050018#include <exception>
Deepak Kodihalli3cd61812020-03-10 06:38:45 -050019#include <filesystem>
Sampa Misraaa8ae722019-12-12 03:20:40 -060020#include <iostream>
Sampa Misra032bd502019-03-06 05:03:22 -060021#include <string>
Sampa Misraa2fa0702019-05-31 01:28:55 -050022#include <variant>
23#include <vector>
Sampa Misra032bd502019-03-06 05:03:22 -060024
25namespace pldm
26{
Jinu Joy Thomasf666db12019-05-29 05:22:31 -050027namespace utils
28{
29
Deepak Kodihalli3cd61812020-03-10 06:38:45 -050030namespace fs = std::filesystem;
Tom Joseph250c4752020-04-15 10:32:45 +053031using Json = nlohmann::json;
Deepak Kodihalli3cd61812020-03-10 06:38:45 -050032
Jinu Joy Thomasf666db12019-05-29 05:22:31 -050033/** @struct CustomFD
34 *
35 * RAII wrapper for file descriptor.
36 */
37struct CustomFD
38{
39 CustomFD(const CustomFD&) = delete;
40 CustomFD& operator=(const CustomFD&) = delete;
41 CustomFD(CustomFD&&) = delete;
42 CustomFD& operator=(CustomFD&&) = delete;
43
44 CustomFD(int fd) : fd(fd)
George Liu6492f522020-06-16 10:34:05 +080045 {}
Jinu Joy Thomasf666db12019-05-29 05:22:31 -050046
47 ~CustomFD()
48 {
49 if (fd >= 0)
50 {
51 close(fd);
52 }
53 }
54
55 int operator()() const
56 {
57 return fd;
58 }
59
60 private:
61 int fd = -1;
62};
63
Sampa Misrab37be312019-07-03 02:26:41 -050064/** @brief Calculate the pad for PLDM data
65 *
66 * @param[in] data - Length of the data
67 * @return - uint8_t - number of pad bytes
68 */
69uint8_t getNumPadBytes(uint32_t data);
70
George Liu83409572019-12-24 18:42:54 +080071/** @brief Convert uint64 to date
72 *
73 * @param[in] data - time date of uint64
74 * @param[out] year - year number in dec
75 * @param[out] month - month number in dec
76 * @param[out] day - day of the month in dec
77 * @param[out] hour - number of hours in dec
78 * @param[out] min - number of minutes in dec
79 * @param[out] sec - number of seconds in dec
80 * @return true if decode success, false if decode faild
81 */
82bool uintToDate(uint64_t data, uint16_t* year, uint8_t* month, uint8_t* day,
83 uint8_t* hour, uint8_t* min, uint8_t* sec);
84
85/** @brief Convert effecter data to structure of set_effecter_state_field
86 *
87 * @param[in] effecterData - the date of effecter
George Liuba4c1fb2020-02-05 14:13:30 +080088 * @param[in] effecterCount - the number of individual sets of effecter
89 * information
90 * @return[out] parse success and get a valid set_effecter_state_field
91 * structure, return nullopt means parse failed
George Liu83409572019-12-24 18:42:54 +080092 */
George Liuba4c1fb2020-02-05 14:13:30 +080093std::optional<std::vector<set_effecter_state_field>>
94 parseEffecterData(const std::vector<uint8_t>& effecterData,
95 uint8_t effecterCount);
Sampa Misra032bd502019-03-06 05:03:22 -060096
97/**
Sampa Misraaa8ae722019-12-12 03:20:40 -060098 * @brief creates an error log
99 * @param[in] errorMsg - the error message
100 */
101void reportError(const char* errorMsg);
102
Sampa Misra032bd502019-03-06 05:03:22 -0600103/** @brief Convert any Decimal number to BCD
104 *
105 * @tparam[in] decimal - Decimal number
106 * @return Corresponding BCD number
107 */
108template <typename T>
109T decimalToBcd(T decimal)
110{
111 T bcd = 0;
112 T rem = 0;
113 auto cnt = 0;
114
115 while (decimal)
116 {
117 rem = decimal % 10;
118 bcd = bcd + (rem << cnt);
119 decimal = decimal / 10;
120 cnt += 4;
121 }
122
123 return bcd;
124}
125
Sampa Misraa2fa0702019-05-31 01:28:55 -0500126constexpr auto dbusProperties = "org.freedesktop.DBus.Properties";
Pavithra Barithaya47180ac2020-10-28 02:12:05 -0500127constexpr auto mapperService = "xyz.openbmc_project.ObjectMapper";
Sampa Misraa2fa0702019-05-31 01:28:55 -0500128
George Liu1e44c732020-02-28 20:20:06 +0800129struct DBusMapping
130{
131 std::string objectPath; //!< D-Bus object path
132 std::string interface; //!< D-Bus interface
133 std::string propertyName; //!< D-Bus property name
134 std::string propertyType; //!< D-Bus property type
135};
136
137using PropertyValue =
138 std::variant<bool, uint8_t, int16_t, uint16_t, int32_t, uint32_t, int64_t,
139 uint64_t, double, std::string>;
Deepak Kodihalli6b1d1ca2020-04-27 07:24:51 -0500140using DbusProp = std::string;
141using DbusChangedProps = std::map<DbusProp, PropertyValue>;
Sampa Misraaea5dde2020-08-31 08:33:47 -0500142using DBusInterfaceAdded = std::vector<
143 std::pair<pldm::dbus::Interface,
144 std::vector<std::pair<pldm::dbus::Property,
145 std::variant<pldm::dbus::Property>>>>>;
George Liu1e44c732020-02-28 20:20:06 +0800146
147/**
148 * @brief The interface for DBusHandler
149 */
150class DBusHandlerInterface
151{
152 public:
153 virtual ~DBusHandlerInterface() = default;
154
George Liu36e81352020-07-01 14:40:30 +0800155 virtual std::string getService(const char* path,
156 const char* interface) const = 0;
157
George Liu1e44c732020-02-28 20:20:06 +0800158 virtual void setDbusProperty(const DBusMapping& dBusMap,
159 const PropertyValue& value) const = 0;
John Wang9e242422020-03-05 08:37:50 +0800160
161 virtual PropertyValue
162 getDbusPropertyVariant(const char* objPath, const char* dbusProp,
163 const char* dbusInterface) const = 0;
George Liu1e44c732020-02-28 20:20:06 +0800164};
165
Sampa Misraa2fa0702019-05-31 01:28:55 -0500166/**
167 * @class DBusHandler
168 *
169 * Wrapper class to handle the D-Bus calls
170 *
171 * This class contains the APIs to handle the D-Bus calls
172 * to cater the request from pldm requester.
173 * A class is created to mock the apis in the test cases
174 */
George Liu1e44c732020-02-28 20:20:06 +0800175class DBusHandler : public DBusHandlerInterface
Sampa Misraa2fa0702019-05-31 01:28:55 -0500176{
177 public:
George Liu0e02c322020-01-01 09:41:51 +0800178 /** @brief Get the bus connection. */
179 static auto& getBus()
180 {
181 static auto bus = sdbusplus::bus::new_default();
182 return bus;
183 }
184
185 /**
186 * @brief Get the DBUS Service name for the input dbus path
John Wang9e242422020-03-05 08:37:50 +0800187 *
George Liu0e02c322020-01-01 09:41:51 +0800188 * @param[in] path - DBUS object path
189 * @param[in] interface - DBUS Interface
John Wang9e242422020-03-05 08:37:50 +0800190 *
George Liu0e02c322020-01-01 09:41:51 +0800191 * @return std::string - the dbus service name
John Wang9e242422020-03-05 08:37:50 +0800192 *
Patrick Williams4fea7a22021-09-02 09:54:12 -0500193 * @throw sdbusplus::exception::exception when it fails
George Liu0e02c322020-01-01 09:41:51 +0800194 */
George Liu36e81352020-07-01 14:40:30 +0800195 std::string getService(const char* path,
196 const char* interface) const override;
George Liu0e02c322020-01-01 09:41:51 +0800197
John Wang9e242422020-03-05 08:37:50 +0800198 /** @brief Get property(type: variant) from the requested dbus
199 *
200 * @param[in] objPath - The Dbus object path
201 * @param[in] dbusProp - The property name to get
202 * @param[in] dbusInterface - The Dbus interface
203 *
204 * @return The value of the property(type: variant)
205 *
Patrick Williams4fea7a22021-09-02 09:54:12 -0500206 * @throw sdbusplus::exception::exception when it fails
John Wang9e242422020-03-05 08:37:50 +0800207 */
208 PropertyValue
209 getDbusPropertyVariant(const char* objPath, const char* dbusProp,
210 const char* dbusInterface) const override;
George Liu0e02c322020-01-01 09:41:51 +0800211
John Wang9e242422020-03-05 08:37:50 +0800212 /** @brief The template function to get property from the requested dbus
213 * path
214 *
215 * @tparam Property - Excepted type of the property on dbus
216 *
217 * @param[in] objPath - The Dbus object path
218 * @param[in] dbusProp - The property name to get
219 * @param[in] dbusInterface - The Dbus interface
220 *
221 * @return The value of the property
222 *
Patrick Williams4fea7a22021-09-02 09:54:12 -0500223 * @throw sdbusplus::exception::exception when dbus request fails
John Wang9e242422020-03-05 08:37:50 +0800224 * std::bad_variant_access when \p Property and property on dbus do
225 * not match
226 */
John Wang92b3c972019-10-17 11:06:41 +0800227 template <typename Property>
228 auto getDbusProperty(const char* objPath, const char* dbusProp,
229 const char* dbusInterface)
230 {
John Wang9e242422020-03-05 08:37:50 +0800231 auto VariantValue =
232 getDbusPropertyVariant(objPath, dbusProp, dbusInterface);
John Wang92b3c972019-10-17 11:06:41 +0800233 return std::get<Property>(VariantValue);
234 }
George Liu1e44c732020-02-28 20:20:06 +0800235
236 /** @brief Set Dbus property
237 *
238 * @param[in] dBusMap - Object path, property name, interface and property
239 * type for the D-Bus object
240 * @param[in] value - The value to be set
John Wang9e242422020-03-05 08:37:50 +0800241 *
Patrick Williams4fea7a22021-09-02 09:54:12 -0500242 * @throw sdbusplus::exception::exception when it fails
George Liu1e44c732020-02-28 20:20:06 +0800243 */
244 void setDbusProperty(const DBusMapping& dBusMap,
245 const PropertyValue& value) const override;
Sampa Misraa2fa0702019-05-31 01:28:55 -0500246};
247
Deepak Kodihalli3cd61812020-03-10 06:38:45 -0500248/** @brief Fetch parent D-Bus object based on pathname
249 *
250 * @param[in] dbusObj - child D-Bus object
251 *
252 * @return std::string - the parent D-Bus object path
253 */
254inline std::string findParent(const std::string& dbusObj)
255{
256 fs::path p(dbusObj);
257 return p.parent_path().string();
258}
259
Pavithra Barithaya51efaf82020-04-02 02:42:27 -0500260/** @brief Read (static) MCTP EID of host firmware from a file
261 *
262 * @return uint8_t - MCTP EID
263 */
264uint8_t readHostEID();
Tom Joseph250c4752020-04-15 10:32:45 +0530265
TOM JOSEPHd4d97a52020-03-23 14:36:34 +0530266/** @brief Convert a value in the JSON to a D-Bus property value
267 *
268 * @param[in] type - type of the D-Bus property
269 * @param[in] value - value in the JSON file
270 *
271 * @return PropertyValue - the D-Bus property value
272 */
273PropertyValue jsonEntryToDbusVal(std::string_view type,
274 const nlohmann::json& value);
Pavithra Barithaya51efaf82020-04-02 02:42:27 -0500275
Pavithra Barithaya0f74c982020-04-27 02:17:10 -0500276/** @brief Find State Effecter PDR
277 * @param[in] tid - PLDM terminus ID.
278 * @param[in] entityID - entity that can be associated with PLDM State set.
279 * @param[in] stateSetId - value that identifies PLDM State set.
280 * @param[in] repo - pointer to BMC's primary PDR repo.
281 * @return array[array[uint8_t]] - StateEffecterPDRs
282 */
283std::vector<std::vector<uint8_t>> findStateEffecterPDR(uint8_t tid,
284 uint16_t entityID,
285 uint16_t stateSetId,
286 const pldm_pdr* repo);
Chicago Duan738e4d82020-05-28 16:39:19 +0800287/** @brief Find State Sensor PDR
288 * @param[in] tid - PLDM terminus ID.
289 * @param[in] entityID - entity that can be associated with PLDM State set.
290 * @param[in] stateSetId - value that identifies PLDM State set.
291 * @param[in] repo - pointer to BMC's primary PDR repo.
292 * @return array[array[uint8_t]] - StateSensorPDRs
293 */
294std::vector<std::vector<uint8_t>> findStateSensorPDR(uint8_t tid,
295 uint16_t entityID,
296 uint16_t stateSetId,
297 const pldm_pdr* repo);
Pavithra Barithaya0f74c982020-04-27 02:17:10 -0500298
Sampa Misra3a0e3b92020-10-21 05:58:00 -0500299/** @brief Find sensor id from a state sensor PDR
300 *
301 * @param[in] pdrRepo - PDR repository
302 * @param[in] tid - terminus id
303 * @param[in] entityType - entity type
304 * @param[in] entityInstance - entity instance number
305 * @param[in] containerId - container id
306 * @param[in] stateSetId - state set id
307 *
308 * @return uint16_t - the sensor id
309 */
310uint16_t findStateSensorId(const pldm_pdr* pdrRepo, uint8_t tid,
311 uint16_t entityType, uint16_t entityInstance,
312 uint16_t containerId, uint16_t stateSetId);
313
Tom Joseph250c4752020-04-15 10:32:45 +0530314/** @brief Find effecter id from a state effecter pdr
315 * @param[in] pdrRepo - PDR repository
316 * @param[in] entityType - entity type
317 * @param[in] entityInstance - entity instance number
318 * @param[in] containerId - container id
319 * @param[in] stateSetId - state set id
Sampa Misraa4a96162020-07-14 05:33:46 -0500320 * @param[in] localOrRemote - true for checking local repo and false for remote
321 * repo
Tom Joseph250c4752020-04-15 10:32:45 +0530322 *
323 * @return uint16_t - the effecter id
324 */
325uint16_t findStateEffecterId(const pldm_pdr* pdrRepo, uint16_t entityType,
326 uint16_t entityInstance, uint16_t containerId,
Sampa Misraa4a96162020-07-14 05:33:46 -0500327 uint16_t stateSetId, bool localOrRemote);
Tom Joseph250c4752020-04-15 10:32:45 +0530328
Chicago Duanfe4d88b2020-06-12 16:44:13 +0800329/** @brief Emit the sensor event signal
330 *
331 * @param[in] tid - the terminus id
332 * @param[in] sensorId - sensorID value of the sensor
333 * @param[in] sensorOffset - Identifies which state sensor within a
334 * composite state sensor the event is being returned for
335 * @param[in] eventState - The event state value from the state change that
336 * triggered the event message
337 * @param[in] previousEventState - The event state value for the state from
338 * which the present event state was entered.
339 * @return PLDM completion code
340 */
341int emitStateSensorEventSignal(uint8_t tid, uint16_t sensorId,
342 uint8_t sensorOffset, uint8_t eventState,
343 uint8_t previousEventState);
344
Sridevi Rameshae28bc72020-12-10 07:21:16 -0600345/** @brief Print the buffer
346 *
347 * @param[in] buffer - Buffer to print
348 * @param[in] pldmVerbose -verbosity flag - true/false
349 *
350 * @return - None
351 */
352void printBuffer(const std::vector<uint8_t>& buffer, bool pldmVerbose);
353
Tom Joseph54922072021-06-19 02:45:46 -0700354/** @brief Convert the buffer to std::string
355 *
356 * If there are characters that are not printable characters, it is replaced
357 * with space(0x20).
358 *
359 * @param[in] var - pointer to data and length of the data
360 *
361 * @return std::string equivalent of variable field
362 */
363std::string toString(const struct variable_field& var);
364
George Liu83409572019-12-24 18:42:54 +0800365} // namespace utils
Sampa Misra032bd502019-03-06 05:03:22 -0600366} // namespace pldm