blob: e53afd3c8e97c697f6a01f152b1711d781466d7d [file] [log] [blame]
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -05001#include "config.h"
2
3#include "dump_offload.hpp"
4
Ravi Tejaff9c4522020-11-13 05:14:51 -06005#include <sys/socket.h>
6#include <sys/types.h>
7#include <sys/un.h>
8#include <unistd.h>
9
10#include <dump_utils.hpp>
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -050011#include <phosphor-logging/elog-errors.hpp>
12#include <phosphor-logging/elog.hpp>
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -050013#include <phosphor-logging/lg2.hpp>
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -050014#include <xyz/openbmc_project/Common/File/error.hpp>
15#include <xyz/openbmc_project/Common/error.hpp>
16
Jayanth Othayoth0af74a52021-04-08 03:55:21 -050017#include <fstream>
Jayanth Othayothceb3e762024-11-25 23:19:37 -060018#include <span>
Jayanth Othayoth0af74a52021-04-08 03:55:21 -050019
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -050020namespace phosphor
21{
22namespace dump
23{
24namespace offload
25{
26
27using namespace sdbusplus::xyz::openbmc_project::Common::Error;
28using namespace phosphor::logging;
29
Ravi Tejaff9c4522020-11-13 05:14:51 -060030/** @brief API to write data on unix socket.
31 *
32 * @param[in] socket - unix socket
33 * @param[in] buf - buffer
34 * @param[in] blockSize - size of data
35 *
36 * @return void
37 */
38void writeOnUnixSocket(const int socket, const char* buf,
39 const uint64_t blockSize)
40{
Jayanth Othayoth17ba8762024-11-25 10:43:24 -060041 ssize_t numOfBytesWrote = 0;
Ravi Tejaff9c4522020-11-13 05:14:51 -060042
43 for (uint64_t i = 0; i < blockSize; i = i + numOfBytesWrote)
44 {
45 numOfBytesWrote = 0;
46 fd_set writeFileDescriptor;
47 struct timeval timeVal;
48 timeVal.tv_sec = 5;
49 timeVal.tv_usec = 0;
50
51 FD_ZERO(&writeFileDescriptor);
52 FD_SET(socket, &writeFileDescriptor);
53 int nextFileDescriptor = socket + 1;
54
Jayanth Othayothad6fb6e2024-11-26 01:31:13 -060055 int retVal = select(nextFileDescriptor, nullptr, &writeFileDescriptor,
56 nullptr, &timeVal);
Ravi Tejaff9c4522020-11-13 05:14:51 -060057 if (retVal <= 0)
58 {
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -050059 lg2::error("writeOnUnixSocket: select() failed, errno: {ERRNO}",
60 "ERRNO", errno);
Ravi Tejaff9c4522020-11-13 05:14:51 -060061 std::string msg = "select() failed " + std::string(strerror(errno));
62 throw std::runtime_error(msg);
63 }
64 if ((retVal > 0) && (FD_ISSET(socket, &writeFileDescriptor)))
65 {
66 numOfBytesWrote = write(socket, buf + i, blockSize - i);
67 if (numOfBytesWrote < 0)
68 {
69 if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR)
70 {
71 numOfBytesWrote = 0;
72 continue;
73 }
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -050074 lg2::error("writeOnUnixSocket: write() failed, errno: {ERRNO}",
75 "ERRNO", errno);
Patrick Williams973b2912024-08-16 15:20:50 -040076 std::string msg =
77 "write() on socket failed " + std::string(strerror(errno));
Ravi Tejaff9c4522020-11-13 05:14:51 -060078 throw std::runtime_error(msg);
79 }
80 }
81 }
82 return;
83}
84
85/**@brief API to setup unix socket.
86 *
87 * @param[in] sockPath - unix socket path
88 *
89 * @return returns returns socket fd on success
90 * and on error exception will be thrown
91 */
92int socketInit(const std::string& sockPath)
93{
94 int unixSocket;
95 struct sockaddr_un socketAddr;
96 memset(&socketAddr, 0, sizeof(socketAddr));
97 socketAddr.sun_family = AF_UNIX;
98 if (strnlen(sockPath.c_str(), sizeof(socketAddr.sun_path)) ==
99 sizeof(socketAddr.sun_path))
100 {
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500101 lg2::error("UNIX socket path too long");
Patrick Williams973b2912024-08-16 15:20:50 -0400102 std::string msg =
103 "UNIX socket path is too long " + std::string(strerror(errno));
Ravi Tejaff9c4522020-11-13 05:14:51 -0600104 throw std::length_error(msg);
105 }
Jayanth Othayothceb3e762024-11-25 23:19:37 -0600106
107 std::span<char> sunPathSpan(reinterpret_cast<char*>(socketAddr.sun_path),
108 sizeof(socketAddr.sun_path));
109 strncpy(sunPathSpan.data(), sockPath.c_str(), sunPathSpan.size() - 1);
110 sunPathSpan[sunPathSpan.size() - 1] = '\0'; // Ensure null-termination
111
Ravi Tejaff9c4522020-11-13 05:14:51 -0600112 if ((unixSocket = socket(AF_UNIX, SOCK_STREAM | SOCK_NONBLOCK, 0)) == -1)
113 {
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500114 lg2::error("socketInit: socket() failed, errno: {ERRNO}", "ERRNO",
115 errno);
Ravi Tejaff9c4522020-11-13 05:14:51 -0600116 std::string msg = "socket() failed " + std::string(strerror(errno));
117 throw std::runtime_error(msg);
118 }
119 if (bind(unixSocket, (struct sockaddr*)&socketAddr, sizeof(socketAddr)) ==
120 -1)
121 {
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500122 lg2::error("socketInit: bind() failed, errno: {ERRNO}", "ERRNO", errno);
Ravi Tejaff9c4522020-11-13 05:14:51 -0600123 close(unixSocket);
124 std::string msg = "socket bind failed " + std::string(strerror(errno));
125 throw std::runtime_error(msg);
126 }
127 if (listen(unixSocket, 1) == -1)
128 {
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500129 lg2::error("socketInit: listen() failed, errno: {ERRNO}", "ERRNO",
130 errno);
Ravi Tejaff9c4522020-11-13 05:14:51 -0600131 close(unixSocket);
132 std::string msg = "listen() failed " + std::string(strerror(errno));
133 throw std::runtime_error(msg);
134 }
135 return unixSocket;
136}
137
Jayanth Othayoth3fc6df42021-04-08 03:45:24 -0500138void requestOffload(std::filesystem::path file, uint32_t dumpId,
139 std::string writePath)
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500140{
141 using namespace sdbusplus::xyz::openbmc_project::Common::File::Error;
142 using ErrnoOpen = xyz::openbmc_project::Common::File::Open::ERRNO;
143 using PathOpen = xyz::openbmc_project::Common::File::Open::PATH;
144 using ErrnoWrite = xyz::openbmc_project::Common::File::Write::ERRNO;
145 using PathWrite = xyz::openbmc_project::Common::File::Write::PATH;
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500146
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500147 try
148 {
Ravi Tejaff9c4522020-11-13 05:14:51 -0600149 CustomFd unixSocket = socketInit(writePath);
150
151 fd_set readFD;
152 struct timeval timeVal;
153 timeVal.tv_sec = 1;
154 timeVal.tv_usec = 0;
155
156 FD_ZERO(&readFD);
157 FD_SET(unixSocket(), &readFD);
158 int numOfFDs = unixSocket() + 1;
159
Jayanth Othayothad6fb6e2024-11-26 01:31:13 -0600160 int retVal = select(numOfFDs, &readFD, nullptr, nullptr, &timeVal);
Ravi Tejaff9c4522020-11-13 05:14:51 -0600161 if (retVal <= 0)
162 {
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500163 lg2::error("select() failed, errno: {ERRNO}, DUMP_ID: {DUMP_ID}",
164 "ERRNO", errno, "DUMP_ID", dumpId);
Ravi Tejaff9c4522020-11-13 05:14:51 -0600165 std::string msg = "select() failed " + std::string(strerror(errno));
166 throw std::runtime_error(msg);
167 }
168 else if ((retVal > 0) && (FD_ISSET(unixSocket(), &readFD)))
169 {
Jayanth Othayothad6fb6e2024-11-26 01:31:13 -0600170 CustomFd socketFD = accept(unixSocket(), nullptr, nullptr);
Ravi Tejaff9c4522020-11-13 05:14:51 -0600171 if (socketFD() < 0)
172 {
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500173 lg2::error(
174 "accept() failed, errno: {ERRNO}, DUMP_ID: {DUMP_ID}",
175 "ERRNO", errno, "DUMP_ID", dumpId);
Patrick Williams973b2912024-08-16 15:20:50 -0400176 std::string msg =
177 "accept() failed " + std::string(strerror(errno));
Ravi Tejaff9c4522020-11-13 05:14:51 -0600178 throw std::runtime_error(msg);
179 }
180
Dhruvaraj Subhashchandranee6f1ba2021-05-04 09:44:57 -0500181 std::ifstream infile{file, std::ios::in | std::ios::binary};
Ravi Tejaff9c4522020-11-13 05:14:51 -0600182 if (!infile.good())
183 {
184 // Unable to open the dump file
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500185 lg2::error("Failed to open the dump from file, errno: {ERRNO}, "
186 "DUMPFILE: {DUMP_FILE}, DUMP_ID: {DUMP_ID}",
187 "ERRNO", errno, "DUMP_FILE", file, "DUMP_ID",
188 dumpId);
Dhruvaraj Subhashchandranee6f1ba2021-05-04 09:44:57 -0500189 elog<Open>(ErrnoOpen(errno), PathOpen(file.c_str()));
Ravi Tejaff9c4522020-11-13 05:14:51 -0600190 }
191
192 infile.exceptions(std::ifstream::failbit | std::ifstream::badbit |
193 std::ifstream::eofbit);
194
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500195 lg2::info("Opening File for RW, FILENAME: {FILENAME}", "FILENAME",
196 file.filename().c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -0600197
198 std::filebuf* pbuf = infile.rdbuf();
199
200 // get file size using buffer's members
201 std::size_t size = pbuf->pubseekoff(0, infile.end, infile.in);
202 pbuf->pubseekpos(0, infile.in);
203
204 // allocate memory to contain file data
205 std::unique_ptr<char[]> buffer(new char[size]);
206 // get file data
Jayanth Othayoth17ba8762024-11-25 10:43:24 -0600207 pbuf->sgetn(buffer.get(), static_cast<std::streamsize>(size));
Ravi Tejaff9c4522020-11-13 05:14:51 -0600208 infile.close();
209
210 writeOnUnixSocket(socketFD(), buffer.get(), size);
211 }
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500212 }
Patrick Williams9d2d7222021-10-06 12:44:44 -0500213 catch (const std::ifstream::failure& oe)
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500214 {
Ravi Tejaff9c4522020-11-13 05:14:51 -0600215 std::remove(writePath.c_str());
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500216 auto err = errno;
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500217 lg2::error("Failed to open, errormsg: {ERROR}, "
218 "OPENINTERFACE: {OPEN_INTERFACE}, DUMP_ID: {DUMP_ID}",
219 "ERROR", oe, "OPEN_INTERFACE", file, "DUMP_ID", dumpId);
Dhruvaraj Subhashchandranee6f1ba2021-05-04 09:44:57 -0500220 elog<Open>(ErrnoOpen(err), PathOpen(file.c_str()));
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500221 }
222 catch (const std::exception& e)
223 {
Ravi Tejaff9c4522020-11-13 05:14:51 -0600224 std::remove(writePath.c_str());
225 auto err = errno;
Dhruvaraj Subhashchandrand1f670f2023-06-05 22:19:25 -0500226 lg2::error("Failed to offload dump, errormsg: {ERROR}, "
227 "DUMPFILE: {DUMP_FILE}, DUMP_ID: {DUMP_ID}",
228 "ERROR", e, "DUMP_FILE", writePath, "DUMP_ID", dumpId);
Ravi Tejaff9c4522020-11-13 05:14:51 -0600229 elog<Write>(ErrnoWrite(err), PathWrite(writePath.c_str()));
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500230 }
Ravi Tejaff9c4522020-11-13 05:14:51 -0600231 std::remove(writePath.c_str());
232 return;
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500233}
234
235} // namespace offload
236} // namespace dump
237} // namespace phosphor