blob: 6fd8fd11411bd134e35ea5ae0a2b628310ac31e0 [file] [log] [blame]
/**
* Copyright © 2022 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 "pcie_card_metadata.hpp"
#include "json_config.hpp"
#include "utils/flight_recorder.hpp"
#include <fmt/format.h>
#include <iostream>
static constexpr auto cardFileName = "pcie_cards.json";
namespace phosphor::fan::control::json
{
namespace fs = std::filesystem;
using namespace phosphor::fan;
PCIeCardMetadata::PCIeCardMetadata(const std::vector<std::string>& systemNames)
{
loadCards(systemNames);
}
void PCIeCardMetadata::loadCards(const std::vector<std::string>& systemNames)
{
const auto defaultPath = fs::path{"control"} / cardFileName;
// Look in the override location first
auto confFile = fs::path{confOverridePath} / defaultPath;
if (!fs::exists(confFile))
{
confFile = fs::path{confBasePath} / defaultPath;
}
if (fs::exists(confFile))
{
FlightRecorder::instance().log(
"main",
fmt::format("Loading configuration from {}", confFile.string()));
load(JsonConfig::load(confFile));
FlightRecorder::instance().log(
"main", fmt::format("Configuration({}) loaded successfully",
confFile.string()));
log<level::INFO>(fmt::format("Configuration({}) loaded successfully",
confFile.string())
.c_str());
}
// Go from least specific to most specific in the system names so files in
// the latter category can override ones in the former.
for (auto nameIt = systemNames.rbegin(); nameIt != systemNames.rend();
++nameIt)
{
const auto basePath = fs::path{"control"} / *nameIt / cardFileName;
// Look in the override location first
auto confFile = fs::path{confOverridePath} / basePath;
if (!fs::exists(confFile))
{
confFile = fs::path{confBasePath} / basePath;
}
if (fs::exists(confFile))
{
FlightRecorder::instance().log(
"main", fmt::format("Loading configuration from {}",
confFile.string()));
load(JsonConfig::load(confFile));
FlightRecorder::instance().log(
"main", fmt::format("Configuration({}) loaded successfully",
confFile.string()));
log<level::INFO>(
fmt::format("Configuration({}) loaded successfully",
confFile.string())
.c_str());
}
}
if (_cards.empty())
{
throw std::runtime_error{
"No valid PCIe card entries found in any JSON"};
}
}
void PCIeCardMetadata::load(const nlohmann::json& json)
{
if (!json.contains("cards") || !json.at("cards").is_array())
{
throw std::runtime_error{
fmt::format("Missing 'cards' array in PCIe card JSON")};
}
for (const auto& card : json.at("cards"))
{
if (!card.contains("vendor_id") || !card.contains("device_id") ||
!card.contains("subsystem_vendor_id") ||
!card.contains("subsystem_id") ||
!(card.contains("has_temp_sensor") || card.contains("floor_index")))
{
throw std::runtime_error{"Invalid PCIe card json"};
}
Metadata data;
data.vendorID = std::stoul(card.at("vendor_id").get<std::string>(),
nullptr, 16);
data.deviceID = std::stoul(card.at("device_id").get<std::string>(),
nullptr, 16);
data.subsystemVendorID = std::stoul(
card.at("subsystem_vendor_id").get<std::string>(), nullptr, 16);
data.subsystemID =
std::stoul(card.at("subsystem_id").get<std::string>(), nullptr, 16);
data.hasTempSensor = card.value("has_temp_sensor", false);
data.floorIndex = card.value("floor_index", -1);
auto iter = std::find(_cards.begin(), _cards.end(), data);
if (iter != _cards.end())
{
iter->vendorID = data.vendorID;
iter->deviceID = data.deviceID;
iter->subsystemVendorID = data.subsystemVendorID;
iter->subsystemID = data.subsystemID;
iter->floorIndex = data.floorIndex;
iter->hasTempSensor = data.hasTempSensor;
}
else
{
_cards.push_back(std::move(data));
}
}
}
void PCIeCardMetadata::dump() const
{
for (const auto& entry : _cards)
{
std::cerr << "--------------------------------------------------"
<< "\n";
std::cerr << "vendorID: " << std::hex << entry.vendorID << "\n";
std::cerr << "deviceID: " << entry.deviceID << "\n";
std::cerr << "subsysVendorID: " << entry.subsystemVendorID << "\n";
std::cerr << "subsystemID: " << entry.subsystemID << "\n";
std::cerr << "hasTempSensor: " << std::dec << entry.hasTempSensor
<< "\n";
std::cerr << "floorIndex: " << entry.floorIndex << "\n";
}
}
std::optional<std::variant<int32_t, bool>>
PCIeCardMetadata::lookup(uint16_t deviceID, uint16_t vendorID,
uint16_t subsystemID,
uint16_t subsystemVendorID) const
{
log<level::DEBUG>(fmt::format("Lookup {:#x} ${:#x} {:#x} {:#x}", deviceID,
vendorID, subsystemID, subsystemVendorID)
.c_str());
auto card = std::find_if(_cards.begin(), _cards.end(),
[&deviceID, &vendorID, &subsystemID,
&subsystemVendorID](const auto& card) {
return (deviceID == card.deviceID) && (vendorID == card.vendorID) &&
(subsystemID == card.subsystemID) &&
(subsystemVendorID == card.subsystemVendorID);
});
if (card != _cards.end())
{
if (card->hasTempSensor)
{
return true;
}
return card->floorIndex;
}
return std::nullopt;
}
} // namespace phosphor::fan::control::json