Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 1 | #include <nlohmann/json.hpp> |
Patrick Williams | 957e03c | 2021-09-02 16:38:42 -0500 | [diff] [blame] | 2 | #include <phosphor-logging/lg2.hpp> |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 3 | #include <sdbusplus/bus.hpp> |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 4 | #include <sdbusplus/message.hpp> |
Vijay Khemka | b38fd58 | 2020-07-23 13:21:23 -0700 | [diff] [blame] | 5 | #include <sdeventplus/clock.hpp> |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 6 | #include <sdeventplus/event.hpp> |
Vijay Khemka | b38fd58 | 2020-07-23 13:21:23 -0700 | [diff] [blame] | 7 | #include <sdeventplus/utility/timer.hpp> |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 8 | #include <xyz/openbmc_project/Association/Definitions/server.hpp> |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 9 | #include <xyz/openbmc_project/Sensor/Threshold/Critical/server.hpp> |
| 10 | #include <xyz/openbmc_project/Sensor/Threshold/Warning/server.hpp> |
| 11 | #include <xyz/openbmc_project/Sensor/Value/server.hpp> |
| 12 | |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 13 | #include <deque> |
Konstantin Aladyshev | a6cd704 | 2021-12-21 15:36:01 +0300 | [diff] [blame] | 14 | #include <limits> |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 15 | #include <map> |
| 16 | #include <string> |
| 17 | |
| 18 | namespace phosphor |
| 19 | { |
| 20 | namespace health |
| 21 | { |
| 22 | |
Sui Chen | 036f161 | 2021-07-22 01:31:49 -0700 | [diff] [blame] | 23 | const char* InventoryPath = "/xyz/openbmc_project/inventory"; |
| 24 | |
| 25 | // Used for identifying the BMC inventory creation signal |
| 26 | const char* BMCActivationPath = "/xyz/openbmc_project/inventory/bmc/activation"; |
| 27 | |
Patrick Williams | bbfe718 | 2022-07-22 19:26:56 -0500 | [diff] [blame] | 28 | bool FindSystemInventoryInObjectMapper(sdbusplus::bus_t& bus) |
Sui Chen | 036f161 | 2021-07-22 01:31:49 -0700 | [diff] [blame] | 29 | { |
Patrick Williams | bbfe718 | 2022-07-22 19:26:56 -0500 | [diff] [blame] | 30 | sdbusplus::message_t msg = |
Sui Chen | 036f161 | 2021-07-22 01:31:49 -0700 | [diff] [blame] | 31 | bus.new_method_call("xyz.openbmc_project.ObjectMapper", |
| 32 | "/xyz/openbmc_project/object_mapper", |
| 33 | "xyz.openbmc_project.ObjectMapper", "GetObject"); |
| 34 | msg.append(InventoryPath); |
| 35 | msg.append(std::vector<std::string>{}); |
| 36 | |
| 37 | try |
| 38 | { |
Patrick Williams | bbfe718 | 2022-07-22 19:26:56 -0500 | [diff] [blame] | 39 | sdbusplus::message_t reply = bus.call(msg, 0); |
Sui Chen | 036f161 | 2021-07-22 01:31:49 -0700 | [diff] [blame] | 40 | return true; |
| 41 | } |
| 42 | catch (const std::exception& e) |
| 43 | {} |
| 44 | return false; |
| 45 | } |
| 46 | |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 47 | using Json = nlohmann::json; |
| 48 | using ValueIface = sdbusplus::xyz::openbmc_project::Sensor::server::Value; |
| 49 | |
| 50 | using CriticalInterface = |
| 51 | sdbusplus::xyz::openbmc_project::Sensor::Threshold::server::Critical; |
| 52 | |
| 53 | using WarningInterface = |
| 54 | sdbusplus::xyz::openbmc_project::Sensor::Threshold::server::Warning; |
| 55 | |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 56 | using AssociationDefinitionInterface = |
| 57 | sdbusplus::xyz::openbmc_project::Association::server::Definitions; |
| 58 | |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 59 | using healthIfaces = |
Patrick Williams | bbfe718 | 2022-07-22 19:26:56 -0500 | [diff] [blame] | 60 | sdbusplus::server::object_t<ValueIface, CriticalInterface, WarningInterface, |
| 61 | AssociationDefinitionInterface>; |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 62 | |
| 63 | using AssociationTuple = std::tuple<std::string, std::string, std::string>; |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 64 | |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 65 | struct HealthConfig |
| 66 | { |
| 67 | std::string name; |
| 68 | uint16_t freq; |
| 69 | uint16_t windowSize; |
Konstantin Aladyshev | a6cd704 | 2021-12-21 15:36:01 +0300 | [diff] [blame] | 70 | double criticalHigh = std::numeric_limits<double>::quiet_NaN(); |
| 71 | double warningHigh = std::numeric_limits<double>::quiet_NaN(); |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 72 | bool criticalLog; |
| 73 | bool warningLog; |
| 74 | std::string criticalTgt; |
| 75 | std::string warningTgt; |
Bruceleequantatw | af9acbd | 2020-10-12 15:21:42 +0800 | [diff] [blame] | 76 | std::string path; |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 77 | }; |
| 78 | |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 79 | class HealthSensor : public healthIfaces |
| 80 | { |
| 81 | public: |
| 82 | HealthSensor() = delete; |
| 83 | HealthSensor(const HealthSensor&) = delete; |
| 84 | HealthSensor& operator=(const HealthSensor&) = delete; |
| 85 | HealthSensor(HealthSensor&&) = delete; |
| 86 | HealthSensor& operator=(HealthSensor&&) = delete; |
| 87 | virtual ~HealthSensor() = default; |
| 88 | |
| 89 | /** @brief Constructs HealthSensor |
| 90 | * |
| 91 | * @param[in] bus - Handle to system dbus |
| 92 | * @param[in] objPath - The Dbus path of health sensor |
| 93 | */ |
Patrick Williams | bbfe718 | 2022-07-22 19:26:56 -0500 | [diff] [blame] | 94 | HealthSensor(sdbusplus::bus_t& bus, const char* objPath, |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 95 | HealthConfig& sensorConfig, |
| 96 | const std::vector<std::string>& bmcIds) : |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 97 | healthIfaces(bus, objPath), |
Vijay Khemka | b38fd58 | 2020-07-23 13:21:23 -0700 | [diff] [blame] | 98 | bus(bus), sensorConfig(sensorConfig), |
| 99 | timerEvent(sdeventplus::Event::get_default()), |
| 100 | readTimer(timerEvent, std::bind(&HealthSensor::readHealthSensor, this)) |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 101 | { |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 102 | initHealthSensor(bmcIds); |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 103 | } |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 104 | |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 105 | /** @brief list of sensor data values */ |
| 106 | std::deque<double> valQueue; |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 107 | /** @brief Initialize sensor, set default value and association */ |
| 108 | void initHealthSensor(const std::vector<std::string>& bmcIds); |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 109 | /** @brief Set sensor value utilization to health sensor D-bus */ |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 110 | void setSensorValueToDbus(const double value); |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 111 | /** @brief Set Sensor Threshold to D-bus at beginning */ |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 112 | void setSensorThreshold(double criticalHigh, double warningHigh); |
Vijay Khemka | b7a7b8a | 2020-07-29 12:22:01 -0700 | [diff] [blame] | 113 | /** @brief Check Sensor threshold and update alarm and log */ |
| 114 | void checkSensorThreshold(const double value); |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 115 | |
| 116 | private: |
Vijay Khemka | b38fd58 | 2020-07-23 13:21:23 -0700 | [diff] [blame] | 117 | /** @brief sdbusplus bus client connection. */ |
Patrick Williams | bbfe718 | 2022-07-22 19:26:56 -0500 | [diff] [blame] | 118 | sdbusplus::bus_t& bus; |
Vijay Khemka | b38fd58 | 2020-07-23 13:21:23 -0700 | [diff] [blame] | 119 | /** @brief Sensor config from config file */ |
Vijay Khemka | 1553776 | 2020-07-22 11:44:56 -0700 | [diff] [blame] | 120 | HealthConfig& sensorConfig; |
Vijay Khemka | b38fd58 | 2020-07-23 13:21:23 -0700 | [diff] [blame] | 121 | /** @brief the Event Loop structure */ |
| 122 | sdeventplus::Event timerEvent; |
| 123 | /** @brief Sensor Read Timer */ |
| 124 | sdeventplus::utility::Timer<sdeventplus::ClockId::Monotonic> readTimer; |
Vijay Khemka | b38fd58 | 2020-07-23 13:21:23 -0700 | [diff] [blame] | 125 | /** @brief Read sensor at regular intrval */ |
| 126 | void readHealthSensor(); |
Potin Lai | 156ecf3 | 2022-07-11 17:09:10 +0800 | [diff] [blame] | 127 | /** @brief Start configured threshold systemd unit */ |
| 128 | void startUnit(const std::string& sysdUnit); |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 129 | }; |
| 130 | |
| 131 | class HealthMon |
| 132 | { |
| 133 | public: |
| 134 | HealthMon() = delete; |
| 135 | HealthMon(const HealthMon&) = delete; |
| 136 | HealthMon& operator=(const HealthMon&) = delete; |
| 137 | HealthMon(HealthMon&&) = delete; |
| 138 | HealthMon& operator=(HealthMon&&) = delete; |
| 139 | virtual ~HealthMon() = default; |
| 140 | |
Sui Chen | 036f161 | 2021-07-22 01:31:49 -0700 | [diff] [blame] | 141 | /** @brief Recreates sensor objects and their association if possible |
| 142 | */ |
| 143 | void recreateSensors(); |
| 144 | |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 145 | /** @brief Constructs HealthMon |
| 146 | * |
| 147 | * @param[in] bus - Handle to system dbus |
| 148 | */ |
Patrick Williams | bbfe718 | 2022-07-22 19:26:56 -0500 | [diff] [blame] | 149 | HealthMon(sdbusplus::bus_t& bus) : bus(bus) |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 150 | { |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 151 | // Read JSON file |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 152 | sensorConfigs = getHealthConfig(); |
Sui Chen | 036f161 | 2021-07-22 01:31:49 -0700 | [diff] [blame] | 153 | recreateSensors(); |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 154 | } |
| 155 | |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 156 | /** @brief Parse Health config JSON file */ |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 157 | Json parseConfigFile(std::string configFile); |
| 158 | |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 159 | /** @brief Read config for each health sensor component */ |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 160 | void getConfigData(Json& data, HealthConfig& cfg); |
| 161 | |
| 162 | /** @brief Map of the object HealthSensor */ |
| 163 | std::unordered_map<std::string, std::shared_ptr<HealthSensor>> |
| 164 | healthSensors; |
| 165 | |
| 166 | /** @brief Create sensors for health monitoring */ |
Sui Chen | 670cc13 | 2021-04-13 09:27:22 -0700 | [diff] [blame] | 167 | void createHealthSensors(const std::vector<std::string>& bmcIds); |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 168 | |
| 169 | private: |
Patrick Williams | bbfe718 | 2022-07-22 19:26:56 -0500 | [diff] [blame] | 170 | sdbusplus::bus_t& bus; |
Vijay Khemka | e279530 | 2020-07-15 17:28:45 -0700 | [diff] [blame] | 171 | std::vector<HealthConfig> sensorConfigs; |
| 172 | std::vector<HealthConfig> getHealthConfig(); |
| 173 | }; |
| 174 | |
| 175 | } // namespace health |
| 176 | } // namespace phosphor |