diff options
-rw-r--r-- | host/include/uhd/rfnoc/CMakeLists.txt | 1 | ||||
-rw-r--r-- | host/include/uhd/rfnoc/radio_control.hpp | 688 | ||||
-rw-r--r-- | host/lib/include/uhdlib/rfnoc/radio_control_impl.hpp | 289 | ||||
-rw-r--r-- | host/lib/rfnoc/CMakeLists.txt | 1 | ||||
-rw-r--r-- | host/lib/rfnoc/radio_control_impl.cpp | 771 | ||||
-rw-r--r-- | host/tests/CMakeLists.txt | 1 |
6 files changed, 1751 insertions, 0 deletions
diff --git a/host/include/uhd/rfnoc/CMakeLists.txt b/host/include/uhd/rfnoc/CMakeLists.txt index 5b81ae6e1..c8308953f 100644 --- a/host/include/uhd/rfnoc/CMakeLists.txt +++ b/host/include/uhd/rfnoc/CMakeLists.txt @@ -41,6 +41,7 @@ if(ENABLE_RFNOC) fir_block_ctrl.hpp null_block_ctrl.hpp radio_ctrl.hpp + radio_control.hpp replay_block_ctrl.hpp siggen_block_ctrl.hpp window_block_ctrl.hpp diff --git a/host/include/uhd/rfnoc/radio_control.hpp b/host/include/uhd/rfnoc/radio_control.hpp new file mode 100644 index 000000000..f966379e1 --- /dev/null +++ b/host/include/uhd/rfnoc/radio_control.hpp @@ -0,0 +1,688 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_LIBUHD_RADIO_BLOCK_CONTROL_HPP +#define INCLUDED_LIBUHD_RADIO_BLOCK_CONTROL_HPP + +#include <uhd/config.hpp> +#include <uhd/rfnoc/noc_block_base.hpp> +#include <uhd/types/device_addr.hpp> +#include <uhd/types/direction.hpp> +#include <uhd/types/eeprom.hpp> +#include <uhd/types/ranges.hpp> +#include <uhd/types/sensors.hpp> +#include <uhd/types/stream_cmd.hpp> + +namespace uhd { namespace rfnoc { + +/*! Parent class for radio block controllers + */ +class radio_control : public noc_block_base +{ +public: + static const std::string ALL_LOS; + static const std::string ALL_GAINS; + static const size_t ALL_CHANS; + + RFNOC_DECLARE_BLOCK(radio_control) + + /************************************************************************** + * Rate-Related API Calls + *************************************************************************/ + //! Set the sample rate + // + // This function will coerce the rate and return the actual, current value. + virtual double set_rate(const double rate) = 0; + + //! Get the sample rate + virtual double get_rate() const = 0; + + //! Return a list of valid rates + virtual uhd::meta_range_t get_rate_range() const = 0; + + /************************************************************************** + * RF-Related API Calls + *************************************************************************/ + /*! Return the selected TX antenna for channel \p chan. + * + * \return The selected antenna. + */ + virtual std::string get_tx_antenna(const size_t chan) const = 0; + + /*! Return a list of valid TX antenna for channel \p chan. + * + * \return The selected antenna. + */ + virtual std::vector<std::string> get_tx_antennas(const size_t chan) const = 0; + + /*! Select RX antenna \p for channel \p chan. + * + * \throws uhd::value_error if \p ant is not a valid value. + */ + virtual void set_tx_antenna(const std::string& ant, const size_t chan) = 0; + + /*! Return the selected RX antenna for channel \p chan. + * + * \return The selected antenna. + */ + virtual std::string get_rx_antenna(const size_t chan) const = 0; + + /*! Return a list of valid RX antenna for channel \p chan. + * + * \return The selected antenna. + */ + virtual std::vector<std::string> get_rx_antennas(const size_t chan) const = 0; + + /*! Select RX antenna \p for channel \p chan. + * + * \throws uhd::value_error if \p ant is not a valid value. + */ + virtual void set_rx_antenna(const std::string& ant, const size_t chan) = 0; + + /*! Return the current transmit LO frequency on channel \p chan. + * + * \return The current LO frequency. + */ + virtual double get_tx_frequency(const size_t chan) = 0; + + /*! Tune the TX frequency for channel \p chan. + * + * This function will attempt to tune as close as possible, and return a + * coerced value of the actual tuning result. + * + * If there is a single LO in this radio, and we're doing direct conversion, + * then this is the LO frequency. + * + * \param freq Frequency in Hz + * \param chan Channel to tune + * + * \return The actual frequency. + */ + virtual double set_tx_frequency(const double freq, size_t chan) = 0; + + /*! Set the TX tune args, if supported by the hardware. + */ + virtual void set_tx_tune_args(const uhd::device_addr_t& args, const size_t chan) = 0; + + /*! Return the range of frequencies that \p chan can be tuned to. + * + * \return The range of frequencies that we can tune the TX chan to + */ + virtual uhd::freq_range_t get_tx_frequency_range(const size_t chan) const = 0; + + /*! Return the current receive LO frequency on channel \p chan. + * + * \return The current LO frequency. + */ + virtual double get_rx_frequency(const size_t chan) = 0; + + /*! Tune the RX frequency for channel \p chan. + * + * This function will attempt to tune as close as possible, and return a + * coerced value of the actual tuning result. + * + * If there is a single LO in this radio, and we're doing direct conversion, + * then this is the LO frequency. + * + * \param freq Requested frequency + * \param chan Channel number. + * \return The actual frequency. + */ + virtual double set_rx_frequency(const double freq, const size_t chan) = 0; + + /*! Set the TX tune args, if supported by the hardware. + */ + virtual void set_rx_tune_args(const uhd::device_addr_t& args, const size_t chan) = 0; + + /*! Return the range of frequencies that \p chan can be tuned to. + * + * \return The range of frequencies that we can tune the RX chan to + */ + virtual uhd::freq_range_t get_rx_frequency_range(const size_t chan) const = 0; + + /*! Return a list of valid TX gain names + */ + virtual std::vector<std::string> get_tx_gain_names(const size_t chan) const = 0; + + /*! Return a range of valid TX gains + */ + virtual uhd::gain_range_t get_tx_gain_range(const size_t chan) const = 0; + + /*! Return a range of valid TX gains + */ + virtual uhd::gain_range_t get_tx_gain_range( + const std::string& name, const size_t chan) const = 0; + + /*! Return the overall transmit gain on channel \p chan + * + * \return The actual gain value + */ + virtual double get_tx_gain(const size_t chan) = 0; + + /*! Return the transmit gain \p name on channel \p chan + * + * \return The actual gain value + */ + virtual double get_tx_gain(const std::string& name, const size_t chan) = 0; + + /*! Set the transmit gain on channel \p chan + * + * This function will attempt to set the gain as close as possible, + * and return a coerced value of the actual gain value. + * + * This method will set the overall gain. To set a specific gain, use + * set_tx_gain(const double, const std::string&, const size_t). + * + * \return The actual gain value + */ + virtual double set_tx_gain(const double gain, const size_t chan) = 0; + + /*! Set the transmit gain \p name on channel \p chan + * + * This function will attempt to set the gain as close as possible, + * and return a coerced value of the actual gain value. + * + * \return The actual gain value + */ + virtual double set_tx_gain( + const double gain, const std::string& name, const size_t chan) = 0; + + /*! Return a list of valid RX gain names + */ + virtual std::vector<std::string> get_rx_gain_names(const size_t chan) const = 0; + + /*! Return a range of valid RX gains + */ + virtual uhd::gain_range_t get_rx_gain_range(const size_t chan) const = 0; + + /*! Return a range of valid RX gains + */ + virtual uhd::gain_range_t get_rx_gain_range( + const std::string& name, const size_t chan) const = 0; + + /*! Return the overall receive gain on channel \p chan + * + * \return The actual gain value + */ + virtual double get_rx_gain(const size_t chan) = 0; + + /*! Return the receive gain \p name on channel \p chan + * + * \return The actual gain value + */ + virtual double get_rx_gain(const std::string& name, const size_t chan) = 0; + + /*! Set the overall receive gain on channel \p chan + * + * This function will attempt to set the gain as close as possible, + * and return a coerced value of the actual gain value. + * + * \return The actual gain value + */ + virtual double set_rx_gain(const double gain, const size_t chan) = 0; + + /*! Set the receive gain \p on channel \p chan + * + * This function will attempt to set the gain as close as possible, + * and return a coerced value of the actual gain value. + * + * \return The actual gain value + */ + virtual double set_rx_gain( + const double gain, const std::string& name, const size_t chan) = 0; + + /*! Enable RX AGC on this radio + * + * \throws uhd::not_implemented_error if this radio doesn't support RX AGC + */ + virtual void set_rx_agc(const bool enable, const size_t chan) = 0; + + /*! Return a list of TX gain profiles for this radio + */ + virtual std::vector<std::string> get_tx_gain_profile_names(const size_t chan) const = 0; + + /*! Return a list of TX gain profiles for this radio + */ + virtual std::vector<std::string> get_rx_gain_profile_names(const size_t chan) const = 0; + + /*! Set the TX gain profile + */ + virtual void set_tx_gain_profile(const std::string& profile, const size_t chan) = 0; + + /*! Set the RX gain profile + */ + virtual void set_rx_gain_profile(const std::string& profile, const size_t chan) = 0; + + /*! Return the TX gain profile + */ + virtual std::string get_tx_gain_profile(const size_t chan) const = 0; + + /*! Set the RX gain profile + */ + virtual std::string get_rx_gain_profile(const size_t chan) const = 0; + + /*! Return a range of valid TX bandwidths + */ + virtual meta_range_t get_tx_bandwidth_range(size_t chan) const = 0; + + /*! Return the analog filter bandwidth channel \p chan + * + * \return The actual bandwidth value + */ + virtual double get_tx_bandwidth(const size_t chan) = 0; + + /*! Set the analog filter bandwidth channel \p chan + * + * This function will attempt to set the analog bandwidth. + * + * \return The actual bandwidth value + */ + virtual double set_tx_bandwidth(const double bandwidth, const size_t chan) = 0; + + /*! Return a range of valid RX bandwidths + */ + virtual meta_range_t get_rx_bandwidth_range(size_t chan) const = 0; + + /*! Return the analog filter bandwidth channel \p chan + * + * \return The actual bandwidth value + */ + virtual double get_rx_bandwidth(const size_t chan) = 0; + + /*! Set the analog filter bandwidth channel \p chan + * + * This function will attempt to set the analog bandwidth. + * + * \return The actual bandwidth value + */ + virtual double set_rx_bandwidth(const double bandwidth, const size_t chan) = 0; + + /************************************************************************** + * LO Controls + *************************************************************************/ + /*! Get a list of possible LO stage names + * + * \param chan the channel index 0 to N-1 + * \return a vector of strings for possible LO names + */ + virtual std::vector<std::string> get_rx_lo_names(const size_t chan) const = 0; + + /*! Get a list of possible LO sources. + * + * Channels which do not have controllable LO sources + * will return "internal". + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + * \return a vector of strings for possible settings + */ + virtual std::vector<std::string> get_rx_lo_sources( + const std::string& name, const size_t chan) const = 0; + + /*! + * Get the LO frequency range of the RX LO. + * If the channel does not have independently configurable LOs + * the rf frequency range will be returned. + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + * \return a frequency range object + */ + virtual freq_range_t get_rx_lo_freq_range( + const std::string& name, const size_t chan) const = 0; + + /*! + * Set the LO source for a channel. + * For usrps that support selectable LOs, this function + * allows switching between them. + * Typical options for source: internal, external. + * \param src a string representing the LO source + * \param name the name of the LO stage to update + * \param chan the channel index 0 to N-1 + */ + virtual void set_rx_lo_source( + const std::string& src, const std::string& name, const size_t chan) = 0; + + /*! + * Get the currently set LO source. + * Channels without controllable LO sources will return + * "internal" + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + * \return the configured LO source + */ + virtual const std::string get_rx_lo_source( + const std::string& name, const size_t chan) = 0; + + /*! + * Set whether the LO used by the usrp device is exported + * For usrps that support exportable LOs, this function + * configures if the LO used by chan is exported or not. + * \param enabled if true then export the LO + * \param name the name of the LO stage to update + * \param chan the channel index 0 to N-1 for the source channel + */ + virtual void set_rx_lo_export_enabled( + bool enabled, const std::string& name, const size_t chan) = 0; + + /*! + * Returns true if the currently selected LO is being exported. + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + */ + virtual bool get_rx_lo_export_enabled( + const std::string& name, const size_t chan) const = 0; + + /*! + * Set the RX LO frequency (Advanced). + * \param freq the frequency to set the LO to + * \param name the name of the LO stage to update + * \param chan the channel index 0 to N-1 + * \return a coerced LO frequency + */ + virtual double set_rx_lo_freq( + double freq, const std::string& name, const size_t chan) = 0; + + /*! + * Get the current RX LO frequency (Advanced). + * If the channel does not have independently configurable LOs + * the current rf frequency will be returned. + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + * \return the configured LO frequency + */ + virtual double get_rx_lo_freq(const std::string& name, const size_t chan) = 0; + + /*! Get a list of possible LO stage names + * + * \param chan the channel index 0 to N-1 + * \return a vector of strings for possible LO names + */ + virtual std::vector<std::string> get_tx_lo_names(const size_t chan) const = 0; + + /*! Get a list of possible LO sources. + * + * Channels which do not have controllable LO sources + * will return "internal". + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + * \return a vector of strings for possible settings + */ + virtual std::vector<std::string> get_tx_lo_sources( + const std::string& name, const size_t chan) = 0; + + /*! + * Get the LO frequency range of the tx LO. + * If the channel does not have independently configurable LOs + * the rf frequency range will be returned. + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + * \return a frequency range object + */ + virtual freq_range_t get_tx_lo_freq_range( + const std::string& name, const size_t chan) = 0; + + /*! + * Set the LO source for a channel. + * For usrps that support selectable LOs, this function + * allows switching between them. + * Typical options for source: internal, external. + * \param src a string representing the LO source + * \param name the name of the LO stage to update + * \param chan the channel index 0 to N-1 + */ + virtual void set_tx_lo_source( + const std::string& src, const std::string& name, const size_t chan) = 0; + + /*! + * Get the currently set LO source. + * Channels without controllable LO sources will return + * "internal" + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + * \return the configured LO source + */ + virtual const std::string get_tx_lo_source( + const std::string& name, const size_t chan) = 0; + + /*! + * Set whether the LO used by the usrp device is exported + * For usrps that support exportable LOs, this function + * configures if the LO used by chan is exported or not. + * \param enabled if true then export the LO + * \param name the name of the LO stage to update + * \param chan the channel index 0 to N-1 for the source channel + */ + virtual void set_tx_lo_export_enabled( + const bool enabled, const std::string& name, const size_t chan) = 0; + + /*! + * Returns true if the currently selected LO is being exported. + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + */ + virtual bool get_tx_lo_export_enabled(const std::string& name, const size_t chan) = 0; + + /*! Set the tx LO frequency (Advanced). + * + * See also multi_usrp::set_tx_lo_freq(). + * + * \param freq the frequency to set the LO to + * \param name the name of the LO stage to update + * \param chan the channel index 0 to N-1 + * \return a coerced LO frequency + */ + virtual double set_tx_lo_freq( + const double freq, const std::string& name, const size_t chan) = 0; + + /*! Get the current TX LO frequency (Advanced). + * + * See also multi_usrp::get_tx_lo_freq() + * + * If the channel does not have independently configurable LOs + * the current RF frequency will be returned. + * + * \param name the name of the LO stage to query + * \param chan the channel index 0 to N-1 + * \return the configured LO frequency + */ + virtual double get_tx_lo_freq(const std::string& name, const size_t chan) = 0; + + /************************************************************************** + * Calibration-Related API Calls + *************************************************************************/ + /*! Set a constant TX DC offset value + * + * The value is complex to control both I and Q. + * + * \param offset the dc offset (1.0 is full-scale) + * \param chan the channel index + */ + virtual void set_tx_dc_offset(const std::complex<double>& offset, size_t chan) = 0; + + /*! Get the valid range for TX DC offset values. + * + * \param chan the channel index + */ + virtual meta_range_t get_tx_dc_offset_range(size_t chan) const = 0; + + /*! Set the TX frontend IQ imbalance correction. + * + * Use this to adjust the magnitude and phase of I and Q. + * + * \param correction the complex correction + * \param chan the channel index 0 to N-1 + */ + virtual void set_tx_iq_balance( + const std::complex<double>& correction, size_t chan) = 0; + + /*! Enable/disable the automatic RX DC offset correction. + * The automatic correction subtracts out the long-run average. + * + * When disabled, the averaging option operation is halted. + * Once halted, the average value will be held constant + * until the user re-enables the automatic correction + * or overrides the value by manually setting the offset. + * + * \param enb true to enable automatic DC offset correction + * \param chan the channel index + */ + virtual void set_rx_dc_offset(const bool enb, size_t chan = ALL_CHANS) = 0; + + /*! Set a constant RX DC offset value. + * + * The value is complex to control both I and Q. + * Only set this when automatic correction is disabled. + * \param offset the dc offset (1.0 is full-scale) + * \param chan the channel index 0 to N-1 + */ + virtual void set_rx_dc_offset(const std::complex<double>& offset, size_t chan) = 0; + + /*! Get the valid range for RX DC offset values. + * + * \param chan the channel index + */ + virtual meta_range_t get_rx_dc_offset_range(size_t chan) const = 0; + + /*! Enable/disable the automatic IQ imbalance correction. + * + * \param enb true to enable automatic IQ balance correction + * \param chan the channel index 0 to N-1 + */ + virtual void set_rx_iq_balance(const bool enb, size_t chan) = 0; + + /*! Enable/disable the automatic IQ imbalance correction. + * + * \param enb true to enable automatic IQ balance correction + * \param chan the channel index 0 to N-1 + */ + virtual void set_rx_iq_balance( + const std::complex<double>& correction, size_t chan) = 0; + + /************************************************************************** + * GPIO Controls + *************************************************************************/ + /*! Returns the list of GPIO banks that are associated with this radio. + * + * \returns list of GPIO bank names + */ + virtual std::vector<std::string> get_gpio_banks() const = 0; + + /*! + * Set a GPIO attribute on a particular GPIO bank. + * Possible attribute names: + * - CTRL - 1 for ATR mode 0 for GPIO mode + * - DDR - 1 for output 0 for input + * - OUT - GPIO output level (not ATR mode) + * - ATR_0X - ATR idle state + * - ATR_RX - ATR receive only state + * - ATR_TX - ATR transmit only state + * - ATR_XX - ATR full duplex state + * \param bank the name of a GPIO bank (e.g., FP0) + * \param attr the name of a GPIO attribute (e.g., CTRL) + * \param value the new value for this GPIO bank + * \param mask the bit mask to effect which pins are changed + */ + virtual void set_gpio_attr(const std::string& bank, + const std::string& attr, + const uint32_t value, + const uint32_t mask) = 0; + + /*! + * Get a GPIO attribute on a particular GPIO bank. + * Possible attribute names: + * - CTRL - 1 for ATR mode 0 for GPIO mode + * - DDR - 1 for output 0 for input + * - OUT - GPIO output level (not ATR mode) + * - ATR_0X - ATR idle state + * - ATR_RX - ATR receive only state + * - ATR_TX - ATR transmit only state + * - ATR_XX - ATR full duplex state + * - READBACK - readback input GPIOs + * \param bank the name of a GPIO bank + * \param attr the name of a GPIO attribute + * \return the value set for this attribute + */ + virtual uint32_t get_gpio_attr(const std::string& bank, const std::string& attr) = 0; + + /************************************************************************** + * Sensor API + *************************************************************************/ + /*! Return a list of RX sensors + * + * The names returned in this list can be used as an input to get_rx_sensor() + */ + virtual std::vector<std::string> get_rx_sensor_names(size_t chan) const = 0; + + /*! Return the sensor value for sensor \p name + * + * \param name Sensor name (e.g. "lo_locked") + * \param chan Channel index + * \throws uhd::key_error if the sensor does not exist + */ + virtual uhd::sensor_value_t get_rx_sensor(const std::string& name, size_t chan) = 0; + + /*! Return a list of TX sensors + * + * The names returned in this list can be used as an input to get_tx_sensor() + */ + virtual std::vector<std::string> get_tx_sensor_names(size_t chan) const = 0; + + /*! Return the sensor value for sensor \p name + * + * \param name Sensor name (e.g. "lo_locked") + * \param chan Channel index + * \throws uhd::key_error if the sensor does not exist + */ + virtual uhd::sensor_value_t get_tx_sensor(const std::string& name, size_t chan) = 0; + + /************************************************************************** + * Streaming-Related API Calls + *************************************************************************/ + /*! Issue stream command: Instruct the RX part of the radio to send samples + * + * \param stream_cmd The actual stream command to execute + * \param port The port for which the stream command is meant + */ + virtual void issue_stream_cmd( + const uhd::stream_cmd_t& stream_cmd, const size_t port) = 0; + + /************************************************************************** + * Radio Identification API Calls + *************************************************************************/ + //! Returns this radio's slot name (typically "A" or "B") + virtual std::string get_slot_name() const = 0; + + //! Return the channel that corresponds to a frontend's name + // + // Example: "0" -> 0 (for UBX), or "A" -> 0 (for E310) + virtual size_t get_chan_from_dboard_fe( + const std::string& fe, const uhd::direction_t direction) const = 0; + + //! Return the frontend name for a channel index + // + // Example: 0 -> "0" (for UBX), or 0 -> "A" (for E310) + virtual std::string get_dboard_fe_from_chan( + const size_t chan, const uhd::direction_t direction) const = 0; + + //! Return the name of the frontend, as given by the dboard driver + virtual std::string get_fe_name( + const size_t chan, const uhd::direction_t direction) const = 0; + + /************************************************************************** + * EEPROM API Calls + *************************************************************************/ + //! Update the daughterboard EEPROM + // + // Note: EEPROMs have finite numbers of write cycles, so don't overuse this + // method! + virtual void set_db_eeprom(const uhd::eeprom_map_t& db_eeprom) = 0; + + //! Return the content of the daughterboard EEPROM + virtual uhd::eeprom_map_t get_db_eeprom() = 0; +}; + +}} // namespace uhd::rfnoc + +#endif /* INCLUDED_LIBUHD_RADIO_BLOCK_CONTROL_HPP */ diff --git a/host/lib/include/uhdlib/rfnoc/radio_control_impl.hpp b/host/lib/include/uhdlib/rfnoc/radio_control_impl.hpp new file mode 100644 index 000000000..c5e990343 --- /dev/null +++ b/host/lib/include/uhdlib/rfnoc/radio_control_impl.hpp @@ -0,0 +1,289 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#include <uhd/rfnoc/defaults.hpp> +#include <uhd/rfnoc/radio_control.hpp> +#include <unordered_map> +#include <mutex> + +#define RFNOC_RADIO_CONSTRUCTOR(CLASS_NAME) \ + CLASS_NAME##_impl(make_args_ptr make_args) : radio_control_impl(std::move(make_args)) + +namespace uhd { namespace rfnoc { + +/*! Base class of radio controllers + * + * All radio control classes should derive from this class (e.g., the X300 radio + * controller, etc.) + * + * Many of the radio_control API calls have virtual (default) implementations + * here, but they can be overridden. + */ +class radio_control_impl : public radio_control +{ +public: + /************************************************************************** + * Structors + *************************************************************************/ + radio_control_impl(make_args_ptr make_args); + + virtual void deinit() {} + + virtual ~radio_control_impl() {} + + /************************************************************************** + * Stream control API calls + *************************************************************************/ + void issue_stream_cmd(const uhd::stream_cmd_t& stream_cmd, const size_t port); + + /************************************************************************** + * Rate-Related API Calls + *************************************************************************/ + virtual double set_rate(const double rate); + virtual double get_rate() const; + virtual meta_range_t get_rate_range() const; + + /************************************************************************** + * RF-specific API calls + *************************************************************************/ + // Setters + virtual void set_tx_antenna(const std::string &ant, const size_t chan); + virtual void set_rx_antenna(const std::string &ant, const size_t chan); + virtual double set_tx_frequency(const double freq, const size_t chan); + virtual double set_rx_frequency(const double freq, const size_t chan); + virtual void set_tx_tune_args(const uhd::device_addr_t&, const size_t chan); + virtual void set_rx_tune_args(const uhd::device_addr_t&, const size_t chan); + virtual double set_tx_gain(const double gain, const size_t chan); + virtual double set_tx_gain(const double gain, const std::string& name, const size_t chan); + virtual double set_rx_gain(const double gain, const size_t chan); + virtual double set_rx_gain(const double gain, const std::string& name, const size_t chan); + virtual void set_rx_agc(const bool enable, const size_t chan); + virtual double set_tx_bandwidth(const double bandwidth, const size_t chan); + virtual double set_rx_bandwidth(const double bandwidth, const size_t chan); + virtual void set_tx_gain_profile(const std::string& profile, const size_t chan); + virtual void set_rx_gain_profile(const std::string& profile, const size_t chan); + + // Getters + virtual std::string get_tx_antenna(const size_t chan) const; + virtual std::string get_rx_antenna(const size_t chan) const; + virtual std::vector<std::string> get_tx_antennas(const size_t chan) const; + virtual std::vector<std::string> get_rx_antennas(const size_t chan) const; + virtual double get_tx_frequency(const size_t); + virtual double get_rx_frequency(const size_t); + virtual uhd::freq_range_t get_tx_frequency_range(const size_t chan) const; + virtual uhd::freq_range_t get_rx_frequency_range(const size_t chan) const; + virtual std::vector<std::string> get_tx_gain_names(const size_t) const; + virtual std::vector<std::string> get_rx_gain_names(const size_t) const; + virtual double get_tx_gain(const size_t); + virtual double get_tx_gain(const std::string&, size_t); + virtual double get_rx_gain(const size_t); + virtual double get_rx_gain(const std::string&, size_t); + virtual uhd::gain_range_t get_tx_gain_range(const size_t) const; + virtual uhd::gain_range_t get_tx_gain_range(const std::string&, const size_t) const; + virtual uhd::gain_range_t get_rx_gain_range(const size_t) const; + virtual uhd::gain_range_t get_rx_gain_range(const std::string&, const size_t) const; + virtual std::vector<std::string> get_tx_gain_profile_names(const size_t chan) const; + virtual std::vector<std::string> get_rx_gain_profile_names(const size_t chan) const; + virtual std::string get_tx_gain_profile(const size_t chan) const; + virtual std::string get_rx_gain_profile(const size_t chan) const; + virtual double get_tx_bandwidth(const size_t); + virtual double get_rx_bandwidth(const size_t); + virtual meta_range_t get_tx_bandwidth_range(size_t chan) const; + virtual meta_range_t get_rx_bandwidth_range(size_t chan) const; + + /************************************************************************** + * LO Controls + *************************************************************************/ + virtual std::vector<std::string> get_rx_lo_names(const size_t chan) const; + virtual std::vector<std::string> get_rx_lo_sources( + const std::string& name, const size_t chan) const; + virtual freq_range_t get_rx_lo_freq_range( + const std::string& name, const size_t chan) const; + virtual void set_rx_lo_source( + const std::string& src, const std::string& name, const size_t chan); + virtual const std::string get_rx_lo_source( + const std::string& name, const size_t chan); + virtual void set_rx_lo_export_enabled( + bool enabled, const std::string& name, const size_t chan); + virtual bool get_rx_lo_export_enabled( + const std::string& name, const size_t chan) const; + virtual double set_rx_lo_freq( + double freq, const std::string& name, const size_t chan); + virtual double get_rx_lo_freq(const std::string& name, const size_t chan); + virtual std::vector<std::string> get_tx_lo_names(const size_t chan) const; + virtual std::vector<std::string> get_tx_lo_sources( + const std::string& name, const size_t chan); + virtual freq_range_t get_tx_lo_freq_range( + const std::string& name, const size_t chan); + virtual void set_tx_lo_source( + const std::string& src, const std::string& name, const size_t chan); + virtual const std::string get_tx_lo_source( + const std::string& name, const size_t chan); + virtual void set_tx_lo_export_enabled( + const bool enabled, const std::string& name, const size_t chan); + virtual bool get_tx_lo_export_enabled(const std::string& name, const size_t chan); + virtual double set_tx_lo_freq( + const double freq, const std::string& name, const size_t chan); + virtual double get_tx_lo_freq(const std::string& name, const size_t chan); + + /************************************************************************** + * Calibration-Related API Calls + *************************************************************************/ + virtual void set_tx_dc_offset(const std::complex<double>& offset, size_t chan); + virtual meta_range_t get_tx_dc_offset_range(size_t chan) const; + virtual void set_tx_iq_balance(const std::complex<double>& correction, size_t chan); + virtual void set_rx_dc_offset(const bool enb, size_t chan = ALL_CHANS); + virtual void set_rx_dc_offset(const std::complex<double>& offset, size_t chan); + virtual meta_range_t get_rx_dc_offset_range(size_t chan) const; + virtual void set_rx_iq_balance(const bool enb, size_t chan); + virtual void set_rx_iq_balance(const std::complex<double>& correction, size_t chan); + + /************************************************************************** + * GPIO Controls + *************************************************************************/ + virtual std::vector<std::string> get_gpio_banks() const; + virtual void set_gpio_attr(const std::string& bank, + const std::string& attr, + const uint32_t value, + const uint32_t mask); + virtual uint32_t get_gpio_attr(const std::string& bank, const std::string& attr); + + /************************************************************************** + * Sensor API + *************************************************************************/ + virtual std::vector<std::string> get_rx_sensor_names(size_t chan) const; + virtual uhd::sensor_value_t get_rx_sensor(const std::string& name, size_t chan); + virtual std::vector<std::string> get_tx_sensor_names(size_t chan) const; + virtual uhd::sensor_value_t get_tx_sensor(const std::string& name, size_t chan); + + /************************************************************************** + * Identification API + *************************************************************************/ + virtual std::string get_fe_name( + const size_t chan, const uhd::direction_t direction) const + { + return get_dboard_fe_from_chan(chan, direction); + } + + /************************************************************************** + * EEPROM API + *************************************************************************/ + virtual void set_db_eeprom(const uhd::eeprom_map_t& db_eeprom); + virtual uhd::eeprom_map_t get_db_eeprom(); + + /*********************************************************************** + * Reg Map + **********************************************************************/ + static const uint16_t MAJOR_COMPAT; + static const uint16_t MINOR_COMPAT; + + /*! Register map common to all radios + * + * See rfnoc_block_radio_regs.vh for details + */ + struct regmap { + static const uint32_t REG_COMPAT_NUM = 0x00; // Compatibility number register offset + static const uint32_t REG_RADIO_WIDTH = 0x1000 + 0x04; // Upper 16 bits is sample width, lower 16 bits is NSPC + + static const uint32_t RADIO_BASE_ADDR = 0x1000; + static const uint32_t REG_CHAN_OFFSET = 128; + static const uint32_t RADIO_ADDR_W = 7; // Address space size per radio + + // General Radio Registers + static const uint32_t REG_LOOPBACK_EN = 0x00; // Loopback enable (connect Tx output to Rx input) + + // Note on the RX and TX Control Registers: These are per-channel, + // which means the values here are offsets. The base address per + // channel is RADIO_BASE_ADDR + i * REG_CHAN_OFFSET, where i is the + // channel index. + + // RX Control Registers + static const uint32_t REG_RX_STATUS = 0x10; // Status of Rx radio + static const uint32_t REG_RX_CMD = 0x14; // The next radio command to execute + static const uint32_t REG_RX_CMD_NUM_WORDS_LO = 0x18; // Number of radio words for the next command (low word) + static const uint32_t REG_RX_CMD_NUM_WORDS_HI = 0x1C; // Number of radio words for the next command (high word) + static const uint32_t REG_RX_CMD_TIME_LO = 0x20; // Time for the next command (low word) + static const uint32_t REG_RX_CMD_TIME_HI = 0x24; // Time for the next command (high word) + static const uint32_t REG_RX_MAX_WORDS_PER_PKT = 0x28; // Maximum packet length to build from Rx data + static const uint32_t REG_RX_ERR_PORT = 0x2C; // Port ID for error reporting + static const uint32_t REG_RX_ERR_REM_PORT = 0x30; // Remote port ID for error reporting + static const uint32_t REG_RX_ERR_REM_EPID = 0x34; // Remote EPID (endpoint ID) for error reporting + static const uint32_t REG_RX_ERR_ADDR = 0x38; // Offset to which to write error code (ADDR+0) and time (ADDR+8) + static const uint32_t REG_RX_DATA = 0x3C; + + // TX Control Registers + static const uint32_t REG_TX_IDLE_VALUE = 0x40; // Value to output when transmitter is idle + static const uint32_t REG_TX_ERROR_POLICY = 0x44; // Tx error policy + static const uint32_t REG_TX_ERR_PORT = 0x48; // Port ID for error reporting + static const uint32_t REG_TX_ERR_REM_PORT = 0x4C; // Remote port ID for error reporting + static const uint32_t REG_TX_ERR_REM_EPID = 0x50; // Remote EPID (endpoint ID) for error reporting + static const uint32_t REG_TX_ERR_ADDR = 0x54; // Offset to which to write error code (ADDR+0) and time (ADDR+8) + + static const uint32_t RX_CMD_STOP = 0; // Stop acquiring at end of next packet + static const uint32_t RX_CMD_FINITE = 1; // Acquire NUM_SAMPS then stop + static const uint32_t RX_CMD_CONTINUOUS = 2; // Acquire until stopped + + static const uint32_t RX_CMD_TIMED_POS = 31; + + static const uint32_t REG_SPI_W = 0x80000 + 168*8; // FIXME + static const uint32_t REG_SPI_R = 0x80000 + 17*8; // FIXME + + static const uint32_t PERIPH_BASE = 0x80000; + static const uint32_t PERIPH_REG_OFFSET = 8; + }; + + //! Tree path to the dboard-specific properties + static const uhd::fs_path DB_PATH; + //! Tree path to the radio frontends' properties + static const uhd::fs_path FE_PATH; + +protected: + //! Properties for samp_rate (one per port) + std::vector<property_t<double>> _samp_rate_in; + //! Properties for samp_rate (one per port) + std::vector<property_t<double>> _samp_rate_out; + +private: + //! Receiver for the async messages + // + // This block will receive all async messages. The following async messages + // are expected to show up: + // - Overrun info + // - Underrun info + // - Late data packets + void async_message_handler(uint32_t addr, const std::vector<uint32_t>& data); + + //! FPGA compat number + const uint32_t _fpga_compat; + + //! Copy of the REG_RADIO_WIDTH register + const uint32_t _radio_width; + + //! Sample width (total width, sc16 == 32 bits per complex sample) + const uint32_t _samp_width; + + //! Samples per cycle + const uint32_t _spc; + + std::vector<property_t<int>> _spp_prop; + //! Properties for type_in (one per port) + std::vector<property_t<io_type_t>> _type_in; + //! Properties for type_out (one per port) + std::vector<property_t<io_type_t>> _type_out; + + mutable std::mutex _cache_mutex; + double _rate = 1.0; + std::unordered_map<size_t, std::string> _tx_antenna; + std::unordered_map<size_t, std::string> _rx_antenna; + std::unordered_map<size_t, double> _tx_freq; + std::unordered_map<size_t, double> _rx_freq; + std::unordered_map<size_t, double> _tx_gain; + std::unordered_map<size_t, double> _rx_gain; + std::unordered_map<size_t, double> _tx_bandwidth; + std::unordered_map<size_t, double> _rx_bandwidth; +}; + +}} // namespace uhd::rfnoc diff --git a/host/lib/rfnoc/CMakeLists.txt b/host/lib/rfnoc/CMakeLists.txt index 366e2d062..127ad6ffd 100644 --- a/host/lib/rfnoc/CMakeLists.txt +++ b/host/lib/rfnoc/CMakeLists.txt @@ -58,6 +58,7 @@ LIBUHD_APPEND_SOURCES( ${CMAKE_CURRENT_SOURCE_DIR}/null_block_ctrl_impl.cpp ${CMAKE_CURRENT_SOURCE_DIR}/window_block_ctrl_impl.cpp ${CMAKE_CURRENT_SOURCE_DIR}/radio_ctrl_impl.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/radio_control_impl.cpp ${CMAKE_CURRENT_SOURCE_DIR}/siggen_block_ctrl_impl.cpp ${CMAKE_CURRENT_SOURCE_DIR}/dma_fifo_block_ctrl_impl.cpp ${CMAKE_CURRENT_SOURCE_DIR}/replay_block_ctrl_impl.cpp diff --git a/host/lib/rfnoc/radio_control_impl.cpp b/host/lib/rfnoc/radio_control_impl.cpp new file mode 100644 index 000000000..d58016b4d --- /dev/null +++ b/host/lib/rfnoc/radio_control_impl.cpp @@ -0,0 +1,771 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#include <uhd/exception.hpp> +#include <uhd/utils/log.hpp> +#include <uhdlib/rfnoc/radio_control_impl.hpp> +#include <uhdlib/utils/compat_check.hpp> +#include <map> +#include <tuple> + +using namespace uhd::rfnoc; + +namespace { + +inline uint32_t get_addr(const uint32_t base_addr, const size_t chan) +{ + return radio_control_impl::regmap::RADIO_BASE_ADDR + base_addr + + radio_control_impl::regmap::REG_CHAN_OFFSET * chan; +} + +const std::string DEFAULT_GAIN_PROFILE("default"); + +} // namespace + +const std::string radio_control::ALL_LOS = "all"; +const std::string radio_control::ALL_GAINS = ""; +const size_t radio_control::ALL_CHANS = size_t(~0); + +const uint16_t radio_control_impl::MAJOR_COMPAT = 0; +const uint16_t radio_control_impl::MINOR_COMPAT = 0; + +const uint32_t radio_control_impl::regmap::REG_COMPAT_NUM; +const uint32_t radio_control_impl::regmap::REG_RADIO_WIDTH; +const uint32_t radio_control_impl::regmap::RADIO_BASE_ADDR; +const uint32_t radio_control_impl::regmap::REG_CHAN_OFFSET; +const uint32_t radio_control_impl::regmap::RADIO_ADDR_W; +const uint32_t radio_control_impl::regmap::REG_LOOPBACK_EN; +const uint32_t radio_control_impl::regmap::REG_RX_STATUS; +const uint32_t radio_control_impl::regmap::REG_RX_CMD; +const uint32_t radio_control_impl::regmap::REG_RX_CMD_NUM_WORDS_LO; +const uint32_t radio_control_impl::regmap::REG_RX_CMD_NUM_WORDS_HI; +const uint32_t radio_control_impl::regmap::REG_RX_CMD_TIME_LO; +const uint32_t radio_control_impl::regmap::REG_RX_CMD_TIME_HI; +const uint32_t radio_control_impl::regmap::REG_RX_MAX_WORDS_PER_PKT; +const uint32_t radio_control_impl::regmap::REG_RX_ERR_PORT; +const uint32_t radio_control_impl::regmap::REG_RX_ERR_REM_PORT; +const uint32_t radio_control_impl::regmap::REG_RX_ERR_REM_EPID; +const uint32_t radio_control_impl::regmap::REG_RX_ERR_ADDR; +const uint32_t radio_control_impl::regmap::REG_TX_IDLE_VALUE; +const uint32_t radio_control_impl::regmap::REG_TX_ERROR_POLICY; +const uint32_t radio_control_impl::regmap::REG_TX_ERR_PORT; +const uint32_t radio_control_impl::regmap::REG_TX_ERR_REM_PORT; +const uint32_t radio_control_impl::regmap::REG_TX_ERR_REM_EPID; +const uint32_t radio_control_impl::regmap::REG_TX_ERR_ADDR; +const uint32_t radio_control_impl::regmap::RX_CMD_STOP; +const uint32_t radio_control_impl::regmap::RX_CMD_FINITE; +const uint32_t radio_control_impl::regmap::RX_CMD_CONTINUOUS; +const uint32_t radio_control_impl::regmap::RX_CMD_TIMED_POS; + +const uhd::fs_path radio_control_impl::DB_PATH("dboard"); +const uhd::fs_path radio_control_impl::FE_PATH("frontends"); + +/**************************************************************************** + * Structors + ***************************************************************************/ +radio_control_impl::radio_control_impl(make_args_ptr make_args) + : radio_control(std::move(make_args)) + , _fpga_compat(regs().peek32(regmap::REG_COMPAT_NUM)) + , _radio_width(regs().peek32(regmap::REG_RADIO_WIDTH)) + , _samp_width(_radio_width >> 16) + , _spc(_radio_width & 0xFFFF) +{ + uhd::assert_fpga_compat(MAJOR_COMPAT, + MINOR_COMPAT, + _fpga_compat, + get_unique_id(), + get_unique_id(), + false /* Let it slide if minors mismatch */ + ); + RFNOC_LOG_TRACE( + "Loading radio with SPC=" << _spc << ", num_inputs=" << get_num_input_ports() + << ", num_outputs=" << get_num_output_ports()); + set_prop_forwarding_policy(forwarding_policy_t::DROP); + set_action_forwarding_policy(forwarding_policy_t::DROP); + regs().register_async_msg_handler( + [this](uint32_t addr, const std::vector<uint32_t>& data) { + this->async_message_handler(addr, data); + }); + register_action_handler(ACTION_KEY_STREAM_CMD, + [this](const res_source_info& src, action_info::sptr action) { + stream_cmd_action_info::sptr stream_cmd_action = + std::dynamic_pointer_cast<stream_cmd_action_info>(action); + if (!stream_cmd_action) { + RFNOC_LOG_WARNING("Received invalid stream command action!"); + return; + } + RFNOC_LOG_TRACE( + "Received stream command: " << stream_cmd_action->stream_cmd.stream_mode + << " to " << src.to_string()); + if (src.type != res_source_info::OUTPUT_EDGE) { + RFNOC_LOG_WARNING("Received stream command, but not to output port! Ignoring."); + return; + } + const size_t port = src.instance; + if (port > get_num_output_ports()) { + RFNOC_LOG_WARNING("Received stream command to invalid output port!"); + return; + } + issue_stream_cmd(stream_cmd_action->stream_cmd, port); + }); + // Register spp properties and resolvers + _spp_prop.reserve(get_num_output_ports()); + _samp_rate_in.reserve(get_num_input_ports()); + _samp_rate_out.reserve(get_num_output_ports()); + _type_in.reserve(get_num_input_ports()); + _type_out.reserve(get_num_output_ports()); + for (size_t chan = 0; chan < get_num_output_ports(); ++chan) { + _spp_prop.push_back(property_t<int>( + PROP_KEY_SPP, DEFAULT_SPP, {res_source_info::USER, chan})); + _samp_rate_in.push_back(property_t<double>( + PROP_KEY_SAMP_RATE, get_tick_rate(), {res_source_info::INPUT_EDGE, chan})); + _samp_rate_out.push_back(property_t<double>( + PROP_KEY_SAMP_RATE, get_tick_rate(), {res_source_info::OUTPUT_EDGE, chan})); + _type_in.push_back(property_t<io_type_t>( + PROP_KEY_TYPE, IO_TYPE_SC16, {res_source_info::INPUT_EDGE, chan})); + _type_out.push_back(property_t<io_type_t>( + PROP_KEY_TYPE, IO_TYPE_SC16, {res_source_info::OUTPUT_EDGE, chan})); + + register_property(&_spp_prop.back(), [this, chan, &spp = _spp_prop.back()]() { + const uint32_t words_per_pkt = spp.get(); + RFNOC_LOG_TRACE( + "Setting words_per_pkt to " << words_per_pkt << " on chan " << chan); + regs().poke32( + get_addr(regmap::REG_RX_MAX_WORDS_PER_PKT, chan), words_per_pkt); + }); + register_property(&_samp_rate_in.back()); + register_property(&_samp_rate_out.back()); + register_property(&_type_in.back()); + register_property(&_type_out.back()); + add_property_resolver({&_spp_prop.back()}, + {&_spp_prop.back()}, + [this, chan, &spp = _spp_prop.back()]() { + RFNOC_LOG_TRACE("Calling resolver for spp@" << chan); + if (spp.get() % _spc) { + spp = spp.get() - (spp.get() % _spc); + RFNOC_LOG_WARNING( + "spp must be a multiple of the block bus width! Coercing to " + << spp.get()); + } + if (spp.get() <= 0) { + spp = DEFAULT_SPP; + RFNOC_LOG_WARNING( + "spp must be greater than zero! Coercing to " << spp.get()); + } + }); + add_property_resolver({&_samp_rate_in.back(), &_samp_rate_out.back()}, + {&_samp_rate_in.back(), &_samp_rate_out.back()}, + [this, chan, + &samp_rate_in = _samp_rate_in.at(chan), + &samp_rate_out = _samp_rate_out.at(chan)]() { + RFNOC_LOG_TRACE("Calling resolver for samp_rate@" << chan); + samp_rate_in = set_rate(samp_rate_in.get()); + samp_rate_out = samp_rate_in.get(); + }); + // Resolvers for type: These are constants + add_property_resolver({&_type_in.back()}, + {&_type_in.back()}, + [& type_in = _type_in.back()]() { type_in.set(IO_TYPE_SC16); }); + add_property_resolver({&_type_out.back()}, + {&_type_out.back()}, + [& type_out = _type_out.back()]() { type_out.set(IO_TYPE_SC16); }); + } +} + +/****************************************************************************** + * Rate-Related API Calls + *****************************************************************************/ +double radio_control_impl::set_rate(const double rate) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + _rate = rate; + return rate; + // FIXME: + ////_tick_rate = rate; + ////_time64->set_tick_rate(_tick_rate); + ////_time64->self_test(); + //// set_command_tick_rate(rate); +} + +double radio_control_impl::get_rate() const +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _rate; +} + +uhd::meta_range_t radio_control_impl::get_rate_range() const +{ + RFNOC_LOG_TRACE("Using default implementation of get_rx_rate_range()"); + uhd::meta_range_t result; + result.push_back(get_rate()); + return result; +} + +/**************************************************************************** + * RF API + ***************************************************************************/ +void radio_control_impl::set_tx_antenna(const std::string& ant, const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + _tx_antenna[chan] = ant; +} + +void radio_control_impl::set_rx_antenna(const std::string& ant, const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + _rx_antenna[chan] = ant; +} + +double radio_control_impl::set_tx_frequency(const double freq, const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _tx_freq[chan] = freq; +} + +void radio_control_impl::set_tx_tune_args(const uhd::device_addr_t&, const size_t) +{ + RFNOC_LOG_TRACE("tune_args not supported by this radio."); +} + +double radio_control_impl::set_rx_frequency(const double freq, const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _rx_freq[chan] = freq; +} + +void radio_control_impl::set_rx_tune_args(const uhd::device_addr_t&, const size_t) +{ + RFNOC_LOG_TRACE("tune_args not supported by this radio."); +} + +std::vector<std::string> radio_control_impl::get_tx_gain_names(const size_t) const +{ + return {ALL_GAINS}; +} + +std::vector<std::string> radio_control_impl::get_rx_gain_names(const size_t) const +{ + return {ALL_GAINS}; +} + +uhd::gain_range_t radio_control_impl::get_tx_gain_range(const size_t chan) const +{ + RFNOC_LOG_DEBUG("Using default implementation of get_tx_gain_range()"); + uhd::gain_range_t result; + std::lock_guard<std::mutex> l(_cache_mutex); + result.push_back(_rx_gain.at(chan)); + return result; +} + +uhd::gain_range_t radio_control_impl::get_tx_gain_range( + const std::string& name, const size_t chan) const +{ + if (name != ALL_GAINS) { + throw uhd::value_error( + std::string("get_tx_gain_range(): Unknown gain name `") + name + "'!"); + } + return get_tx_gain_range(chan); +} + +uhd::gain_range_t radio_control_impl::get_rx_gain_range(const size_t chan) const +{ + RFNOC_LOG_DEBUG("Using default implementation of get_rx_gain_range()"); + uhd::gain_range_t result; + std::lock_guard<std::mutex> l(_cache_mutex); + result.push_back(_rx_gain.at(chan)); + return result; +} + +uhd::gain_range_t radio_control_impl::get_rx_gain_range(const std::string& name, const size_t chan) const +{ + if (name != ALL_GAINS) { + throw uhd::value_error( + std::string("get_rx_gain_range(): Unknown gain name `") + name + "'!"); + } + return get_rx_gain_range(chan); +} + +double radio_control_impl::set_tx_gain(const double gain, const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + _tx_gain[chan] = gain; + return gain; +} + +double radio_control_impl::set_tx_gain(const double gain, const std::string& name, const size_t chan) +{ + if (name != ALL_GAINS) { + throw uhd::key_error( + std::string("set_tx_gain(): Gain name `") + name + "' is not defined!"); + } + return set_tx_gain(gain, chan); +} + +double radio_control_impl::set_rx_gain(const double gain, const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + _rx_gain[chan] = gain; + return gain; +} + +double radio_control_impl::set_rx_gain(const double gain, const std::string& name, const size_t chan) +{ + if (name != ALL_GAINS) { + throw uhd::key_error( + std::string("set_rx_gain(): Gain name `") + name + "' is not defined!"); + } + return set_rx_gain(gain, chan); +} + +void radio_control_impl::set_rx_agc(const bool, const size_t) +{ + throw uhd::not_implemented_error("set_rx_agc() is not supported on this radio!"); +} + +void radio_control_impl::set_tx_gain_profile(const std::string& profile, const size_t) +{ + if (profile != DEFAULT_GAIN_PROFILE) { + throw uhd::value_error( + std::string("set_tx_gain_profile(): Unknown gain profile: `") + profile + + "'"); + } +} + +void radio_control_impl::set_rx_gain_profile(const std::string& profile, const size_t) +{ + if (profile != DEFAULT_GAIN_PROFILE) { + throw uhd::value_error( + std::string("set_rx_gain_profile(): Unknown gain profile: `") + profile + + "'"); + } +} + +std::vector<std::string> radio_control_impl::get_tx_gain_profile_names(const size_t) const +{ + return {DEFAULT_GAIN_PROFILE}; +} + +std::vector<std::string> radio_control_impl::get_rx_gain_profile_names(const size_t) const +{ + return {DEFAULT_GAIN_PROFILE}; +} + +std::string radio_control_impl::get_tx_gain_profile(const size_t) const +{ + return DEFAULT_GAIN_PROFILE; +} + +std::string radio_control_impl::get_rx_gain_profile(const size_t) const +{ + return DEFAULT_GAIN_PROFILE; +} + +double radio_control_impl::set_tx_bandwidth(const double bandwidth, const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _tx_bandwidth[chan] = bandwidth; +} + +double radio_control_impl::set_rx_bandwidth(const double bandwidth, const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _rx_bandwidth[chan] = bandwidth; +} + +//void radio_control_impl::set_time_sync(const uhd::time_spec_t& time) +//{ + //// FIXME +//} + +std::string radio_control_impl::get_tx_antenna(const size_t chan) const +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _tx_antenna.at(chan); +} + +std::string radio_control_impl::get_rx_antenna(const size_t chan) const +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _rx_antenna.at(chan); +} + +std::vector<std::string> radio_control_impl::get_tx_antennas(const size_t chan) const +{ + RFNOC_LOG_DEBUG("get_tx_antennas(): Using default implementation."); + std::lock_guard<std::mutex> l(_cache_mutex); + return {_tx_antenna.at(chan)}; +} + +std::vector<std::string> radio_control_impl::get_rx_antennas(const size_t chan) const +{ + RFNOC_LOG_DEBUG("get_rx_antennas(): Using default implementation."); + std::lock_guard<std::mutex> l(_cache_mutex); + return {_rx_antenna.at(chan)}; +} + +double radio_control_impl::get_tx_frequency(const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _tx_freq.at(chan); +} + +double radio_control_impl::get_rx_frequency(const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _rx_freq.at(chan); +} + +uhd::freq_range_t radio_control_impl::get_tx_frequency_range(const size_t) const +{ + RFNOC_LOG_WARNING( + "get_tx_frequency_range() not implemented! Returning current rate only."); + uhd::freq_range_t result; + result.push_back(get_rate()); + return result; +} + +uhd::freq_range_t radio_control_impl::get_rx_frequency_range(const size_t) const +{ + RFNOC_LOG_WARNING( + "get_rx_frequency_range() not implemented! Returning current rate only."); + uhd::freq_range_t result; + result.push_back(get_rate()); + return result; +} + +double radio_control_impl::get_tx_gain(const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _tx_gain.at(chan); +} + +double radio_control_impl::get_rx_gain(const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _rx_gain.at(chan); +} + +double radio_control_impl::get_tx_gain(const std::string& name, const size_t chan) +{ + if (name != ALL_GAINS) { + throw uhd::value_error(std::string("get_tx_gain(): Unknown gain name `") + name + "'"); + } + return get_tx_gain(chan); +} + +double radio_control_impl::get_rx_gain(const std::string& name, const size_t chan) +{ + if (name != ALL_GAINS) { + throw uhd::value_error(std::string("get_rx_gain(): Unknown gain name `") + name + "'"); + } + return get_rx_gain(chan); +} + +double radio_control_impl::get_tx_bandwidth(const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _tx_bandwidth.at(chan); +} + +double radio_control_impl::get_rx_bandwidth(const size_t chan) +{ + std::lock_guard<std::mutex> l(_cache_mutex); + return _rx_bandwidth.at(chan); +} + +uhd::meta_range_t radio_control_impl::get_tx_bandwidth_range(size_t chan) const +{ + RFNOC_LOG_DEBUG("get_tx_bandwidth_range(): Using default implementation."); + uhd::meta_range_t result; + std::lock_guard<std::mutex> l(_cache_mutex); + result.push_back(_rx_bandwidth.at(chan)); + return result; +} + +uhd::meta_range_t radio_control_impl::get_rx_bandwidth_range(size_t chan) const +{ + RFNOC_LOG_DEBUG("get_tx_bandwidth_range(): Using default implementation."); + uhd::meta_range_t result; + std::lock_guard<std::mutex> l(_cache_mutex); + result.push_back(_rx_bandwidth.at(chan)); + return result; +} + +/****************************************************************************** + * LO Default API + *****************************************************************************/ +std::vector<std::string> radio_control_impl::get_rx_lo_names(const size_t) const +{ + return {}; +} + +std::vector<std::string> radio_control_impl::get_rx_lo_sources( + const std::string&, const size_t) const +{ + return {"internal"}; +} + +uhd::freq_range_t radio_control_impl::get_rx_lo_freq_range( + const std::string&, const size_t) const +{ + return uhd::freq_range_t(); +} + +void radio_control_impl::set_rx_lo_source( + const std::string&, const std::string&, const size_t) +{ + throw uhd::not_implemented_error("set_rx_lo_source is not supported on this radio"); +} + +const std::string radio_control_impl::get_rx_lo_source(const std::string&, const size_t) +{ + return "internal"; +} + +void radio_control_impl::set_rx_lo_export_enabled(bool, const std::string&, const size_t) +{ + throw uhd::not_implemented_error( + "set_rx_lo_export_enabled is not supported on this radio"); +} + +bool radio_control_impl::get_rx_lo_export_enabled(const std::string&, const size_t) const +{ + return false; +} +double radio_control_impl::set_rx_lo_freq(double, const std::string&, const size_t) +{ + throw uhd::not_implemented_error("set_rx_lo_freq is not supported on this radio"); +} + +double radio_control_impl::get_rx_lo_freq(const std::string&, const size_t chan) +{ + return get_rx_frequency(chan); +} + +std::vector<std::string> radio_control_impl::get_tx_lo_names(const size_t) const +{ + return {}; +} + +std::vector<std::string> radio_control_impl::get_tx_lo_sources(const std::string&, const size_t) +{ + return {"internal"}; +} + +uhd::freq_range_t radio_control_impl::get_tx_lo_freq_range(const std::string&, const size_t) +{ + return uhd::freq_range_t(); +} + +void radio_control_impl::set_tx_lo_source( + const std::string&, const std::string&, const size_t) +{ + throw uhd::not_implemented_error("set_tx_lo_source is not supported on this radio"); +} +const std::string radio_control_impl::get_tx_lo_source(const std::string&, const size_t) +{ + return "internal"; +} + +void radio_control_impl::set_tx_lo_export_enabled( + const bool, const std::string&, const size_t) +{ + throw uhd::not_implemented_error( + "set_tx_lo_export_enabled is not supported on this radio"); +} + +bool radio_control_impl::get_tx_lo_export_enabled(const std::string&, const size_t) +{ + return false; +} + +double radio_control_impl::set_tx_lo_freq(const double, const std::string&, const size_t) +{ + throw uhd::not_implemented_error("set_tx_lo_freq is not supported on this radio"); +} + +double radio_control_impl::get_tx_lo_freq(const std::string&, const size_t chan) +{ + return get_tx_frequency(chan); +} + +/****************************************************************************** + * Calibration-Related API Calls + *****************************************************************************/ +void radio_control_impl::set_tx_dc_offset(const std::complex<double>&, size_t) +{ + throw uhd::not_implemented_error("set_tx_dc_offset() is not supported on this radio"); +} + +uhd::meta_range_t radio_control_impl::get_tx_dc_offset_range(size_t) const +{ + return uhd::meta_range_t(0, 0); +} + +void radio_control_impl::set_tx_iq_balance(const std::complex<double>&, size_t) +{ + throw uhd::not_implemented_error( + "set_tx_iq_balance() is not supported on this radio"); +} + +void radio_control_impl::set_rx_dc_offset(const bool enb, size_t) +{ + RFNOC_LOG_DEBUG("set_rx_dc_offset() has no effect on this radio"); + if (enb) { + throw uhd::not_implemented_error( + "set_rx_dc_offset() is not supported on this radio"); + } +} + +void radio_control_impl::set_rx_dc_offset(const std::complex<double>&, size_t) +{ + throw uhd::not_implemented_error("set_rx_dc_offset() is not supported on this radio"); +} + +uhd::meta_range_t radio_control_impl::get_rx_dc_offset_range(size_t) const +{ + return uhd::meta_range_t(0, 0); +} + +void radio_control_impl::set_rx_iq_balance(const bool enb, size_t) +{ + RFNOC_LOG_DEBUG("set_rx_iq_balance() has no effect on this radio"); + if (enb) { + throw uhd::not_implemented_error( + "set_rx_iq_balance() is not supported on this radio"); + } +} + +void radio_control_impl::set_rx_iq_balance(const std::complex<double>&, size_t) +{ + throw uhd::not_implemented_error( + "set_rx_iq_balance() is not supported on this radio"); +} + +/****************************************************************************** + * GPIO Controls + *****************************************************************************/ +std::vector<std::string> radio_control_impl::get_gpio_banks() const +{ + return {}; +} + +void radio_control_impl::set_gpio_attr( + const std::string&, const std::string&, const uint32_t, const uint32_t) +{ + throw uhd::not_implemented_error("set_gpio_attr() not implemented on this radio!"); +} + +uint32_t radio_control_impl::get_gpio_attr(const std::string&, const std::string&) +{ + throw uhd::not_implemented_error("get_gpio_attr() not implemented on this radio!"); +} + +/************************************************************************** + * Sensor API + *************************************************************************/ +std::vector<std::string> radio_control_impl::get_rx_sensor_names(size_t) const +{ + return {}; +} + +uhd::sensor_value_t radio_control_impl::get_rx_sensor(const std::string& name, size_t) +{ + throw uhd::key_error(std::string("Unknown RX sensor: ") + name); +} + +std::vector<std::string> radio_control_impl::get_tx_sensor_names(size_t) const +{ + return {}; +} + +uhd::sensor_value_t radio_control_impl::get_tx_sensor(const std::string& name, size_t) +{ + throw uhd::key_error(std::string("Unknown TX sensor: ") + name); +} + +/************************************************************************** + * EEPROM API + *************************************************************************/ +void radio_control_impl::set_db_eeprom(const uhd::eeprom_map_t&) +{ + throw uhd::not_implemented_error("set_db_eeprom() not implemented for this radio!"); +} + +uhd::eeprom_map_t radio_control_impl::get_db_eeprom() +{ + return {}; +} + +/**************************************************************************** + * Streaming API + ***************************************************************************/ +void radio_control_impl::issue_stream_cmd( + const uhd::stream_cmd_t& stream_cmd, const size_t chan) +{ + // std::lock_guard<std::mutex> lock(_mutex); + RFNOC_LOG_TRACE("radio_control_impl::issue_stream_cmd(chan=" + << chan << ", mode=" << char(stream_cmd.stream_mode) << ")"); + //_continuous_streaming[chan] = + //(stream_cmd.stream_mode == stream_cmd_t::STREAM_MODE_START_CONTINUOUS); + + // calculate the command word + const std::unordered_map<stream_cmd_t::stream_mode_t, uint32_t, std::hash<size_t>> + stream_mode_to_cmd_word{ + {stream_cmd_t::STREAM_MODE_START_CONTINUOUS, regmap::RX_CMD_CONTINUOUS}, + {stream_cmd_t::STREAM_MODE_STOP_CONTINUOUS, regmap::RX_CMD_STOP}, + {stream_cmd_t::STREAM_MODE_NUM_SAMPS_AND_DONE, regmap::RX_CMD_FINITE}, + {stream_cmd_t::STREAM_MODE_NUM_SAMPS_AND_MORE, regmap::RX_CMD_FINITE}}; + const uint32_t cmd_bits = stream_mode_to_cmd_word.at(stream_cmd.stream_mode); + const uint32_t cmd_word = + cmd_bits + | (uint32_t((stream_cmd.stream_now) ? 0 : 1) << regmap::RX_CMD_TIMED_POS); + + if (cmd_bits == regmap::RX_CMD_FINITE) { + if (stream_cmd.num_samps == 0) { + throw uhd::value_error("When requesting a finite number of samples, the " + "number of samples must be greater than zero."); + } + // FIXME: The num words might be different from num_samps, check the + // radio width + const uint64_t num_words = stream_cmd.num_samps; + constexpr uint64_t max_num_words = 0x00FFFFFFFFFFFF; // 48 bits + if (num_words > max_num_words) { + RFNOC_LOG_ERROR("Requesting too many samples in a single burst! " + "Requested " + + std::to_string(stream_cmd.num_samps) + + ", maximum " + "is " + + std::to_string(max_num_words) + "."); // FIXME + RFNOC_LOG_INFO( + "Note that a decimation block will increase the number of samples " + "per burst by the decimation factor. Your application may have " + "requested fewer samples."); + throw uhd::value_error("Requested too many samples in a single burst."); + } + regs().poke32( + get_addr(regmap::REG_RX_CMD_NUM_WORDS_HI, chan), uint32_t(num_words >> 32)); + regs().poke32(get_addr(regmap::REG_RX_CMD_NUM_WORDS_LO, chan), + uint32_t(num_words & 0xFFFFFFFF)); + } + if (!stream_cmd.stream_now) { + const uint64_t ticks = stream_cmd.time_spec.to_ticks(get_tick_rate()); + regs().poke32(get_addr(regmap::REG_RX_CMD_TIME_HI, chan), uint32_t(ticks >> 32)); + regs().poke32(get_addr(regmap::REG_RX_CMD_TIME_LO, chan), uint32_t(ticks >> 0)); + } + regs().poke32(get_addr(regmap::REG_RX_CMD, chan), cmd_word); +} + +/****************************************************************************** + * Private methods + *****************************************************************************/ +void radio_control_impl::async_message_handler( + uint32_t addr, const std::vector<uint32_t>& data) +{ + RFNOC_LOG_TRACE( + str(boost::format("Received async message to addr 0x%08X, data length %d words.") + % addr % data.size())); +} diff --git a/host/tests/CMakeLists.txt b/host/tests/CMakeLists.txt index bd0964def..d443a4db1 100644 --- a/host/tests/CMakeLists.txt +++ b/host/tests/CMakeLists.txt @@ -249,6 +249,7 @@ UHD_ADD_NONAPI_TEST( ${CMAKE_SOURCE_DIR}/lib/rfnoc/block_id.cpp ${CMAKE_SOURCE_DIR}/lib/utils/compat_check.cpp ${CMAKE_SOURCE_DIR}/lib/rfnoc/ddc_block_control.cpp + ${CMAKE_SOURCE_DIR}/lib/rfnoc/radio_control_impl.cpp ${CMAKE_SOURCE_DIR}/lib/usrp/cores/dsp_core_utils.cpp ) |