From 2d7676e8cf2cc4fb59e59185864c8804ca77e15b Mon Sep 17 00:00:00 2001 From: Martin Braun Date: Wed, 15 Apr 2020 15:07:53 -0700 Subject: 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. --- .../lib/include/uhdlib/usrp/common/pwr_cal_mgr.hpp | 151 +++++++++++++++++++++ 1 file changed, 151 insertions(+) create mode 100644 host/lib/include/uhdlib/usrp/common/pwr_cal_mgr.hpp (limited to 'host/lib/include/uhdlib') diff --git a/host/lib/include/uhdlib/usrp/common/pwr_cal_mgr.hpp b/host/lib/include/uhdlib/usrp/common/pwr_cal_mgr.hpp new file mode 100644 index 000000000..4d063825b --- /dev/null +++ b/host/lib/include/uhdlib/usrp/common/pwr_cal_mgr.hpp @@ -0,0 +1,151 @@ +// +// Copyright 2020 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#pragma once + +#include +#include +#include +#include +#include +#include + +namespace uhd { namespace usrp { + +/*! Power Calibration Manager: Helper class to set track power vs. gain settings + * + * This class can be plugged into USRP device drivers (or daughterboard drivers). + * It will manage loading the calibration data, and translating power settings + * into gain settings, and vice versa (i.e. read the power from a gain setting). + * + * This class mostly communicates with the device driver via callbacks. This + * allows separating the power control from the device driver without having to + * implement any device-specific code in this manager. + * + * The actual control of gains is also done via callbacks, but they are + * enapsulated into a gain group. The gain group must separate hardware gains + * from other gains that can be used for more fine-grained correction; the + * existing gain groups in device drivers may or may not be adequate. + * + * For example, the X3x0 has two gain stages on the RX side, there's an ADC gain + * as well as the analog gain of the daughterboards. The calibration is performed + * over the entire gain range, including the ADC gain, which means that the + * hardware gains *include* the ADC gain and must therefore not be a separate gain + * stage for this class. + */ +class pwr_cal_mgr +{ +public: + using sptr = std::shared_ptr; + using get_double_type = std::function; + using get_str_type = std::function; + //! The current power/gain tracking mode. When the device does something + // that will cause the output power to change, we have an option of keeping + // the gain constant, or keeping the power constant. + enum class tracking_mode { + TRACK_GAIN, //!< In this mode, we keep the gain after retuning. + TRACK_POWER //!< In this mode, we keep the power after retuning + }; + + //! Helper: Sanitize antenna names (e.g. TX/RX -> tx_rx) + // + // Note: Argument is not const std::string&; we make use of C++'s short + // string optimization for an easier implementation + static std::string sanitize_antenna_name(std::string antenna_name); + + //! Helper: Check if an antenna name is valid for calibration (e.g., CAL and + // LOCAL are not) + static bool is_valid_antenna(const std::string& antenna); + + /*! Factory + * + * \param serial The serial string used for the cal database lookup + * \param log_id A string used for logging + * \param get_freq A function object to read back the current frequency from the + * underlying device + * \param get_key A function object to read back the current calibraton key + * from the underlying device + * \param gain_group A gain group. The first gain stage (with the + * higher priority) must be the same gain as used in the + * calibration data; usually, it's the overall hardware + * gain. The second is any ancillary gain that can be used + * for correction. This could be a baseband gain. + */ + static sptr make(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); + + //! Update the gain group (see make()); + // + // Not thread-safe: Don't call at the same time as set_power() + virtual void set_gain_group(uhd::gain_group::sptr gain_group) = 0; + + //! Return true if there is power cal data for the currently selected port + virtual bool has_power_data() = 0; + + //! Add a property tree node (ref_power/range and ref_power/value) + // + // For non-RFNoC devices (e.g. B200), this must be called in order to + // expose the power APIs to multi_usrp. + virtual void populate_subtree(uhd::property_tree::sptr subtree) = 0; + + /*! Set the power to \p power_dbm + * + * This function will coerce both the valid power, and the resulting gain + * value, and will return the actual, current power. + * + * Note: Calling this will automatically set the tracking mode to TRACK_POWER. + */ + virtual double set_power(const double power_dbm) = 0; + + //! Return the current power + virtual double get_power() = 0; + + /*! Reapply the power setting from the last set_power() call + * + * This is useful for reapplying settings, e.g., after tuning. It can be + * set as a callback or subscriber to events that require a power reset. + * This will do nothing if the current tracking mode is set to + * tracking_mode::KEEP_GAIN + */ + virtual void update_power() = 0; + + /*! Return the valid range of powers + * + * This will query the device state and return currently valid settings. + */ + virtual uhd::meta_range_t get_power_range() = 0; + + //! Update the temperature in Celsius + // + // Because reading the temperature can be an invasive operation, we leave it + // up to the device implementation to update the temperature at sensible + // intervals instead of using a callback. + virtual void set_temperature(const int temp_C) = 0; + + /*! Set the current power tracking mode + */ + virtual void set_tracking_mode(const tracking_mode) = 0; + + //! Return the currently active calibration data key + virtual std::string get_key() = 0; + + //! Return the calibration serial + virtual std::string get_serial() const = 0; + + //! Update serial + // + // This may be called for example when the hardware is hot-pluggable, or + // if the calibration key changes at runtime. + // Calling it will not only set the serial number, but will also force a + // reload of the calibration data. The existing calibration data is + // discarded. + virtual void set_serial(const std::string& serial) = 0; +}; + +}} // namespace uhd::usrp -- cgit v1.2.3