diff options
| author | Martin Braun <martin.braun@ettus.com> | 2019-05-31 15:08:27 -0700 | 
|---|---|---|
| committer | Martin Braun <martin.braun@ettus.com> | 2019-11-26 11:49:22 -0800 | 
| commit | 6730a0b72b3a6a5b1b94a5d09bfb395d837672da (patch) | |
| tree | 643ee003afd92489d8d0bad217e5e3a8655693ca /host/lib | |
| parent | 911b89e4f6e2df66d355448c452af0b2fc058fe6 (diff) | |
| download | uhd-6730a0b72b3a6a5b1b94a5d09bfb395d837672da.tar.gz uhd-6730a0b72b3a6a5b1b94a5d09bfb395d837672da.tar.bz2 uhd-6730a0b72b3a6a5b1b94a5d09bfb395d837672da.zip | |
rfnoc: Add radio block controller
Diffstat (limited to 'host/lib')
| -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 | 
3 files changed, 1061 insertions, 0 deletions
| 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())); +} | 
