aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/usrp
diff options
context:
space:
mode:
authorMartin Braun <martin.braun@ettus.com>2020-04-15 15:07:53 -0700
committerAaron Rossetto <aaron.rossetto@ni.com>2020-05-19 13:24:59 -0500
commit2d7676e8cf2cc4fb59e59185864c8804ca77e15b (patch)
tree765965e19b86f37077b19b4ea4eff46c8b68c77c /host/lib/usrp
parent676c3a37867f0e19476e9dc8495127bb3208a1a0 (diff)
downloaduhd-2d7676e8cf2cc4fb59e59185864c8804ca77e15b.tar.gz
uhd-2d7676e8cf2cc4fb59e59185864c8804ca77e15b.tar.bz2
uhd-2d7676e8cf2cc4fb59e59185864c8804ca77e15b.zip
lib: Add power cal manager
This is a utility class that can be used by USRP or daughterboard drivers to tie power calibration into their respective drivers.
Diffstat (limited to 'host/lib/usrp')
-rw-r--r--host/lib/usrp/common/CMakeLists.txt1
-rw-r--r--host/lib/usrp/common/pwr_cal_mgr.cpp274
2 files changed, 275 insertions, 0 deletions
diff --git a/host/lib/usrp/common/CMakeLists.txt b/host/lib/usrp/common/CMakeLists.txt
index 7a6a4fe27..a61738743 100644
--- a/host/lib/usrp/common/CMakeLists.txt
+++ b/host/lib/usrp/common/CMakeLists.txt
@@ -35,6 +35,7 @@ LIBUHD_APPEND_SOURCES(
${CMAKE_CURRENT_SOURCE_DIR}/recv_packet_demuxer.cpp
${CMAKE_CURRENT_SOURCE_DIR}/io_service_mgr.cpp
${CMAKE_CURRENT_SOURCE_DIR}/io_service_args.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/pwr_cal_mgr.cpp
)
if(ENABLE_DPDK)
diff --git a/host/lib/usrp/common/pwr_cal_mgr.cpp b/host/lib/usrp/common/pwr_cal_mgr.cpp
new file mode 100644
index 000000000..1fdd15645
--- /dev/null
+++ b/host/lib/usrp/common/pwr_cal_mgr.cpp
@@ -0,0 +1,274 @@
+//
+// Copyright 2020 Ettus Research, a National Instruments Brand
+//
+// SPDX-License-Identifier: GPL-3.0-or-later
+//
+
+#include <uhd/cal/database.hpp>
+#include <uhd/cal/pwr_cal.hpp>
+#include <uhd/types/direction.hpp>
+#include <uhd/utils/log.hpp>
+#include <uhd/utils/math.hpp>
+#include <uhdlib/usrp/common/pwr_cal_mgr.hpp>
+#include <unordered_map>
+#include <boost/algorithm/string.hpp>
+#include <boost/range/adaptor/filtered.hpp>
+#include <algorithm>
+#include <set>
+
+using namespace uhd::usrp;
+
+namespace {
+
+// List of antenna names that are globally known to never have their own cal data
+const std::set<std::string> INVALID_ANTENNAS{"CAL", "LOCAL"};
+
+} // namespace
+
+std::string pwr_cal_mgr::sanitize_antenna_name(std::string antenna_name)
+{
+ std::replace(antenna_name.begin(), antenna_name.end(), '/', '+');
+ boost::to_lower(antenna_name);
+ return antenna_name;
+}
+
+// Shorthand for filtering against INVALID_ANTENNAS
+bool pwr_cal_mgr::is_valid_antenna(const std::string& antenna)
+{
+ return !INVALID_ANTENNAS.count(antenna);
+}
+
+class pwr_cal_mgr_impl : public pwr_cal_mgr
+{
+public:
+ pwr_cal_mgr_impl(const std::string& serial,
+ const std::string& log_id,
+ get_double_type&& get_freq,
+ get_str_type&& get_key,
+ uhd::gain_group::sptr gain_group)
+ : _log_id(log_id)
+ , _get_freq(std::move(get_freq))
+ , _get_key(std::move(get_key))
+ , _gain_group(gain_group)
+ , _hw_gain_name(gain_group->get_names().at(0))
+ {
+ set_serial(serial);
+ }
+
+ void set_gain_group(uhd::gain_group::sptr gain_group)
+ {
+ _gain_group = gain_group;
+ }
+
+ bool has_power_data()
+ {
+ const std::string key = _get_key();
+ _load_cal_data(key);
+ return _cal_data.count(key) && bool(_cal_data.at(key));
+ }
+
+ void populate_subtree(uhd::property_tree::sptr subtree)
+ {
+ subtree->create<std::string>(uhd::fs_path("ref_power/key"))
+ .set_coercer([](const std::string&) -> std::string {
+ throw uhd::runtime_error("Cannot overwrite power cal key!");
+ })
+ .set_publisher([this]() { return _get_key(); });
+ subtree->create<std::string>(uhd::fs_path("ref_power/serial"))
+ .set_coercer([](const std::string&) -> std::string {
+ throw uhd::runtime_error("Cannot overwrite cal serial!");
+ })
+ .set_publisher([this]() { return _serial; });
+ if (!has_power_data()) {
+ return;
+ }
+ subtree->create<double>(uhd::fs_path("ref_power/value"))
+ .set_coercer(
+ [this](const double power_dbm) { return this->set_power(power_dbm); })
+ .set_publisher([this]() { return this->get_power(); });
+ subtree->create<uhd::meta_range_t>(uhd::fs_path("ref_power/range"))
+ .set_coercer([](const uhd::meta_range_t&) -> uhd::meta_range_t {
+ throw uhd::runtime_error("Cannot overwrite power range!");
+ })
+ .set_publisher([this]() { return this->get_power_range(); });
+ }
+
+ double set_power(const double power_dbm)
+ {
+ const std::string key = _get_key();
+ _load_cal_data(key);
+ UHD_ASSERT_THROW(_cal_data.count(key));
+ _desired_power = power_dbm;
+ const uint64_t freq = static_cast<uint64_t>(_get_freq());
+ auto& cal_data = _cal_data.at(key);
+ if (!cal_data) {
+ const std::string err_msg = std::string("Attempting to set power for key ")
+ + key + ", but no cal data available!";
+ UHD_LOG_ERROR(_log_id, err_msg);
+ throw uhd::runtime_error(err_msg);
+ }
+
+ const double desired_hw_gain = cal_data->get_gain(power_dbm, freq);
+ // This sets all the gains
+ _gain_group->set_value(desired_hw_gain);
+ const double coerced_hw_gain = _gain_group->get_value(_hw_gain_name);
+ const double coerced_hw_power = cal_data->get_power(coerced_hw_gain, freq);
+ const double coerced_total_gain = _gain_group->get_value();
+ const double coerced_total_power =
+ coerced_hw_power + coerced_total_gain - coerced_hw_gain;
+ UHD_LOG_TRACE(_log_id,
+ "Desired power: " << power_dbm << " dBm -> desired gain: " << desired_hw_gain
+ << " dB; Actual HW power: " << coerced_hw_power
+ << " dBm -> actual HW gain: " << coerced_hw_gain
+ << " dB, Actual total power: " << coerced_total_power
+ << " dBm -> actual total gain: " << coerced_total_gain
+ << " dB");
+ // We directly scale the power with the residual gain
+ return coerced_total_power;
+ }
+
+ double get_power()
+ {
+ const std::string key = _get_key();
+ _load_cal_data(key);
+ UHD_ASSERT_THROW(_cal_data.count(key));
+ auto& cal_data = _cal_data.at(key);
+ if (!cal_data) {
+ const std::string err_msg = std::string("Attempting to get power for key ")
+ + key + ", but no cal data available!";
+ UHD_LOG_ERROR(_log_id, err_msg);
+ throw uhd::runtime_error(err_msg);
+ }
+
+ const uint64_t freq = static_cast<uint64_t>(_get_freq());
+ const double hw_gain = _gain_group->get_value(_hw_gain_name);
+ const double hw_power = cal_data->get_power(hw_gain, freq);
+ // We directly scale the power with the residual gain
+ return hw_power + (_gain_group->get_value() - hw_gain);
+ }
+
+ void update_power()
+ {
+ if (_mode == tracking_mode::TRACK_POWER) {
+ set_power(_desired_power);
+ }
+ }
+
+ uhd::meta_range_t get_power_range()
+ {
+ const std::string key = _get_key();
+ _load_cal_data(key);
+ UHD_ASSERT_THROW(_cal_data.count(key));
+ auto& cal_data = _cal_data.at(key);
+ if (!cal_data) {
+ const std::string err_msg = std::string("Attempting to get power range for key ")
+ + key + ", but no cal data available!";
+ UHD_LOG_ERROR(_log_id, err_msg);
+ throw uhd::runtime_error(err_msg);
+ }
+ const uint64_t freq = static_cast<uint64_t>(_get_freq());
+ return cal_data->get_power_limits(freq);
+ }
+
+ void set_temperature(const int temp_C)
+ {
+ for (auto& cal_data : _cal_data) {
+ if (cal_data.second) {
+ cal_data.second->set_temperature(temp_C);
+ }
+ }
+ }
+
+ void set_tracking_mode(const tracking_mode mode)
+ {
+ _mode = mode;
+ }
+
+ void set_serial(const std::string& serial)
+ {
+ if (serial == _serial || serial.empty()) {
+ return;
+ }
+ _serial = serial;
+ _cal_data.clear();
+ }
+
+ void _load_cal_data(const std::string& key)
+ {
+ if (_cal_data.count(key)) {
+ return;
+ }
+ cal::pwr_cal::sptr cal_data(nullptr);
+ UHD_LOG_TRACE(
+ _log_id, "Looking for power cal data for " << key << ", serial " << _serial);
+ bool cal_data_found = false;
+ if (cal::database::has_cal_data(key, _serial)) {
+ try {
+ cal_data = cal::container::make<cal::pwr_cal>(
+ cal::database::read_cal_data(key, _serial));
+ cal_data_found = true;
+ } catch (const uhd::exception& ex) {
+ UHD_LOG_WARNING(_log_id, "Error loading cal data: " << ex.what());
+ }
+ }
+ _cal_data.insert({key, cal_data});
+ UHD_LOG_TRACE(_log_id,
+ (bool(cal_data) ? "" : "No ") << "power cal data found for key " << key
+ << ", key " << key << ", serial " << _serial);
+ if (cal_data_found) {
+ // If we found cal data, check that all the other antennas/keys also
+ // have cal data
+ if (std::any_of(_cal_data.cbegin(),
+ _cal_data.cend(),
+ [](const cal_data_map_type::value_type& data) {
+ return bool(data.second);
+ })) {
+ UHD_LOG_WARNING(_log_id,
+ "Some ports for " << _serial
+ << " have power cal data, others do not. This "
+ "will cause inconsistent behaviour across "
+ "antenna ports when setting power levels.");
+ }
+ }
+ }
+
+ std::string get_serial() const
+ {
+ return _serial;
+ }
+
+ std::string get_key()
+ {
+ return _get_key();
+ }
+
+private:
+ const std::string _log_id;
+ std::string _serial;
+
+ get_double_type _get_freq;
+ get_str_type _get_key;
+ uhd::gain_group::sptr _gain_group;
+ const std::string _hw_gain_name;
+
+ //! Store the cal data for every cal key
+ using cal_data_map_type =
+ std::unordered_map<std::string /* key */, uhd::usrp::cal::pwr_cal::sptr>;
+ cal_data_map_type _cal_data;
+
+ double _desired_power = 0;
+ tracking_mode _mode = tracking_mode::TRACK_GAIN;
+};
+
+pwr_cal_mgr::sptr pwr_cal_mgr::make(const std::string& serial,
+ const std::string& log_id,
+ pwr_cal_mgr::get_double_type&& get_freq,
+ pwr_cal_mgr::get_str_type&& get_key,
+ uhd::gain_group::sptr gain_group)
+{
+ return std::make_shared<pwr_cal_mgr_impl>(serial,
+ log_id,
+ std::move(get_freq),
+ std::move(get_key),
+ gain_group);
+}