blob: e0eadfcb61bbb7255069826a628c8c85b0f4ab5d [file] [log] [blame]
Dawid Fryckia642a942018-06-12 10:44:23 -07001/* Copyright 2018 Intel
2 *
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16#include "ipmbbridged.hpp"
17
18#include "ipmbdefines.hpp"
19#include "ipmbutils.hpp"
20
Qiang XU8edcf1a2019-06-14 22:18:15 +080021#include <boost/algorithm/string/replace.hpp>
22#include <filesystem>
Amithash Prasad314862d2019-03-26 11:14:03 -070023#include <fstream>
24#include <nlohmann/json.hpp>
Dawid Fryckia642a942018-06-12 10:44:23 -070025#include <phosphor-logging/log.hpp>
26#include <tuple>
Amithash Prasad314862d2019-03-26 11:14:03 -070027#include <unordered_map>
Dawid Fryckia642a942018-06-12 10:44:23 -070028
Dawid Fryckia642a942018-06-12 10:44:23 -070029/**
30 * @brief Dbus
31 */
32static constexpr const char *ipmbBus = "xyz.openbmc_project.Ipmi.Channel.Ipmb";
33static constexpr const char *ipmbObj = "/xyz/openbmc_project/Ipmi/Channel/Ipmb";
Dawid Fryckia642a942018-06-12 10:44:23 -070034static constexpr const char *ipmbDbusIntf = "org.openbmc.Ipmb";
35
36boost::asio::io_service io;
37auto conn = std::make_shared<sdbusplus::asio::connection>(io);
38
Dawid Fryckia642a942018-06-12 10:44:23 -070039static std::list<IpmbChannel> ipmbChannels;
Amithash Prasad314862d2019-03-26 11:14:03 -070040static const std::unordered_map<std::string, ipmbChannelType>
41 ipmbChannelTypeMap = {{"me", ipmbChannelType::me},
42 {"ipmb", ipmbChannelType::ipmb}};
Dawid Fryckia642a942018-06-12 10:44:23 -070043
44/**
45 * @brief Ipmb request class methods
46 */
47IpmbRequest::IpmbRequest()
48{
49 data.reserve(ipmbMaxDataSize);
50}
51
52IpmbRequest::IpmbRequest(uint8_t address, uint8_t netFn, uint8_t rsLun,
53 uint8_t rqSA, uint8_t seq, uint8_t rqLun, uint8_t cmd,
Dawid Frycki8188d762019-04-01 18:03:48 -070054 const std::vector<uint8_t> &inputData) :
Dawid Fryckia642a942018-06-12 10:44:23 -070055 address(address),
56 netFn(netFn), rsLun(rsLun), rqSA(rqSA), seq(seq), rqLun(rqLun), cmd(cmd),
57 timer(io)
58{
59 data.reserve(ipmbMaxDataSize);
60 state = ipmbRequestState::invalid;
61
62 if (inputData.size() > 0)
63 {
64 data = std::move(inputData);
65 }
66}
67
Dawid Fryckia642a942018-06-12 10:44:23 -070068void IpmbRequest::i2cToIpmbConstruct(IPMB_HEADER *ipmbBuffer,
69 size_t bufferLength)
70{
71 // constructing ipmb request from i2c buffer
72 netFn = ipmbNetFnGet(ipmbBuffer->Header.Req.rsNetFnLUN);
73 rsLun = ipmbLunFromNetFnLunGet(ipmbBuffer->Header.Req.rsNetFnLUN);
74 rqSA = ipmbBuffer->Header.Req.rqSA;
75 seq = ipmbSeqGet(ipmbBuffer->Header.Req.rqSeqLUN);
76 rqLun = ipmbLunFromSeqLunGet(ipmbBuffer->Header.Req.rqSeqLUN);
77 cmd = ipmbBuffer->Header.Req.cmd;
78
79 size_t dataLength =
80 bufferLength - (ipmbConnectionHeaderLength +
81 ipmbRequestDataHeaderLength + ipmbChecksumSize);
82
83 if (dataLength > 0)
84 {
85 data.insert(data.end(), ipmbBuffer->Header.Req.data,
86 &ipmbBuffer->Header.Req.data[dataLength]);
87 }
88}
89
90int IpmbRequest::ipmbToi2cConstruct(std::vector<uint8_t> &buffer)
91{
Vijay Khemka37a7eac2019-12-06 13:52:28 -080092 /* Add one byte for length byte as per required by driver */
93 size_t bufferLength = 1 + data.size() + ipmbRequestDataHeaderLength +
Dawid Fryckia642a942018-06-12 10:44:23 -070094 ipmbConnectionHeaderLength + ipmbChecksumSize;
95
96 if (bufferLength > ipmbMaxFrameLength)
97 {
98 return -1;
99 }
100
101 buffer.resize(bufferLength);
102 static_assert(ipmbMaxFrameLength >= sizeof(IPMB_HEADER));
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800103 IPMB_PKT *ipmbPkt = reinterpret_cast<IPMB_PKT *>(buffer.data());
104 ipmbPkt->len = bufferLength - 1;
105 IPMB_HEADER *ipmbBuffer = &(ipmbPkt->hdr);
Dawid Fryckia642a942018-06-12 10:44:23 -0700106
107 // constructing buffer from ipmb request
108 ipmbBuffer->Header.Req.address = address;
109 ipmbBuffer->Header.Req.rsNetFnLUN = ipmbNetFnLunSet(netFn, rsLun);
110 ipmbBuffer->Header.Req.rqSA = rqSA;
111 ipmbBuffer->Header.Req.rqSeqLUN = ipmbSeqLunSet(seq, rqLun);
112 ipmbBuffer->Header.Req.cmd = cmd;
113
Qiang XUbbfd00a2019-06-27 21:10:06 +0800114 ipmbBuffer->Header.Req.checksum1 = ipmbChecksumCompute(
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800115 (uint8_t *)ipmbBuffer, ipmbConnectionHeaderLength - ipmbChecksumSize);
Dawid Fryckia642a942018-06-12 10:44:23 -0700116
117 if (data.size() > 0)
118 {
119 std::copy(data.begin(), data.end(), ipmbBuffer->Header.Req.data);
120 }
121
122 buffer[bufferLength - ipmbChecksumSize] =
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800123 ipmbChecksumCompute((uint8_t *)ipmbBuffer + ipmbChecksum2StartOffset,
Dawid Fryckia642a942018-06-12 10:44:23 -0700124 (ipmbRequestDataHeaderLength + data.size()));
125
126 return 0;
127}
128
129std::tuple<int, uint8_t, uint8_t, uint8_t, uint8_t, std::vector<uint8_t>>
130 IpmbRequest::returnMatchedResponse()
131{
132 return std::make_tuple(
133 static_cast<int>(ipmbResponseStatus::success), matchedResponse->netFn,
134 matchedResponse->rsLun, matchedResponse->cmd,
135 matchedResponse->completionCode, matchedResponse->data);
136}
137
138static std::tuple<int, uint8_t, uint8_t, uint8_t, uint8_t, std::vector<uint8_t>>
139 returnStatus(ipmbResponseStatus status)
140{
141 // we only want to send status here, other fields are not relevant
142 return std::make_tuple(static_cast<int>(status), 0, 0, 0, 0,
143 std::vector<uint8_t>(0));
144}
145
Dawid Fryckia642a942018-06-12 10:44:23 -0700146/**
147 * @brief Ipmb response class methods
148 */
149IpmbResponse::IpmbResponse()
150{
151 data.reserve(ipmbMaxDataSize);
152}
153
154IpmbResponse::IpmbResponse(uint8_t address, uint8_t netFn, uint8_t rqLun,
155 uint8_t rsSA, uint8_t seq, uint8_t rsLun,
156 uint8_t cmd, uint8_t completionCode,
Dawid Frycki8188d762019-04-01 18:03:48 -0700157 const std::vector<uint8_t> &inputData) :
Dawid Fryckia642a942018-06-12 10:44:23 -0700158 address(address),
159 netFn(netFn), rqLun(rqLun), rsSA(rsSA), seq(seq), rsLun(rsLun), cmd(cmd),
160 completionCode(completionCode)
161{
162 data.reserve(ipmbMaxDataSize);
163
164 if (inputData.size() > 0)
165 {
166 data = std::move(inputData);
167 }
168}
169
170void IpmbResponse::i2cToIpmbConstruct(IPMB_HEADER *ipmbBuffer,
171 size_t bufferLength)
172{
173 netFn = ipmbNetFnGet(ipmbBuffer->Header.Resp.rqNetFnLUN);
174 rqLun = ipmbLunFromNetFnLunGet(ipmbBuffer->Header.Resp.rqNetFnLUN);
175 rsSA = ipmbBuffer->Header.Resp.rsSA;
176 seq = ipmbSeqGet(ipmbBuffer->Header.Resp.rsSeqLUN);
177 rsLun = ipmbLunFromSeqLunGet(ipmbBuffer->Header.Resp.rsSeqLUN);
178 cmd = ipmbBuffer->Header.Resp.cmd;
179 completionCode = ipmbBuffer->Header.Resp.completionCode;
180
181 size_t dataLength =
182 bufferLength - (ipmbConnectionHeaderLength +
183 ipmbResponseDataHeaderLength + ipmbChecksumSize);
184
185 if (dataLength > 0)
186 {
187 data.insert(data.end(), ipmbBuffer->Header.Resp.data,
188 &ipmbBuffer->Header.Resp.data[dataLength]);
189 }
190}
191
Dawid Frycki8188d762019-04-01 18:03:48 -0700192std::shared_ptr<std::vector<uint8_t>> IpmbResponse::ipmbToi2cConstruct()
Dawid Fryckia642a942018-06-12 10:44:23 -0700193{
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800194 /* Add one byte for length byte as per required by driver */
195 size_t bufferLength = 1 + data.size() + ipmbResponseDataHeaderLength +
Dawid Fryckia642a942018-06-12 10:44:23 -0700196 ipmbConnectionHeaderLength + ipmbChecksumSize;
197
198 if (bufferLength > ipmbMaxFrameLength)
199 {
Dawid Frycki8188d762019-04-01 18:03:48 -0700200 return nullptr;
Dawid Fryckia642a942018-06-12 10:44:23 -0700201 }
202
Dawid Frycki8188d762019-04-01 18:03:48 -0700203 std::shared_ptr<std::vector<uint8_t>> buffer =
204 std::make_shared<std::vector<uint8_t>>(bufferLength);
205
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800206 IPMB_PKT *ipmbPkt = reinterpret_cast<IPMB_PKT *>(buffer->data());
207 ipmbPkt->len = bufferLength - 1;
208 IPMB_HEADER *ipmbBuffer = &(ipmbPkt->hdr);
Dawid Fryckia642a942018-06-12 10:44:23 -0700209
210 ipmbBuffer->Header.Resp.address = address;
211 ipmbBuffer->Header.Resp.rqNetFnLUN = ipmbNetFnLunSet(netFn, rqLun);
212 ipmbBuffer->Header.Resp.rsSA = rsSA;
213 ipmbBuffer->Header.Resp.rsSeqLUN = ipmbSeqLunSet(seq, rsLun);
214 ipmbBuffer->Header.Resp.cmd = cmd;
215 ipmbBuffer->Header.Resp.completionCode = completionCode;
216
217 ipmbBuffer->Header.Resp.checksum1 = ipmbChecksumCompute(
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800218 (uint8_t *)ipmbBuffer, ipmbConnectionHeaderLength - ipmbChecksumSize);
Dawid Fryckia642a942018-06-12 10:44:23 -0700219
220 if (data.size() > 0)
221 {
222 std::copy(data.begin(), data.end(), ipmbBuffer->Header.Resp.data);
223 }
224
Dawid Frycki8188d762019-04-01 18:03:48 -0700225 (*buffer)[bufferLength - ipmbChecksumSize] =
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800226 ipmbChecksumCompute((uint8_t *)ipmbBuffer + ipmbChecksum2StartOffset,
Dawid Fryckia642a942018-06-12 10:44:23 -0700227 (ipmbResponseDataHeaderLength + data.size()));
228
Dawid Frycki8188d762019-04-01 18:03:48 -0700229 return buffer;
Dawid Fryckia642a942018-06-12 10:44:23 -0700230}
231
232bool IpmbCommandFilter::isBlocked(const uint8_t reqNetFn, const uint8_t cmd)
233{
234 auto blockedCmd = unhandledCommands.find({reqNetFn, cmd});
235
236 if (blockedCmd != unhandledCommands.end())
237 {
238 return true;
239 }
240
241 return false;
242}
243
244void IpmbCommandFilter::addFilter(const uint8_t reqNetFn, const uint8_t cmd)
245{
246 if (unhandledCommands.insert({reqNetFn, cmd}).second)
247 {
248 phosphor::logging::log<phosphor::logging::level::INFO>(
249 "addFilter: added command to filter",
250 phosphor::logging::entry("netFn = %d", reqNetFn),
251 phosphor::logging::entry("cmd = %d", cmd));
252 }
253}
254
255/**
256 * @brief Ipmb channel
257 */
Qiang XUbbfd00a2019-06-27 21:10:06 +0800258void IpmbChannel::ipmbSendI2cFrame(std::shared_ptr<std::vector<uint8_t>> buffer,
Dawid Fryckia642a942018-06-12 10:44:23 -0700259 size_t retriesAttempted = 0)
260{
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800261 IPMB_PKT *ipmbPkt = reinterpret_cast<IPMB_PKT *>(buffer->data());
262 uint8_t targetAddr = ipmbIsResponse(&(ipmbPkt->hdr))
263 ? ipmbPkt->hdr.Header.Resp.address
264 : ipmbPkt->hdr.Header.Req.address;
265 boost::asio::async_write(
266 i2cSlaveDescriptor, boost::asio::buffer(*buffer),
267 [this, buffer, retriesAttempted,
268 targetAddr](const boost::system::error_code &ec, size_t bytesSent) {
269 if (ec)
Dawid Fryckia642a942018-06-12 10:44:23 -0700270 {
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800271 size_t currentRetryCnt = retriesAttempted;
272
273 if (currentRetryCnt > ipmbI2cNumberOfRetries)
274 {
275 std::string msgToLog =
276 "ipmbSendI2cFrame: send to I2C failed after retries."
277 " busId=" +
278 std::to_string(ipmbBusId) +
279 ", targetAddr=" + std::to_string(targetAddr) +
280 ", error=" + ec.message();
281 phosphor::logging::log<phosphor::logging::level::ERR>(
282 msgToLog.c_str());
283 return;
284 }
285 currentRetryCnt++;
286 ipmbSendI2cFrame(buffer, currentRetryCnt);
Dawid Fryckia642a942018-06-12 10:44:23 -0700287 }
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800288 });
Dawid Fryckia642a942018-06-12 10:44:23 -0700289}
290
291/**
292 * @brief Ipmb Outstanding Requests
293 */
294void IpmbChannel::makeRequestInvalid(IpmbRequest &request)
295{
296 // change request state to invalid and remove it from outstanding requests
297 // list
298 request.state = ipmbRequestState::invalid;
299 outstandingRequests[request.seq] = nullptr;
300}
301
302void IpmbChannel::makeRequestValid(std::shared_ptr<IpmbRequest> request)
303{
304 // change request state to valid and add it to outstanding requests list
305 request->state = ipmbRequestState::valid;
306 outstandingRequests[request->seq] = request;
307}
308
309bool IpmbChannel::seqNumGet(uint8_t &seq)
310{
311 static uint8_t seqNum = 0;
312
313 for (int i = 0; i < ipmbMaxOutstandingRequestsCount; i++)
314 {
315 seqNum = ++seqNum & ipmbSeqMask;
316 if (seqNum == ipmbMaxOutstandingRequestsCount)
317 {
318 seqNum = 0;
319 }
320
321 if (outstandingRequests[seqNum] == nullptr)
322 {
323 seq = seqNum;
324 return true;
325 }
326 }
327
328 return false;
329}
330
331void IpmbChannel::responseMatch(std::unique_ptr<IpmbResponse> &response)
332{
333 std::shared_ptr<IpmbRequest> request = outstandingRequests[response->seq];
334
335 if (request != nullptr)
336 {
337 if (((ipmbRespNetFn(request->netFn)) == (response->netFn)) &&
338 ((request->rqLun) == (response->rqLun)) &&
339 ((request->rsLun) == (response->rsLun)) &&
340 ((request->cmd) == (response->cmd)))
341 {
342 // match, response is corresponding to previously sent request
343 request->state = ipmbRequestState::matched;
344 request->timer->cancel();
345 request->matchedResponse = std::move(response);
346 }
347 }
348}
349
350void IpmbChannel::processI2cEvent()
351{
352 std::array<uint8_t, ipmbMaxFrameLength> buffer{};
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800353 IPMB_PKT *ipmbPkt = reinterpret_cast<IPMB_PKT *>(buffer.data());
354 IPMB_HEADER *ipmbFrame = &(ipmbPkt->hdr);
Dawid Fryckia642a942018-06-12 10:44:23 -0700355
356 lseek(ipmbi2cSlaveFd, 0, SEEK_SET);
357 int r = read(ipmbi2cSlaveFd, buffer.data(), ipmbMaxFrameLength);
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800358
359 /* Substract first byte len size from total frame length */
360 r--;
361
Dawid Fryckia642a942018-06-12 10:44:23 -0700362 if ((r < ipmbMinFrameLength) || (r > ipmbMaxFrameLength))
363 {
364 goto end;
365 }
366
367 // valiate the frame
368 if (!isFrameValid(ipmbFrame, r))
369 {
370 goto end;
371 }
372
Qiang XUbbfd00a2019-06-27 21:10:06 +0800373 // if it is broadcast message from ipmb channel, send out dbus signal
374 if (ipmbFrame->Header.Req.address == broadcastAddress &&
375 getChannelType() == ipmbChannelType::ipmb)
376 {
377 auto ipmbMessageReceived = IpmbRequest();
378 ipmbMessageReceived.i2cToIpmbConstruct(ipmbFrame, r);
379 sdbusplus::message::message msg =
380 conn->new_signal(ipmbObj, ipmbDbusIntf, "receiveBroadcast");
381 msg.append(ipmbMessageReceived.netFn, ipmbMessageReceived.cmd,
382 ipmbMessageReceived.data);
383 msg.signal_send();
384 }
385
Dawid Fryckia642a942018-06-12 10:44:23 -0700386 // copy frame to ipmib message buffer
Qiang XUbbfd00a2019-06-27 21:10:06 +0800387 else if (ipmbIsResponse(ipmbFrame))
Dawid Fryckia642a942018-06-12 10:44:23 -0700388 {
389 std::unique_ptr<IpmbResponse> ipmbMessageReceived =
390 std::make_unique<IpmbResponse>();
391
392 ipmbMessageReceived->i2cToIpmbConstruct(ipmbFrame, r);
393
394 // try to match response with outstanding request
395 responseMatch(ipmbMessageReceived);
396 }
397 else
398 {
399 // if command is blocked - respond with 'invalid command'
400 // completion code
401 if (commandFilter)
402 {
403 uint8_t netFn = ipmbNetFnGet(ipmbFrame->Header.Req.rsNetFnLUN);
404 uint8_t cmd = ipmbFrame->Header.Req.cmd;
Qiang XUbbfd00a2019-06-27 21:10:06 +0800405 uint8_t rqSA = ipmbFrame->Header.Req.rqSA;
Dawid Fryckia642a942018-06-12 10:44:23 -0700406
407 if (commandFilter->isBlocked(netFn, cmd))
408 {
409 uint8_t seq = ipmbSeqGet(ipmbFrame->Header.Req.rqSeqLUN);
410 uint8_t lun =
411 ipmbLunFromSeqLunGet(ipmbFrame->Header.Req.rqSeqLUN);
Dawid Fryckia642a942018-06-12 10:44:23 -0700412
413 // prepare generic response
Qiang XUbbfd00a2019-06-27 21:10:06 +0800414 auto ipmbResponse = IpmbResponse(
415 rqSA, ipmbRespNetFn(netFn), lun, ipmbBmcSlaveAddress, seq,
416 ipmbRsLun, cmd, ipmbIpmiInvalidCmd, {});
Dawid Fryckia642a942018-06-12 10:44:23 -0700417
Dawid Frycki8188d762019-04-01 18:03:48 -0700418 auto buffer = ipmbResponse.ipmbToi2cConstruct();
419 if (buffer)
Dawid Fryckia642a942018-06-12 10:44:23 -0700420 {
Qiang XUbbfd00a2019-06-27 21:10:06 +0800421 ipmbSendI2cFrame(buffer);
Dawid Fryckia642a942018-06-12 10:44:23 -0700422 }
423
424 goto end;
425 }
426 }
427
428 auto ipmbMessageReceived = IpmbRequest();
Dawid Fryckia642a942018-06-12 10:44:23 -0700429 ipmbMessageReceived.i2cToIpmbConstruct(ipmbFrame, r);
430
Dawid Frycki8188d762019-04-01 18:03:48 -0700431 std::map<std::string, std::variant<int>> options{
Qiang XUbbfd00a2019-06-27 21:10:06 +0800432 {"rqSA", ipmbAddressTo7BitSet(ipmbMessageReceived.rqSA)}};
Dawid Frycki8188d762019-04-01 18:03:48 -0700433 using IpmiDbusRspType = std::tuple<uint8_t, uint8_t, uint8_t, uint8_t,
434 std::vector<uint8_t>>;
435 conn->async_method_call(
436 [this, rqLun{ipmbMessageReceived.rqLun},
Qiang XUbbfd00a2019-06-27 21:10:06 +0800437 seq{ipmbMessageReceived.seq}, address{ipmbMessageReceived.rqSA}](
438 const boost::system::error_code &ec,
439 const IpmiDbusRspType &response) {
Dawid Frycki8188d762019-04-01 18:03:48 -0700440 const auto &[netfn, lun, cmd, cc, payload] = response;
441 if (ec)
442 {
443 phosphor::logging::log<phosphor::logging::level::ERR>(
444 "processI2cEvent: error getting response from IPMI");
445 return;
446 }
Dawid Fryckia642a942018-06-12 10:44:23 -0700447
Dawid Frycki8188d762019-04-01 18:03:48 -0700448 uint8_t bmcSlaveAddress = getBmcSlaveAddress();
449
450 if (payload.size() > ipmbMaxDataSize)
451 {
452 phosphor::logging::log<phosphor::logging::level::ERR>(
453 "processI2cEvent: response exceeding maximum size");
454
455 // prepare generic response
456 auto ipmbResponse = IpmbResponse(
Qiang XUbbfd00a2019-06-27 21:10:06 +0800457 address, netfn, rqLun, bmcSlaveAddress, seq, ipmbRsLun,
458 cmd, ipmbIpmiCmdRespNotProvided, {});
Dawid Frycki8188d762019-04-01 18:03:48 -0700459
460 auto buffer = ipmbResponse.ipmbToi2cConstruct();
461 if (buffer)
462 {
Qiang XUbbfd00a2019-06-27 21:10:06 +0800463 ipmbSendI2cFrame(buffer);
Dawid Frycki8188d762019-04-01 18:03:48 -0700464 }
465
466 return;
467 }
468
469 if (!(netfn & ipmbNetFnResponseMask))
470 {
471 // we are not expecting request here
472 phosphor::logging::log<phosphor::logging::level::ERR>(
473 "processI2cEvent: got a request instead of response");
474 return;
475 }
476
477 // if command is not supported, add it to filter
478 if (cc == ipmbIpmiInvalidCmd)
479 {
480 addFilter(ipmbReqNetFnFromRespNetFn(netfn), cmd);
481 }
482
483 // payload is empty after constructor invocation
484 auto ipmbResponse =
Qiang XUbbfd00a2019-06-27 21:10:06 +0800485 IpmbResponse(address, netfn, rqLun, bmcSlaveAddress, seq,
486 lun, cmd, cc, payload);
Dawid Frycki8188d762019-04-01 18:03:48 -0700487
488 auto buffer = ipmbResponse.ipmbToi2cConstruct();
489 if (!buffer)
490 {
491 phosphor::logging::log<phosphor::logging::level::ERR>(
492 "processI2cEvent: error constructing a request");
493 return;
494 }
495
Qiang XUbbfd00a2019-06-27 21:10:06 +0800496 ipmbSendI2cFrame(buffer);
Dawid Frycki8188d762019-04-01 18:03:48 -0700497 },
498 "xyz.openbmc_project.Ipmi.Host", "/xyz/openbmc_project/Ipmi",
499 "xyz.openbmc_project.Ipmi.Server", "execute",
500 ipmbMessageReceived.netFn, ipmbMessageReceived.rsLun,
501 ipmbMessageReceived.cmd, ipmbMessageReceived.data, options);
Dawid Fryckia642a942018-06-12 10:44:23 -0700502 }
503
504end:
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800505 i2cSlaveDescriptor.async_wait(
506 boost::asio::posix::descriptor_base::wait_read,
Dawid Fryckia642a942018-06-12 10:44:23 -0700507 [this](const boost::system::error_code &ec) {
508 if (ec)
509 {
510 phosphor::logging::log<phosphor::logging::level::ERR>(
511 "Error: processI2cEvent()");
512 return;
513 }
514
515 processI2cEvent();
516 });
517}
518
519IpmbChannel::IpmbChannel(boost::asio::io_service &io,
520 uint8_t ipmbBmcSlaveAddress,
521 uint8_t ipmbRqSlaveAddress, ipmbChannelType type,
522 std::shared_ptr<IpmbCommandFilter> commandFilter) :
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800523 i2cSlaveDescriptor(io),
524 ipmbBmcSlaveAddress(ipmbBmcSlaveAddress),
Dawid Fryckia642a942018-06-12 10:44:23 -0700525 ipmbRqSlaveAddress(ipmbRqSlaveAddress), type(type),
526 commandFilter(commandFilter)
527{
528}
529
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800530int IpmbChannel::ipmbChannelInit(const char *ipmbI2cSlave)
Dawid Fryckia642a942018-06-12 10:44:23 -0700531{
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800532 // extract bus id from slave path and save
533 std::string ipmbI2cSlaveStr(ipmbI2cSlave);
534 auto findHyphen = ipmbI2cSlaveStr.find("-");
535 std::string busStr = ipmbI2cSlaveStr.substr(findHyphen + 1);
Qiang XU8edcf1a2019-06-14 22:18:15 +0800536 try
537 {
538 ipmbBusId = std::stoi(busStr);
539 }
540 catch (std::invalid_argument)
541 {
542 phosphor::logging::log<phosphor::logging::level::ERR>(
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800543 "ipmbChannelInit: invalid bus id in slave-path config");
Qiang XU8edcf1a2019-06-14 22:18:15 +0800544 return -1;
545 }
546
547 // Check if sysfs has device. If not, enable I2C slave driver by command
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800548 // echo "ipmb-dev 0x1010" > /sys/bus/i2c/devices/i2c-0/new_device
Qiang XU8edcf1a2019-06-14 22:18:15 +0800549 bool hasSysfs = std::filesystem::exists(ipmbI2cSlave);
550 if (!hasSysfs)
551 {
552 std::string deviceFileName =
553 "/sys/bus/i2c/devices/i2c-" + busStr + "/new_device";
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800554 std::string para = "ipmb-dev 0x1010"; // init with BMC addr 0x20
Qiang XU8edcf1a2019-06-14 22:18:15 +0800555 std::fstream deviceFile;
556 deviceFile.open(deviceFileName, std::ios::out);
557 if (!deviceFile.good())
558 {
559 phosphor::logging::log<phosphor::logging::level::ERR>(
560 "ipmbChannelInit: error opening deviceFile");
561 return -1;
562 }
563 deviceFile << para;
564 deviceFile.close();
565 }
566
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800567 // open fd to i2c slave device for read write
568 ipmbi2cSlaveFd = open(ipmbI2cSlave, O_RDWR | O_NONBLOCK | O_CLOEXEC);
Dawid Fryckia642a942018-06-12 10:44:23 -0700569 if (ipmbi2cSlaveFd < 0)
570 {
571 phosphor::logging::log<phosphor::logging::level::ERR>(
572 "ipmbChannelInit: error opening ipmbI2cSlave");
573 return -1;
574 }
575
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800576 i2cSlaveDescriptor.assign(ipmbi2cSlaveFd);
Dawid Fryckia642a942018-06-12 10:44:23 -0700577
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800578 i2cSlaveDescriptor.async_wait(
579 boost::asio::posix::descriptor_base::wait_read,
Dawid Fryckia642a942018-06-12 10:44:23 -0700580 [this](const boost::system::error_code &ec) {
581 if (ec)
582 {
583 phosphor::logging::log<phosphor::logging::level::ERR>(
584 "Error: processI2cEvent()");
585 return;
586 }
587
588 processI2cEvent();
589 });
590
591 return 0;
592}
593
Qiang XU8edcf1a2019-06-14 22:18:15 +0800594int IpmbChannel::ipmbChannelUpdateSlaveAddress(const uint8_t newBmcSlaveAddr)
595{
596 if (ipmbi2cSlaveFd > 0)
597 {
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800598 i2cSlaveDescriptor.close();
Qiang XU8edcf1a2019-06-14 22:18:15 +0800599 close(ipmbi2cSlaveFd);
600 ipmbi2cSlaveFd = 0;
601 }
602
603 // disable old I2C slave driver by command:
604 // echo "0x1010" > /sys/bus/i2c/devices/i2c-0/delete_device
605 std::string deviceFileName;
606 std::string para;
607 std::fstream deviceFile;
608 deviceFileName = "/sys/bus/i2c/devices/i2c-" + std::to_string(ipmbBusId) +
609 "/delete_device";
610 para = "0x1010"; // align with removed ipmb0 definition in dts file
611 deviceFile.open(deviceFileName, std::ios::out);
612 if (!deviceFile.good())
613 {
614 phosphor::logging::log<phosphor::logging::level::ERR>(
615 "ipmbChannelUpdateSlaveAddress: error opening deviceFile to delete "
616 "sysfs");
617 return -1;
618 }
619 deviceFile << para;
620 deviceFile.close();
621
622 // enable new I2C slave driver by command:
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800623 // echo "ipmb-dev 0x1012" > /sys/bus/i2c/devices/i2c-0/new_device
Qiang XU8edcf1a2019-06-14 22:18:15 +0800624 deviceFileName =
625 "/sys/bus/i2c/devices/i2c-" + std::to_string(ipmbBusId) + "/new_device";
626 std::ostringstream hex;
627 uint16_t addr = 0x1000 + (newBmcSlaveAddr >> 1);
628 hex << std::hex << static_cast<uint16_t>(addr);
629 const std::string &addressHexStr = hex.str();
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800630 para = "ipmb-dev 0x" + addressHexStr;
Qiang XU8edcf1a2019-06-14 22:18:15 +0800631 deviceFile.open(deviceFileName, std::ios::out);
632 if (!deviceFile.good())
633 {
634 phosphor::logging::log<phosphor::logging::level::ERR>(
635 "ipmbChannelUpdateSlaveAddress: error opening deviceFile to create "
636 "sysfs");
637 return -1;
638 }
639 deviceFile << para;
640 deviceFile.close();
641
642 // open fd to i2c slave device
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800643 std::string ipmbI2cSlaveStr = "/dev/ipmb-" + std::to_string(ipmbBusId);
644 ipmbi2cSlaveFd = open(ipmbI2cSlaveStr.c_str(), O_RDWR | O_NONBLOCK);
Qiang XU8edcf1a2019-06-14 22:18:15 +0800645 if (ipmbi2cSlaveFd < 0)
646 {
647 phosphor::logging::log<phosphor::logging::level::ERR>(
648 "ipmbChannelInit: error opening ipmbI2cSlave");
649 return -1;
650 }
651
652 // start to receive i2c data as slave
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800653 i2cSlaveDescriptor.assign(ipmbi2cSlaveFd);
654 i2cSlaveDescriptor.async_wait(
655 boost::asio::posix::descriptor_base::wait_read,
Qiang XU8edcf1a2019-06-14 22:18:15 +0800656 [this](const boost::system::error_code &ec) {
657 if (ec)
658 {
659 phosphor::logging::log<phosphor::logging::level::ERR>(
660 "Error: processI2cEvent()");
661 return;
662 }
663
664 processI2cEvent();
665 });
666
Qiang XUbbfd00a2019-06-27 21:10:06 +0800667 ipmbBmcSlaveAddress = newBmcSlaveAddr;
668
Qiang XU8edcf1a2019-06-14 22:18:15 +0800669 return 0;
670}
671
672uint8_t IpmbChannel::getBusId()
673{
674 return ipmbBusId;
675}
676
Dawid Fryckia642a942018-06-12 10:44:23 -0700677uint8_t IpmbChannel::getBmcSlaveAddress()
678{
679 return ipmbBmcSlaveAddress;
680}
681
682uint8_t IpmbChannel::getRqSlaveAddress()
683{
684 return ipmbRqSlaveAddress;
685}
686
687ipmbChannelType IpmbChannel::getChannelType()
688{
689 return type;
690}
691
692void IpmbChannel::addFilter(const uint8_t respNetFn, const uint8_t cmd)
693{
694 if (commandFilter)
695 {
696 commandFilter->addFilter(respNetFn, cmd);
697 }
698}
699
700std::tuple<int, uint8_t, uint8_t, uint8_t, uint8_t, std::vector<uint8_t>>
701 IpmbChannel::requestAdd(boost::asio::yield_context &yield,
702 std::shared_ptr<IpmbRequest> request)
703{
704 makeRequestValid(request);
705
706 std::vector<uint8_t> buffer(0);
707 if (request->ipmbToi2cConstruct(buffer) != 0)
708 {
709 return returnStatus(ipmbResponseStatus::error);
710 }
711
712 for (int i = 0; i < ipmbNumberOfTries; i++)
713 {
714 boost::system::error_code ec;
Jae Hyun Yoo25e85c72019-03-05 14:28:13 -0800715 int i2cRetryCnt = 0;
Dawid Fryckia642a942018-06-12 10:44:23 -0700716
Jae Hyun Yoo25e85c72019-03-05 14:28:13 -0800717 for (; i2cRetryCnt < ipmbI2cNumberOfRetries; i2cRetryCnt++)
Dawid Fryckia642a942018-06-12 10:44:23 -0700718 {
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800719 boost::asio::async_write(i2cSlaveDescriptor,
720 boost::asio::buffer(buffer), yield[ec]);
Dawid Fryckia642a942018-06-12 10:44:23 -0700721
722 if (ec)
723 {
Jae Hyun Yoo25e85c72019-03-05 14:28:13 -0800724 continue; // retry
Dawid Fryckia642a942018-06-12 10:44:23 -0700725 }
726 break;
727 }
728
Jae Hyun Yoo25e85c72019-03-05 14:28:13 -0800729 if (i2cRetryCnt == ipmbI2cNumberOfRetries)
730 {
Qiang XUbbfd00a2019-06-27 21:10:06 +0800731 std::string msgToLog =
732 "requestAdd: Sent to I2C failed after retries."
733 " busId=" +
734 std::to_string(ipmbBusId) + ", error=" + ec.message();
Jae Hyun Yoo25e85c72019-03-05 14:28:13 -0800735 phosphor::logging::log<phosphor::logging::level::INFO>(
Qiang XUbbfd00a2019-06-27 21:10:06 +0800736 msgToLog.c_str());
Jae Hyun Yoo25e85c72019-03-05 14:28:13 -0800737 }
738
Dawid Fryckia642a942018-06-12 10:44:23 -0700739 request->timer->expires_after(
740 std::chrono::milliseconds(ipmbRequestRetryTimeout));
741 request->timer->async_wait(yield[ec]);
742
743 if (ec && ec != boost::asio::error::operation_aborted)
744 {
745 // unexpected error - invalidate request and return generic error
746 phosphor::logging::log<phosphor::logging::level::ERR>(
747 "requestAdd: async_wait error");
748 makeRequestInvalid(*request);
749 return returnStatus(ipmbResponseStatus::error);
750 }
751
752 if (request->state == ipmbRequestState::matched)
753 {
754 // matched response, send it to client application
755 makeRequestInvalid(*request);
756 return request->returnMatchedResponse();
757 }
758 }
759
760 makeRequestInvalid(*request);
761 return returnStatus(ipmbResponseStatus::timeout);
762}
763
764static IpmbChannel *getChannel(ipmbChannelType channelType)
765{
766 auto channel =
767 std::find_if(ipmbChannels.begin(), ipmbChannels.end(),
768 [channelType](IpmbChannel &channel) {
769 return channel.getChannelType() == channelType;
770 });
771 if (channel != ipmbChannels.end())
772 {
773 return &(*channel);
774 }
775
776 return nullptr;
777}
778
779static int initializeChannels()
780{
781 std::shared_ptr<IpmbCommandFilter> commandFilter =
782 std::make_shared<IpmbCommandFilter>();
783
Amithash Prasad314862d2019-03-26 11:14:03 -0700784 constexpr const char *configFilePath =
785 "/usr/share/ipmbbridge/ipmb-channels.json";
786 std::ifstream configFile(configFilePath);
787 if (!configFile.is_open())
Dawid Fryckia642a942018-06-12 10:44:23 -0700788 {
Amithash Prasad314862d2019-03-26 11:14:03 -0700789 phosphor::logging::log<phosphor::logging::level::ERR>(
790 "initializeChannels: Cannot open config path");
791 return -1;
792 }
793 try
794 {
795 auto data = nlohmann::json::parse(configFile, nullptr);
796 for (const auto &channelConfig : data["channels"])
Dawid Fryckia642a942018-06-12 10:44:23 -0700797 {
Amithash Prasad314862d2019-03-26 11:14:03 -0700798 const std::string &typeConfig = channelConfig["type"];
Amithash Prasad314862d2019-03-26 11:14:03 -0700799 const std::string &slavePath = channelConfig["slave-path"];
800 uint8_t bmcAddr = channelConfig["bmc-addr"];
801 uint8_t reqAddr = channelConfig["remote-addr"];
802 ipmbChannelType type = ipmbChannelTypeMap.at(typeConfig);
803
804 auto channel = ipmbChannels.emplace(ipmbChannels.end(), io, bmcAddr,
805 reqAddr, type, commandFilter);
Vijay Khemka37a7eac2019-12-06 13:52:28 -0800806 if (channel->ipmbChannelInit(slavePath.c_str()) < 0)
Amithash Prasad314862d2019-03-26 11:14:03 -0700807 {
808 phosphor::logging::log<phosphor::logging::level::ERR>(
809 "initializeChannels: channel initialization failed");
810 return -1;
811 }
Dawid Fryckia642a942018-06-12 10:44:23 -0700812 }
813 }
Amithash Prasad314862d2019-03-26 11:14:03 -0700814 catch (nlohmann::json::exception &e)
815 {
816 phosphor::logging::log<phosphor::logging::level::ERR>(
817 "initializeChannels: Error parsing config file");
818 return -1;
819 }
820 catch (std::out_of_range &e)
821 {
822 phosphor::logging::log<phosphor::logging::level::ERR>(
823 "initializeChannels: Error invalid type");
824 return -1;
825 }
Dawid Fryckia642a942018-06-12 10:44:23 -0700826 return 0;
827}
828
Dawid Fryckia642a942018-06-12 10:44:23 -0700829auto ipmbHandleRequest = [](boost::asio::yield_context yield,
830 uint8_t reqChannel, uint8_t netfn, uint8_t lun,
831 uint8_t cmd, std::vector<uint8_t> dataReceived) {
832 IpmbChannel *channel = getChannel(static_cast<ipmbChannelType>(reqChannel));
833 if (channel == nullptr)
834 {
835 phosphor::logging::log<phosphor::logging::level::ERR>(
836 "ipmbHandleRequest: requested channel does not exist");
837 return returnStatus(ipmbResponseStatus::invalid_param);
838 }
839
840 // check outstanding request list for valid sequence number
841 uint8_t seqNum = 0;
842 bool seqValid = channel->seqNumGet(seqNum);
843 if (!seqValid)
844 {
845 phosphor::logging::log<phosphor::logging::level::WARNING>(
846 "ipmbHandleRequest: cannot add more requests to the list");
847 return returnStatus(ipmbResponseStatus::busy);
848 }
849
850 uint8_t bmcSlaveAddress = channel->getBmcSlaveAddress();
851 uint8_t rqSlaveAddress = channel->getRqSlaveAddress();
852
853 // construct the request to add it to outstanding request list
854 std::shared_ptr<IpmbRequest> request = std::make_shared<IpmbRequest>(
855 rqSlaveAddress, netfn, ipmbRsLun, bmcSlaveAddress, seqNum, lun, cmd,
856 dataReceived);
857
858 if (!request->timer)
859 {
860 phosphor::logging::log<phosphor::logging::level::ERR>(
861 "ipmbHandleRequest: timer object does not exist");
862 return returnStatus(ipmbResponseStatus::error);
863 }
864
865 return channel->requestAdd(yield, request);
866};
867
Qiang XU8edcf1a2019-06-14 22:18:15 +0800868void addUpdateSlaveAddrHandler()
869{
870 // callback to handle dbus signal of updating slave addr
871 std::function<void(sdbusplus::message::message &)> updateSlaveAddrHandler =
872 [](sdbusplus::message::message &message) {
873 uint8_t reqChannel, busId, slaveAddr;
874
875 // valid source of signal, check whether from multi-node manager
876 std::string pathName = message.get_path();
877 if (pathName != "/xyz/openbmc_project/MultiNode/Status")
878 {
879 phosphor::logging::log<phosphor::logging::level::ERR>(
880 "addUpdateSlaveAddrHandler: invalid obj path");
881 return;
882 }
883
884 message.read(reqChannel, busId, slaveAddr);
885
886 IpmbChannel *channel =
887 getChannel(static_cast<ipmbChannelType>(reqChannel));
888 if (channel == nullptr ||
889 reqChannel != static_cast<uint8_t>(ipmbChannelType::ipmb))
890 {
891 phosphor::logging::log<phosphor::logging::level::ERR>(
892 "addUpdateSlaveAddrHandler: invalid channel");
893 return;
894 }
895 if (busId != channel->getBusId())
896 {
897 phosphor::logging::log<phosphor::logging::level::ERR>(
898 "addUpdateSlaveAddrHandler: invalid busId");
899 return;
900 }
901 if (channel->getBmcSlaveAddress() == slaveAddr)
902 {
903 phosphor::logging::log<phosphor::logging::level::INFO>(
904 "addUpdateSlaveAddrHandler: channel bmc slave addr is "
905 "unchanged, do nothing");
906 return;
907 }
908
909 channel->ipmbChannelUpdateSlaveAddress(slaveAddr);
910 };
911
912 static auto match = std::make_unique<sdbusplus::bus::match::match>(
913 static_cast<sdbusplus::bus::bus &>(*conn),
914 "type='signal',member='updateBmcSlaveAddr',", updateSlaveAddrHandler);
915}
916
Qiang XUbbfd00a2019-06-27 21:10:06 +0800917void addSendBroadcastHandler()
918{
919 // callback to handle dbus signal of sending broadcast message
920 std::function<void(sdbusplus::message::message &)> sendBroadcastHandler =
921 [](sdbusplus::message::message &message) {
922 uint8_t reqChannel, netFn, lun, cmd;
923 std::vector<uint8_t> dataReceived;
924 message.read(reqChannel, netFn, lun, cmd, dataReceived);
925
926 IpmbChannel *channel =
927 getChannel(static_cast<ipmbChannelType>(reqChannel));
928 if (channel == nullptr)
929 {
930 phosphor::logging::log<phosphor::logging::level::ERR>(
931 "addSendBroadcastMsgHandler: requested channel does not "
932 "exist");
933 return;
934 }
935
936 uint8_t bmcSlaveAddress = channel->getBmcSlaveAddress();
937 uint8_t seqNum = 0; // seqNum is not used in broadcast msg
938 uint8_t targetAddr = broadcastAddress;
939
940 std::shared_ptr<IpmbRequest> request =
941 std::make_shared<IpmbRequest>(targetAddr, netFn, ipmbRsLun,
942 bmcSlaveAddress, seqNum, lun, cmd,
943 dataReceived);
944
945 std::shared_ptr<std::vector<uint8_t>> buffer =
946 std::make_shared<std::vector<uint8_t>>();
947
948 if (request->ipmbToi2cConstruct(*buffer) != 0)
949 {
950 return;
951 }
952
953 channel->ipmbSendI2cFrame(buffer);
954 };
955
956 static auto match = std::make_unique<sdbusplus::bus::match::match>(
957 static_cast<sdbusplus::bus::bus &>(*conn),
958 "type='signal',member='sendBroadcast',", sendBroadcastHandler);
959}
960
Dawid Fryckia642a942018-06-12 10:44:23 -0700961/**
962 * @brief Main
963 */
964int main(int argc, char *argv[])
965{
966 conn->request_name(ipmbBus);
967
968 auto server = sdbusplus::asio::object_server(conn);
969
Dawid Fryckia642a942018-06-12 10:44:23 -0700970 std::shared_ptr<sdbusplus::asio::dbus_interface> ipmbIface =
971 server.add_interface(ipmbObj, ipmbDbusIntf);
972
Dawid Fryckia642a942018-06-12 10:44:23 -0700973 ipmbIface->register_method("sendRequest", std::move(ipmbHandleRequest));
Dawid Fryckia642a942018-06-12 10:44:23 -0700974 ipmbIface->initialize();
975
976 if (initializeChannels() < 0)
977 {
978 phosphor::logging::log<phosphor::logging::level::ERR>(
979 "Error initializeChannels");
980 return -1;
981 }
982
Qiang XU8edcf1a2019-06-14 22:18:15 +0800983 addUpdateSlaveAddrHandler();
984
Qiang XUbbfd00a2019-06-27 21:10:06 +0800985 addSendBroadcastHandler();
986
Dawid Fryckia642a942018-06-12 10:44:23 -0700987 io.run();
988 return 0;
989}