blob: 40bb0572eb42219e7c9a12dd9d5db23d50dbe8ef [file] [log] [blame]
Pavithra Barithaya51efaf82020-04-02 02:42:27 -05001#include "host_pdr_handler.hpp"
2
Andrew Jeffery4668f5c2024-01-15 14:59:17 +10303#include <libpldm/fru.h>
George Liu682ee182020-12-25 15:24:33 +08004#ifdef OEM_IBM
Andrew Jeffery21f128d2024-01-15 15:34:26 +10305#include <libpldm/oem/ibm/fru.h>
Sagar Srinivas3687e2b2023-04-10 05:08:28 -05006#endif
Kamalkumar Patel14107a12024-06-19 08:50:01 -05007#include "dbus/custom_dbus.hpp"
George Liu682ee182020-12-25 15:24:33 +08008
Deepak Kodihalli87514cc2020-04-16 09:08:38 -05009#include <nlohmann/json.hpp>
Riya Dixit49cfb132023-03-02 04:26:53 -060010#include <phosphor-logging/lg2.hpp>
sampmisr6decfc12021-03-02 11:07:36 +053011#include <sdeventplus/clock.hpp>
12#include <sdeventplus/exception.hpp>
13#include <sdeventplus/source/io.hpp>
14#include <sdeventplus/source/time.hpp>
Deepak Kodihalli87514cc2020-04-16 09:08:38 -050015
Pavithra Barithayab3b84b42024-08-23 11:43:57 +053016#include <cassert>
George Liu6492f522020-06-16 10:34:05 +080017#include <fstream>
George Liu96af8cb2021-07-31 15:23:45 +080018#include <type_traits>
Pavithra Barithaya51efaf82020-04-02 02:42:27 -050019
Riya Dixit49cfb132023-03-02 04:26:53 -060020PHOSPHOR_LOG2_USING;
21
Pavithra Barithaya51efaf82020-04-02 02:42:27 -050022namespace pldm
23{
Brad Bishop5079ac42021-08-19 18:35:06 -040024using namespace pldm::responder::events;
Deepak Kodihalli6b1d1ca2020-04-27 07:24:51 -050025using namespace pldm::utils;
26using namespace sdbusplus::bus::match::rules;
Sagar Srinivas3687e2b2023-04-10 05:08:28 -050027using namespace pldm::responder::pdr_utils;
Kamalkumar Patel516122e2024-05-07 04:39:32 -050028using namespace pldm::hostbmc::utils;
Deepak Kodihalli87514cc2020-04-16 09:08:38 -050029using Json = nlohmann::json;
30namespace fs = std::filesystem;
George Liu682ee182020-12-25 15:24:33 +080031using namespace pldm::dbus;
Deepak Kodihalli87514cc2020-04-16 09:08:38 -050032const Json emptyJson{};
Deepak Kodihalli87514cc2020-04-16 09:08:38 -050033
Manojkiran Eda3ca40452021-10-04 22:51:37 +053034template <typename T>
35uint16_t extractTerminusHandle(std::vector<uint8_t>& pdr)
36{
37 T* var = nullptr;
38 if (std::is_same<T, pldm_pdr_fru_record_set>::value)
39 {
40 var = (T*)(pdr.data() + sizeof(pldm_pdr_hdr));
41 }
42 else
43 {
44 var = (T*)(pdr.data());
45 }
46 if (var != nullptr)
47 {
48 return var->terminus_handle;
49 }
50 return TERMINUS_HANDLE;
51}
52
George Liu96af8cb2021-07-31 15:23:45 +080053template <typename T>
54void updateContainerId(pldm_entity_association_tree* entityTree,
55 std::vector<uint8_t>& pdr)
56{
57 T* t = nullptr;
58 if (entityTree == nullptr)
59 {
60 return;
61 }
62 if (std::is_same<T, pldm_pdr_fru_record_set>::value)
63 {
64 t = (T*)(pdr.data() + sizeof(pldm_pdr_hdr));
65 }
66 else
67 {
68 t = (T*)(pdr.data());
69 }
70 if (t == nullptr)
71 {
72 return;
73 }
74
75 pldm_entity entity{t->entity_type, t->entity_instance, t->container_id};
Patrick Williams16c2a0a2024-08-16 15:20:59 -040076 auto node = pldm_entity_association_tree_find_with_locality(
77 entityTree, &entity, true);
George Liu96af8cb2021-07-31 15:23:45 +080078 if (node)
79 {
80 pldm_entity e = pldm_entity_extract(node);
81 t->container_id = e.entity_container_id;
82 }
83}
84
Tom Joseph74f27c72021-05-16 07:58:53 -070085HostPDRHandler::HostPDRHandler(
Andrew Jeffery3dd444d2024-07-25 22:00:27 +093086 int /* mctp_fd */, uint8_t mctp_eid, sdeventplus::Event& event,
87 pldm_pdr* repo, const std::string& eventsJsonsDir,
88 pldm_entity_association_tree* entityTree,
Andrew Jefferya330b2f2023-05-04 14:55:37 +093089 pldm_entity_association_tree* bmcEntityTree,
90 pldm::InstanceIdDb& instanceIdDb,
George Liua881c172021-06-21 18:28:11 +080091 pldm::requester::Handler<pldm::requester::Request>* handler) :
Patrick Williams16c2a0a2024-08-16 15:20:59 -040092 mctp_eid(mctp_eid), event(event), repo(repo),
93 stateSensorHandler(eventsJsonsDir), entityTree(entityTree),
94 instanceIdDb(instanceIdDb), handler(handler),
Kamalkumar Pateleb43d6c2024-05-01 06:11:31 -050095 entityMaps(parseEntityMap(ENTITY_MAP_JSON)), oemUtilsHandler(nullptr)
Deepak Kodihalli87514cc2020-04-16 09:08:38 -050096{
George Liuacf2c8c2021-05-10 14:08:52 +080097 mergedHostParents = false;
Patrick Williams84b790c2022-07-22 19:26:56 -050098 hostOffMatch = std::make_unique<sdbusplus::bus::match_t>(
Deepak Kodihalli6b1d1ca2020-04-27 07:24:51 -050099 pldm::utils::DBusHandler::getBus(),
100 propertiesChanged("/xyz/openbmc_project/state/host0",
101 "xyz.openbmc_project.State.Host"),
Patrick Williams84b790c2022-07-22 19:26:56 -0500102 [this, repo, entityTree, bmcEntityTree](sdbusplus::message_t& msg) {
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400103 DbusChangedProps props{};
104 std::string intf;
105 msg.read(intf, props);
106 const auto itr = props.find("CurrentHostState");
107 if (itr != props.end())
Deepak Kodihalli6b1d1ca2020-04-27 07:24:51 -0500108 {
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400109 PropertyValue value = itr->second;
110 auto propVal = std::get<std::string>(value);
111 if (propVal == "xyz.openbmc_project.State.Host.HostState.Off")
112 {
113 // Delete all the remote terminus information
114 std::erase_if(tlPDRInfo, [](const auto& item) {
115 const auto& [key, value] = item;
116 return key != TERMINUS_HANDLE;
117 });
118 pldm_pdr_remove_remote_pdrs(repo);
119 pldm_entity_association_tree_destroy_root(entityTree);
120 pldm_entity_association_tree_copy_root(bmcEntityTree,
121 entityTree);
122 this->sensorMap.clear();
123 this->responseReceived = false;
124 this->mergedHostParents = false;
125 }
Deepak Kodihalli6b1d1ca2020-04-27 07:24:51 -0500126 }
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400127 });
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500128}
129
Deepak Kodihalli7246e0c2020-07-08 06:40:18 -0500130void HostPDRHandler::fetchPDR(PDRRecordHandles&& recordHandles)
Pavithra Barithaya51efaf82020-04-02 02:42:27 -0500131{
Deepak Kodihalli8cb6f662020-04-10 02:55:43 -0500132 pdrRecordHandles.clear();
Pavithra Barithayaae5c97e2022-08-29 02:57:59 -0500133 modifiedPDRRecordHandles.clear();
134
135 if (isHostPdrModified)
136 {
137 modifiedPDRRecordHandles = std::move(recordHandles);
138 }
139 else
140 {
141 pdrRecordHandles = std::move(recordHandles);
142 }
Deepak Kodihalli8cb6f662020-04-10 02:55:43 -0500143
144 // Defer the actual fetch of PDRs from the host (by queuing the call on the
145 // main event loop). That way, we can respond to the platform event msg from
146 // the host firmware.
147 pdrFetchEvent = std::make_unique<sdeventplus::source::Defer>(
148 event, std::bind(std::mem_fn(&HostPDRHandler::_fetchPDR), this,
149 std::placeholders::_1));
150}
151
152void HostPDRHandler::_fetchPDR(sdeventplus::source::EventBase& /*source*/)
153{
Sampa Misrac0c79482021-06-02 08:01:54 -0500154 getHostPDR();
155}
156
157void HostPDRHandler::getHostPDR(uint32_t nextRecordHandle)
158{
Deepak Kodihalli8cb6f662020-04-10 02:55:43 -0500159 pdrFetchEvent.reset();
160
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400161 std::vector<uint8_t> requestMsg(
162 sizeof(pldm_msg_hdr) + PLDM_GET_PDR_REQ_BYTES);
Pavithra Barithaya51efaf82020-04-02 02:42:27 -0500163 auto request = reinterpret_cast<pldm_msg*>(requestMsg.data());
Deepak Kodihalli7246e0c2020-07-08 06:40:18 -0500164 uint32_t recordHandle{};
Pavithra Barithayaae5c97e2022-08-29 02:57:59 -0500165 if (!nextRecordHandle && (!modifiedPDRRecordHandles.empty()) &&
166 isHostPdrModified)
Deepak Kodihalli7246e0c2020-07-08 06:40:18 -0500167 {
Pavithra Barithayaae5c97e2022-08-29 02:57:59 -0500168 recordHandle = modifiedPDRRecordHandles.front();
169 modifiedPDRRecordHandles.pop_front();
170 }
171 else if (!nextRecordHandle && (!pdrRecordHandles.empty()))
172 {
173 recordHandle = pdrRecordHandles.front();
174 pdrRecordHandles.pop_front();
Sampa Misrac0c79482021-06-02 08:01:54 -0500175 }
176 else
177 {
178 recordHandle = nextRecordHandle;
179 }
Andrew Jefferya330b2f2023-05-04 14:55:37 +0930180 auto instanceId = instanceIdDb.next(mctp_eid);
Sampa Misrac0c79482021-06-02 08:01:54 -0500181
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400182 auto rc =
183 encode_get_pdr_req(instanceId, recordHandle, 0, PLDM_GET_FIRSTPART,
184 UINT16_MAX, 0, request, PLDM_GET_PDR_REQ_BYTES);
Sampa Misrac0c79482021-06-02 08:01:54 -0500185 if (rc != PLDM_SUCCESS)
186 {
Andrew Jefferya330b2f2023-05-04 14:55:37 +0930187 instanceIdDb.free(mctp_eid, instanceId);
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500188 error("Failed to encode get pdr request, response code '{RC}'", "RC",
189 rc);
Sampa Misrac0c79482021-06-02 08:01:54 -0500190 return;
191 }
Deepak Kodihalli7246e0c2020-07-08 06:40:18 -0500192
Sampa Misrac0c79482021-06-02 08:01:54 -0500193 rc = handler->registerRequest(
194 mctp_eid, instanceId, PLDM_PLATFORM, PLDM_GET_PDR,
195 std::move(requestMsg),
Andrew Jefferyc14fb4b2024-07-25 22:13:09 +0930196 std::bind_front(&HostPDRHandler::processHostPDRs, this));
Sampa Misrac0c79482021-06-02 08:01:54 -0500197 if (rc)
Pavithra Barithaya51efaf82020-04-02 02:42:27 -0500198 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500199 error(
200 "Failed to send the getPDR request to remote terminus, response code '{RC}'",
201 "RC", rc);
Pavithra Barithayae8beb892020-04-14 23:24:25 -0500202 }
Pavithra Barithaya51efaf82020-04-02 02:42:27 -0500203}
204
Pavithra Barithaya3aec9972020-12-14 01:55:44 -0600205int HostPDRHandler::handleStateSensorEvent(const StateSensorEntry& entry,
206 pdr::EventState state)
207{
208 auto rc = stateSensorHandler.eventAction(entry, state);
209 if (rc != PLDM_SUCCESS)
210 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500211 error("Failed to fetch and update D-bus property, response code '{RC}'",
212 "RC", rc);
Pavithra Barithaya3aec9972020-12-14 01:55:44 -0600213 return rc;
214 }
215 return PLDM_SUCCESS;
216}
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500217
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500218void HostPDRHandler::mergeEntityAssociations(
219 const std::vector<uint8_t>& pdr, [[maybe_unused]] const uint32_t& size,
220 [[maybe_unused]] const uint32_t& record_handle)
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500221{
222 size_t numEntities{};
223 pldm_entity* entities = nullptr;
224 bool merged = false;
225 auto entityPdr = reinterpret_cast<pldm_pdr_entity_association*>(
226 const_cast<uint8_t*>(pdr.data()) + sizeof(pldm_pdr_hdr));
227
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500228 if (oemPlatformHandler &&
229 oemPlatformHandler->checkRecordHandleInRange(record_handle))
230 {
231 // Adding the remote range PDRs to the repo before merging it
232 uint32_t handle = record_handle;
Andrew Jeffery5a945bd2024-08-01 13:15:36 +0000233 pldm_pdr_add(repo, pdr.data(), size, true, 0xFFFF, &handle);
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500234 }
235
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500236 pldm_entity_association_pdr_extract(pdr.data(), pdr.size(), &numEntities,
237 &entities);
George Liuacf2c8c2021-05-10 14:08:52 +0800238 if (numEntities > 0)
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500239 {
George Liuacf2c8c2021-05-10 14:08:52 +0800240 pldm_entity_node* pNode = nullptr;
241 if (!mergedHostParents)
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500242 {
George Liuacf2c8c2021-05-10 14:08:52 +0800243 pNode = pldm_entity_association_tree_find_with_locality(
244 entityTree, &entities[0], false);
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500245 }
George Liuacf2c8c2021-05-10 14:08:52 +0800246 else
247 {
248 pNode = pldm_entity_association_tree_find_with_locality(
249 entityTree, &entities[0], true);
250 }
251 if (!pNode)
252 {
253 return;
254 }
255
George Liudf9a6d32020-12-22 16:27:16 +0800256 Entities entityAssoc;
257 entityAssoc.push_back(pNode);
George Liuacf2c8c2021-05-10 14:08:52 +0800258 for (size_t i = 1; i < numEntities; ++i)
259 {
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500260 bool isUpdateContainerId = true;
261 if (oemPlatformHandler)
262 {
263 isUpdateContainerId =
Sagar Srinivas5db6e872023-12-01 10:03:30 -0600264 checkIfLogicalBitSet(entities[i].entity_container_id);
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500265 }
George Liudf9a6d32020-12-22 16:27:16 +0800266 auto node = pldm_entity_association_tree_add_entity(
George Liuacf2c8c2021-05-10 14:08:52 +0800267 entityTree, &entities[i], entities[i].entity_instance_num,
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500268 pNode, entityPdr->association_type, true, isUpdateContainerId,
269 0xFFFF);
270 if (!node)
271 {
272 continue;
273 }
George Liuacf2c8c2021-05-10 14:08:52 +0800274 merged = true;
George Liudf9a6d32020-12-22 16:27:16 +0800275 entityAssoc.push_back(node);
George Liuacf2c8c2021-05-10 14:08:52 +0800276 }
277
278 mergedHostParents = true;
George Liudf9a6d32020-12-22 16:27:16 +0800279 if (merged)
280 {
281 entityAssociations.push_back(entityAssoc);
282 }
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500283 }
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500284
285 if (merged)
286 {
Pavithra Barithayae8beb892020-04-14 23:24:25 -0500287 // Update our PDR repo with the merged entity association PDRs
Sampa Misra719ed392021-06-04 05:15:13 -0500288 pldm_entity_node* node = nullptr;
289 pldm_find_entity_ref_in_tree(entityTree, entities[0], &node);
290 if (node == nullptr)
291 {
Manojkiran Eda2576aec2024-06-17 12:05:17 +0530292 error("Failed to find reference of the entity in the tree");
Sampa Misra719ed392021-06-04 05:15:13 -0500293 }
294 else
295 {
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500296 int rc = 0;
297 if (oemPlatformHandler)
298 {
299 auto record = oemPlatformHandler->fetchLastBMCRecord(repo);
300
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400301 uint32_t record_handle =
302 pldm_pdr_get_record_handle(repo, record);
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500303
304 rc =
305 pldm_entity_association_pdr_add_from_node_with_record_handle(
306 node, repo, &entities, numEntities, true,
307 TERMINUS_HANDLE, (record_handle + 1));
308 }
309 else
310 {
Andrew Jeffery7761bd22024-08-01 13:15:36 +0000311 rc = pldm_entity_association_pdr_add_from_node(
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500312 node, repo, &entities, numEntities, true, TERMINUS_HANDLE);
313 }
314
Andrew Jeffery1fd3c512023-07-03 13:02:03 +0930315 if (rc)
316 {
317 error(
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500318 "Failed to add entity association PDR from node, response code '{RC}'",
319 "RC", rc);
Andrew Jeffery1fd3c512023-07-03 13:02:03 +0930320 }
Sampa Misra719ed392021-06-04 05:15:13 -0500321 }
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500322 }
Sampa Misra719ed392021-06-04 05:15:13 -0500323 free(entities);
Deepak Kodihalli87514cc2020-04-16 09:08:38 -0500324}
325
Pavithra Barithayae8beb892020-04-14 23:24:25 -0500326void HostPDRHandler::sendPDRRepositoryChgEvent(std::vector<uint8_t>&& pdrTypes,
327 uint8_t eventDataFormat)
328{
329 assert(eventDataFormat == FORMAT_IS_PDR_HANDLES);
330
331 // Extract from the PDR repo record handles of PDRs we want the host
332 // to pull up.
333 std::vector<uint8_t> eventDataOps{PLDM_RECORDS_ADDED};
334 std::vector<uint8_t> numsOfChangeEntries(1);
335 std::vector<std::vector<ChangeEntry>> changeEntries(
336 numsOfChangeEntries.size());
337 for (auto pdrType : pdrTypes)
338 {
339 const pldm_pdr_record* record{};
340 do
341 {
342 record = pldm_pdr_find_record_by_type(repo, pdrType, record,
343 nullptr, nullptr);
344 if (record && pldm_pdr_record_is_remote(record))
345 {
346 changeEntries[0].push_back(
347 pldm_pdr_get_record_handle(repo, record));
348 }
349 } while (record);
350 }
351 if (changeEntries.empty())
352 {
353 return;
354 }
355 numsOfChangeEntries[0] = changeEntries[0].size();
356
357 // Encode PLDM platform event msg to indicate a PDR repo change.
358 size_t maxSize = PLDM_PDR_REPOSITORY_CHG_EVENT_MIN_LENGTH +
359 PLDM_PDR_REPOSITORY_CHANGE_RECORD_MIN_LENGTH +
360 changeEntries[0].size() * sizeof(uint32_t);
361 std::vector<uint8_t> eventDataVec{};
362 eventDataVec.resize(maxSize);
363 auto eventData =
364 reinterpret_cast<struct pldm_pdr_repository_chg_event_data*>(
365 eventDataVec.data());
366 size_t actualSize{};
367 auto firstEntry = changeEntries[0].data();
368 auto rc = encode_pldm_pdr_repository_chg_event_data(
369 eventDataFormat, 1, eventDataOps.data(), numsOfChangeEntries.data(),
370 &firstEntry, eventData, &actualSize, maxSize);
371 if (rc != PLDM_SUCCESS)
372 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500373 error(
374 "Failed to encode pldm pdr repository change event data, response code '{RC}'",
375 "RC", rc);
Pavithra Barithayae8beb892020-04-14 23:24:25 -0500376 return;
377 }
Andrew Jefferya330b2f2023-05-04 14:55:37 +0930378 auto instanceId = instanceIdDb.next(mctp_eid);
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400379 std::vector<uint8_t> requestMsg(
380 sizeof(pldm_msg_hdr) + PLDM_PLATFORM_EVENT_MESSAGE_MIN_REQ_BYTES +
381 actualSize);
Pavithra Barithayae8beb892020-04-14 23:24:25 -0500382 auto request = reinterpret_cast<pldm_msg*>(requestMsg.data());
383 rc = encode_platform_event_message_req(
ArchanaKakani6c39c7a2022-12-05 04:36:35 -0600384 instanceId, 1, TERMINUS_ID, PLDM_PDR_REPOSITORY_CHG_EVENT,
385 eventDataVec.data(), actualSize, request,
Christian Geddes3bdb3c22020-05-01 14:55:39 -0500386 actualSize + PLDM_PLATFORM_EVENT_MESSAGE_MIN_REQ_BYTES);
Pavithra Barithayae8beb892020-04-14 23:24:25 -0500387 if (rc != PLDM_SUCCESS)
388 {
Andrew Jefferya330b2f2023-05-04 14:55:37 +0930389 instanceIdDb.free(mctp_eid, instanceId);
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500390 error(
391 "Failed to encode platform event message request, response code '{RC}'",
392 "RC", rc);
Pavithra Barithayae8beb892020-04-14 23:24:25 -0500393 return;
394 }
395
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400396 auto platformEventMessageResponseHandler = [](mctp_eid_t /*eid*/,
397 const pldm_msg* response,
398 size_t respMsgLen) {
Tom Joseph74f27c72021-05-16 07:58:53 -0700399 if (response == nullptr || !respMsgLen)
400 {
Riya Dixit49cfb132023-03-02 04:26:53 -0600401 error(
402 "Failed to receive response for the PDR repository changed event");
Tom Joseph74f27c72021-05-16 07:58:53 -0700403 return;
404 }
405
406 uint8_t completionCode{};
407 uint8_t status{};
408 auto responsePtr = reinterpret_cast<const struct pldm_msg*>(response);
Pavithra Barithaya54b5a562021-09-27 06:07:10 -0500409 auto rc = decode_platform_event_message_resp(responsePtr, respMsgLen,
410 &completionCode, &status);
Sampa Misrac0c79482021-06-02 08:01:54 -0500411 if (rc || completionCode)
Tom Joseph74f27c72021-05-16 07:58:53 -0700412 {
Riya Dixit49cfb132023-03-02 04:26:53 -0600413 error(
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500414 "Failed to decode platform event message response, response code '{RC}' and completion code '{CC}'",
415 "RC", rc, "CC", completionCode);
Tom Joseph74f27c72021-05-16 07:58:53 -0700416 }
417 };
418
Sampa Misrac0c79482021-06-02 08:01:54 -0500419 rc = handler->registerRequest(
Sampa Misra626c5652021-08-11 10:28:48 -0500420 mctp_eid, instanceId, PLDM_PLATFORM, PLDM_PLATFORM_EVENT_MESSAGE,
Tom Joseph74f27c72021-05-16 07:58:53 -0700421 std::move(requestMsg), std::move(platformEventMessageResponseHandler));
Sampa Misrac0c79482021-06-02 08:01:54 -0500422 if (rc)
Pavithra Barithayae8beb892020-04-14 23:24:25 -0500423 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500424 error(
425 "Failed to send the PDR repository changed event request, response code '{RC}'",
426 "RC", rc);
Pavithra Barithayae8beb892020-04-14 23:24:25 -0500427 }
428}
429
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530430void HostPDRHandler::parseStateSensorPDRs(const PDRList& stateSensorPDRs)
Sampa Misra868c8792020-05-26 03:12:13 -0500431{
432 for (const auto& pdr : stateSensorPDRs)
433 {
434 SensorEntry sensorEntry{};
435 const auto& [terminusHandle, sensorID, sensorInfo] =
436 responder::pdr_utils::parseStateSensorPDR(pdr);
437 sensorEntry.sensorID = sensorID;
438 try
439 {
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530440 sensorEntry.terminusID = std::get<0>(tlPDRInfo.at(terminusHandle));
Sampa Misra868c8792020-05-26 03:12:13 -0500441 }
442 // If there is no mapping for terminusHandle assign the reserved TID
443 // value of 0xFF to indicate that.
Kamalkumar Patel58cbcaf2023-10-06 03:48:25 -0500444 catch (const std::out_of_range&)
Sampa Misra868c8792020-05-26 03:12:13 -0500445 {
446 sensorEntry.terminusID = PLDM_TID_RESERVED;
447 }
448 sensorMap.emplace(sensorEntry, std::move(sensorInfo));
449 }
450}
451
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400452void HostPDRHandler::processHostPDRs(
453 mctp_eid_t /*eid*/, const pldm_msg* response, size_t respMsgLen)
Sampa Misrac0c79482021-06-02 08:01:54 -0500454{
455 static bool merged = false;
456 static PDRList stateSensorPDRs{};
George Liu682ee182020-12-25 15:24:33 +0800457 static PDRList fruRecordSetPDRs{};
Sampa Misrac0c79482021-06-02 08:01:54 -0500458 uint32_t nextRecordHandle{};
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600459 uint8_t tlEid = 0;
460 bool tlValid = true;
461 uint32_t rh = 0;
462 uint16_t terminusHandle = 0;
Manojkiran Eda3ca40452021-10-04 22:51:37 +0530463 uint16_t pdrTerminusHandle = 0;
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600464 uint8_t tid = 0;
465
Sampa Misrac0c79482021-06-02 08:01:54 -0500466 uint8_t completionCode{};
467 uint32_t nextDataTransferHandle{};
468 uint8_t transferFlag{};
469 uint16_t respCount{};
470 uint8_t transferCRC{};
471 if (response == nullptr || !respMsgLen)
472 {
Riya Dixit49cfb132023-03-02 04:26:53 -0600473 error("Failed to receive response for the GetPDR command");
Riya Dixit478e71d2024-06-21 14:10:14 -0500474 pldm::utils::reportError(
475 "xyz.openbmc_project.PLDM.Error.GetPDR.PDRExchangeFailure");
Sampa Misrac0c79482021-06-02 08:01:54 -0500476 return;
477 }
478
479 auto rc = decode_get_pdr_resp(
480 response, respMsgLen /*- sizeof(pldm_msg_hdr)*/, &completionCode,
481 &nextRecordHandle, &nextDataTransferHandle, &transferFlag, &respCount,
482 nullptr, 0, &transferCRC);
483 std::vector<uint8_t> responsePDRMsg;
484 responsePDRMsg.resize(respMsgLen + sizeof(pldm_msg_hdr));
485 memcpy(responsePDRMsg.data(), response, respMsgLen + sizeof(pldm_msg_hdr));
Sampa Misrac0c79482021-06-02 08:01:54 -0500486 if (rc != PLDM_SUCCESS)
487 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500488 error(
489 "Failed to decode getPDR response for next record handle '{NEXT_RECORD_HANDLE}', response code '{RC}'",
490 "NEXT_RECORD_HANDLE", nextRecordHandle, "RC", rc);
Sampa Misrac0c79482021-06-02 08:01:54 -0500491 return;
492 }
493 else
494 {
495 std::vector<uint8_t> pdr(respCount, 0);
496 rc = decode_get_pdr_resp(response, respMsgLen, &completionCode,
497 &nextRecordHandle, &nextDataTransferHandle,
498 &transferFlag, &respCount, pdr.data(),
499 respCount, &transferCRC);
500 if (rc != PLDM_SUCCESS || completionCode != PLDM_SUCCESS)
501 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500502 error(
503 "Failed to decode getPDR response for next record handle '{NEXT_RECORD_HANDLE}', next data transfer handle '{DATA_TRANSFER_HANDLE}' and transfer flag '{FLAG}', response code '{RC}' and completion code '{CC}'",
504 "NEXT_RECORD_HANDLE", nextRecordHandle, "DATA_TRANSFER_HANDLE",
505 nextDataTransferHandle, "FLAG", transferFlag, "RC", rc, "CC",
506 completionCode);
Sampa Misrac0c79482021-06-02 08:01:54 -0500507 return;
508 }
509 else
510 {
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600511 // when nextRecordHandle is 0, we need the recordHandle of the last
512 // PDR and not 0-1.
513 if (!nextRecordHandle)
514 {
515 rh = nextRecordHandle;
516 }
517 else
518 {
519 rh = nextRecordHandle - 1;
520 }
521
Sampa Misrac0c79482021-06-02 08:01:54 -0500522 auto pdrHdr = reinterpret_cast<pldm_pdr_hdr*>(pdr.data());
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600523 if (!rh)
524 {
525 rh = pdrHdr->record_handle;
526 }
527
Sampa Misrac0c79482021-06-02 08:01:54 -0500528 if (pdrHdr->type == PLDM_PDR_ENTITY_ASSOCIATION)
529 {
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500530 this->mergeEntityAssociations(pdr, respCount, rh);
Sampa Misrac0c79482021-06-02 08:01:54 -0500531 merged = true;
532 }
533 else
534 {
535 if (pdrHdr->type == PLDM_TERMINUS_LOCATOR_PDR)
536 {
Manojkiran Eda3ca40452021-10-04 22:51:37 +0530537 pdrTerminusHandle =
538 extractTerminusHandle<pldm_terminus_locator_pdr>(pdr);
Sampa Misrac0c79482021-06-02 08:01:54 -0500539 auto tlpdr =
540 reinterpret_cast<const pldm_terminus_locator_pdr*>(
541 pdr.data());
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600542
543 terminusHandle = tlpdr->terminus_handle;
544 tid = tlpdr->tid;
545 auto terminus_locator_type = tlpdr->terminus_locator_type;
546 if (terminus_locator_type ==
547 PLDM_TERMINUS_LOCATOR_TYPE_MCTP_EID)
548 {
549 auto locatorValue = reinterpret_cast<
550 const pldm_terminus_locator_type_mctp_eid*>(
551 tlpdr->terminus_locator_value);
552 tlEid = static_cast<uint8_t>(locatorValue->eid);
553 }
554 if (tlpdr->validity == 0)
555 {
556 tlValid = false;
557 }
Pavithra Barithaya52aad392022-08-02 04:18:52 -0500558 for (const auto& terminusMap : tlPDRInfo)
559 {
560 if ((terminusHandle == (terminusMap.first)) &&
561 (get<1>(terminusMap.second) == tlEid) &&
562 (get<2>(terminusMap.second) == tlpdr->validity))
563 {
564 // TL PDR already present with same validity don't
565 // add the PDR to the repo just return
566 return;
567 }
568 }
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530569 tlPDRInfo.insert_or_assign(
570 tlpdr->terminus_handle,
571 std::make_tuple(tlpdr->tid, tlEid, tlpdr->validity));
Sampa Misrac0c79482021-06-02 08:01:54 -0500572 }
573 else if (pdrHdr->type == PLDM_STATE_SENSOR_PDR)
574 {
Manojkiran Eda3ca40452021-10-04 22:51:37 +0530575 pdrTerminusHandle =
576 extractTerminusHandle<pldm_state_sensor_pdr>(pdr);
George Liu96af8cb2021-07-31 15:23:45 +0800577 updateContainerId<pldm_state_sensor_pdr>(entityTree, pdr);
Sampa Misrac0c79482021-06-02 08:01:54 -0500578 stateSensorPDRs.emplace_back(pdr);
579 }
Manojkiran Eda3ca40452021-10-04 22:51:37 +0530580 else if (pdrHdr->type == PLDM_PDR_FRU_RECORD_SET)
581 {
582 pdrTerminusHandle =
583 extractTerminusHandle<pldm_pdr_fru_record_set>(pdr);
George Liu96af8cb2021-07-31 15:23:45 +0800584 updateContainerId<pldm_pdr_fru_record_set>(entityTree, pdr);
George Liu682ee182020-12-25 15:24:33 +0800585 fruRecordSetPDRs.emplace_back(pdr);
Manojkiran Eda3ca40452021-10-04 22:51:37 +0530586 }
587 else if (pdrHdr->type == PLDM_STATE_EFFECTER_PDR)
588 {
589 pdrTerminusHandle =
590 extractTerminusHandle<pldm_state_effecter_pdr>(pdr);
George Liu96af8cb2021-07-31 15:23:45 +0800591 updateContainerId<pldm_state_effecter_pdr>(entityTree, pdr);
Manojkiran Eda3ca40452021-10-04 22:51:37 +0530592 }
593 else if (pdrHdr->type == PLDM_NUMERIC_EFFECTER_PDR)
594 {
595 pdrTerminusHandle =
596 extractTerminusHandle<pldm_numeric_effecter_value_pdr>(
597 pdr);
George Liu96af8cb2021-07-31 15:23:45 +0800598 updateContainerId<pldm_numeric_effecter_value_pdr>(
599 entityTree, pdr);
Manojkiran Eda3ca40452021-10-04 22:51:37 +0530600 }
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600601 // if the TLPDR is invalid update the repo accordingly
602 if (!tlValid)
603 {
604 pldm_pdr_update_TL_pdr(repo, terminusHandle, tid, tlEid,
605 tlValid);
Sagar Srinivas3687e2b2023-04-10 05:08:28 -0500606
607 if (!isHostUp())
608 {
609 // The terminus PDR becomes invalid when the terminus
610 // itself is down. We don't need to do PDR exchange in
611 // that case, so setting the next record handle to 0.
612 nextRecordHandle = 0;
613 }
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600614 }
615 else
616 {
Andrew Jeffery5a945bd2024-08-01 13:15:36 +0000617 rc = pldm_pdr_add(repo, pdr.data(), respCount, true,
618 pdrTerminusHandle, &rh);
Andrew Jeffery64f37fe2023-07-03 15:41:13 +0930619 if (rc)
620 {
621 // pldm_pdr_add() assert()ed on failure to add a PDR.
622 throw std::runtime_error("Failed to add PDR");
623 }
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600624 }
Sampa Misrac0c79482021-06-02 08:01:54 -0500625 }
626 }
627 }
628 if (!nextRecordHandle)
629 {
Kamalkumar Patel516122e2024-05-07 04:39:32 -0500630 updateEntityAssociation(entityAssociations, entityTree, objPathMap,
Kamalkumar Patel15ce5a12024-05-07 11:45:11 -0500631 entityMaps, oemPlatformHandler);
Kamalkumar Pateleb43d6c2024-05-01 06:11:31 -0500632 if (oemUtilsHandler)
633 {
634 oemUtilsHandler->setCoreCount(entityAssociations, entityMaps);
635 }
Sampa Misrac0c79482021-06-02 08:01:54 -0500636 /*received last record*/
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530637 this->parseStateSensorPDRs(stateSensorPDRs);
George Liu682ee182020-12-25 15:24:33 +0800638 this->createDbusObjects(fruRecordSetPDRs);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600639 if (isHostUp())
640 {
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530641 this->setHostSensorState(stateSensorPDRs);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600642 }
Sampa Misrac0c79482021-06-02 08:01:54 -0500643 stateSensorPDRs.clear();
George Liu682ee182020-12-25 15:24:33 +0800644 fruRecordSetPDRs.clear();
George Liudf9a6d32020-12-22 16:27:16 +0800645 entityAssociations.clear();
646
Sampa Misrac0c79482021-06-02 08:01:54 -0500647 if (merged)
648 {
649 merged = false;
650 deferredPDRRepoChgEvent =
651 std::make_unique<sdeventplus::source::Defer>(
652 event,
653 std::bind(
654 std::mem_fn((&HostPDRHandler::_processPDRRepoChgEvent)),
655 this, std::placeholders::_1));
656 }
657 }
658 else
659 {
Pavithra Barithayaae5c97e2022-08-29 02:57:59 -0500660 if (modifiedPDRRecordHandles.empty() && isHostPdrModified)
661 {
662 isHostPdrModified = false;
663 }
664 else
665 {
666 deferredFetchPDREvent =
667 std::make_unique<sdeventplus::source::Defer>(
668 event,
669 std::bind(
670 std::mem_fn((&HostPDRHandler::_processFetchPDREvent)),
671 this, nextRecordHandle, std::placeholders::_1));
672 }
Sampa Misrac0c79482021-06-02 08:01:54 -0500673 }
674}
675
676void HostPDRHandler::_processPDRRepoChgEvent(
677 sdeventplus::source::EventBase& /*source */)
678{
679 deferredPDRRepoChgEvent.reset();
680 this->sendPDRRepositoryChgEvent(
Andrew Jefferyc14fb4b2024-07-25 22:13:09 +0930681 std::vector<uint8_t>(1, PLDM_PDR_ENTITY_ASSOCIATION),
Sampa Misrac0c79482021-06-02 08:01:54 -0500682 FORMAT_IS_PDR_HANDLES);
683}
684
685void HostPDRHandler::_processFetchPDREvent(
686 uint32_t nextRecordHandle, sdeventplus::source::EventBase& /*source */)
687{
688 deferredFetchPDREvent.reset();
689 if (!this->pdrRecordHandles.empty())
690 {
691 nextRecordHandle = this->pdrRecordHandles.front();
692 this->pdrRecordHandles.pop_front();
693 }
Pavithra Barithayaae5c97e2022-08-29 02:57:59 -0500694 if (isHostPdrModified && (!this->modifiedPDRRecordHandles.empty()))
695 {
696 nextRecordHandle = this->modifiedPDRRecordHandles.front();
697 this->modifiedPDRRecordHandles.pop_front();
698 }
Sampa Misrac0c79482021-06-02 08:01:54 -0500699 this->getHostPDR(nextRecordHandle);
700}
701
Sampa Misraf9ba8c12021-08-06 00:33:47 -0500702void HostPDRHandler::setHostFirmwareCondition()
sampmisr6decfc12021-03-02 11:07:36 +0530703{
sampmisr6decfc12021-03-02 11:07:36 +0530704 responseReceived = false;
Andrew Jefferya330b2f2023-05-04 14:55:37 +0930705 auto instanceId = instanceIdDb.next(mctp_eid);
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400706 std::vector<uint8_t> requestMsg(
707 sizeof(pldm_msg_hdr) + PLDM_GET_VERSION_REQ_BYTES);
Sampa Misraf9ba8c12021-08-06 00:33:47 -0500708 auto request = reinterpret_cast<pldm_msg*>(requestMsg.data());
709 auto rc = encode_get_version_req(instanceId, 0, PLDM_GET_FIRSTPART,
710 PLDM_BASE, request);
711 if (rc != PLDM_SUCCESS)
sampmisr6decfc12021-03-02 11:07:36 +0530712 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500713 error("Failed to encode GetPLDMVersion, response code {RC}", "RC",
Riya Dixit49cfb132023-03-02 04:26:53 -0600714 lg2::hex, rc);
Andrew Jefferya330b2f2023-05-04 14:55:37 +0930715 instanceIdDb.free(mctp_eid, instanceId);
sampmisr6decfc12021-03-02 11:07:36 +0530716 return;
717 }
718
Sampa Misraf9ba8c12021-08-06 00:33:47 -0500719 auto getPLDMVersionHandler = [this](mctp_eid_t /*eid*/,
720 const pldm_msg* response,
721 size_t respMsgLen) {
722 if (response == nullptr || !respMsgLen)
sampmisr6decfc12021-03-02 11:07:36 +0530723 {
Riya Dixit49cfb132023-03-02 04:26:53 -0600724 error(
725 "Failed to receive response for getPLDMVersion command, Host seems to be off");
sampmisr6decfc12021-03-02 11:07:36 +0530726 return;
727 }
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500728 info("Getting the response code '{RC}'", "RC", lg2::hex,
Riya Dixit1e5c81e2024-05-03 07:54:00 -0500729 response->payload[0]);
Sampa Misraf9ba8c12021-08-06 00:33:47 -0500730 this->responseReceived = true;
sampmisr6decfc12021-03-02 11:07:36 +0530731 };
Sampa Misraf9ba8c12021-08-06 00:33:47 -0500732 rc = handler->registerRequest(mctp_eid, instanceId, PLDM_BASE,
733 PLDM_GET_PLDM_VERSION, std::move(requestMsg),
734 std::move(getPLDMVersionHandler));
735 if (rc)
sampmisr6decfc12021-03-02 11:07:36 +0530736 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500737 error(
738 "Failed to discover remote terminus state. Assuming remote terminus as off");
sampmisr6decfc12021-03-02 11:07:36 +0530739 }
740}
741
742bool HostPDRHandler::isHostUp()
743{
744 return responseReceived;
745}
746
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530747void HostPDRHandler::setHostSensorState(const PDRList& stateSensorPDRs)
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600748{
749 for (const auto& stateSensorPDR : stateSensorPDRs)
750 {
751 auto pdr = reinterpret_cast<const pldm_state_sensor_pdr*>(
752 stateSensorPDR.data());
753
754 if (!pdr)
755 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500756 error("Failed to get state sensor PDR");
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600757 pldm::utils::reportError(
Manojkiran Eda92fb0b52024-04-17 10:48:17 +0530758 "xyz.openbmc_project.bmc.pldm.InternalFailure");
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600759 return;
760 }
761
762 uint16_t sensorId = pdr->sensor_id;
763
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530764 for (const auto& [terminusHandle, terminusInfo] : tlPDRInfo)
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600765 {
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530766 if (terminusHandle == pdr->terminus_handle)
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600767 {
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530768 if (std::get<2>(terminusInfo) == PLDM_TL_PDR_VALID)
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600769 {
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530770 mctp_eid = std::get<1>(terminusInfo);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600771 }
772
773 bitfield8_t sensorRearm;
774 sensorRearm.byte = 0;
Manojkiran Eda60e1fe92021-10-08 15:58:16 +0530775 uint8_t tid = std::get<0>(terminusInfo);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600776
Andrew Jefferya330b2f2023-05-04 14:55:37 +0930777 auto instanceId = instanceIdDb.next(mctp_eid);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600778 std::vector<uint8_t> requestMsg(
779 sizeof(pldm_msg_hdr) +
780 PLDM_GET_STATE_SENSOR_READINGS_REQ_BYTES);
781 auto request = reinterpret_cast<pldm_msg*>(requestMsg.data());
782 auto rc = encode_get_state_sensor_readings_req(
783 instanceId, sensorId, sensorRearm, 0, request);
784
785 if (rc != PLDM_SUCCESS)
786 {
Andrew Jefferya330b2f2023-05-04 14:55:37 +0930787 instanceIdDb.free(mctp_eid, instanceId);
Riya Dixit49cfb132023-03-02 04:26:53 -0600788 error(
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500789 "Failed to encode get state sensor readings request for sensorID '{SENSOR_ID}' and instanceID '{INSTANCE}', response code '{RC}'",
790 "SENSOR_ID", sensorId, "INSTANCE", instanceId, "RC",
791 rc);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600792 pldm::utils::reportError(
Manojkiran Eda92fb0b52024-04-17 10:48:17 +0530793 "xyz.openbmc_project.bmc.pldm.InternalFailure");
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600794 return;
795 }
796
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400797 auto getStateSensorReadingRespHandler = [=, this](
798 mctp_eid_t /*eid*/,
799 const pldm_msg*
800 response,
801 size_t respMsgLen) {
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600802 if (response == nullptr || !respMsgLen)
803 {
Riya Dixit49cfb132023-03-02 04:26:53 -0600804 error(
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500805 "Failed to receive response for get state sensor reading command for sensorID '{SENSOR_ID}' and instanceID '{INSTANCE}'",
806 "SENSOR_ID", sensorId, "INSTANCE", instanceId);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600807 return;
808 }
809 std::array<get_sensor_state_field, 8> stateField{};
810 uint8_t completionCode = 0;
811 uint8_t comp_sensor_count = 0;
812
813 auto rc = decode_get_state_sensor_readings_resp(
814 response, respMsgLen, &completionCode,
815 &comp_sensor_count, stateField.data());
816
817 if (rc != PLDM_SUCCESS || completionCode != PLDM_SUCCESS)
818 {
Riya Dixit49cfb132023-03-02 04:26:53 -0600819 error(
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500820 "Failed to decode get state sensor readings response for sensorID '{SENSOR_ID}' and instanceID '{INSTANCE}', response code'{RC}' and completion code '{CC}'",
821 "SENSOR_ID", sensorId, "INSTANCE", instanceId, "RC",
822 rc, "CC", completionCode);
Manojkiran Eda92fb0b52024-04-17 10:48:17 +0530823 pldm::utils::reportError(
824 "xyz.openbmc_project.bmc.pldm.InternalFailure");
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600825 }
826
827 uint8_t eventState;
828 uint8_t previousEventState;
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600829
RIYA DIXITbb5fda42022-09-27 06:48:08 -0500830 for (uint8_t sensorOffset = 0;
831 sensorOffset < comp_sensor_count; sensorOffset++)
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600832 {
RIYA DIXITbb5fda42022-09-27 06:48:08 -0500833 eventState = stateField[sensorOffset].present_state;
834 previousEventState =
835 stateField[sensorOffset].previous_state;
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600836
837 emitStateSensorEventSignal(tid, sensorId, sensorOffset,
838 eventState,
839 previousEventState);
840
841 SensorEntry sensorEntry{tid, sensorId};
842
843 pldm::pdr::EntityInfo entityInfo{};
844 pldm::pdr::CompositeSensorStates
845 compositeSensorStates{};
Sagar Srinivase3607a32024-02-16 03:50:53 -0600846 std::vector<pldm::pdr::StateSetId> stateSetIds{};
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600847
848 try
849 {
Sagar Srinivase3607a32024-02-16 03:50:53 -0600850 std::tie(entityInfo, compositeSensorStates,
851 stateSetIds) =
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600852 lookupSensorInfo(sensorEntry);
853 }
Kamalkumar Patel58cbcaf2023-10-06 03:48:25 -0500854 catch (const std::out_of_range&)
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600855 {
856 try
857 {
858 sensorEntry.terminusID = PLDM_TID_RESERVED;
Sagar Srinivase3607a32024-02-16 03:50:53 -0600859 std::tie(entityInfo, compositeSensorStates,
860 stateSetIds) =
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600861 lookupSensorInfo(sensorEntry);
862 }
Kamalkumar Patel58cbcaf2023-10-06 03:48:25 -0500863 catch (const std::out_of_range&)
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600864 {
Riya Dixit49cfb132023-03-02 04:26:53 -0600865 error("No mapping for the events");
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600866 }
867 }
868
Manojkiran Eda07a07e22024-04-24 19:15:10 +0530869 if ((compositeSensorStates.size() > 1) &&
870 (sensorOffset > (compositeSensorStates.size() - 1)))
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600871 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500872 error(
873 "Error Invalid data, Invalid sensor offset '{SENSOR_OFFSET}'",
874 "SENSOR_OFFSET", sensorOffset);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600875 return;
876 }
877
878 const auto& possibleStates =
879 compositeSensorStates[sensorOffset];
880 if (possibleStates.find(eventState) ==
881 possibleStates.end())
882 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500883 error(
884 "Error invalid_data, Invalid event state '{STATE}'",
885 "STATE", eventState);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600886 return;
887 }
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400888 const auto& [containerId, entityType, entityInstance] =
889 entityInfo;
Sagar Srinivase3607a32024-02-16 03:50:53 -0600890 auto stateSetId = stateSetIds[sensorOffset];
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600891 pldm::responder::events::StateSensorEntry
Manojkiran Edafa084702024-05-27 10:20:30 +0530892 stateSensorEntry{containerId, entityType,
Sagar Srinivase3607a32024-02-16 03:50:53 -0600893 entityInstance, sensorOffset,
Manojkiran Edafa084702024-05-27 10:20:30 +0530894 stateSetId, false};
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600895 handleStateSensorEvent(stateSensorEntry, eventState);
896 }
897 };
898
899 rc = handler->registerRequest(
900 mctp_eid, instanceId, PLDM_PLATFORM,
901 PLDM_GET_STATE_SENSOR_READINGS, std::move(requestMsg),
902 std::move(getStateSensorReadingRespHandler));
903
904 if (rc != PLDM_SUCCESS)
905 {
Riya Dixit49cfb132023-03-02 04:26:53 -0600906 error(
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500907 "Failed to send request to get state sensor reading on remote terminus for sensorID '{SENSOR_ID}' and instanceID '{INSTANCE}', response code '{RC}'",
908 "SENSOR_ID", sensorId, "INSTANCE", instanceId, "RC",
909 rc);
Pavithra Barithaya4f2538a2021-03-05 07:32:15 -0600910 }
911 }
912 }
913 }
914}
George Liu682ee182020-12-25 15:24:33 +0800915
916void HostPDRHandler::getFRURecordTableMetadataByRemote(
917 const PDRList& fruRecordSetPDRs)
918{
919 auto instanceId = instanceIdDb.next(mctp_eid);
920 std::vector<uint8_t> requestMsg(
921 sizeof(pldm_msg_hdr) + PLDM_GET_FRU_RECORD_TABLE_METADATA_REQ_BYTES);
922
923 // GetFruRecordTableMetadata
924 auto request = reinterpret_cast<pldm_msg*>(requestMsg.data());
925 auto rc = encode_get_fru_record_table_metadata_req(
926 instanceId, request, requestMsg.size() - sizeof(pldm_msg_hdr));
927 if (rc != PLDM_SUCCESS)
928 {
929 instanceIdDb.free(mctp_eid, instanceId);
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500930 error(
931 "Failed to encode get fru record table metadata request, response code '{RC}'",
George Liu682ee182020-12-25 15:24:33 +0800932 "RC", lg2::hex, rc);
933 return;
934 }
935
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400936 auto getFruRecordTableMetadataResponseHandler = [this, fruRecordSetPDRs](
937 mctp_eid_t /*eid*/,
938 const pldm_msg*
939 response,
940 size_t respMsgLen) {
George Liu682ee182020-12-25 15:24:33 +0800941 if (response == nullptr || !respMsgLen)
942 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500943 error(
944 "Failed to receive response for the get fru record table metadata");
George Liu682ee182020-12-25 15:24:33 +0800945 return;
946 }
947
948 uint8_t cc = 0;
949 uint8_t fru_data_major_version, fru_data_minor_version;
950 uint32_t fru_table_maximum_size, fru_table_length;
951 uint16_t total_record_set_identifiers;
952 uint16_t total;
953 uint32_t checksum;
954
955 auto rc = decode_get_fru_record_table_metadata_resp(
956 response, respMsgLen, &cc, &fru_data_major_version,
957 &fru_data_minor_version, &fru_table_maximum_size, &fru_table_length,
958 &total_record_set_identifiers, &total, &checksum);
959
960 if (rc != PLDM_SUCCESS || cc != PLDM_SUCCESS)
961 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500962 error(
963 "Failed to decode get fru record table metadata response, response code '{RC}' and completion code '{CC}'",
George Liu682ee182020-12-25 15:24:33 +0800964 "RC", lg2::hex, rc, "CC", cc);
965 return;
966 }
967
968 // pass total to getFRURecordTableByRemote
969 this->getFRURecordTableByRemote(fruRecordSetPDRs, total);
970 };
971
972 rc = handler->registerRequest(
973 mctp_eid, instanceId, PLDM_FRU, PLDM_GET_FRU_RECORD_TABLE_METADATA,
974 std::move(requestMsg),
975 std::move(getFruRecordTableMetadataResponseHandler));
976 if (rc != PLDM_SUCCESS)
977 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500978 error(
979 "Failed to send the the set state effecter states request, response code '{RC}'",
980 "RC", rc);
George Liu682ee182020-12-25 15:24:33 +0800981 }
982
983 return;
984}
985
986void HostPDRHandler::getFRURecordTableByRemote(const PDRList& fruRecordSetPDRs,
987 uint16_t totalTableRecords)
988{
989 fruRecordData.clear();
990
991 if (!totalTableRecords)
992 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -0500993 error("Failed to get fru record table");
George Liu682ee182020-12-25 15:24:33 +0800994 return;
995 }
996
997 auto instanceId = instanceIdDb.next(mctp_eid);
Patrick Williams16c2a0a2024-08-16 15:20:59 -0400998 std::vector<uint8_t> requestMsg(
999 sizeof(pldm_msg_hdr) + PLDM_GET_FRU_RECORD_TABLE_REQ_BYTES);
George Liu682ee182020-12-25 15:24:33 +08001000
1001 // send the getFruRecordTable command
1002 auto request = reinterpret_cast<pldm_msg*>(requestMsg.data());
1003 auto rc = encode_get_fru_record_table_req(
1004 instanceId, 0, PLDM_GET_FIRSTPART, request,
1005 requestMsg.size() - sizeof(pldm_msg_hdr));
1006 if (rc != PLDM_SUCCESS)
1007 {
1008 instanceIdDb.free(mctp_eid, instanceId);
Riya Dixitd6e10ad2024-03-28 19:44:16 -05001009 error(
1010 "Failed to encode get fru record table request, response code '{RC}'",
1011 "RC", lg2::hex, rc);
George Liu682ee182020-12-25 15:24:33 +08001012 return;
1013 }
1014
Patrick Williams16c2a0a2024-08-16 15:20:59 -04001015 auto getFruRecordTableResponseHandler = [totalTableRecords, this,
1016 fruRecordSetPDRs](
1017 mctp_eid_t /*eid*/,
1018 const pldm_msg* response,
1019 size_t respMsgLen) {
George Liu682ee182020-12-25 15:24:33 +08001020 if (response == nullptr || !respMsgLen)
1021 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -05001022 error("Failed to receive response for the get fru record table");
George Liu682ee182020-12-25 15:24:33 +08001023 return;
1024 }
1025
1026 uint8_t cc = 0;
1027 uint32_t next_data_transfer_handle = 0;
1028 uint8_t transfer_flag = 0;
1029 size_t fru_record_table_length = 0;
Patrick Williams16c2a0a2024-08-16 15:20:59 -04001030 std::vector<uint8_t> fru_record_table_data(
1031 respMsgLen - sizeof(pldm_msg_hdr));
George Liu682ee182020-12-25 15:24:33 +08001032 auto responsePtr = reinterpret_cast<const struct pldm_msg*>(response);
1033 auto rc = decode_get_fru_record_table_resp(
Manojkiran Eda33a9cd02024-01-22 11:06:59 +05301034 responsePtr, respMsgLen, &cc, &next_data_transfer_handle,
1035 &transfer_flag, fru_record_table_data.data(),
1036 &fru_record_table_length);
George Liu682ee182020-12-25 15:24:33 +08001037
1038 if (rc != PLDM_SUCCESS || cc != PLDM_SUCCESS)
1039 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -05001040 error(
1041 "Failed to decode get fru record table resp, response code '{RC}' and completion code '{CC}'",
George Liu682ee182020-12-25 15:24:33 +08001042 "RC", lg2::hex, rc, "CC", cc);
1043 return;
1044 }
1045
1046 fruRecordData = responder::pdr_utils::parseFruRecordTable(
1047 fru_record_table_data.data(), fru_record_table_length);
1048
1049 if (totalTableRecords != fruRecordData.size())
1050 {
1051 fruRecordData.clear();
1052
Manojkiran Eda2576aec2024-06-17 12:05:17 +05301053 error("Failed to parse fru record data format.");
George Liu682ee182020-12-25 15:24:33 +08001054 return;
1055 }
1056
1057 this->setFRUDataOnDBus(fruRecordSetPDRs, fruRecordData);
1058 };
1059
1060 rc = handler->registerRequest(
1061 mctp_eid, instanceId, PLDM_FRU, PLDM_GET_FRU_RECORD_TABLE,
1062 std::move(requestMsg), std::move(getFruRecordTableResponseHandler));
1063 if (rc != PLDM_SUCCESS)
1064 {
Riya Dixitd6e10ad2024-03-28 19:44:16 -05001065 error("Failed to send the the set state effecter states request");
George Liu682ee182020-12-25 15:24:33 +08001066 }
1067}
1068
1069std::optional<uint16_t> HostPDRHandler::getRSI(const PDRList& fruRecordSetPDRs,
1070 const pldm_entity& entity)
1071{
1072 for (const auto& pdr : fruRecordSetPDRs)
1073 {
1074 auto fruPdr = reinterpret_cast<const pldm_pdr_fru_record_set*>(
1075 const_cast<uint8_t*>(pdr.data()) + sizeof(pldm_pdr_hdr));
1076
1077 if (fruPdr->entity_type == entity.entity_type &&
Sagar Srinivasebf8bb52023-10-06 01:15:55 -05001078 fruPdr->entity_instance == entity.entity_instance_num &&
1079 fruPdr->container_id == entity.entity_container_id)
George Liu682ee182020-12-25 15:24:33 +08001080 {
1081 return fruPdr->fru_rsi;
1082 }
1083 }
1084
1085 return std::nullopt;
1086}
1087
1088void HostPDRHandler::setFRUDataOnDBus(
Patrick Williamsd310f822023-10-07 19:01:19 -05001089 [[maybe_unused]] const PDRList& fruRecordSetPDRs,
1090 [[maybe_unused]] const std::vector<
1091 responder::pdr_utils::FruRecordDataFormat>& fruRecordData)
George Liu682ee182020-12-25 15:24:33 +08001092{
Patrick Williamsd310f822023-10-07 19:01:19 -05001093#ifdef OEM_IBM
George Liu682ee182020-12-25 15:24:33 +08001094 for (const auto& entity : objPathMap)
1095 {
1096 pldm_entity node = pldm_entity_extract(entity.second);
1097 auto fruRSI = getRSI(fruRecordSetPDRs, node);
1098
1099 for (const auto& data : fruRecordData)
1100 {
1101 if (!fruRSI || *fruRSI != data.fruRSI)
1102 {
1103 continue;
1104 }
1105
1106 if (data.fruRecType == PLDM_FRU_RECORD_TYPE_OEM)
1107 {
1108 for (const auto& tlv : data.fruTLV)
1109 {
1110 if (tlv.fruFieldType ==
1111 PLDM_OEM_FRU_FIELD_TYPE_LOCATION_CODE)
1112 {
1113 CustomDBus::getCustomDBus().setLocationCode(
1114 entity.first,
1115 std::string(reinterpret_cast<const char*>(
1116 tlv.fruFieldValue.data()),
1117 tlv.fruFieldLen));
1118 }
1119 }
1120 }
1121 }
1122 }
Patrick Williamsd310f822023-10-07 19:01:19 -05001123#endif
George Liu682ee182020-12-25 15:24:33 +08001124}
1125void HostPDRHandler::createDbusObjects(const PDRList& fruRecordSetPDRs)
1126{
1127 // TODO: Creating and Refreshing dbus hosted by remote PLDM entity Fru PDRs
Kamalkumar Patel56da5742024-05-23 04:53:07 -05001128 for (const auto& entity : objPathMap)
1129 {
1130 pldm_entity node = pldm_entity_extract(entity.second);
1131 switch (node.entity_type)
1132 {
1133 case PLDM_ENTITY_PROC | 0x8000:
1134 CustomDBus::getCustomDBus().implementCpuCoreInterface(
1135 entity.first);
1136 break;
Archana Kakanibf1fd272024-06-05 13:25:53 -05001137 case PLDM_ENTITY_SLOT:
1138 CustomDBus::getCustomDBus().implementPCIeSlotInterface(
1139 entity.first);
1140 break;
Archana Kakani733b39d2024-06-05 21:05:20 -05001141 case PLDM_ENTITY_CARD:
1142 CustomDBus::getCustomDBus().implementPCIeDeviceInterface(
1143 entity.first);
1144 break;
Kamalkumar Patel2ed986c2024-05-08 02:20:47 -05001145 case PLDM_ENTITY_SYS_BOARD:
1146 CustomDBus::getCustomDBus().implementMotherboardInterface(
1147 entity.first);
1148 break;
Archana Kakani733b39d2024-06-05 21:05:20 -05001149 default:
1150 break;
Kamalkumar Patel56da5742024-05-23 04:53:07 -05001151 }
1152 }
George Liu682ee182020-12-25 15:24:33 +08001153 getFRURecordTableMetadataByRemote(fruRecordSetPDRs);
1154}
1155
Pavithra Barithaya51efaf82020-04-02 02:42:27 -05001156} // namespace pldm