From bf986d32632199917dc811f00465ecf409c64a76 Mon Sep 17 00:00:00 2001 From: Martin Braun Date: Mon, 29 Jul 2019 14:06:39 -0700 Subject: rfnoc: Add DUC block controller --- host/include/uhd/rfnoc/CMakeLists.txt | 1 + host/include/uhd/rfnoc/defaults.hpp | 1 + host/include/uhd/rfnoc/duc_block_control.hpp | 123 +++++++++++++++++++++++++++ 3 files changed, 125 insertions(+) create mode 100644 host/include/uhd/rfnoc/duc_block_control.hpp (limited to 'host/include') diff --git a/host/include/uhd/rfnoc/CMakeLists.txt b/host/include/uhd/rfnoc/CMakeLists.txt index f5cb4006b..d65332dac 100644 --- a/host/include/uhd/rfnoc/CMakeLists.txt +++ b/host/include/uhd/rfnoc/CMakeLists.txt @@ -36,6 +36,7 @@ if(ENABLE_RFNOC) tick_node_ctrl.hpp # Block controllers ddc_block_control.hpp + duc_block_control.hpp ddc_block_ctrl.hpp dma_fifo_block_ctrl.hpp duc_block_ctrl.hpp diff --git a/host/include/uhd/rfnoc/defaults.hpp b/host/include/uhd/rfnoc/defaults.hpp index 99ac9791f..008847091 100644 --- a/host/include/uhd/rfnoc/defaults.hpp +++ b/host/include/uhd/rfnoc/defaults.hpp @@ -15,6 +15,7 @@ namespace uhd { namespace rfnoc { static const std::string CLOCK_KEY_GRAPH("__graph__"); static const std::string PROP_KEY_DECIM("decim"); +static const std::string PROP_KEY_INTERP("interp"); static const std::string PROP_KEY_SAMP_RATE("samp_rate"); static const std::string PROP_KEY_SCALING("scaling"); static const std::string PROP_KEY_TYPE("type"); diff --git a/host/include/uhd/rfnoc/duc_block_control.hpp b/host/include/uhd/rfnoc/duc_block_control.hpp new file mode 100644 index 000000000..8b0325515 --- /dev/null +++ b/host/include/uhd/rfnoc/duc_block_control.hpp @@ -0,0 +1,123 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_LIBUHD_DUC_BLOCK_CONTROL_HPP +#define INCLUDED_LIBUHD_DUC_BLOCK_CONTROL_HPP + +#include +#include +#include +#include +#include + +namespace uhd { namespace rfnoc { + +/*! DUC Block Control Class + * + * The DUC Block is a multi-channel digital upconverter (DUC) with built-in + * frequency shift. The number of channels as well as the maximum interpolation + * is configurable in the FPGA, the block controller will read out registers to + * identify the capabilities of this block. + * + * This block has two user properties per channel: + * - `freq`: The frequency shift at the input. Note: A convenience method + * set_freq() is provided to set this property. It also takes care of the + * command time, which set_property() does not, and thus should be preferred. + * - `interp`: The interpolation value + */ +class UHD_API duc_block_control : public noc_block_base +{ +public: + RFNOC_DECLARE_BLOCK(duc_block_control) + + static const uint16_t MAJOR_COMPAT; + static const uint16_t MINOR_COMPAT; + // Readback addresses + static const uint32_t RB_COMPAT_NUM; + static const uint32_t RB_NUM_HB; + static const uint32_t RB_CIC_MAX_INTERP; + // Write addresses + static const uint32_t SR_N_ADDR; + static const uint32_t SR_M_ADDR; + static const uint32_t SR_CONFIG_ADDR; + static const uint32_t SR_FREQ_ADDR; + static const uint32_t SR_SCALE_IQ_ADDR; + static const uint32_t SR_INTERP_ADDR; + + /*! Set the DDS frequency + * + * This block will shift the signal at the input by this frequency before + * decimation. The frequency is given in Hz, it is not a relative frequency + * to the input sampling rate. + * + * Note: When the rate is modified, the frequency is kept constant. Because + * the FPGA internally uses a relative phase increment, changing the input + * sampling rate will trigger a property propagation to recalculate the + * phase increment based off of this value. + * + * This function will coerce the frequency to a valid value, and return the + * coerced value. + * + * \param freq The frequency shift in Hz + * \param chan The channel to which this change shall be applied + * \param time When to apply the new frequency + * \returns The coerced, actual current frequency of the DDS + */ + virtual double set_freq(const double freq, + const size_t chan, + const boost::optional time = boost::none) = 0; + + /*! Return the current DDS frequency + * + * \returns The current frequency of the DDS + */ + virtual double get_freq(const size_t chan) const = 0; + + /*! Return the range of frequencies that \p chan can be set to. + * + * \return The range of frequencies that the DUC can shift the input by + */ + virtual uhd::freq_range_t get_frequency_range(const size_t chan) const = 0; + + /*! Return the sampling rate at this block's input + * + * \param chan The channel for which the rate is being queried + * \returns the sampling rate at this block's input + */ + virtual double get_input_rate(const size_t chan) const = 0; + + /*! Return the sampling rate at this block's output + * + * This is equivalent to calling get_input_rate() multiplied by the interpolation + * + * \param chan The channel for which the rate is being queried + * \returns the sampling rate at this block's input + */ + virtual double get_output_rate(const size_t chan) const = 0; + + /*! Return a range of valid input rates, based on the current output rate + * + * Note the return value is only valid as long as the output rate does not + * change. + */ + virtual uhd::meta_range_t get_input_rates(const size_t chan) const = 0; + + /*! Attempt to set the input rate of this block + * + * This will set the interpolation such that the output rate is untouched, and + * that the output rate divided by the new interpolation is as close as + * possible to the requested \p rate. + * + * \param rate The requested rate + * \param chan The channel for which the rate is being queried + * \returns the coerced sampling rate at this block's output + */ + virtual double set_input_rate(const double rate, const size_t chan) = 0; +}; + +}} // namespace uhd::rfnoc + +#endif /* INCLUDED_LIBUHD_DUC_BLOCK_CONTROL_HPP */ -- cgit v1.2.3