blob: 67637680066228d80d6a9a7744788256661e90cf [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
12std::string getService(sdbusplus::bus::bus& bus, const std::string& path,
13 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 Williams4ce901c2021-09-02 09:34:45 -050035 catch (const sdbusplus::exception::exception& 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
George Liufc025e12021-11-09 19:29:12 +080044const PropertyValue getProperty(sdbusplus::bus::bus& bus,
45 const std::string& objectPath,
46 const std::string& interface,
47 const std::string& propertyName)
48{
49 PropertyValue value{};
50 auto service = getService(bus, objectPath, interface);
51 if (service.empty())
52 {
53 return value;
54 }
55
56 auto method = bus.new_method_call(service.c_str(), objectPath.c_str(),
57 "org.freedesktop.DBus.Properties", "Get");
58 method.append(interface, propertyName);
59
60 auto reply = bus.call(method);
61 reply.read(value);
62
63 return value;
64}
65
66void setProperty(sdbusplus::bus::bus& bus, const std::string& objectPath,
67 const std::string& interface, const std::string& propertyName,
68 const PropertyValue& value)
69{
70 auto service = getService(bus, objectPath, interface);
71 if (service.empty())
72 {
73 return;
74 }
75
76 auto method = bus.new_method_call(service.c_str(), objectPath.c_str(),
77 "org.freedesktop.DBus.Properties", "Set");
78 method.append(interface.c_str(), propertyName.c_str(), value);
79
80 bus.call_noreply(method);
81}
82
Lei YU0cd6d842021-12-27 11:56:02 +080083void mergeFiles(const std::vector<std::string>& srcFiles,
84 const std::string& dstFile)
George Liu0a06e972020-12-17 09:17:04 +080085{
86 std::ofstream outFile(dstFile, std::ios::out);
Lei YU0cd6d842021-12-27 11:56:02 +080087 for (const auto& file : srcFiles)
George Liu0a06e972020-12-17 09:17:04 +080088 {
89 std::ifstream inFile;
90 inFile.open(file, std::ios_base::in);
91 if (!inFile)
92 {
93 continue;
94 }
95
96 inFile.peek();
97 if (inFile.eof())
98 {
99 inFile.close();
100 continue;
101 }
102
103 outFile << inFile.rdbuf();
104 inFile.close();
105 }
106 outFile.close();
107}
108
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600109namespace internal
110{
111
112/* @brief Helper function to build a string from command arguments */
113static std::string buildCommandStr(const char* name, char** args)
114{
115 std::string command = name;
116 for (int i = 0; args[i]; i++)
117 {
118 command += " ";
119 command += args[i];
120 }
121 return command;
122}
123
124int executeCmd(const char* path, char** args)
125{
126 pid_t pid = fork();
127 if (pid == 0)
128 {
129 execv(path, args);
130
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500131 // execv only retruns on err
132 auto err = errno;
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600133 auto command = buildCommandStr(path, args);
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500134 error("Failed ({ERRNO}) to execute command: {COMMAND}", "ERRNO", err,
135 "COMMAND", command);
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600136 return -1;
137 }
138 else if (pid > 0)
139 {
140 int status;
141 if (waitpid(pid, &status, 0) < 0)
142 {
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500143 error("Error ({ERRNO}) during waitpid.", "ERRNO", errno);
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600144 return -1;
145 }
146 else if (WEXITSTATUS(status) != 0)
147 {
148 auto command = buildCommandStr(path, args);
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500149 error("Error ({STATUS}) occurred when executing command: {COMMAND}",
150 "STATUS", status, "COMMAND", command);
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600151 return -1;
152 }
153 }
154 else
155 {
Patrick Williamsc9bb6422021-08-27 06:18:35 -0500156 error("Error ({ERRNO}) during fork.", "ERRNO", errno);
Adriana Kobylak8a5ccbb2021-01-20 10:57:05 -0600157 return -1;
158 }
159
160 return 0;
161}
162
163} // namespace internal
164
Jayashankar Padatha0135602019-04-22 16:22:58 +0530165} // namespace utils