blob: 91857fc5d1bfbe546eb9b9afa69b771dba9ee6d1 [file] [log] [blame]
Adriana Kobylak5ed9b2d2018-09-06 13:15:34 -05001#include "utils.hpp"
2
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -06003#include <unistd.h>
4
Patrick Williamsc9bb6422021-08-27 06:18:35 -05005#include <phosphor-logging/lg2.hpp>
Adriana Kobylak5ed9b2d2018-09-06 13:15:34 -05006
Jayashankar Padatha0135602019-04-22 16:22:58 +05307namespace utils
Adriana Kobylak5ed9b2d2018-09-06 13:15:34 -05008{
Adriana Kobylak5ed9b2d2018-09-06 13:15:34 -05009
Patrick Williamsc9bb6422021-08-27 06:18:35 -050010PHOSPHOR_LOG2_USING;
Jayashankar Padatha0135602019-04-22 16:22:58 +053011
Patrick Williamsbf2bb2b2022-07-22 19:26:52 -050012std::string getService(sdbusplus::bus_t& bus, const std::string& path,
Jayashankar Padatha0135602019-04-22 16:22:58 +053013 const std::string& interface)
14{
15 auto method = bus.new_method_call(MAPPER_BUSNAME, MAPPER_PATH,
16 MAPPER_BUSNAME, "GetObject");
17
18 method.append(path);
19 method.append(std::vector<std::string>({interface}));
20
21 std::vector<std::pair<std::string, std::vector<std::string>>> response;
22
23 try
Adriana Kobylak5ed9b2d2018-09-06 13:15:34 -050024 {
Jayashankar Padatha0135602019-04-22 16:22:58 +053025 auto reply = bus.call(method);
26 reply.read(response);
27 if (response.empty())
28 {
Patrick Williamsc9bb6422021-08-27 06:18:35 -050029 error(
30 "Empty response from mapper for getting service name: {PATH} {INTERFACE}",
31 "PATH", path, "INTERFACE", interface);
Jayashankar Padatha0135602019-04-22 16:22:58 +053032 return std::string{};
33 }
Adriana Kobylak5ed9b2d2018-09-06 13:15:34 -050034 }
Patrick Williamsbf2bb2b2022-07-22 19:26:52 -050035 catch (const sdbusplus::exception_t& e)
Jayashankar Padatha0135602019-04-22 16:22:58 +053036 {
Patrick Williamsc9bb6422021-08-27 06:18:35 -050037 error("Error in mapper method call for ({PATH}, {INTERFACE}: {ERROR}",
38 "ERROR", e, "PATH", path, "INTERFACE", interface);
Jayashankar Padatha0135602019-04-22 16:22:58 +053039 return std::string{};
40 }
41 return response[0].first;
Adriana Kobylak5ed9b2d2018-09-06 13:15:34 -050042}
43
Patrick Williamsbf2bb2b2022-07-22 19:26:52 -050044void setProperty(sdbusplus::bus_t& bus, const std::string& objectPath,
George Liufc025e12021-11-09 19:29:12 +080045 const std::string& interface, const std::string& propertyName,
46 const PropertyValue& value)
47{
48 auto service = getService(bus, objectPath, interface);
49 if (service.empty())
50 {
51 return;
52 }
53
54 auto method = bus.new_method_call(service.c_str(), objectPath.c_str(),
55 "org.freedesktop.DBus.Properties", "Set");
56 method.append(interface.c_str(), propertyName.c_str(), value);
57
58 bus.call_noreply(method);
59}
60
Lei YU0cd6d842021-12-27 11:56:02 +080061void mergeFiles(const std::vector<std::string>& srcFiles,
62 const std::string& dstFile)
George Liu0a06e972020-12-17 09:17:04 +080063{
64 std::ofstream outFile(dstFile, std::ios::out);
Lei YU0cd6d842021-12-27 11:56:02 +080065 for (const auto& file : srcFiles)
George Liu0a06e972020-12-17 09:17:04 +080066 {
67 std::ifstream inFile;
68 inFile.open(file, std::ios_base::in);
69 if (!inFile)
70 {
71 continue;
72 }
73
74 inFile.peek();
75 if (inFile.eof())
76 {
77 inFile.close();
78 continue;
79 }
80
81 outFile << inFile.rdbuf();
82 inFile.close();
83 }
84 outFile.close();
85}
86
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -060087namespace internal
88{
89
90/* @brief Helper function to build a string from command arguments */
91static std::string buildCommandStr(const char* name, char** args)
92{
93 std::string command = name;
94 for (int i = 0; args[i]; i++)
95 {
96 command += " ";
97 command += args[i];
98 }
99 return command;
100}
101
102int executeCmd(const char* path, char** args)
103{
104 pid_t pid = fork();
105 if (pid == 0)
106 {
107 execv(path, args);
108
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500109 // execv only retruns on err
110 auto err = errno;
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600111 auto command = buildCommandStr(path, args);
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500112 error("Failed ({ERRNO}) to execute command: {COMMAND}", "ERRNO", err,
113 "COMMAND", command);
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600114 return -1;
115 }
116 else if (pid > 0)
117 {
118 int status;
119 if (waitpid(pid, &status, 0) < 0)
120 {
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500121 error("Error ({ERRNO}) during waitpid.", "ERRNO", errno);
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600122 return -1;
123 }
124 else if (WEXITSTATUS(status) != 0)
125 {
126 auto command = buildCommandStr(path, args);
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500127 error("Error ({STATUS}) occurred when executing command: {COMMAND}",
128 "STATUS", status, "COMMAND", command);
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600129 return -1;
130 }
131 }
132 else
133 {
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500134 error("Error ({ERRNO}) during fork.", "ERRNO", errno);
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600135 return -1;
136 }
137
138 return 0;
139}
140
141} // namespace internal
142
Jayashankar Padatha0135602019-04-22 16:22:58 +0530143} // namespace utils