blob: 003763f4b583c31edfcfbf6e01245604d31e7360 [file] [log] [blame]
George Liu362c18d2020-05-14 09:46:36 +08001#pragma once
2
George Liu362c18d2020-05-14 09:46:36 +08003#include "common/utils.hpp"
Pavithra Barithaya699534f2024-07-29 11:39:15 +05304#include "host-bmc/dbus_to_event_handler.hpp"
George Liu362c18d2020-05-14 09:46:36 +08005#include "libpldmresponder/pdr.hpp"
6#include "pdr_utils.hpp"
7#include "pldmd/handler.hpp"
8
George Liuc453e162022-12-21 17:16:23 +08009#include <libpldm/platform.h>
10#include <libpldm/states.h>
11
Riya Dixit49cfb132023-03-02 04:26:53 -060012#include <phosphor-logging/lg2.hpp>
13
George Liu362c18d2020-05-14 09:46:36 +080014#include <cstdint>
15#include <map>
16
Riya Dixit49cfb132023-03-02 04:26:53 -060017PHOSPHOR_LOG2_USING;
18
George Liu362c18d2020-05-14 09:46:36 +080019namespace pldm
20{
21namespace responder
22{
23namespace platform_state_sensor
24{
George Liu362c18d2020-05-14 09:46:36 +080025/** @brief Function to get the sensor state
26 *
27 * @tparam[in] DBusInterface - DBus interface type
28 * @param[in] dBusIntf - The interface object of DBusInterface
29 * @param[in] stateToDbusValue - Map of DBus property State to attribute value
30 * @param[in] dbusMapping - The d-bus object
31 *
32 * @return - Enumeration of SensorState
33 */
34template <class DBusInterface>
35uint8_t getStateSensorEventState(
36 const DBusInterface& dBusIntf,
Brad Bishop5079ac42021-08-19 18:35:06 -040037 const std::map<pldm::responder::pdr_utils::State,
38 pldm::utils::PropertyValue>& stateToDbusValue,
39 const pldm::utils::DBusMapping& dbusMapping)
George Liu362c18d2020-05-14 09:46:36 +080040{
41 try
42 {
43 auto propertyValue = dBusIntf.getDbusPropertyVariant(
44 dbusMapping.objectPath.c_str(), dbusMapping.propertyName.c_str(),
45 dbusMapping.interface.c_str());
46
47 for (const auto& stateValue : stateToDbusValue)
48 {
49 if (stateValue.second == propertyValue)
50 {
51 return stateValue.first;
52 }
53 }
54 }
55 catch (const std::exception& e)
56 {
Riya Dixit49cfb132023-03-02 04:26:53 -060057 error(
Riya Dixit89644442024-03-31 05:39:59 -050058 "Failed to get state sensor event state from dbus interface '{PATH}', error - {ERROR}.",
Riya Dixit1e5c81e2024-05-03 07:54:00 -050059 "PATH", dbusMapping.objectPath, "ERROR", e);
George Liu362c18d2020-05-14 09:46:36 +080060 }
61
George Liu916808c2021-01-19 17:56:42 +080062 return PLDM_SENSOR_UNKNOWN;
George Liu362c18d2020-05-14 09:46:36 +080063}
64
65/** @brief Function to get the state sensor readings requested by pldm requester
66 *
67 * @tparam[in] DBusInterface - DBus interface type
68 * @tparam[in] Handler - pldm::responder::platform::Handler
69 * @param[in] dBusIntf - The interface object of DBusInterface
70 * @param[in] handler - The interface object of
71 * pldm::responder::platform::Handler
72 * @param[in] sensorId - Sensor ID sent by the requester to act on
73 * @param[in] sensorRearmCnt - Each bit location in this field corresponds to a
74 * particular sensor within the state sensor
75 * @param[out] compSensorCnt - composite sensor count
76 * @param[out] stateField - The state field data for each of the states,
77 * equal to composite sensor count in number
78 * @return - Success or failure in setting the states. Returns failure in
Manojkiran Eda2576aec2024-06-17 12:05:17 +053079 * terms of PLDM completion codes if at least one state fails to be set
George Liu362c18d2020-05-14 09:46:36 +080080 */
81template <class DBusInterface, class Handler>
82int getStateSensorReadingsHandler(
83 const DBusInterface& dBusIntf, Handler& handler, uint16_t sensorId,
84 uint8_t sensorRearmCnt, uint8_t& compSensorCnt,
Manojkiran Edaae933cc2024-02-21 17:19:21 +053085 std::vector<get_sensor_state_field>& stateField,
86 const stateSensorCacheMaps& sensorCache)
George Liu362c18d2020-05-14 09:46:36 +080087{
88 using namespace pldm::responder::pdr;
89 using namespace pldm::utils;
90
91 pldm_state_sensor_pdr* pdr = nullptr;
92
93 std::unique_ptr<pldm_pdr, decltype(&pldm_pdr_destroy)> stateSensorPdrRepo(
94 pldm_pdr_init(), pldm_pdr_destroy);
Andrew Jefferyacb20292023-06-30 11:47:44 +093095 if (!stateSensorPdrRepo)
96 {
97 throw std::runtime_error(
98 "Failed to instantiate state sensor PDR repository");
99 }
Brad Bishop5079ac42021-08-19 18:35:06 -0400100 pldm::responder::pdr_utils::Repo stateSensorPDRs(stateSensorPdrRepo.get());
George Liu362c18d2020-05-14 09:46:36 +0800101 getRepoByType(handler.getRepo(), stateSensorPDRs, PLDM_STATE_SENSOR_PDR);
102 if (stateSensorPDRs.empty())
103 {
Riya Dixit89644442024-03-31 05:39:59 -0500104 error("Failed to get StateSensorPDR record.");
George Liu362c18d2020-05-14 09:46:36 +0800105 return PLDM_PLATFORM_INVALID_SENSOR_ID;
106 }
107
Brad Bishop5079ac42021-08-19 18:35:06 -0400108 pldm::responder::pdr_utils::PdrEntry pdrEntry{};
George Liu362c18d2020-05-14 09:46:36 +0800109 auto pdrRecord = stateSensorPDRs.getFirstRecord(pdrEntry);
110 while (pdrRecord)
111 {
112 pdr = reinterpret_cast<pldm_state_sensor_pdr*>(pdrEntry.data);
Jayanth Othayoth7c14fc42024-12-17 10:24:47 -0600113 assert(pdr != nullptr);
George Liu362c18d2020-05-14 09:46:36 +0800114 if (pdr->sensor_id != sensorId)
115 {
116 pdr = nullptr;
117 pdrRecord = stateSensorPDRs.getNextRecord(pdrRecord, pdrEntry);
118 continue;
119 }
120
121 compSensorCnt = pdr->composite_sensor_count;
122 if (sensorRearmCnt > compSensorCnt)
123 {
Riya Dixit49cfb132023-03-02 04:26:53 -0600124 error(
Riya Dixit89644442024-03-31 05:39:59 -0500125 "The requester sent wrong sensor rearm count '{SENSOR_REARM_COUNT}' for the sensor ID '{SENSORID}'",
126 "SENSORID", sensorId, "SENSOR_REARM_COUNT", sensorRearmCnt);
George Liu362c18d2020-05-14 09:46:36 +0800127 return PLDM_PLATFORM_REARM_UNAVAILABLE_IN_PRESENT_STATE;
128 }
George Liu916808c2021-01-19 17:56:42 +0800129
130 if (sensorRearmCnt == 0)
131 {
132 sensorRearmCnt = compSensorCnt;
133 stateField.resize(sensorRearmCnt);
134 }
135
George Liu362c18d2020-05-14 09:46:36 +0800136 break;
137 }
138
139 if (!pdr)
140 {
141 return PLDM_PLATFORM_INVALID_SENSOR_ID;
142 }
143
144 int rc = PLDM_SUCCESS;
145 try
146 {
Brad Bishop5079ac42021-08-19 18:35:06 -0400147 const auto& [dbusMappings, dbusValMaps] = handler.getDbusObjMaps(
148 sensorId, pldm::responder::pdr_utils::TypeId::PLDM_SENSOR_ID);
George Liu362c18d2020-05-14 09:46:36 +0800149
Manojkiran Eda3daf7a12024-04-17 20:33:44 +0530150 if (dbusMappings.empty() || dbusValMaps.empty())
151 {
Riya Dixit89644442024-03-31 05:39:59 -0500152 error("DbusMappings for sensor ID '{SENSOR_ID}' is missing",
Manojkiran Eda3daf7a12024-04-17 20:33:44 +0530153 "SENSOR_ID", sensorId);
154 return PLDM_ERROR;
155 }
Manojkiran Edaae933cc2024-02-21 17:19:21 +0530156 pldm::responder::pdr_utils::EventStates sensorCacheforSensor{};
157 if (sensorCache.contains(sensorId))
158 {
159 sensorCacheforSensor = sensorCache.at(sensorId);
160 }
George Liu362c18d2020-05-14 09:46:36 +0800161 stateField.clear();
Manojkiran Eda3daf7a12024-04-17 20:33:44 +0530162 for (std::size_t offset{0};
163 offset < sensorRearmCnt && offset < dbusMappings.size() &&
164 offset < dbusValMaps.size();
165 offset++)
George Liu362c18d2020-05-14 09:46:36 +0800166 {
Manojkiran Eda3daf7a12024-04-17 20:33:44 +0530167 auto& dbusMapping = dbusMappings[offset];
George Liu362c18d2020-05-14 09:46:36 +0800168
George Liu916808c2021-01-19 17:56:42 +0800169 uint8_t sensorEvent = getStateSensorEventState<DBusInterface>(
Manojkiran Eda3daf7a12024-04-17 20:33:44 +0530170 dBusIntf, dbusValMaps[offset], dbusMapping);
George Liu916808c2021-01-19 17:56:42 +0800171
Manojkiran Edaae933cc2024-02-21 17:19:21 +0530172 uint8_t previousState = PLDM_SENSOR_UNKNOWN;
173
174 // if sensor cache is empty, then its the first
175 // get_state_sensor_reading on this sensor, set the previous state
176 // as the current state
177
Manojkiran Eda3daf7a12024-04-17 20:33:44 +0530178 if (sensorCacheforSensor.at(offset) == PLDM_SENSOR_UNKNOWN)
Manojkiran Edaae933cc2024-02-21 17:19:21 +0530179 {
180 previousState = sensorEvent;
Manojkiran Eda3daf7a12024-04-17 20:33:44 +0530181 handler.updateSensorCache(sensorId, offset, previousState);
Manojkiran Edaae933cc2024-02-21 17:19:21 +0530182 }
183 else
184 {
185 // sensor cache is not empty, so get the previous state from
186 // the sensor cache
Manojkiran Eda3daf7a12024-04-17 20:33:44 +0530187 previousState = sensorCacheforSensor[offset];
Manojkiran Edaae933cc2024-02-21 17:19:21 +0530188 }
George Liu916808c2021-01-19 17:56:42 +0800189 uint8_t opState = PLDM_SENSOR_ENABLED;
190 if (sensorEvent == PLDM_SENSOR_UNKNOWN)
191 {
192 opState = PLDM_SENSOR_UNAVAILABLE;
193 }
194
Manojkiran Edaae933cc2024-02-21 17:19:21 +0530195 stateField.push_back(
196 {opState, PLDM_SENSOR_NORMAL, previousState, sensorEvent});
George Liu362c18d2020-05-14 09:46:36 +0800197 }
198 }
199 catch (const std::out_of_range& e)
200 {
Riya Dixit89644442024-03-31 05:39:59 -0500201 error("The sensor ID '{SENSORID}' does not exist, error - {ERROR}",
202 "SENSORID", sensorId, "ERROR", e);
George Liu362c18d2020-05-14 09:46:36 +0800203 rc = PLDM_ERROR;
204 }
205
206 return rc;
207}
208
209} // namespace platform_state_sensor
210} // namespace responder
211} // namespace pldm