uhd/host/lib/usrp/mpmd/mpmd_prop_tree.cpp
Martin Braun be2e33ced0 mpmd: Add device DNA to prop tree if available
If an MPM device's info includes the key 'device_dna', then there will
now be a property tree path at /mboards/$i/device_dna (where $i is the
motherboard number).
2024-06-11 10:20:07 +02:00

226 lines
9.8 KiB
C++

//
// Copyright 2018 Ettus Research, a National Instruments Company
// Copyright 2019 Ettus Research, a National Instruments Brand
//
// SPDX-License-Identifier: GPL-3.0-or-later
//
// property tree initialization code
#include "mpmd_impl.hpp"
#include <uhd/property_tree.hpp>
#include <uhd/types/component_file.hpp>
#include <uhd/types/eeprom.hpp>
#include <uhd/types/sensors.hpp>
#include <uhd/usrp/mboard_eeprom.hpp>
#include <uhd/utils/cast.hpp>
#include <boost/algorithm/string/case_conv.hpp>
using namespace uhd;
using namespace uhd::mpmd;
namespace {
/*! Update a component using all required files. For example, when updating the FPGA image
* (.bit or .bin), users can provide a new overlay image (DTS) to apply in addition.
*
* \param comps Vector of component files to be updated
* \param mb Reference to the actual device
*/
uhd::usrp::component_files_t _update_component(
const uhd::usrp::component_files_t& comps, mpmd_mboard_impl* mb)
{
// Construct the arguments to update component
std::vector<std::vector<uint8_t>> all_data;
std::vector<std::map<std::string, std::string>> all_metadata;
// Also construct a copy of just the metadata to store in the property tree
uhd::usrp::component_files_t all_comps_copy;
bool just_reload = false;
for (const auto& comp : comps) {
// Make a map for update components args
std::map<std::string, std::string> metadata;
// Make a component copy to add to the property tree
uhd::usrp::component_file_t comp_copy;
// Copy the metadata
for (const auto& key : comp.metadata.keys()) {
metadata[key] = comp.metadata[key];
comp_copy.metadata[key] = comp.metadata[key];
if (key == "just_reload") {
just_reload = just_reload
| uhd::cast::from_str<bool>(metadata.at("just_reload"));
//| (boost::to_lower_copy(metadata.at("just_reload")) == "true");
}
}
// Copy to the update component args
all_data.push_back(comp.data);
all_metadata.push_back(metadata);
// Copy to the property tree
all_comps_copy.push_back(comp_copy);
}
// If reset is specified we presume that the fpga/dts
// components were updated in the last uhd::image_loader::load()
// call and just reload the fpga/dts by resetting the peripheral
// manager.
if (just_reload) {
mb->rpc->notify_with_token(MPMD_DEFAULT_INIT_TIMEOUT, "reset_timer_and_mgr");
return all_comps_copy;
}
// Now call update component
mb->rpc->notify_with_token(
MPMD_DEFAULT_INIT_TIMEOUT, "update_component", all_metadata, all_data);
return all_comps_copy;
}
/*
* Query the device to get the metadata for desired component
*
* \param comp_name String component name
* \param mb Reference to the actual device
* \return component files containing the component metadata
*/
uhd::usrp::component_files_t _get_component_info(
const std::string& comp_name, mpmd_mboard_impl* mb)
{
UHD_LOG_TRACE("MPMD", "Getting component info for " << comp_name);
const auto component_metadata = mb->rpc->request<std::map<std::string, std::string>>(
"get_component_info", comp_name);
// Copy the contents of the component metadata into a object we can return
uhd::usrp::component_file_t return_component;
auto& return_metadata = return_component.metadata;
for (auto item : component_metadata) {
return_metadata[item.first] = item.second;
}
return uhd::usrp::component_files_t{return_component};
}
} // namespace
void mpmd_impl::init_property_tree(
uhd::property_tree::sptr tree, fs_path mb_path, mpmd_mboard_impl* mb)
{
/*** Device info ****************************************************/
if (not tree->exists("/name")) {
tree->create<std::string>("/name").set(
mb->device_info.get("description", "Unknown MPM device"));
}
tree->create<std::string>(mb_path / "name")
.set(mb->device_info.get("name", "UNKNOWN"));
tree->create<std::string>(mb_path / "serial")
.set(mb->device_info.get("serial", "n/a"));
tree->create<std::string>(mb_path / "connection")
.set(mb->device_info.get("connection", "UNKNOWN"));
tree->create<size_t>(mb_path / "link_max_rate").set(125000000);
tree->create<std::string>(mb_path / "mpm_version")
.set(mb->device_info.get("mpm_version", "UNKNOWN"));
tree->create<std::string>(mb_path / "fpga_version")
.set(mb->device_info.get("fpga_version", "UNKNOWN"));
tree->create<std::string>(mb_path / "fpga_version_hash")
.set(mb->device_info.get("fpga_version_hash", "UNKNOWN"));
tree->create<std::string>(mb_path / "token").set(mb->get_token());
tree->create<uhd::device_addr_t>(mb_path / "args").set(mb->mb_args);
tree->create<std::string>(mb_path / "mender_artifact")
.set(mb->device_info.get("mender_artifact", "UNKNOWN"));
tree->create<std::string>(mb_path / "mpm_sw_version")
.set(mb->device_info.get("mpm_sw_version", "UNKNOWN"));
tree->create<std::string>(mb_path / "fs_version")
.set(mb->device_info.get("fs_version", "UNKNOWN"));
if (mb->device_info.has_key("device_dna")) {
tree->create<std::string>(mb_path / "device_dna")
.set(mb->device_info.get("device_dna", "UNKNOWN"));
}
/*** Clocking *******************************************************/
tree->create<std::string>(mb_path / "clock_source/value")
.add_coerced_subscriber([mb](const std::string& clock_source) {
mb->rpc->notify_with_token(
MPMD_DEFAULT_INIT_TIMEOUT, "set_clock_source", clock_source);
})
.set_publisher([mb]() {
return mb->rpc->request_with_token<std::string>("get_clock_source");
});
tree->create<std::vector<std::string>>(mb_path / "clock_source/options")
.set_publisher([mb]() {
return mb->rpc->request_with_token<std::vector<std::string>>(
"get_clock_sources");
});
tree->create<std::string>(mb_path / "time_source/value")
.add_coerced_subscriber([mb](const std::string& time_source) {
mb->rpc->notify_with_token(
MPMD_DEFAULT_INIT_TIMEOUT, "set_time_source", time_source);
})
.set_publisher([mb]() {
return mb->rpc->request_with_token<std::string>("get_time_source");
});
tree->create<std::vector<std::string>>(mb_path / "time_source/options")
.set_publisher([mb]() {
return mb->rpc->request_with_token<std::vector<std::string>>(
"get_time_sources");
});
/*** Sensors ********************************************************/
auto sensor_list =
mb->rpc->request_with_token<std::vector<std::string>>("get_mb_sensors");
UHD_LOG_DEBUG("MPMD", "Found " << sensor_list.size() << " motherboard sensors.");
for (const auto& sensor_name : sensor_list) {
UHD_LOG_TRACE("MPMD", "Adding motherboard sensor `" << sensor_name << "'");
tree->create<sensor_value_t>(mb_path / "sensors" / sensor_name)
.set_publisher([mb, sensor_name]() {
auto sensor_val = sensor_value_t(
mb->rpc->request_with_token<sensor_value_t::sensor_map_t>(
MPMD_DEFAULT_INIT_TIMEOUT, "get_mb_sensor", sensor_name));
return sensor_val;
})
.set_coercer([](const sensor_value_t&) {
throw uhd::runtime_error("Trying to write read-only sensor value!");
return sensor_value_t("", "", "");
});
}
/*** EEPROM *********************************************************/
tree->create<uhd::usrp::mboard_eeprom_t>(mb_path / "eeprom")
.add_coerced_subscriber([mb](const uhd::usrp::mboard_eeprom_t& mb_eeprom) {
eeprom_map_t eeprom_map;
for (const auto& key : mb_eeprom.keys()) {
eeprom_map[key] =
std::vector<uint8_t>(mb_eeprom[key].cbegin(), mb_eeprom[key].cend());
}
mb->rpc->notify_with_token(
MPMD_DEFAULT_INIT_TIMEOUT, "set_mb_eeprom", eeprom_map);
})
.set_publisher([mb]() {
auto mb_eeprom =
mb->rpc->request_with_token<std::map<std::string, std::string>>(
"get_mb_eeprom");
uhd::usrp::mboard_eeprom_t mb_eeprom_dict(
mb_eeprom.cbegin(), mb_eeprom.cend());
return mb_eeprom_dict;
});
/*** Updateable Components ******************************************/
std::vector<std::string> updateable_components =
mb->rpc->request<std::vector<std::string>>("list_updateable_components");
// TODO: Check the 'id' against the registered property
UHD_LOG_DEBUG("MPMD",
"Found " << updateable_components.size()
<< " updateable motherboard components.");
for (const auto& comp_name : updateable_components) {
UHD_LOG_TRACE("MPMD", "Adding motherboard component: " << comp_name);
tree->create<uhd::usrp::component_files_t>(mb_path / "components" / comp_name)
.set_coercer([mb, comp_name](const uhd::usrp::component_files_t& comp_files) {
auto comp_info = _get_component_info(comp_name, mb)[0];
if (comp_info.metadata.get("reset", "") == "True") {
UHD_LOG_DEBUG(
"MPMD", "Bracing for potential loss of RPC server connection.");
mb->allow_claim_failure(true);
}
auto result = _update_component(comp_files, mb);
mb->allow_claim_failure(false);
return result;
})
.set_publisher([mb, comp_name]() {
return _get_component_info(comp_name, mb);
}); // Done adding component to property tree
}
}