xref: /openbmc/phosphor-fan-presence/control/json/utils/pcie_card_metadata.cpp (revision 3d0c36967c7bdbf695f4765045274c01cd2e3c93)
1  /**
2   * Copyright © 2022 IBM 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 "pcie_card_metadata.hpp"
18  
19  #include "json_config.hpp"
20  #include "utils/flight_recorder.hpp"
21  
22  #include <format>
23  #include <iostream>
24  
25  static constexpr auto cardFileName = "pcie_cards.json";
26  
27  namespace phosphor::fan::control::json
28  {
29  
30  namespace fs = std::filesystem;
31  using namespace phosphor::fan;
32  
PCIeCardMetadata(const std::vector<std::string> & systemNames)33  PCIeCardMetadata::PCIeCardMetadata(const std::vector<std::string>& systemNames)
34  {
35      loadCards(systemNames);
36  }
37  
loadCards(const std::vector<std::string> & systemNames)38  void PCIeCardMetadata::loadCards(const std::vector<std::string>& systemNames)
39  {
40      const auto defaultPath = fs::path{"control"} / cardFileName;
41  
42      // First look in /etc/phosphor-fan-presence/control/
43      auto confFile = fs::path{confOverridePath} / defaultPath;
44  
45      // Next look in the system subdirectories, first under /etc then /usr.
46      if (!fs::exists(confFile))
47      {
48          for (const auto& systemName : systemNames)
49          {
50              const auto basePath =
51                  fs::path{"control"} / systemName / cardFileName;
52  
53              // Look in the /etc override location first
54              confFile = fs::path{confOverridePath} / basePath;
55  
56              if (!fs::exists(confFile))
57              {
58                  confFile = fs::path{confBasePath} / basePath;
59              }
60  
61              if (fs::exists(confFile))
62              {
63                  break;
64              }
65          }
66      }
67  
68      // Next look in /usr/share/phosphor-fan-presence/control/
69      if (!fs::exists(confFile))
70      {
71          confFile = fs::path{confBasePath} / defaultPath;
72      }
73  
74      if (fs::exists(confFile))
75      {
76          FlightRecorder::instance().log(
77              "main",
78              std::format("Loading configuration from {}", confFile.string()));
79          load(JsonConfig::load(confFile));
80          FlightRecorder::instance().log(
81              "main", std::format("Configuration({}) loaded successfully",
82                                  confFile.string()));
83          log<level::INFO>(std::format("Configuration({}) loaded successfully",
84                                       confFile.string())
85                               .c_str());
86      }
87  }
88  
load(const nlohmann::json & json)89  void PCIeCardMetadata::load(const nlohmann::json& json)
90  {
91      if (!json.contains("cards") || !json.at("cards").is_array())
92      {
93          throw std::runtime_error{
94              std::format("Missing 'cards' array in PCIe card JSON")};
95      }
96  
97      for (const auto& card : json.at("cards"))
98      {
99          if (!card.contains("vendor_id") || !card.contains("device_id") ||
100              !card.contains("subsystem_vendor_id") ||
101              !card.contains("subsystem_id") ||
102              !(card.contains("has_temp_sensor") || card.contains("floor_index")))
103          {
104              throw std::runtime_error{"Invalid PCIe card json"};
105          }
106  
107          Metadata data;
108          data.vendorID =
109              std::stoul(card.at("vendor_id").get<std::string>(), nullptr, 16);
110          data.deviceID =
111              std::stoul(card.at("device_id").get<std::string>(), nullptr, 16);
112          data.subsystemVendorID = std::stoul(
113              card.at("subsystem_vendor_id").get<std::string>(), nullptr, 16);
114          data.subsystemID =
115              std::stoul(card.at("subsystem_id").get<std::string>(), nullptr, 16);
116  
117          data.hasTempSensor = card.value("has_temp_sensor", false);
118          data.floorIndex = card.value("floor_index", -1);
119  
120          auto iter = std::find(_cards.begin(), _cards.end(), data);
121          if (iter != _cards.end())
122          {
123              iter->vendorID = data.vendorID;
124              iter->deviceID = data.deviceID;
125              iter->subsystemVendorID = data.subsystemVendorID;
126              iter->subsystemID = data.subsystemID;
127              iter->floorIndex = data.floorIndex;
128              iter->hasTempSensor = data.hasTempSensor;
129          }
130          else
131          {
132              _cards.push_back(std::move(data));
133          }
134      }
135  }
136  
dump() const137  void PCIeCardMetadata::dump() const
138  {
139      for (const auto& entry : _cards)
140      {
141          std::cerr << "--------------------------------------------------"
142                    << "\n";
143          std::cerr << "vendorID: " << std::hex << entry.vendorID << "\n";
144          std::cerr << "deviceID: " << entry.deviceID << "\n";
145          std::cerr << "subsysVendorID: " << entry.subsystemVendorID << "\n";
146          std::cerr << "subsystemID: " << entry.subsystemID << "\n";
147          std::cerr << "hasTempSensor: " << std::dec << entry.hasTempSensor
148                    << "\n";
149          std::cerr << "floorIndex: " << entry.floorIndex << "\n";
150      }
151  }
152  
lookup(uint16_t deviceID,uint16_t vendorID,uint16_t subsystemID,uint16_t subsystemVendorID) const153  std::optional<std::variant<int32_t, bool>> PCIeCardMetadata::lookup(
154      uint16_t deviceID, uint16_t vendorID, uint16_t subsystemID,
155      uint16_t subsystemVendorID) const
156  {
157      log<level::DEBUG>(std::format("Lookup {:#x} ${:#x} {:#x} {:#x}", deviceID,
158                                    vendorID, subsystemID, subsystemVendorID)
159                            .c_str());
160      auto card = std::find_if(
161          _cards.begin(), _cards.end(),
162          [&deviceID, &vendorID, &subsystemID,
163           &subsystemVendorID](const auto& card) {
164              return (deviceID == card.deviceID) && (vendorID == card.vendorID) &&
165                     (subsystemID == card.subsystemID) &&
166                     (subsystemVendorID == card.subsystemVendorID);
167          });
168  
169      if (card != _cards.end())
170      {
171          if (card->hasTempSensor)
172          {
173              return true;
174          }
175          return card->floorIndex;
176      }
177      return std::nullopt;
178  }
179  
180  } // namespace phosphor::fan::control::json
181