blob: 0d7f896c4b4ba32b8dc4841c419fb734454c113d [file] [log] [blame]
James Feist6714a252018-09-10 15:26:18 -07001/*
2// Copyright (c) 2018 Intel Corporation
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15*/
16
17#include <fcntl.h>
James Feist6714a252018-09-10 15:26:18 -070018
Thu Nguyen255da6b2022-07-29 10:05:52 +070019#include <IntelCPUSensor.hpp>
Ed Tanous8a57ec02020-10-09 12:46:52 -070020#include <Utils.hpp>
21#include <VariantVisitors.hpp>
James Feist6714a252018-09-10 15:26:18 -070022#include <boost/algorithm/string/replace.hpp>
Patrick Venture96e97db2019-10-31 13:44:38 -070023#include <boost/container/flat_map.hpp>
James Feist6714a252018-09-10 15:26:18 -070024#include <boost/container/flat_set.hpp>
James Feist38fb5982020-05-28 10:09:54 -070025#include <sdbusplus/asio/connection.hpp>
26#include <sdbusplus/asio/object_server.hpp>
27#include <sdbusplus/bus/match.hpp>
28
29#include <array>
James Feist24f02f22019-04-15 11:05:39 -070030#include <filesystem>
James Feist6714a252018-09-10 15:26:18 -070031#include <fstream>
Patrick Venture96e97db2019-10-31 13:44:38 -070032#include <functional>
33#include <memory>
James Feist6714a252018-09-10 15:26:18 -070034#include <regex>
Patrick Venture96e97db2019-10-31 13:44:38 -070035#include <sstream>
36#include <stdexcept>
37#include <string>
38#include <utility>
39#include <variant>
40#include <vector>
James Feist6714a252018-09-10 15:26:18 -070041
James Feistf87dc4c2018-12-05 14:39:51 -080042// clang-format off
43// this needs to be included last or we'll have build issues
Ed Tanous74cffa82022-01-25 13:00:28 -080044extern "C" {
James Feistf87dc4c2018-12-05 14:39:51 -080045#include <linux/peci-ioctl.h>
Ed Tanous74cffa82022-01-25 13:00:28 -080046}
Jae Hyun Yoo201c8d92019-02-27 15:41:56 -080047#if !defined(PECI_MBX_INDEX_DDR_DIMM_TEMP)
48#define PECI_MBX_INDEX_DDR_DIMM_TEMP MBX_INDEX_DDR_DIMM_TEMP
49#endif
James Feistf87dc4c2018-12-05 14:39:51 -080050// clang-format on
51
Ed Tanous8a57ec02020-10-09 12:46:52 -070052static constexpr bool debug = false;
James Feist6714a252018-09-10 15:26:18 -070053
Thu Nguyen255da6b2022-07-29 10:05:52 +070054boost::container::flat_map<std::string, std::shared_ptr<IntelCPUSensor>>
55 gCpuSensors;
James Feistc140e202019-11-14 15:23:51 -080056boost::container::flat_map<std::string,
57 std::shared_ptr<sdbusplus::asio::dbus_interface>>
58 inventoryIfaces;
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -080059
James Feist6714a252018-09-10 15:26:18 -070060enum State
61{
62 OFF, // host powered down
63 ON, // host powered on
64 READY // host powered on and mem test passed - fully ready
65};
66
67struct CPUConfig
68{
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -070069 CPUConfig(const uint64_t& bus, const uint64_t& addr,
70 const std::string& name, const State& state) :
71 bus(bus),
72 addr(addr), name(name), state(state)
James Feist38fb5982020-05-28 10:09:54 -070073 {}
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -070074 int bus;
James Feist6714a252018-09-10 15:26:18 -070075 int addr;
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -070076 std::string name;
James Feist6714a252018-09-10 15:26:18 -070077 State state;
78
79 bool operator<(const CPUConfig& rhs) const
80 {
Andrew Jeffery92b96292021-05-27 16:41:31 +093081 // NOLINTNEXTLINE
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -070082 return (name < rhs.name);
James Feist6714a252018-09-10 15:26:18 -070083 }
84};
85
Jae Hyun Yoo9c55e6a2018-10-26 10:09:01 -070086static constexpr const char* peciDev = "/dev/peci-";
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -070087static constexpr const unsigned int rankNumMax = 8;
James Feist6714a252018-09-10 15:26:18 -070088
James Feistcf3bce62019-01-08 10:07:19 -080089namespace fs = std::filesystem;
James Feist3eb82622019-02-08 13:10:22 -080090
Zev Weiss054aad82022-08-18 01:37:34 -070091static constexpr auto sensorTypes{std::to_array<const char*>({"XeonCPU"})};
Brandon Kim66558232021-11-09 16:53:08 -080092static constexpr auto hiddenProps{std::to_array<const char*>(
Thu Nguyen255da6b2022-07-29 10:05:52 +070093 {IntelCPUSensor::labelTcontrol, "Tthrottle", "Tjmax"})};
James Feist6714a252018-09-10 15:26:18 -070094
Jae Hyun Yood64262b2018-11-01 13:31:16 -070095void detectCpuAsync(
Ed Tanous9b4a20e2022-09-06 08:47:11 -070096 boost::asio::steady_timer& pingTimer,
97 boost::asio::steady_timer& creationTimer, boost::asio::io_service& io,
Jae Hyun Yood64262b2018-11-01 13:31:16 -070098 sdbusplus::asio::object_server& objectServer,
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -080099 std::shared_ptr<sdbusplus::asio::connection>& dbusConnection,
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800100 boost::container::flat_set<CPUConfig>& cpuConfigs,
101 ManagedObjectType& sensorConfigs);
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700102
Zbigniew Kurzynski0eee0c12020-06-18 14:20:08 +0200103std::string createSensorName(const std::string& label, const std::string& item,
104 const int& cpuId)
105{
106 std::string sensorName = label;
Ed Tanous2049bd22022-07-09 07:20:26 -0700107 if (item != "input")
Zbigniew Kurzynski0eee0c12020-06-18 14:20:08 +0200108 {
109 sensorName += " " + item;
110 }
111 sensorName += " CPU" + std::to_string(cpuId);
112 // converting to Upper Camel case whole name
113 bool isWordEnd = true;
114 std::transform(sensorName.begin(), sensorName.end(), sensorName.begin(),
115 [&isWordEnd](int c) {
Ed Tanous2049bd22022-07-09 07:20:26 -0700116 if (std::isspace(c) != 0)
Ed Tanousbb679322022-05-16 16:10:00 -0700117 {
118 isWordEnd = true;
119 }
120 else
121 {
122 if (isWordEnd)
123 {
124 isWordEnd = false;
125 return std::toupper(c);
126 }
127 }
128 return c;
129 });
Zbigniew Kurzynski0eee0c12020-06-18 14:20:08 +0200130 return sensorName;
131}
132
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800133bool createSensors(boost::asio::io_service& io,
134 sdbusplus::asio::object_server& objectServer,
135 std::shared_ptr<sdbusplus::asio::connection>& dbusConnection,
136 boost::container::flat_set<CPUConfig>& cpuConfigs,
137 ManagedObjectType& sensorConfigs)
James Feist6714a252018-09-10 15:26:18 -0700138{
139 bool available = false;
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800140 for (const CPUConfig& cpu : cpuConfigs)
James Feist6714a252018-09-10 15:26:18 -0700141 {
142 if (cpu.state != State::OFF)
143 {
144 available = true;
Zev Weiss9702c9d2021-04-21 22:41:51 -0500145 std::shared_ptr<sdbusplus::asio::dbus_interface>& iface =
James Feistc140e202019-11-14 15:23:51 -0800146 inventoryIfaces[cpu.name];
147 if (iface != nullptr)
148 {
149 continue;
150 }
151 iface = objectServer.add_interface(
152 cpuInventoryPath + std::string("/") + cpu.name,
153 "xyz.openbmc_project.Inventory.Item");
154 iface->register_property("PrettyName", cpu.name);
155 iface->register_property("Present", true);
156 iface->initialize();
James Feist6714a252018-09-10 15:26:18 -0700157 }
158 }
159 if (!available)
160 {
Yoo, Jae Hyunf78d0a42018-10-10 11:03:11 -0700161 return false;
James Feist6714a252018-09-10 15:26:18 -0700162 }
163
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800164 if (sensorConfigs.empty())
James Feist6714a252018-09-10 15:26:18 -0700165 {
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800166 return false;
James Feist6714a252018-09-10 15:26:18 -0700167 }
168
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700169 std::vector<fs::path> hwmonNamePaths;
Lei YU0b207a62021-10-20 13:41:51 +0800170 if (!findFiles(fs::path(R"(/sys/bus/peci/devices/peci-0)"),
171 R"(\d+-.+/peci-.+/hwmon/hwmon\d+/name$)", hwmonNamePaths, 5))
James Feist6714a252018-09-10 15:26:18 -0700172 {
173 std::cerr << "No CPU sensors in system\n";
Yoo, Jae Hyunf78d0a42018-10-10 11:03:11 -0700174 return true;
James Feist6714a252018-09-10 15:26:18 -0700175 }
176
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700177 boost::container::flat_set<std::string> scannedDirectories;
178 boost::container::flat_set<std::string> createdSensors;
179
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800180 for (const fs::path& hwmonNamePath : hwmonNamePaths)
James Feist6714a252018-09-10 15:26:18 -0700181 {
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700182 auto hwmonDirectory = hwmonNamePath.parent_path();
183
184 auto ret = scannedDirectories.insert(hwmonDirectory.string());
185 if (!ret.second)
James Feist6714a252018-09-10 15:26:18 -0700186 {
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700187 continue; // already searched this path
188 }
189
190 fs::path::iterator it = hwmonNamePath.begin();
191 std::advance(it, 6); // pick the 6th part for a PECI client device name
192 std::string deviceName = *it;
Ed Tanous8a57ec02020-10-09 12:46:52 -0700193 auto findHyphen = deviceName.find('-');
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700194 if (findHyphen == std::string::npos)
195 {
196 std::cerr << "found bad device " << deviceName << "\n";
James Feist6714a252018-09-10 15:26:18 -0700197 continue;
198 }
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700199 std::string busStr = deviceName.substr(0, findHyphen);
200 std::string addrStr = deviceName.substr(findHyphen + 1);
201
202 size_t bus = 0;
203 size_t addr = 0;
204 try
205 {
206 bus = std::stoi(busStr);
Ed Tanous8a57ec02020-10-09 12:46:52 -0700207 addr = std::stoi(addrStr, nullptr, 16);
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700208 }
Patrick Williams26601e82021-10-06 12:43:25 -0500209 catch (const std::invalid_argument&)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700210 {
211 continue;
212 }
213
214 std::ifstream nameFile(hwmonNamePath);
215 if (!nameFile.good())
216 {
217 std::cerr << "Failure reading " << hwmonNamePath << "\n";
218 continue;
219 }
220 std::string hwmonName;
221 std::getline(nameFile, hwmonName);
James Feist6714a252018-09-10 15:26:18 -0700222 nameFile.close();
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800223 if (hwmonName.empty())
James Feist6714a252018-09-10 15:26:18 -0700224 {
225 // shouldn't have an empty name file
226 continue;
227 }
Ed Tanous8a57ec02020-10-09 12:46:52 -0700228 if (debug)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700229 {
230 std::cout << "Checking: " << hwmonNamePath << ": " << hwmonName
231 << "\n";
232 }
James Feist6714a252018-09-10 15:26:18 -0700233
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700234 std::string sensorType;
James Feist6714a252018-09-10 15:26:18 -0700235 const SensorData* sensorData = nullptr;
236 const std::string* interfacePath = nullptr;
Jae Hyun Yooffa07e22019-07-17 10:47:18 -0700237 const SensorBaseConfiguration* baseConfiguration = nullptr;
James Feist6714a252018-09-10 15:26:18 -0700238
Zev Weiss8908b3c2022-08-12 18:21:01 -0700239 for (const auto& [path, cfgData] : sensorConfigs)
James Feist6714a252018-09-10 15:26:18 -0700240 {
Zev Weiss8908b3c2022-08-12 18:21:01 -0700241 sensorData = &cfgData;
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -0700242 for (const char* type : sensorTypes)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700243 {
Zev Weiss26fb1492022-08-17 15:33:46 -0700244 sensorType = type;
Zev Weiss054aad82022-08-18 01:37:34 -0700245 auto sensorBase =
246 sensorData->find(configInterfaceName(sensorType));
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700247 if (sensorBase != sensorData->end())
248 {
249 baseConfiguration = &(*sensorBase);
250 break;
251 }
252 }
253 if (baseConfiguration == nullptr)
254 {
255 std::cerr << "error finding base configuration for" << hwmonName
256 << "\n";
257 continue;
258 }
259 auto configurationBus = baseConfiguration->second.find("Bus");
260 auto configurationAddress =
261 baseConfiguration->second.find("Address");
262
263 if (configurationBus == baseConfiguration->second.end() ||
264 configurationAddress == baseConfiguration->second.end())
265 {
266 std::cerr << "error finding bus or address in configuration";
267 continue;
268 }
269
James Feist3eb82622019-02-08 13:10:22 -0800270 if (std::get<uint64_t>(configurationBus->second) != bus ||
271 std::get<uint64_t>(configurationAddress->second) != addr)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700272 {
273 continue;
274 }
275
Zev Weiss8908b3c2022-08-12 18:21:01 -0700276 interfacePath = &path.str;
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700277 break;
278 }
279 if (interfacePath == nullptr)
280 {
281 std::cerr << "failed to find match for " << hwmonName << "\n";
James Feist6714a252018-09-10 15:26:18 -0700282 continue;
283 }
284
285 auto findCpuId = baseConfiguration->second.find("CpuID");
286 if (findCpuId == baseConfiguration->second.end())
287 {
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700288 std::cerr << "could not determine CPU ID for " << hwmonName << "\n";
James Feist6714a252018-09-10 15:26:18 -0700289 continue;
290 }
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700291 int cpuId =
James Feist3eb82622019-02-08 13:10:22 -0800292 std::visit(VariantToUnsignedIntVisitor(), findCpuId->second);
James Feist6714a252018-09-10 15:26:18 -0700293
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700294 auto directory = hwmonNamePath.parent_path();
James Feist6714a252018-09-10 15:26:18 -0700295 std::vector<fs::path> inputPaths;
Zbigniew Kurzynski8d8d8d72020-05-29 19:21:24 +0200296 if (!findFiles(directory, R"((temp|power)\d+_(input|average|cap)$)",
Zbigniew Kurzynski0a4c4802020-04-01 11:22:27 +0200297 inputPaths, 0))
James Feist6714a252018-09-10 15:26:18 -0700298 {
299 std::cerr << "No temperature sensors in system\n";
300 continue;
301 }
302
303 // iterate through all found temp sensors
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800304 for (const auto& inputPath : inputPaths)
James Feist6714a252018-09-10 15:26:18 -0700305 {
Zbigniew Kurzynski63f38662020-06-09 13:02:11 +0200306 auto fileParts = splitFileName(inputPath);
Zbigniew Kurzynski0a4c4802020-04-01 11:22:27 +0200307 if (!fileParts)
308 {
309 continue;
310 }
Zbigniew Kurzynskidbfd4662020-09-28 18:06:00 +0200311 auto& [type, nr, item] = *fileParts;
James Feist6714a252018-09-10 15:26:18 -0700312 auto inputPathStr = inputPath.string();
313 auto labelPath =
Zbigniew Kurzynski0a4c4802020-04-01 11:22:27 +0200314 boost::replace_all_copy(inputPathStr, item, "label");
James Feist6714a252018-09-10 15:26:18 -0700315 std::ifstream labelFile(labelPath);
316 if (!labelFile.good())
317 {
318 std::cerr << "Failure reading " << labelPath << "\n";
319 continue;
320 }
321 std::string label;
322 std::getline(labelFile, label);
323 labelFile.close();
Jae Hyun Yoo13f48882019-02-19 13:37:07 -0800324
Zbigniew Kurzynski0eee0c12020-06-18 14:20:08 +0200325 std::string sensorName = createSensorName(label, item, cpuId);
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700326
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800327 auto findSensor = gCpuSensors.find(sensorName);
328 if (findSensor != gCpuSensors.end())
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700329 {
Ed Tanous8a57ec02020-10-09 12:46:52 -0700330 if (debug)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700331 {
332 std::cout << "Skipped: " << inputPath << ": " << sensorName
333 << " is already created\n";
334 }
335 continue;
336 }
337
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800338 // check hidden properties
339 bool show = true;
340 for (const char* prop : hiddenProps)
341 {
342 if (label == prop)
343 {
344 show = false;
345 break;
346 }
347 }
348
Vijay Khemka86dea2b2019-06-06 11:14:37 -0700349 /*
350 * Find if there is DtsCritOffset is configured in config file
351 * set it if configured or else set it to 0
352 */
353 double dtsOffset = 0;
354 if (label == "DTS")
355 {
356 auto findThrOffset =
357 baseConfiguration->second.find("DtsCritOffset");
358 if (findThrOffset != baseConfiguration->second.end())
359 {
360 dtsOffset = std::visit(VariantToDoubleVisitor(),
361 findThrOffset->second);
362 }
363 }
364
James Feist6714a252018-09-10 15:26:18 -0700365 std::vector<thresholds::Threshold> sensorThresholds;
Ed Tanous8a57ec02020-10-09 12:46:52 -0700366 std::string labelHead = label.substr(0, label.find(' '));
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -0700367 parseThresholdsFromConfig(*sensorData, sensorThresholds,
Yoo, Jae Hyun81a464c2018-10-09 16:38:58 -0700368 &labelHead);
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800369 if (sensorThresholds.empty())
James Feist6714a252018-09-10 15:26:18 -0700370 {
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -0700371 if (!parseThresholdsFromAttr(sensorThresholds, inputPathStr,
Thu Nguyen255da6b2022-07-29 10:05:52 +0700372 IntelCPUSensor::sensorScaleFactor,
Vijay Khemka86dea2b2019-06-06 11:14:37 -0700373 dtsOffset))
James Feist6714a252018-09-10 15:26:18 -0700374 {
Yoo, Jae Hyun81a464c2018-10-09 16:38:58 -0700375 std::cerr << "error populating thresholds for "
376 << sensorName << "\n";
James Feist6714a252018-09-10 15:26:18 -0700377 }
378 }
James Feistc140e202019-11-14 15:23:51 -0800379 auto& sensorPtr = gCpuSensors[sensorName];
380 // make sure destructor fires before creating a new one
381 sensorPtr = nullptr;
Thu Nguyen255da6b2022-07-29 10:05:52 +0700382 sensorPtr = std::make_shared<IntelCPUSensor>(
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700383 inputPathStr, sensorType, objectServer, dbusConnection, io,
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800384 sensorName, std::move(sensorThresholds), *interfacePath, cpuId,
Vijay Khemka86dea2b2019-06-06 11:14:37 -0700385 show, dtsOffset);
Arun P. Mohanan04d05062021-10-29 20:30:26 +0530386 sensorPtr->setupRead();
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700387 createdSensors.insert(sensorName);
Ed Tanous8a57ec02020-10-09 12:46:52 -0700388 if (debug)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700389 {
James Feist6714a252018-09-10 15:26:18 -0700390 std::cout << "Mapped: " << inputPath << " to " << sensorName
391 << "\n";
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700392 }
James Feist6714a252018-09-10 15:26:18 -0700393 }
394 }
Yoo, Jae Hyunf78d0a42018-10-10 11:03:11 -0700395
Ed Tanous2049bd22022-07-09 07:20:26 -0700396 if (static_cast<unsigned int>(!createdSensors.empty()) != 0U)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700397 {
398 std::cout << "Sensor" << (createdSensors.size() == 1 ? " is" : "s are")
399 << " created\n";
400 }
Yoo, Jae Hyuna441f3c2018-10-09 16:43:03 -0700401
Yoo, Jae Hyunf78d0a42018-10-10 11:03:11 -0700402 return true;
James Feist6714a252018-09-10 15:26:18 -0700403}
404
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700405void exportDevice(const CPUConfig& config)
James Feist6714a252018-09-10 15:26:18 -0700406{
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700407 std::ostringstream hex;
408 hex << std::hex << config.addr;
409 const std::string& addrHexStr = hex.str();
410 std::string busStr = std::to_string(config.bus);
411
412 std::string parameters = "peci-client 0x" + addrHexStr;
413 std::string device = "/sys/bus/peci/devices/peci-" + busStr + "/new_device";
414
James Feistcf3bce62019-01-08 10:07:19 -0800415 std::filesystem::path devicePath(device);
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700416 const std::string& dir = devicePath.parent_path().string();
James Feistcf3bce62019-01-08 10:07:19 -0800417 for (const auto& path : std::filesystem::directory_iterator(dir))
James Feist6714a252018-09-10 15:26:18 -0700418 {
James Feistcf3bce62019-01-08 10:07:19 -0800419 if (!std::filesystem::is_directory(path))
James Feist6714a252018-09-10 15:26:18 -0700420 {
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700421 continue;
James Feist6714a252018-09-10 15:26:18 -0700422 }
423
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700424 const std::string& directoryName = path.path().filename();
Zev Weiss6c106d62022-08-17 20:50:00 -0700425 if (directoryName.starts_with(busStr) &&
426 directoryName.ends_with(addrHexStr))
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700427 {
Ed Tanous8a57ec02020-10-09 12:46:52 -0700428 if (debug)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700429 {
430 std::cout << parameters << " on bus " << busStr
431 << " is already exported\n";
432 }
433 return;
434 }
James Feist6714a252018-09-10 15:26:18 -0700435 }
436
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700437 std::ofstream deviceFile(device);
438 if (!deviceFile.good())
James Feist6714a252018-09-10 15:26:18 -0700439 {
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700440 std::cerr << "Error writing " << device << "\n";
James Feist6714a252018-09-10 15:26:18 -0700441 return;
442 }
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700443 deviceFile << parameters;
444 deviceFile.close();
445
446 std::cout << parameters << " on bus " << busStr << " is exported\n";
James Feist6714a252018-09-10 15:26:18 -0700447}
448
Ed Tanous9b4a20e2022-09-06 08:47:11 -0700449void detectCpu(boost::asio::steady_timer& pingTimer,
450 boost::asio::steady_timer& creationTimer,
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800451 boost::asio::io_service& io,
452 sdbusplus::asio::object_server& objectServer,
453 std::shared_ptr<sdbusplus::asio::connection>& dbusConnection,
454 boost::container::flat_set<CPUConfig>& cpuConfigs,
455 ManagedObjectType& sensorConfigs)
James Feist6714a252018-09-10 15:26:18 -0700456{
James Feist6714a252018-09-10 15:26:18 -0700457 size_t rescanDelaySeconds = 0;
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700458 static bool keepPinging = false;
Jae Hyun Yoo9c55e6a2018-10-26 10:09:01 -0700459
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800460 for (CPUConfig& config : cpuConfigs)
James Feist6714a252018-09-10 15:26:18 -0700461 {
Jae Hyun Yoo9c55e6a2018-10-26 10:09:01 -0700462 std::string peciDevPath = peciDev + std::to_string(config.bus);
Ed Tanous99c44092022-01-14 09:59:09 -0800463
464 // NOLINTNEXTLINE(cppcoreguidelines-pro-type-vararg)
Jae Hyun Yoo9c55e6a2018-10-26 10:09:01 -0700465 auto file = open(peciDevPath.c_str(), O_RDWR | O_CLOEXEC);
466 if (file < 0)
467 {
468 std::cerr << "unable to open " << peciDevPath << "\n";
469 std::exit(EXIT_FAILURE);
470 }
471
Ed Tanousa771f6a2022-01-14 09:36:51 -0800472 State newState = State::OFF;
473 struct peci_ping_msg msg
474 {};
James Feist6714a252018-09-10 15:26:18 -0700475 msg.addr = config.addr;
Ed Tanous99c44092022-01-14 09:59:09 -0800476
477 // NOLINTNEXTLINE(cppcoreguidelines-pro-type-vararg)
Ed Tanous2049bd22022-07-09 07:20:26 -0700478 if (ioctl(file, PECI_IOC_PING, &msg) == 0)
James Feist6714a252018-09-10 15:26:18 -0700479 {
480 bool dimmReady = false;
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -0700481 for (unsigned int rank = 0; rank < rankNumMax; rank++)
James Feist6714a252018-09-10 15:26:18 -0700482 {
Ed Tanousa771f6a2022-01-14 09:36:51 -0800483 struct peci_rd_pkg_cfg_msg msg
484 {};
James Feist6714a252018-09-10 15:26:18 -0700485 msg.addr = config.addr;
Jae Hyun Yoo201c8d92019-02-27 15:41:56 -0800486 msg.index = PECI_MBX_INDEX_DDR_DIMM_TEMP;
James Feist6714a252018-09-10 15:26:18 -0700487 msg.param = rank;
488 msg.rx_len = 4;
Ed Tanous99c44092022-01-14 09:59:09 -0800489
490 // NOLINTNEXTLINE(cppcoreguidelines-pro-type-vararg)
Ed Tanous2049bd22022-07-09 07:20:26 -0700491 if (ioctl(file, PECI_IOC_RD_PKG_CFG, &msg) == 0)
James Feist6714a252018-09-10 15:26:18 -0700492 {
Ed Tanous2049bd22022-07-09 07:20:26 -0700493 if ((msg.pkg_config[0] != 0U) ||
494 (msg.pkg_config[1] != 0U) || (msg.pkg_config[2] != 0U))
James Feist6714a252018-09-10 15:26:18 -0700495 {
496 dimmReady = true;
497 break;
498 }
499 }
500 else
501 {
502 break;
503 }
504 }
Yoo, Jae Hyuna441f3c2018-10-09 16:43:03 -0700505
James Feist6714a252018-09-10 15:26:18 -0700506 if (dimmReady)
507 {
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700508 newState = State::READY;
James Feist6714a252018-09-10 15:26:18 -0700509 }
510 else
511 {
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700512 newState = State::ON;
James Feist6714a252018-09-10 15:26:18 -0700513 }
514 }
James Feist6714a252018-09-10 15:26:18 -0700515
Jae Hyun Yoo9c55e6a2018-10-26 10:09:01 -0700516 close(file);
517
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700518 if (config.state != newState)
James Feist6714a252018-09-10 15:26:18 -0700519 {
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700520 if (newState != State::OFF)
James Feist6714a252018-09-10 15:26:18 -0700521 {
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700522 if (config.state == State::OFF)
523 {
524 std::cout << config.name << " is detected\n";
525 exportDevice(config);
526 }
James Feist6714a252018-09-10 15:26:18 -0700527
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700528 if (newState == State::ON)
529 {
530 rescanDelaySeconds = 3;
531 }
532 else if (newState == State::READY)
533 {
534 rescanDelaySeconds = 5;
535 std::cout << "DIMM(s) on " << config.name
536 << " is/are detected\n";
537 }
Yoo, Jae Hyunf78d0a42018-10-10 11:03:11 -0700538 }
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700539
540 config.state = newState;
Yoo, Jae Hyunf78d0a42018-10-10 11:03:11 -0700541 }
542
543 if (config.state != State::READY)
James Feist6714a252018-09-10 15:26:18 -0700544 {
545 keepPinging = true;
546 }
547
Ed Tanous8a57ec02020-10-09 12:46:52 -0700548 if (debug)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700549 {
550 std::cout << config.name << ", state: " << config.state << "\n";
551 }
James Feist6714a252018-09-10 15:26:18 -0700552 }
553
Ed Tanous2049bd22022-07-09 07:20:26 -0700554 if (rescanDelaySeconds != 0U)
James Feist6714a252018-09-10 15:26:18 -0700555 {
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700556 creationTimer.expires_from_now(
Ed Tanous9b4a20e2022-09-06 08:47:11 -0700557 std::chrono::seconds(rescanDelaySeconds));
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700558 creationTimer.async_wait([&](const boost::system::error_code& ec) {
559 if (ec == boost::asio::error::operation_aborted)
560 {
561 return; // we're being canceled
562 }
563
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800564 if (!createSensors(io, objectServer, dbusConnection, cpuConfigs,
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700565 sensorConfigs) ||
566 keepPinging)
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700567 {
568 detectCpuAsync(pingTimer, creationTimer, io, objectServer,
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800569 dbusConnection, cpuConfigs, sensorConfigs);
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700570 }
571 });
James Feist6714a252018-09-10 15:26:18 -0700572 }
Jae Hyun Yoo18ae22f2019-04-02 10:11:30 -0700573 else if (keepPinging)
James Feist6714a252018-09-10 15:26:18 -0700574 {
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800575 detectCpuAsync(pingTimer, creationTimer, io, objectServer,
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800576 dbusConnection, cpuConfigs, sensorConfigs);
James Feist6714a252018-09-10 15:26:18 -0700577 }
578}
579
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700580void detectCpuAsync(
Ed Tanous9b4a20e2022-09-06 08:47:11 -0700581 boost::asio::steady_timer& pingTimer,
582 boost::asio::steady_timer& creationTimer, boost::asio::io_service& io,
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700583 sdbusplus::asio::object_server& objectServer,
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800584 std::shared_ptr<sdbusplus::asio::connection>& dbusConnection,
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800585 boost::container::flat_set<CPUConfig>& cpuConfigs,
586 ManagedObjectType& sensorConfigs)
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700587{
Ed Tanous9b4a20e2022-09-06 08:47:11 -0700588 pingTimer.expires_from_now(std::chrono::seconds(1));
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700589 pingTimer.async_wait([&](const boost::system::error_code& ec) {
590 if (ec == boost::asio::error::operation_aborted)
591 {
592 return; // we're being canceled
593 }
594
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800595 detectCpu(pingTimer, creationTimer, io, objectServer, dbusConnection,
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800596 cpuConfigs, sensorConfigs);
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700597 });
598}
599
James Feistc140e202019-11-14 15:23:51 -0800600bool getCpuConfig(const std::shared_ptr<sdbusplus::asio::connection>& systemBus,
601 boost::container::flat_set<CPUConfig>& cpuConfigs,
602 ManagedObjectType& sensorConfigs,
603 sdbusplus::asio::object_server& objectServer)
James Feist6714a252018-09-10 15:26:18 -0700604{
James Feist6714a252018-09-10 15:26:18 -0700605 bool useCache = false;
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800606 sensorConfigs.clear();
James Feist6714a252018-09-10 15:26:18 -0700607 // use new data the first time, then refresh
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -0700608 for (const char* type : sensorTypes)
James Feist6714a252018-09-10 15:26:18 -0700609 {
Zev Weiss26fb1492022-08-17 15:33:46 -0700610 if (!getSensorConfiguration(type, systemBus, sensorConfigs, useCache))
James Feist6714a252018-09-10 15:26:18 -0700611 {
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700612 return false;
James Feist6714a252018-09-10 15:26:18 -0700613 }
614 useCache = true;
615 }
616
Jae Hyun Yooffa07e22019-07-17 10:47:18 -0700617 // check PECI client addresses and names from CPU configuration
James Feist6714a252018-09-10 15:26:18 -0700618 // before starting ping operation
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -0700619 for (const char* type : sensorTypes)
James Feist6714a252018-09-10 15:26:18 -0700620 {
Zev Weiss8908b3c2022-08-12 18:21:01 -0700621 for (const auto& [path, cfgData] : sensorConfigs)
James Feist6714a252018-09-10 15:26:18 -0700622 {
Zev Weiss8908b3c2022-08-12 18:21:01 -0700623 for (const auto& [intf, cfg] : cfgData)
James Feist6714a252018-09-10 15:26:18 -0700624 {
Zev Weiss054aad82022-08-18 01:37:34 -0700625 if (intf != configInterfaceName(type))
James Feist6714a252018-09-10 15:26:18 -0700626 {
627 continue;
628 }
629
Zev Weiss8908b3c2022-08-12 18:21:01 -0700630 auto findName = cfg.find("Name");
631 if (findName == cfg.end())
James Feist6714a252018-09-10 15:26:18 -0700632 {
633 continue;
634 }
James Feist3eb82622019-02-08 13:10:22 -0800635 std::string nameRaw =
636 std::visit(VariantToStringVisitor(), findName->second);
James Feist6714a252018-09-10 15:26:18 -0700637 std::string name =
Jae Hyun Yoo9ced0a32018-10-25 10:42:39 -0700638 std::regex_replace(nameRaw, illegalDbusRegex, "_");
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700639
James Feistc140e202019-11-14 15:23:51 -0800640 auto present = std::optional<bool>();
641 // if we can't detect it via gpio, we set presence later
Zev Weiss8908b3c2022-08-12 18:21:01 -0700642 for (const auto& [suppIntf, suppCfg] : cfgData)
James Feist58295ad2019-05-30 15:01:41 -0700643 {
Zev Weiss8908b3c2022-08-12 18:21:01 -0700644 if (suppIntf.find("PresenceGpio") != std::string::npos)
James Feist58295ad2019-05-30 15:01:41 -0700645 {
Zev Weiss8908b3c2022-08-12 18:21:01 -0700646 present = cpuIsPresent(suppCfg);
Jae Hyun Yooffa07e22019-07-17 10:47:18 -0700647 break;
James Feist58295ad2019-05-30 15:01:41 -0700648 }
Jae Hyun Yooffa07e22019-07-17 10:47:18 -0700649 }
650
James Feistc140e202019-11-14 15:23:51 -0800651 if (inventoryIfaces.find(name) == inventoryIfaces.end() &&
652 present)
Jae Hyun Yooffa07e22019-07-17 10:47:18 -0700653 {
654 auto iface = objectServer.add_interface(
655 cpuInventoryPath + std::string("/") + name,
656 "xyz.openbmc_project.Inventory.Item");
657 iface->register_property("PrettyName", name);
James Feistc140e202019-11-14 15:23:51 -0800658 iface->register_property("Present", *present);
Jae Hyun Yooffa07e22019-07-17 10:47:18 -0700659 iface->initialize();
660 inventoryIfaces[name] = std::move(iface);
661 }
James Feist58295ad2019-05-30 15:01:41 -0700662
Zev Weiss8908b3c2022-08-12 18:21:01 -0700663 auto findBus = cfg.find("Bus");
664 if (findBus == cfg.end())
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700665 {
666 std::cerr << "Can't find 'Bus' setting in " << name << "\n";
667 continue;
668 }
James Feist3eb82622019-02-08 13:10:22 -0800669 uint64_t bus =
670 std::visit(VariantToUnsignedIntVisitor(), findBus->second);
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700671
Zev Weiss8908b3c2022-08-12 18:21:01 -0700672 auto findAddress = cfg.find("Address");
673 if (findAddress == cfg.end())
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700674 {
675 std::cerr << "Can't find 'Address' setting in " << name
676 << "\n";
677 continue;
678 }
James Feist3eb82622019-02-08 13:10:22 -0800679 uint64_t addr = std::visit(VariantToUnsignedIntVisitor(),
680 findAddress->second);
James Feist6714a252018-09-10 15:26:18 -0700681
Ed Tanous8a57ec02020-10-09 12:46:52 -0700682 if (debug)
James Feist6714a252018-09-10 15:26:18 -0700683 {
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700684 std::cout << "bus: " << bus << "\n";
James Feist6714a252018-09-10 15:26:18 -0700685 std::cout << "addr: " << addr << "\n";
686 std::cout << "name: " << name << "\n";
687 std::cout << "type: " << type << "\n";
James Feist6714a252018-09-10 15:26:18 -0700688 }
689
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800690 cpuConfigs.emplace(bus, addr, name, State::OFF);
James Feist6714a252018-09-10 15:26:18 -0700691 }
692 }
693 }
Yoo, Jae Hyuna441f3c2018-10-09 16:43:03 -0700694
Ed Tanous2049bd22022-07-09 07:20:26 -0700695 if (static_cast<unsigned int>(!cpuConfigs.empty()) != 0U)
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700696 {
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800697 std::cout << "CPU config" << (cpuConfigs.size() == 1 ? " is" : "s are")
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700698 << " parsed\n";
699 return true;
700 }
701
702 return false;
James Feist6714a252018-09-10 15:26:18 -0700703}
704
James Feistb6c0b912019-07-09 12:21:44 -0700705int main()
James Feist6714a252018-09-10 15:26:18 -0700706{
707 boost::asio::io_service io;
708 auto systemBus = std::make_shared<sdbusplus::asio::connection>(io);
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800709 boost::container::flat_set<CPUConfig> cpuConfigs;
James Feist6714a252018-09-10 15:26:18 -0700710
James Feist6714a252018-09-10 15:26:18 -0700711 sdbusplus::asio::object_server objectServer(systemBus);
Ed Tanous9b4a20e2022-09-06 08:47:11 -0700712 boost::asio::steady_timer pingTimer(io);
713 boost::asio::steady_timer creationTimer(io);
714 boost::asio::steady_timer filterTimer(io);
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800715 ManagedObjectType sensorConfigs;
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700716
Ed Tanous9b4a20e2022-09-06 08:47:11 -0700717 filterTimer.expires_from_now(std::chrono::seconds(1));
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700718 filterTimer.async_wait([&](const boost::system::error_code& ec) {
719 if (ec == boost::asio::error::operation_aborted)
720 {
Jae Hyun Yood64262b2018-11-01 13:31:16 -0700721 return; // we're being canceled
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700722 }
723
James Feistc140e202019-11-14 15:23:51 -0800724 if (getCpuConfig(systemBus, cpuConfigs, sensorConfigs, objectServer))
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700725 {
Jae Hyun Yooe8b60d02019-01-14 15:27:13 -0800726 detectCpuAsync(pingTimer, creationTimer, io, objectServer,
Jae Hyun Yoo73ca5512019-02-28 21:20:17 -0800727 systemBus, cpuConfigs, sensorConfigs);
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700728 }
729 });
730
Patrick Williams92f8f512022-07-22 19:26:55 -0500731 std::function<void(sdbusplus::message_t&)> eventHandler =
732 [&](sdbusplus::message_t& message) {
Ed Tanousbb679322022-05-16 16:10:00 -0700733 if (message.is_method_error())
734 {
735 std::cerr << "callback method error\n";
736 return;
737 }
738
739 if (debug)
740 {
741 std::cout << message.get_path() << " is changed\n";
742 }
743
744 // this implicitly cancels the timer
Ed Tanous9b4a20e2022-09-06 08:47:11 -0700745 filterTimer.expires_from_now(std::chrono::seconds(1));
Ed Tanousbb679322022-05-16 16:10:00 -0700746 filterTimer.async_wait([&](const boost::system::error_code& ec) {
747 if (ec == boost::asio::error::operation_aborted)
James Feist6714a252018-09-10 15:26:18 -0700748 {
Ed Tanousbb679322022-05-16 16:10:00 -0700749 return; // we're being canceled
James Feist6714a252018-09-10 15:26:18 -0700750 }
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700751
Ed Tanousbb679322022-05-16 16:10:00 -0700752 if (getCpuConfig(systemBus, cpuConfigs, sensorConfigs,
753 objectServer))
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700754 {
Ed Tanousbb679322022-05-16 16:10:00 -0700755 detectCpuAsync(pingTimer, creationTimer, io, objectServer,
756 systemBus, cpuConfigs, sensorConfigs);
Jae Hyun Yoo8d9886d2018-10-22 15:24:29 -0700757 }
Ed Tanousbb679322022-05-16 16:10:00 -0700758 });
759 };
James Feist6714a252018-09-10 15:26:18 -0700760
Zev Weiss214d9712022-08-12 12:54:31 -0700761 std::vector<std::unique_ptr<sdbusplus::bus::match_t>> matches =
762 setupPropertiesChangedMatches(*systemBus, sensorTypes, eventHandler);
James Feist6714a252018-09-10 15:26:18 -0700763
Thu Nguyen255da6b2022-07-29 10:05:52 +0700764 systemBus->request_name("xyz.openbmc_project.IntelCPUSensor");
Bruce Lee1263c3d2021-06-04 15:16:33 +0800765
766 setupManufacturingModeMatch(*systemBus);
James Feist6714a252018-09-10 15:26:18 -0700767 io.run();
Zhikui Ren8685b172021-06-29 15:16:52 -0700768 return 0;
James Feist6714a252018-09-10 15:26:18 -0700769}