// // Copyright 2018 Ettus Research, a National Instruments Company // // SPDX-License-Identifier: GPL-3.0-or-later // #ifndef INCLUDED_LIBUHD_RFNOC_RHODIUM_RADIO_CTRL_IMPL_HPP #define INCLUDED_LIBUHD_RFNOC_RHODIUM_RADIO_CTRL_IMPL_HPP #include "rhodium_cpld_ctrl.hpp" #include "rhodium_cpld_regs.hpp" #include #include #include #include #include #include #include #include #include #include namespace uhd { namespace rfnoc { /*! \brief Provide access to an Rhodium radio. */ class rhodium_radio_ctrl_impl : public radio_ctrl_impl, public rpc_block_ctrl { public: typedef boost::shared_ptr sptr; //! Frequency bands for RX. Bands are a function of the analog filter banks enum class rx_band { RX_BAND_INVALID, RX_BAND_0, RX_BAND_1, RX_BAND_2, RX_BAND_3, RX_BAND_4, RX_BAND_5, RX_BAND_6, RX_BAND_7 }; //! Frequency bands for TX. Bands are a function of the analog filter banks enum class tx_band { TX_BAND_INVALID, TX_BAND_0, TX_BAND_1, TX_BAND_2, TX_BAND_3, TX_BAND_4, TX_BAND_5, TX_BAND_6, TX_BAND_7 }; /************************************************************************ * Structors ***********************************************************************/ UHD_RFNOC_RADIO_BLOCK_CONSTRUCTOR_DECL(rhodium_radio_ctrl) virtual ~rhodium_radio_ctrl_impl(); /************************************************************************ * API calls ***********************************************************************/ // Note: We use the cached values in radio_ctrl_impl, so most getters are // not reimplemented here double set_rate(const double rate); void set_tx_antenna(const std::string &ant, const size_t chan); void set_rx_antenna(const std::string &ant, const size_t chan); double set_tx_frequency(const double freq, const size_t chan); double set_rx_frequency(const double freq, const size_t chan); double set_tx_bandwidth(const double bandwidth, const size_t chan); double set_rx_bandwidth(const double bandwidth, const size_t chan); double set_tx_gain(const double gain, const size_t chan); double set_rx_gain(const double gain, const size_t chan); // LO Property Getters std::vector get_tx_lo_names(const size_t chan); std::vector get_rx_lo_names(const size_t chan); std::vector get_tx_lo_sources(const std::string& name, const size_t chan); std::vector get_rx_lo_sources(const std::string& name, const size_t chan); freq_range_t get_tx_lo_freq_range(const std::string& name, const size_t chan); freq_range_t get_rx_lo_freq_range(const std::string& name, const size_t chan); // LO Frequency Control double set_tx_lo_freq(const double freq, const std::string& name, const size_t chan); double set_rx_lo_freq(const double freq, const std::string& name, const size_t chan); double get_tx_lo_freq(const std::string& name, const size_t chan); double get_rx_lo_freq(const std::string& name, const size_t chan); // LO Source Control void set_tx_lo_source(const std::string& src, const std::string& name, const size_t chan); void set_rx_lo_source(const std::string& src, const std::string& name, const size_t chan); const std::string get_tx_lo_source(const std::string& name, const size_t chan); const std::string get_rx_lo_source(const std::string& name, const size_t chan); // LO Export Control void set_tx_lo_export_enabled(const bool enabled, const std::string& name, const size_t chan); void set_rx_lo_export_enabled(const bool enabled, const std::string& name, const size_t chan); bool get_tx_lo_export_enabled(const std::string& name, const size_t chan); bool get_rx_lo_export_enabled(const std::string& name, const size_t chan); // LO Distribution Control void set_tx_lo_output_enabled(const bool enabled, const std::string& port_name, const size_t chan); void set_rx_lo_output_enabled(const bool enabled, const std::string& port_name, const size_t chan); bool get_tx_lo_output_enabled(const std::string& port_name, const size_t chan); bool get_rx_lo_output_enabled(const std::string& port_name, const size_t chan); // LO Gain Control //! Set the external gain for a TX LO // Out of range values will be coerced double set_tx_lo_gain(const double gain, const std::string &name, const size_t chan); //! Set the external gain for an RX LO // Out of range values will be coerced double set_rx_lo_gain(const double gain, const std::string &name, const size_t chan); double get_tx_lo_gain(const std::string &name, const size_t chan); double get_rx_lo_gain(const std::string &name, const size_t chan); // LO Output Power Control //! Set the output power setting of a TX LO // Out of range values will be coerced double set_tx_lo_power(const double power, const std::string &name, const size_t chan); //! Set the output power setting of a RX LO // Out of range values will be coerced double set_rx_lo_power(const double power, const std::string &name, const size_t chan); double get_tx_lo_power(const std::string &name, const size_t chan); double get_rx_lo_power(const std::string &name, const size_t chan); size_t get_chan_from_dboard_fe(const std::string &fe, const direction_t dir); std::string get_dboard_fe_from_chan(const size_t chan, const direction_t dir); void set_rpc_client( uhd::rpc_client::sptr rpcc, const uhd::device_addr_t &block_args ); private: /************************************************************************** * Helpers *************************************************************************/ //! Initialize all the peripherals connected to this block void _init_peripherals(); //! Set state of this class to sensible defaults void _init_defaults(); //! Init a subtree for the RF frontends void _init_frontend_subtree( uhd::property_tree::sptr subtree, const size_t chan_idx ); //! Initialize property tree void _init_prop_tree(); //! Discover and initialize any mpm sensors void _init_mpm_sensors( const direction_t dir, const size_t chan_idx ); //! Get the frequency range for an LO freq_range_t _get_lo_freq_range(const std::string &name) const; //! Get the current lowband intermediate frequency double _get_lowband_lo_freq() const; //! Configure LO1's export void _set_lo1_export_enabled( const bool enabled, const direction_t dir ); //! Validate that port_name is valid, and that LO distribution functions // can be called in this instance void _validate_output_port( const std::string& port_name, const std::string& function_name ); //! Configure LO Distribution board's termination switches void _set_lo_output_enabled( const bool enabled, const std::string& port_name, const direction_t dir ); bool _get_lo_output_enabled( const std::string& port_name, const direction_t dir ); //! Configure LO1's output power // Out of range values will be coerced to [0-63] double _set_lo1_power( const double power, const direction_t dir ); //! Configure ATR registers and update the cached antenna value from the // new antenna value. // ATR registers control SW10 and the frontend LEDs. void _update_atr(const std::string& ant, const direction_t dir); //! Map a frequency in Hz to an rx_band value. Will return // rx_band::INVALID_BAND if the frequency is out of range. static rx_band _map_freq_to_rx_band(const double freq); //! Map a frequency in Hz to an tx_band value. Will return // tx_band::INVALID_BAND if the frequency is out of range. static tx_band _map_freq_to_tx_band(const double freq); //! Return if the given rx frequency is in lowband // NOTE: Returns false if frequency is out of Rh's rx frequency range static bool _is_rx_lowband(const double freq); //! Return if the given tx frequency is in lowband // NOTE: Returns false if frequency is out of Rh's tx frequency range static bool _is_tx_lowband(const double freq); //! Return the gain range for dir static uhd::gain_range_t _get_gain_range(const direction_t dir); //! Return the gain range of the LMX LO static uhd::gain_range_t _get_lo_gain_range(); //! Return the power setting range of the LMX LO static uhd::gain_range_t _get_lo_power_range(); //! Lookup the LO DSA setting from LO frequency int _get_lo_dsa_setting(const double freq, const direction_t dir); //! Lookup the LO output power setting from LO frequency unsigned int _get_lo_power_setting(const double freq); bool _get_spur_dodging_enabled(const uhd::direction_t dir) const; double _get_spur_dodging_threshold(const uhd::direction_t dir) const; bool _get_highband_spur_reduction_enabled(const uhd::direction_t dir) const; bool _get_timed_command_enabled() const; /************************************************************************** * Sensors *************************************************************************/ //! Return LO lock status. Factors in current band (low/high) and // direction (TX/RX) bool get_lo_lock_status(const direction_t dir) const; /************************************************************************** * Frontend Controls *************************************************************************/ void _set_tx_fe_connection(const std::string &conn); void _set_rx_fe_connection(const std::string &conn); std::string _get_tx_fe_connection() const; std::string _get_rx_fe_connection() const; /************************************************************************** * CPLD Controls (implemented in rhodium_radio_ctrl_cpld.cpp) *************************************************************************/ void _update_rx_freq_switches( const double freq ); void _update_tx_freq_switches( const double freq ); void _update_rx_input_switches( const std::string &input ); void _update_tx_output_switches( const std::string &output ); /************************************************************************** * Private attributes *************************************************************************/ //! Locks access to the antenna cached values std::mutex _ant_mutex; //! Letter representation of the radio we're currently running std::string _radio_slot; //! Prepended for all dboard RPC calls std::string _rpc_prefix; //! Additional block args; gets set during set_rpc_client() uhd::device_addr_t _block_args; //! Reference to the RPC client uhd::rpc_client::sptr _rpcc; //! Reference to the SPI core uhd::spi_iface::sptr _spi; //! Reference to the TX LO lmx2592_iface::sptr _tx_lo; //! Reference to the RX LO lmx2592_iface::sptr _rx_lo; //! Reference to the CPLD controls. Even if there's multiple radios, // there's only one CPLD control. std::shared_ptr _cpld; //! ATR controls. These control the external DSA and the AD9371 gain // up/down bits. They do *not* control the ATR state of the CPLD, the // tx/rx run states are hooked up directly to the CPLD. // // Every radio channel gets its own ATR state register. usrp::gpio_atr::gpio_atr_3000::sptr _gpio; //! Front panel GPIO controller. Note that only one radio block per // module can be the FP-GPIO master. usrp::gpio_atr::gpio_atr_3000::sptr _fp_gpio; //! One DSP core per channel rx_frontend_core_3000::sptr _rx_fe_core; tx_frontend_core_200::sptr _tx_fe_core; //! Sampling rate, and also ref clock frequency for the lowband LOs. double _master_clock_rate = 1.0; //! Saved frontend connection for DSP core std::string _rx_fe_connection; std::string _tx_fe_connection; //! Desired RF frequency std::map _desired_rf_freq = { {RX_DIRECTION, 2.44e9}, {TX_DIRECTION, 2.44e9} }; //! Frequency at which gain setting was last applied. The CPLD requires a new gain // control write when switching between lowband and highband frequencies, so save // the frequency when sending a gain control command. double _tx_frequency_at_last_gain_write = 0.0; double _rx_frequency_at_last_gain_write = 0.0; //! LO gain double _lo_rx_gain = 0.0; double _lo_tx_gain = 0.0; //! LO output power double _lo_rx_power = 0.0; double _lo_tx_power = 0.0; //! Gain profile std::map _gain_profile = { {RX_DIRECTION, "default"}, {TX_DIRECTION, "default"} }; //! LO source std::string _rx_lo_source = "internal"; std::string _tx_lo_source = "internal"; //! LO export enabled bool _rx_lo_exported = false; bool _tx_lo_exported = false; //! LO state frequency double _rx_lo_freq = 0.0; double _tx_lo_freq = 0.0; //! LO Distribution board bool _lo_dist_present = false; //! LO Distribution board output status bool _lo_dist_rx_out_enabled[4] = { false, false, false, false }; bool _lo_dist_tx_out_enabled[4] = { false, false, false, false }; }; /* class radio_ctrl_impl */ }} /* namespace uhd::rfnoc */ #endif /* INCLUDED_LIBUHD_RFNOC_RHODIUM_RADIO_CTRL_IMPL_HPP */ // vim: sw=4 et: