| /** |
| * Copyright © 2024 IBM Corporation |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "pmbus_driver_device.hpp" |
| |
| #include <ctype.h> // for tolower() |
| |
| #include <algorithm> |
| #include <exception> |
| #include <filesystem> |
| #include <format> |
| #include <regex> |
| #include <stdexcept> |
| |
| namespace phosphor::power::sequencer |
| { |
| |
| using namespace pmbus; |
| namespace fs = std::filesystem; |
| |
| std::vector<int> PMBusDriverDevice::getGPIOValues(Services& services) |
| { |
| // Get lower case version of device name to use as chip label |
| std::string label{name}; |
| std::transform(label.begin(), label.end(), label.begin(), ::tolower); |
| |
| // Read the GPIO values by specifying the chip label |
| std::vector<int> values; |
| try |
| { |
| values = services.getGPIOValues(label); |
| } |
| catch (const std::exception& e) |
| { |
| throw std::runtime_error{std::format( |
| "Unable to read GPIO values from device {} using label {}: {}", |
| name, label, e.what())}; |
| } |
| return values; |
| } |
| |
| uint16_t PMBusDriverDevice::getStatusWord(uint8_t page) |
| { |
| uint16_t value{0}; |
| try |
| { |
| std::string fileName = std::format("status{:d}", page); |
| value = pmbusInterface->read(fileName, Type::Debug); |
| } |
| catch (const std::exception& e) |
| { |
| throw std::runtime_error{std::format( |
| "Unable to read STATUS_WORD for PAGE {:d} of device {}: {}", page, |
| name, e.what())}; |
| } |
| return value; |
| } |
| |
| uint8_t PMBusDriverDevice::getStatusVout(uint8_t page) |
| { |
| uint8_t value{0}; |
| try |
| { |
| std::string fileName = std::format("status{:d}_vout", page); |
| value = pmbusInterface->read(fileName, Type::Debug); |
| } |
| catch (const std::exception& e) |
| { |
| throw std::runtime_error{std::format( |
| "Unable to read STATUS_VOUT for PAGE {:d} of device {}: {}", page, |
| name, e.what())}; |
| } |
| return value; |
| } |
| |
| double PMBusDriverDevice::getReadVout(uint8_t page) |
| { |
| double volts{0.0}; |
| try |
| { |
| unsigned int fileNumber = getFileNumber(page); |
| std::string fileName = std::format("in{}_input", fileNumber); |
| std::string millivoltsStr = |
| pmbusInterface->readString(fileName, Type::Hwmon); |
| unsigned long millivolts = std::stoul(millivoltsStr); |
| volts = millivolts / 1000.0; |
| } |
| catch (const std::exception& e) |
| { |
| throw std::runtime_error{std::format( |
| "Unable to read READ_VOUT for PAGE {:d} of device {}: {}", page, |
| name, e.what())}; |
| } |
| return volts; |
| } |
| |
| double PMBusDriverDevice::getVoutUVFaultLimit(uint8_t page) |
| { |
| double volts{0.0}; |
| try |
| { |
| unsigned int fileNumber = getFileNumber(page); |
| std::string fileName = std::format("in{}_lcrit", fileNumber); |
| std::string millivoltsStr = |
| pmbusInterface->readString(fileName, Type::Hwmon); |
| unsigned long millivolts = std::stoul(millivoltsStr); |
| volts = millivolts / 1000.0; |
| } |
| catch (const std::exception& e) |
| { |
| throw std::runtime_error{std::format( |
| "Unable to read VOUT_UV_FAULT_LIMIT for PAGE {:d} of device {}: {}", |
| page, name, e.what())}; |
| } |
| return volts; |
| } |
| |
| unsigned int PMBusDriverDevice::getFileNumber(uint8_t page) |
| { |
| if (pageToFileNumber.empty()) |
| { |
| buildPageToFileNumberMap(); |
| } |
| |
| auto it = pageToFileNumber.find(page); |
| if (it == pageToFileNumber.end()) |
| { |
| throw std::runtime_error{std::format( |
| "Unable to find hwmon file number for PAGE {:d} of device {}", page, |
| name)}; |
| } |
| |
| return it->second; |
| } |
| |
| void PMBusDriverDevice::buildPageToFileNumberMap() |
| { |
| // Clear any existing mappings |
| pageToFileNumber.clear(); |
| |
| // Build mappings using voltage label files in hwmon directory |
| try |
| { |
| fs::path hwmonDir = pmbusInterface->getPath(Type::Hwmon); |
| if (fs::is_directory(hwmonDir)) |
| { |
| // Loop through all files in hwmon directory |
| std::string fileName; |
| unsigned int fileNumber; |
| std::optional<uint8_t> page; |
| for (const auto& f : fs::directory_iterator{hwmonDir}) |
| { |
| // If this is a voltage label file |
| fileName = f.path().filename().string(); |
| if (isLabelFile(fileName, fileNumber)) |
| { |
| // Read PMBus PAGE number from label file contents |
| page = readPageFromLabelFile(fileName); |
| if (page) |
| { |
| // Add mapping from PAGE number to file number |
| pageToFileNumber.emplace(*page, fileNumber); |
| } |
| } |
| } |
| } |
| } |
| catch (const std::exception& e) |
| { |
| throw std::runtime_error{ |
| std::format("Unable to map PMBus PAGE numbers to hwmon file " |
| "numbers for device {}: {}", |
| name, e.what())}; |
| } |
| } |
| |
| bool PMBusDriverDevice::isLabelFile(const std::string& fileName, |
| unsigned int& fileNumber) |
| { |
| bool isLabel{false}; |
| try |
| { |
| // Check if file name has expected pattern for voltage label file |
| std::regex regex{"in(\\d+)_label"}; |
| std::smatch results; |
| if (std::regex_match(fileName, results, regex)) |
| { |
| // Verify 2 match results: entire match and one sub-match |
| if (results.size() == 2) |
| { |
| // Get sub-match that contains the file number |
| std::string fileNumberStr = results.str(1); |
| fileNumber = std::stoul(fileNumberStr); |
| isLabel = true; |
| } |
| } |
| } |
| catch (...) |
| { |
| // Ignore error. If this file is needed for pgood fault detection, an |
| // error will occur later when the necessary mapping is missing. Avoid |
| // logging unnecessary errors for files that may not be required. |
| } |
| return isLabel; |
| } |
| |
| std::optional<uint8_t> |
| PMBusDriverDevice::readPageFromLabelFile(const std::string& fileName) |
| { |
| std::optional<uint8_t> page; |
| try |
| { |
| // Read voltage label file contents |
| std::string contents = |
| pmbusInterface->readString(fileName, Type::Hwmon); |
| |
| // Check if file contents match the expected pattern |
| std::regex regex{"vout(\\d+)"}; |
| std::smatch results; |
| if (std::regex_match(contents, results, regex)) |
| { |
| // Verify 2 match results: entire match and one sub-match |
| if (results.size() == 2) |
| { |
| // Get sub-match that contains the page number + 1 |
| std::string pageStr = results.str(1); |
| page = std::stoul(pageStr) - 1; |
| } |
| } |
| } |
| catch (...) |
| { |
| // Ignore error. If this file is needed for pgood fault detection, an |
| // error will occur later when the necessary mapping is missing. Avoid |
| // logging unnecessary errors for files that may not be required. |
| } |
| return page; |
| } |
| |
| } // namespace phosphor::power::sequencer |