blob: 3c4b834e034a824046861059042d2e3691c52fe1 [file] [log] [blame]
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -05001#include "config.h"
2
3#include "dump_offload.hpp"
4
George Liu858fbb22021-07-01 12:25:44 +08005#include <fmt/core.h>
Ravi Tejaff9c4522020-11-13 05:14:51 -06006#include <sys/socket.h>
7#include <sys/types.h>
8#include <sys/un.h>
9#include <unistd.h>
10
11#include <dump_utils.hpp>
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -050012#include <phosphor-logging/elog-errors.hpp>
13#include <phosphor-logging/elog.hpp>
14#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>
18
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -050019namespace phosphor
20{
21namespace dump
22{
23namespace offload
24{
25
26using namespace sdbusplus::xyz::openbmc_project::Common::Error;
27using namespace phosphor::logging;
28
Ravi Tejaff9c4522020-11-13 05:14:51 -060029/** @brief API to write data on unix socket.
30 *
31 * @param[in] socket - unix socket
32 * @param[in] buf - buffer
33 * @param[in] blockSize - size of data
34 *
35 * @return void
36 */
37void writeOnUnixSocket(const int socket, const char* buf,
38 const uint64_t blockSize)
39{
40 int numOfBytesWrote = 0;
41
42 for (uint64_t i = 0; i < blockSize; i = i + numOfBytesWrote)
43 {
44 numOfBytesWrote = 0;
45 fd_set writeFileDescriptor;
46 struct timeval timeVal;
47 timeVal.tv_sec = 5;
48 timeVal.tv_usec = 0;
49
50 FD_ZERO(&writeFileDescriptor);
51 FD_SET(socket, &writeFileDescriptor);
52 int nextFileDescriptor = socket + 1;
53
54 int retVal = select(nextFileDescriptor, NULL, &writeFileDescriptor,
55 NULL, &timeVal);
56 if (retVal <= 0)
57 {
George Liu858fbb22021-07-01 12:25:44 +080058 log<level::ERR>(
59 fmt::format("writeOnUnixSocket: select() failed, errno({})",
60 errno)
61 .c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -060062 std::string msg = "select() failed " + std::string(strerror(errno));
63 throw std::runtime_error(msg);
64 }
65 if ((retVal > 0) && (FD_ISSET(socket, &writeFileDescriptor)))
66 {
67 numOfBytesWrote = write(socket, buf + i, blockSize - i);
68 if (numOfBytesWrote < 0)
69 {
70 if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR)
71 {
72 numOfBytesWrote = 0;
73 continue;
74 }
George Liu858fbb22021-07-01 12:25:44 +080075 log<level::ERR>(
76 fmt::format("writeOnUnixSocket: write() failed, errno({})",
77 errno)
78 .c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -060079 std::string msg =
80 "write() on socket failed " + std::string(strerror(errno));
81 throw std::runtime_error(msg);
82 }
83 }
84 }
85 return;
86}
87
88/**@brief API to setup unix socket.
89 *
90 * @param[in] sockPath - unix socket path
91 *
92 * @return returns returns socket fd on success
93 * and on error exception will be thrown
94 */
95int socketInit(const std::string& sockPath)
96{
97 int unixSocket;
98 struct sockaddr_un socketAddr;
99 memset(&socketAddr, 0, sizeof(socketAddr));
100 socketAddr.sun_family = AF_UNIX;
101 if (strnlen(sockPath.c_str(), sizeof(socketAddr.sun_path)) ==
102 sizeof(socketAddr.sun_path))
103 {
104 log<level::ERR>("UNIX socket path too long");
105 std::string msg =
106 "UNIX socket path is too long " + std::string(strerror(errno));
107 throw std::length_error(msg);
108 }
109 strncpy(socketAddr.sun_path, sockPath.c_str(),
110 sizeof(socketAddr.sun_path) - 1);
111 if ((unixSocket = socket(AF_UNIX, SOCK_STREAM | SOCK_NONBLOCK, 0)) == -1)
112 {
George Liu858fbb22021-07-01 12:25:44 +0800113 log<level::ERR>(
114 fmt::format("socketInit: socket() failed, errno({})", errno)
115 .c_str());
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 {
George Liu858fbb22021-07-01 12:25:44 +0800122 log<level::ERR>(
123 fmt::format("socketInit: bind() failed, errno({})", errno).c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -0600124 close(unixSocket);
125 std::string msg = "socket bind failed " + std::string(strerror(errno));
126 throw std::runtime_error(msg);
127 }
128 if (listen(unixSocket, 1) == -1)
129 {
George Liu858fbb22021-07-01 12:25:44 +0800130 log<level::ERR>(
131 fmt::format("socketInit: listen() failed, errno({})", errno)
132 .c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -0600133 close(unixSocket);
134 std::string msg = "listen() failed " + std::string(strerror(errno));
135 throw std::runtime_error(msg);
136 }
137 return unixSocket;
138}
139
Jayanth Othayoth3fc6df42021-04-08 03:45:24 -0500140void requestOffload(std::filesystem::path file, uint32_t dumpId,
141 std::string writePath)
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500142{
143 using namespace sdbusplus::xyz::openbmc_project::Common::File::Error;
144 using ErrnoOpen = xyz::openbmc_project::Common::File::Open::ERRNO;
145 using PathOpen = xyz::openbmc_project::Common::File::Open::PATH;
146 using ErrnoWrite = xyz::openbmc_project::Common::File::Write::ERRNO;
147 using PathWrite = xyz::openbmc_project::Common::File::Write::PATH;
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500148 // open a dump file for a transfer.
Jayanth Othayoth3fc6df42021-04-08 03:45:24 -0500149 std::filesystem::path dumpPath(BMC_DUMP_PATH);
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500150 dumpPath /= std::to_string(dumpId);
151 dumpPath /= file.filename();
152
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500153 try
154 {
Ravi Tejaff9c4522020-11-13 05:14:51 -0600155
156 CustomFd unixSocket = socketInit(writePath);
157
158 fd_set readFD;
159 struct timeval timeVal;
160 timeVal.tv_sec = 1;
161 timeVal.tv_usec = 0;
162
163 FD_ZERO(&readFD);
164 FD_SET(unixSocket(), &readFD);
165 int numOfFDs = unixSocket() + 1;
166
167 int retVal = select(numOfFDs, &readFD, NULL, NULL, &timeVal);
168 if (retVal <= 0)
169 {
George Liu858fbb22021-07-01 12:25:44 +0800170 log<level::ERR>(
171 fmt::format("select() failed, errno({}), DUMP_ID({})", errno,
172 dumpId)
173 .c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -0600174 std::string msg = "select() failed " + std::string(strerror(errno));
175 throw std::runtime_error(msg);
176 }
177 else if ((retVal > 0) && (FD_ISSET(unixSocket(), &readFD)))
178 {
179 CustomFd socketFD = accept(unixSocket(), NULL, NULL);
180 if (socketFD() < 0)
181 {
George Liu858fbb22021-07-01 12:25:44 +0800182 log<level::ERR>(
183 fmt::format("accept() failed, errno({}), DUMP_ID({})",
184 errno, dumpId)
185 .c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -0600186 std::string msg =
187 "accept() failed " + std::string(strerror(errno));
188 throw std::runtime_error(msg);
189 }
190
191 std::ifstream infile{dumpPath, std::ios::in | std::ios::binary};
192 if (!infile.good())
193 {
194 // Unable to open the dump file
George Liu858fbb22021-07-01 12:25:44 +0800195 log<level::ERR>(
196 fmt::format("Failed to open the dump from file, errno({}), "
197 "DUMPFILE({}), DUMP_ID({})",
198 errno, dumpPath.c_str(), dumpId)
199 .c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -0600200 elog<Open>(ErrnoOpen(errno), PathOpen(dumpPath.c_str()));
201 }
202
203 infile.exceptions(std::ifstream::failbit | std::ifstream::badbit |
204 std::ifstream::eofbit);
205
George Liu858fbb22021-07-01 12:25:44 +0800206 log<level::INFO>(fmt::format("Opening File for RW, FILENAME({})",
207 file.filename().c_str())
208 .c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -0600209
210 std::filebuf* pbuf = infile.rdbuf();
211
212 // get file size using buffer's members
213 std::size_t size = pbuf->pubseekoff(0, infile.end, infile.in);
214 pbuf->pubseekpos(0, infile.in);
215
216 // allocate memory to contain file data
217 std::unique_ptr<char[]> buffer(new char[size]);
218 // get file data
219 pbuf->sgetn(buffer.get(), size);
220 infile.close();
221
222 writeOnUnixSocket(socketFD(), buffer.get(), size);
223 }
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500224 }
Ravi Tejaff9c4522020-11-13 05:14:51 -0600225 catch (std::ifstream::failure& oe)
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500226 {
Ravi Tejaff9c4522020-11-13 05:14:51 -0600227 std::remove(writePath.c_str());
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500228 auto err = errno;
George Liu858fbb22021-07-01 12:25:44 +0800229 log<level::ERR>(
230 fmt::format(
231 "Failed to open, errormsg({}), OPENINTERFACE({}), DUMP_ID({})",
232 oe.what(), dumpPath.c_str(), dumpId)
233 .c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -0600234 elog<Open>(ErrnoOpen(err), PathOpen(dumpPath.c_str()));
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500235 }
236 catch (const std::exception& e)
237 {
Ravi Tejaff9c4522020-11-13 05:14:51 -0600238 std::remove(writePath.c_str());
239 auto err = errno;
George Liu858fbb22021-07-01 12:25:44 +0800240 log<level::ERR>(fmt::format("Failed to offload dump, errormsg({}), "
241 "DUMPFILE({}), DUMP_ID({})",
242 e.what(), writePath.c_str(), dumpId)
243 .c_str());
Ravi Tejaff9c4522020-11-13 05:14:51 -0600244 elog<Write>(ErrnoWrite(err), PathWrite(writePath.c_str()));
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500245 }
Ravi Tejaff9c4522020-11-13 05:14:51 -0600246 std::remove(writePath.c_str());
247 return;
Dhruvaraj Subhashchandran580d91d2020-04-22 12:29:18 -0500248}
249
250} // namespace offload
251} // namespace dump
252} // namespace phosphor