1 #include "event_manager.hpp"
2
3 #include "libpldm/platform.h"
4 #include "libpldm/utils.h"
5
6 #include "terminus_manager.hpp"
7
8 #include <phosphor-logging/lg2.hpp>
9 #include <xyz/openbmc_project/Logging/Entry/server.hpp>
10
11 #include <cerrno>
12 #include <memory>
13
14 PHOSPHOR_LOG2_USING;
15
16 namespace pldm
17 {
18 namespace platform_mc
19 {
20 namespace fs = std::filesystem;
21
handlePlatformEvent(pldm_tid_t tid,uint16_t eventId,uint8_t eventClass,const uint8_t * eventData,size_t eventDataSize)22 int EventManager::handlePlatformEvent(
23 pldm_tid_t tid, uint16_t eventId, uint8_t eventClass,
24 const uint8_t* eventData, size_t eventDataSize)
25 {
26 /* Only handle the event of the discovered termini*/
27 if (!termini.contains(tid))
28 {
29 lg2::error("Terminus ID {TID} is not in the managing list.", "TID",
30 tid);
31 return PLDM_ERROR;
32 }
33
34 /* EventClass sensorEvent `Table 11 - PLDM Event Types` DSP0248 */
35 if (eventClass == PLDM_SENSOR_EVENT)
36 {
37 uint16_t sensorId = 0;
38 uint8_t sensorEventClassType = 0;
39 size_t eventClassDataOffset = 0;
40 auto rc = decode_sensor_event_data(eventData, eventDataSize, &sensorId,
41 &sensorEventClassType,
42 &eventClassDataOffset);
43 if (rc)
44 {
45 lg2::error(
46 "Failed to decode sensor event data from terminus ID {TID}, event class {CLASS}, event ID {EVENTID} with return code {RC}.",
47 "TID", tid, "CLASS", eventClass, "EVENTID", eventId, "RC", rc);
48 return rc;
49 }
50 switch (sensorEventClassType)
51 {
52 case PLDM_NUMERIC_SENSOR_STATE:
53 {
54 const uint8_t* sensorData = eventData + eventClassDataOffset;
55 size_t sensorDataLength = eventDataSize - eventClassDataOffset;
56 return processNumericSensorEvent(tid, sensorId, sensorData,
57 sensorDataLength);
58 }
59 case PLDM_STATE_SENSOR_STATE:
60 case PLDM_SENSOR_OP_STATE:
61 default:
62 lg2::info(
63 "Unsupported class type {CLASSTYPE} for the sensor event from terminus ID {TID} sensorId {SID}",
64 "CLASSTYPE", sensorEventClassType, "TID", tid, "SID",
65 sensorId);
66 return PLDM_ERROR;
67 }
68 }
69
70 /* EventClass CPEREvent as `Table 11 - PLDM Event Types` DSP0248 V1.3.0 */
71 if (eventClass == PLDM_CPER_EVENT)
72 {
73 return processCperEvent(tid, eventId, eventData, eventDataSize);
74 }
75
76 /* EventClass pldmMessagePollEvent `Table 11 - PLDM Event Types` DSP0248 */
77 if (eventClass == PLDM_MESSAGE_POLL_EVENT)
78 {
79 lg2::info("Received pldmMessagePollEvent for terminus {TID}", "TID",
80 tid);
81 pldm_message_poll_event poll_event{};
82 auto rc = decode_pldm_message_poll_event_data(eventData, eventDataSize,
83 &poll_event);
84 if (rc)
85 {
86 lg2::error(
87 "Failed to decode PldmMessagePollEvent event, error {RC} ",
88 "RC", rc);
89 return rc;
90 }
91
92 auto it = termini.find(tid);
93 if (it != termini.end())
94 {
95 auto& terminus = it->second; // Reference for clarity
96 terminus->pollEvent = true;
97 terminus->pollEventId = poll_event.event_id;
98 terminus->pollDataTransferHandle = poll_event.data_transfer_handle;
99 }
100
101 return PLDM_SUCCESS;
102 }
103
104 lg2::info("Unsupported class type {CLASSTYPE}", "CLASSTYPE", eventClass);
105
106 return PLDM_ERROR;
107 }
108
processNumericSensorEvent(pldm_tid_t tid,uint16_t sensorId,const uint8_t * sensorData,size_t sensorDataLength)109 int EventManager::processNumericSensorEvent(pldm_tid_t tid, uint16_t sensorId,
110 const uint8_t* sensorData,
111 size_t sensorDataLength)
112 {
113 uint8_t eventState = 0;
114 uint8_t previousEventState = 0;
115 uint8_t sensorDataSize = 0;
116 uint32_t presentReading;
117 auto rc = decode_numeric_sensor_data(
118 sensorData, sensorDataLength, &eventState, &previousEventState,
119 &sensorDataSize, &presentReading);
120 if (rc)
121 {
122 lg2::error(
123 "Failed to decode numericSensorState event for terminus ID {TID}, error {RC} ",
124 "TID", tid, "RC", rc);
125 return rc;
126 }
127
128 double value = static_cast<double>(presentReading);
129 lg2::error(
130 "processNumericSensorEvent tid {TID}, sensorID {SID} value {VAL} previousState {PSTATE} eventState {ESTATE}",
131 "TID", tid, "SID", sensorId, "VAL", value, "PSTATE", previousEventState,
132 "ESTATE", eventState);
133
134 if (!termini.contains(tid) || !termini[tid])
135 {
136 lg2::error("Terminus ID {TID} is not in the managing list.", "TID",
137 tid);
138 return PLDM_ERROR;
139 }
140
141 auto& terminus = termini[tid];
142
143 auto sensor = terminus->getSensorObject(sensorId);
144 if (!sensor)
145 {
146 lg2::error(
147 "Terminus ID {TID} has no sensor object with sensor ID {SID}.",
148 "TID", tid, "SID", sensorId);
149 return PLDM_ERROR;
150 }
151
152 switch (previousEventState)
153 {
154 case PLDM_SENSOR_UNKNOWN:
155 case PLDM_SENSOR_NORMAL:
156 {
157 switch (eventState)
158 {
159 case PLDM_SENSOR_UPPERFATAL:
160 case PLDM_SENSOR_UPPERCRITICAL:
161 {
162 sensor->triggerThresholdEvent(pldm::utils::Level::WARNING,
163 pldm::utils::Direction::HIGH,
164 value, true, true);
165 return sensor->triggerThresholdEvent(
166 pldm::utils::Level::CRITICAL,
167 pldm::utils::Direction::HIGH, value, true, true);
168 }
169 case PLDM_SENSOR_UPPERWARNING:
170 {
171 return sensor->triggerThresholdEvent(
172 pldm::utils::Level::WARNING,
173 pldm::utils::Direction::HIGH, value, true, true);
174 }
175 case PLDM_SENSOR_NORMAL:
176 break;
177 case PLDM_SENSOR_LOWERWARNING:
178 {
179 return sensor->triggerThresholdEvent(
180 pldm::utils::Level::WARNING,
181 pldm::utils::Direction::LOW, value, true, true);
182 }
183 case PLDM_SENSOR_LOWERCRITICAL:
184 case PLDM_SENSOR_LOWERFATAL:
185 {
186 sensor->triggerThresholdEvent(pldm::utils::Level::WARNING,
187 pldm::utils::Direction::LOW,
188 value, true, true);
189 return sensor->triggerThresholdEvent(
190 pldm::utils::Level::CRITICAL,
191 pldm::utils::Direction::LOW, value, true, true);
192 }
193 default:
194 break;
195 }
196 break;
197 }
198 case PLDM_SENSOR_LOWERWARNING:
199 {
200 switch (eventState)
201 {
202 case PLDM_SENSOR_UPPERFATAL:
203 case PLDM_SENSOR_UPPERCRITICAL:
204 break;
205 case PLDM_SENSOR_UPPERWARNING:
206 {
207 sensor->triggerThresholdEvent(pldm::utils::Level::WARNING,
208 pldm::utils::Direction::LOW,
209 value, false, false);
210 return sensor->triggerThresholdEvent(
211 pldm::utils::Level::WARNING,
212 pldm::utils::Direction::HIGH, value, true, true);
213 }
214 case PLDM_SENSOR_NORMAL:
215 {
216 return sensor->triggerThresholdEvent(
217 pldm::utils::Level::WARNING,
218 pldm::utils::Direction::LOW, value, false, false);
219 }
220 case PLDM_SENSOR_LOWERWARNING:
221 break;
222 case PLDM_SENSOR_LOWERCRITICAL:
223 case PLDM_SENSOR_LOWERFATAL:
224 {
225 return sensor->triggerThresholdEvent(
226 pldm::utils::Level::CRITICAL,
227 pldm::utils::Direction::LOW, value, true, true);
228 }
229 default:
230 break;
231 }
232 break;
233 }
234 case PLDM_SENSOR_LOWERCRITICAL:
235 case PLDM_SENSOR_LOWERFATAL:
236 {
237 switch (eventState)
238 {
239 case PLDM_SENSOR_UPPERFATAL:
240 case PLDM_SENSOR_UPPERCRITICAL:
241 case PLDM_SENSOR_UPPERWARNING:
242 break;
243 case PLDM_SENSOR_NORMAL:
244 {
245 sensor->triggerThresholdEvent(pldm::utils::Level::CRITICAL,
246 pldm::utils::Direction::LOW,
247 value, false, false);
248 sensor->triggerThresholdEvent(pldm::utils::Level::WARNING,
249 pldm::utils::Direction::LOW,
250 value, true, true);
251 return sensor->triggerThresholdEvent(
252 pldm::utils::Level::WARNING,
253 pldm::utils::Direction::LOW, value, false, false);
254 }
255 case PLDM_SENSOR_LOWERWARNING:
256 {
257 sensor->triggerThresholdEvent(pldm::utils::Level::CRITICAL,
258 pldm::utils::Direction::LOW,
259 value, false, false);
260 return sensor->triggerThresholdEvent(
261 pldm::utils::Level::WARNING,
262 pldm::utils::Direction::LOW, value, true, true);
263 }
264 case PLDM_SENSOR_LOWERCRITICAL:
265 case PLDM_SENSOR_LOWERFATAL:
266 default:
267 break;
268 }
269 break;
270 }
271 case PLDM_SENSOR_UPPERFATAL:
272 case PLDM_SENSOR_UPPERCRITICAL:
273 {
274 switch (eventState)
275 {
276 case PLDM_SENSOR_UPPERFATAL:
277 case PLDM_SENSOR_UPPERCRITICAL:
278 break;
279 case PLDM_SENSOR_UPPERWARNING:
280 {
281 sensor->triggerThresholdEvent(pldm::utils::Level::CRITICAL,
282 pldm::utils::Direction::HIGH,
283 value, false, false);
284 return sensor->triggerThresholdEvent(
285 pldm::utils::Level::WARNING,
286 pldm::utils::Direction::HIGH, value, true, true);
287 }
288 case PLDM_SENSOR_NORMAL:
289 {
290 sensor->triggerThresholdEvent(pldm::utils::Level::CRITICAL,
291 pldm::utils::Direction::HIGH,
292 value, false, false);
293 sensor->triggerThresholdEvent(pldm::utils::Level::WARNING,
294 pldm::utils::Direction::HIGH,
295 value, true, true);
296 return sensor->triggerThresholdEvent(
297 pldm::utils::Level::WARNING,
298 pldm::utils::Direction::HIGH, value, false, false);
299 }
300 case PLDM_SENSOR_LOWERWARNING:
301 case PLDM_SENSOR_LOWERCRITICAL:
302 case PLDM_SENSOR_LOWERFATAL:
303 default:
304 break;
305 }
306 break;
307 }
308 case PLDM_SENSOR_UPPERWARNING:
309 {
310 switch (eventState)
311 {
312 case PLDM_SENSOR_UPPERFATAL:
313 case PLDM_SENSOR_UPPERCRITICAL:
314 {
315 return sensor->triggerThresholdEvent(
316 pldm::utils::Level::CRITICAL,
317 pldm::utils::Direction::HIGH, value, true, true);
318 }
319 case PLDM_SENSOR_UPPERWARNING:
320 break;
321 case PLDM_SENSOR_NORMAL:
322 {
323 return sensor->triggerThresholdEvent(
324 pldm::utils::Level::WARNING,
325 pldm::utils::Direction::HIGH, value, false, false);
326 }
327 case PLDM_SENSOR_LOWERWARNING:
328 {
329 sensor->triggerThresholdEvent(pldm::utils::Level::WARNING,
330 pldm::utils::Direction::HIGH,
331 value, false, false);
332 return sensor->triggerThresholdEvent(
333 pldm::utils::Level::WARNING,
334 pldm::utils::Direction::LOW, value, true, true);
335 }
336 case PLDM_SENSOR_LOWERCRITICAL:
337 case PLDM_SENSOR_LOWERFATAL:
338 default:
339 break;
340 }
341 break;
342 }
343 default:
344 break;
345 }
346
347 return PLDM_SUCCESS;
348 }
349
processCperEvent(pldm_tid_t tid,uint16_t eventId,const uint8_t * eventData,const size_t eventDataSize)350 int EventManager::processCperEvent(pldm_tid_t tid, uint16_t eventId,
351 const uint8_t* eventData,
352 const size_t eventDataSize)
353 {
354 if (eventDataSize < PLDM_PLATFORM_CPER_EVENT_MIN_LENGTH)
355 {
356 lg2::error(
357 "Error : Invalid CPER Event data length for eventId {EVENTID}.",
358 "EVENTID", eventId);
359 return PLDM_ERROR;
360 }
361 const size_t cperEventDataSize =
362 eventDataSize - PLDM_PLATFORM_CPER_EVENT_MIN_LENGTH;
363 const size_t msgDataLen =
364 sizeof(pldm_platform_cper_event) + cperEventDataSize;
365 std::string terminusName = "";
366 auto msgData = std::make_unique<unsigned char[]>(msgDataLen);
367 auto cperEvent = new (msgData.get()) pldm_platform_cper_event;
368
369 auto rc = decode_pldm_platform_cper_event(eventData, eventDataSize,
370 cperEvent, msgDataLen);
371
372 if (rc)
373 {
374 lg2::error(
375 "Failed to decode CPER event for eventId {EVENTID} of terminus ID {TID} error {RC}.",
376 "EVENTID", eventId, "TID", tid, "RC", rc);
377 return rc;
378 }
379
380 if (termini.contains(tid) && termini[tid])
381 {
382 auto tmp = termini[tid]->getTerminusName();
383 if (tmp && !tmp.value().empty())
384 {
385 terminusName = static_cast<std::string>(tmp.value());
386 }
387 }
388 else
389 {
390 lg2::error("Terminus ID {TID} is not in the managing list.", "TID",
391 tid);
392 return PLDM_ERROR;
393 }
394
395 // Save event data to file
396 std::filesystem::path dirName{"/var/cper"};
397 if (!std::filesystem::exists(dirName))
398 {
399 try
400 {
401 std::filesystem::create_directory(dirName);
402 }
403 catch (const std::filesystem::filesystem_error& e)
404 {
405 lg2::error("Failed to create /var/cper directory: {ERROR}", "ERROR",
406 e);
407 return PLDM_ERROR;
408 }
409 }
410
411 std::string fileName{dirName.string() + "/cper-XXXXXX"};
412 auto fd = mkstemp(fileName.data());
413 if (fd < 0)
414 {
415 lg2::error("Failed to generate temp file, error {ERRORNO}", "ERRORNO",
416 std::strerror(errno));
417 return PLDM_ERROR;
418 }
419 close(fd);
420
421 std::ofstream ofs;
422 ofs.exceptions(std::ofstream::failbit | std::ofstream::badbit |
423 std::ofstream::eofbit);
424
425 try
426 {
427 ofs.open(fileName);
428 ofs.write(reinterpret_cast<const char*>(
429 pldm_platform_cper_event_event_data(cperEvent)),
430 cperEvent->event_data_length);
431 if (cperEvent->format_type == PLDM_PLATFORM_CPER_EVENT_WITH_HEADER)
432 {
433 rc = createCperDumpEntry("CPER", fileName, terminusName);
434 }
435 else
436 {
437 rc = createCperDumpEntry("CPERSection", fileName, terminusName);
438 }
439 ofs.close();
440 }
441 catch (const std::ofstream::failure& e)
442 {
443 lg2::error("Failed to save CPER to '{FILENAME}', error - {ERROR}.",
444 "FILENAME", fileName, "ERROR", e);
445 return PLDM_ERROR;
446 }
447 return rc;
448 }
449
createCperDumpEntry(const std::string & dataType,const std::string & dataPath,const std::string & typeName)450 int EventManager::createCperDumpEntry(const std::string& dataType,
451 const std::string& dataPath,
452 const std::string& typeName)
453 {
454 auto createDump =
455 [](std::map<std::string, std::variant<std::string, uint64_t>>&
456 addData) {
457 static constexpr auto dumpObjPath =
458 "/xyz/openbmc_project/dump/faultlog";
459 static constexpr auto dumpInterface =
460 "xyz.openbmc_project.Dump.Create";
461 auto& bus = pldm::utils::DBusHandler::getBus();
462
463 try
464 {
465 auto service = pldm::utils::DBusHandler().getService(
466 dumpObjPath, dumpInterface);
467 auto method = bus.new_method_call(service.c_str(), dumpObjPath,
468 dumpInterface, "CreateDump");
469 method.append(addData);
470 bus.call_noreply(method);
471 }
472 catch (const std::exception& e)
473 {
474 lg2::error(
475 "Failed to create D-Bus Dump entry, error - {ERROR}.",
476 "ERROR", e);
477 }
478 };
479
480 std::map<std::string, std::variant<std::string, uint64_t>> addData;
481 addData["Type"] = dataType;
482 addData["PrimaryLogId"] = dataPath;
483 addData["AdditionalTypeName"] = typeName;
484 createDump(addData);
485 return PLDM_SUCCESS;
486 }
487
getNextPartParameters(uint16_t eventId,std::vector<uint8_t> eventMessage,uint8_t transferFlag,uint32_t eventDataIntegrityChecksum,uint32_t nextDataTransferHandle,uint8_t * transferOperationFlag,uint32_t * dataTransferHandle,uint32_t * eventIdToAcknowledge)488 int EventManager::getNextPartParameters(
489 uint16_t eventId, std::vector<uint8_t> eventMessage, uint8_t transferFlag,
490 uint32_t eventDataIntegrityChecksum, uint32_t nextDataTransferHandle,
491 uint8_t* transferOperationFlag, uint32_t* dataTransferHandle,
492 uint32_t* eventIdToAcknowledge)
493 {
494 if (transferFlag != PLDM_PLATFORM_TRANSFER_START_AND_END &&
495 transferFlag != PLDM_PLATFORM_TRANSFER_END)
496 {
497 *transferOperationFlag = PLDM_GET_NEXTPART;
498 *dataTransferHandle = nextDataTransferHandle;
499 *eventIdToAcknowledge = PLDM_PLATFORM_EVENT_ID_FRAGMENT;
500 return PLDM_SUCCESS;
501 }
502
503 if (transferFlag == PLDM_PLATFORM_TRANSFER_END)
504 {
505 if (eventDataIntegrityChecksum !=
506 crc32(eventMessage.data(), eventMessage.size()))
507 {
508 lg2::error("pollForPlatformEventMessage invalid checksum.");
509 return PLDM_ERROR_INVALID_DATA;
510 }
511 }
512
513 /* End of one event. Set request transfer flag to ACK */
514 *transferOperationFlag = PLDM_ACKNOWLEDGEMENT_ONLY;
515 *dataTransferHandle = 0;
516 *eventIdToAcknowledge = eventId;
517
518 return PLDM_SUCCESS;
519 }
520
callPolledEventHandlers(pldm_tid_t tid,uint8_t eventClass,uint16_t eventId,std::vector<uint8_t> & eventMessage)521 void EventManager::callPolledEventHandlers(pldm_tid_t tid, uint8_t eventClass,
522 uint16_t eventId,
523 std::vector<uint8_t>& eventMessage)
524 {
525 try
526 {
527 const auto& handlers = eventHandlers.at(eventClass);
528 for (const auto& handler : handlers)
529 {
530 auto rc =
531 handler(tid, eventId, eventMessage.data(), eventMessage.size());
532 if (rc != PLDM_SUCCESS)
533 {
534 lg2::error(
535 "Failed to handle platform event msg for terminus {TID}, event {EVENTID} return {RET}",
536 "TID", tid, "EVENTID", eventId, "RET", rc);
537 }
538 }
539 }
540 catch (const std::out_of_range& e)
541 {
542 lg2::error(
543 "Failed to handle platform event msg for terminus {TID}, event {EVENTID} error - {ERROR}",
544 "TID", tid, "EVENTID", eventId, "ERROR", e);
545 }
546 }
547
pollForPlatformEventTask(pldm_tid_t tid,uint32_t pollDataTransferHandle)548 exec::task<int> EventManager::pollForPlatformEventTask(
549 pldm_tid_t tid, uint32_t pollDataTransferHandle)
550 {
551 uint8_t rc = 0;
552 // Set once, doesn't need resetting
553 uint8_t transferOperationFlag = PLDM_GET_FIRSTPART;
554 uint32_t dataTransferHandle = pollDataTransferHandle;
555 uint32_t eventIdToAcknowledge = PLDM_PLATFORM_EVENT_ID_NULL;
556 uint8_t formatVersion = 0x1; // Constant, no need to reset
557 uint16_t eventId = PLDM_PLATFORM_EVENT_ID_ACK;
558 uint16_t polledEventId = PLDM_PLATFORM_EVENT_ID_NONE;
559 pldm_tid_t polledEventTid = 0;
560 uint8_t polledEventClass = 0;
561
562 std::vector<uint8_t> eventMessage{};
563
564 // Reset and mark terminus as available
565 updateAvailableState(tid, true);
566
567 while (eventId != PLDM_PLATFORM_EVENT_ID_NONE)
568 {
569 uint8_t completionCode = 0;
570 pldm_tid_t eventTid = PLDM_PLATFORM_EVENT_ID_NONE;
571 eventId = PLDM_PLATFORM_EVENT_ID_NONE;
572 uint32_t nextDataTransferHandle = 0;
573 uint8_t transferFlag = 0;
574 uint8_t eventClass = 0;
575 uint32_t eventDataSize = 0;
576 uint8_t* eventData = nullptr;
577 uint32_t eventDataIntegrityChecksum = 0;
578
579 /* Stop event polling */
580 if (!getAvailableState(tid))
581 {
582 lg2::info(
583 "Terminus ID {TID} is not available for PLDM request from {NOW}.",
584 "TID", tid, "NOW", pldm::utils::getCurrentSystemTime());
585 co_await stdexec::just_stopped();
586 }
587
588 rc = co_await pollForPlatformEventMessage(
589 tid, formatVersion, transferOperationFlag, dataTransferHandle,
590 eventIdToAcknowledge, completionCode, eventTid, eventId,
591 nextDataTransferHandle, transferFlag, eventClass, eventDataSize,
592 eventData, eventDataIntegrityChecksum);
593 if (rc || completionCode != PLDM_SUCCESS)
594 {
595 lg2::error(
596 "Failed to pollForPlatformEventMessage for terminus {TID}, event {EVENTID}, error {RC}, complete code {CC}",
597 "TID", tid, "EVENTID", eventId, "RC", rc, "CC", completionCode);
598 co_return rc;
599 }
600
601 if (eventDataSize > 0)
602 {
603 eventMessage.insert(eventMessage.end(), eventData,
604 eventData + eventDataSize);
605 }
606
607 if (transferOperationFlag == PLDM_ACKNOWLEDGEMENT_ONLY)
608 {
609 /* Handle the polled event after finish ACK it */
610 if (eventHandlers.contains(polledEventClass))
611 {
612 callPolledEventHandlers(polledEventTid, polledEventClass,
613 polledEventId, eventMessage);
614 }
615 eventMessage.clear();
616
617 if (eventId == PLDM_PLATFORM_EVENT_ID_ACK)
618 {
619 transferOperationFlag = PLDM_GET_FIRSTPART;
620 dataTransferHandle = 0;
621 eventIdToAcknowledge = PLDM_PLATFORM_EVENT_ID_NULL;
622 }
623 }
624 else
625 {
626 auto ret = getNextPartParameters(
627 eventId, eventMessage, transferFlag, eventDataIntegrityChecksum,
628 nextDataTransferHandle, &transferOperationFlag,
629 &dataTransferHandle, &eventIdToAcknowledge);
630 if (ret)
631 {
632 lg2::error(
633 "Failed to process data of pollForPlatformEventMessage for terminus {TID}, event {EVENTID} return {RET}",
634 "TID", tid, "EVENTID", eventId, "RET", ret);
635 co_return PLDM_ERROR_INVALID_DATA;
636 }
637
638 /* Store the polled event INFO to handle after ACK */
639 if ((transferFlag == PLDM_PLATFORM_TRANSFER_START_AND_END) ||
640 (transferFlag == PLDM_PLATFORM_TRANSFER_END))
641 {
642 polledEventTid = eventTid;
643 polledEventId = eventId;
644 polledEventClass = eventClass;
645 }
646 }
647 }
648
649 co_return PLDM_SUCCESS;
650 }
651
pollForPlatformEventMessage(pldm_tid_t tid,uint8_t formatVersion,uint8_t transferOperationFlag,uint32_t dataTransferHandle,uint16_t eventIdToAcknowledge,uint8_t & completionCode,uint8_t & eventTid,uint16_t & eventId,uint32_t & nextDataTransferHandle,uint8_t & transferFlag,uint8_t & eventClass,uint32_t & eventDataSize,uint8_t * & eventData,uint32_t & eventDataIntegrityChecksum)652 exec::task<int> EventManager::pollForPlatformEventMessage(
653 pldm_tid_t tid, uint8_t formatVersion, uint8_t transferOperationFlag,
654 uint32_t dataTransferHandle, uint16_t eventIdToAcknowledge,
655 uint8_t& completionCode, uint8_t& eventTid, uint16_t& eventId,
656 uint32_t& nextDataTransferHandle, uint8_t& transferFlag,
657 uint8_t& eventClass, uint32_t& eventDataSize, uint8_t*& eventData,
658 uint32_t& eventDataIntegrityChecksum)
659 {
660 Request request(
661 sizeof(pldm_msg_hdr) + PLDM_POLL_FOR_PLATFORM_EVENT_MESSAGE_REQ_BYTES);
662 auto requestMsg = new (request.data()) pldm_msg;
663 auto rc = encode_poll_for_platform_event_message_req(
664 0, formatVersion, transferOperationFlag, dataTransferHandle,
665 eventIdToAcknowledge, requestMsg, request.size());
666 if (rc)
667 {
668 lg2::error(
669 "Failed to encode request PollForPlatformEventMessage for terminus ID {TID}, error {RC} ",
670 "TID", tid, "RC", rc);
671 co_return rc;
672 }
673
674 /* Stop event polling */
675 if (!getAvailableState(tid))
676 {
677 lg2::info(
678 "Terminus ID {TID} is not available for PLDM request from {NOW}.",
679 "TID", tid, "NOW", pldm::utils::getCurrentSystemTime());
680 co_await stdexec::just_stopped();
681 }
682
683 const pldm_msg* responseMsg = nullptr;
684 size_t responseLen = 0;
685 rc = co_await terminusManager.sendRecvPldmMsg(tid, request, &responseMsg,
686 &responseLen);
687 if (rc)
688 {
689 lg2::error(
690 "Failed to send PollForPlatformEventMessage message for terminus {TID}, error {RC}",
691 "TID", tid, "RC", rc);
692 co_return rc;
693 }
694
695 rc = decode_poll_for_platform_event_message_resp(
696 responseMsg, responseLen, &completionCode, &eventTid, &eventId,
697 &nextDataTransferHandle, &transferFlag, &eventClass, &eventDataSize,
698 (void**)&eventData, &eventDataIntegrityChecksum);
699 if (rc)
700 {
701 lg2::error(
702 "Failed to decode response PollForPlatformEventMessage for terminus ID {TID}, error {RC} ",
703 "TID", tid, "RC", rc);
704 co_return rc;
705 }
706 if (completionCode != PLDM_SUCCESS)
707 {
708 lg2::error(
709 "Error : PollForPlatformEventMessage for terminus ID {TID}, complete code {CC}.",
710 "TID", tid, "CC", completionCode);
711 co_return rc;
712 }
713
714 co_return completionCode;
715 }
716
717 } // namespace platform_mc
718 } // namespace pldm
719