diff options
Diffstat (limited to 'host')
-rw-r--r-- | host/include/uhd/rfnoc/blocks/radio_eiscat.xml | 36 | ||||
-rw-r--r-- | host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.cpp | 136 | ||||
-rw-r--r-- | host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.hpp | 38 |
3 files changed, 183 insertions, 27 deletions
diff --git a/host/include/uhd/rfnoc/blocks/radio_eiscat.xml b/host/include/uhd/rfnoc/blocks/radio_eiscat.xml index 3d3ef7abc..a7077d503 100644 --- a/host/include/uhd/rfnoc/blocks/radio_eiscat.xml +++ b/host/include/uhd/rfnoc/blocks/radio_eiscat.xml @@ -14,7 +14,13 @@ <address>159</address> </setreg> <setreg> - <!--1-Bit register. Are we sending the upper or lower 5 beam contributions? 2 means we ignore contributions --> + <!--4-Bit register. + [0] Are we sending the upper or lower 5 beam contirbutions? + [1] = 1, beams_to_here goes directly to output (bypass neighbour contributions, single USRP) + [2] = 1, input to rx_receive direct from jesd_core (bypassing beamforming) + [3] = 1, if bypassing beamforming, instead output counter 00-FF + to output counter [3:0] = 0d10. + to output jesd streams directly [3:0] = 6 --> <name>SR_BEAMS_TO_NEIGHBOR</name> <address>202</address> <value>2</value> @@ -86,24 +92,32 @@ <check_message>use_prev must be 0 or 1.</check_message> <action>SR_WRITE("SR_PREV_OR_NULL", $use_prev) AND SR_WRITE("SR_PREV_OR_NULL", $use_prev)</action> </arg> - <!--4-Bit register. - [0] Are we sending the upper or lower 5 beam contirbutions? - [1] = 1, beams_to_here goes directly to output (bypass neighbour contributions, single USRP) - [2] = 1, input to rx_receive direct from jesd_core (bypassing beamforming) - [3] = 1, if bypassing beamforming, instead output counter 00-FF - to output counter [3:0] = 0d10. - to output jesd streams directly [3:0] = 6 --> + <!--Direct access to the beams register--> + <arg> + <name>configure_beams</name> + <type>int</type> + </arg> + <!--0: Upper 1: Lower 2: no neighbours--> <arg> <name>choose_beams</name> <type>int</type> - <value>6</value> - <action>SR_WRITE("SR_BEAMS_TO_NEIGHBOR", $choose_beams)</action> + <value>2</value> + </arg> + <arg> + <name>enable_firs</name> + <type>int</type> + <value>1</value> + </arg> + <arg> + <name>enable_counter</name> + <type>int</type> + <value>0</value> </arg> <arg> <name>channel_enable</name> <type>int</type> <value>1</value> - <action>SR_WRITE("SR_RX_STREAM_ENABLE", 1)</action> + <action>SR_WRITE("SR_RX_STREAM_ENABLE", $channel_enable)</action> </arg> <arg> <name>gain</name> diff --git a/host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.cpp b/host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.cpp index dfee25acd..df7fdba8e 100644 --- a/host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.cpp +++ b/host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.cpp @@ -33,6 +33,8 @@ using namespace uhd::rfnoc; namespace { const size_t SR_ANTENNA_GAIN_BASE = 204; const size_t SR_ANTENNA_SELECT_BASE = 192; // Note: On other dboards, 192 is DB_GPIO address space + const size_t RB_CHOOSE_BEAMS = 10; + const double EISCAT_TICK_RATE = 208e6; // Hz const double EISCAT_RADIO_RATE = 104e6; // Hz @@ -53,7 +55,13 @@ namespace { const size_t EISCAT_NUM_FIR_TAPS = 10; const size_t EISCAT_NUM_FIR_SETS = 1024; // BRAM must be at least EISCAT_NUM_FIR_TAPS * EISCAT_NUM_FIR_SETS const size_t EISCAT_FIR_INDEX_IMPULSE = 1002; - const size_t EISCAT_FIR_INDEX_ZEROS = 1003; // FIXME + const size_t EISCAT_FIR_INDEX_ZEROS = 1003; + + const uint32_t EISCAT_CONTRIB_LOWER = 0<<0; + const uint32_t EISCAT_CONTRIB_UPPER = 1<<0; + const uint32_t EISCAT_SKIP_NEIGHBOURS = 1<<1; + const uint32_t EISCAT_BYPASS_MATRIX = 1<<2; + const uint32_t EISCAT_OUTPUT_COUNTER = 1<<3; }; @@ -77,11 +85,38 @@ UHD_RFNOC_RADIO_BLOCK_CONSTRUCTOR(eiscat_radio_ctrl) for (size_t i = 0; i < EISCAT_NUM_ANTENNAS; i++) { _tree->access<double>(get_arg_path("gain", i) / "value") .set_coercer([](double gain){ return std::max(-1.0, std::min(1.0, gain)); }) - .add_coerced_subscriber([this, i](double gain){ this->set_antenna_gain(i, gain); }) + .add_coerced_subscriber([this, i](double gain){ + this->set_antenna_gain(i, gain); + }) .set(EISCAT_DEFAULT_NORM_GAIN) ; } + /**** Add subscribers for our special properties ************************/ + _tree->access<int>(get_arg_path("choose_beams", 0) / "value") + .add_coerced_subscriber([this](int choose_beams){ + this->set_beam_selection(choose_beams); + }) + .update() + ; + _tree->access<bool>(get_arg_path("enable_firs", 0) / "value") + .add_coerced_subscriber([this](int enable){ + this->enable_firs(bool(enable)); + }) + .update() + ; + _tree->access<bool>(get_arg_path("enable_counter", 0) / "value") + .add_coerced_subscriber([this](int enable){ + this->enable_counter(bool(enable)); + }) + .update() + ; + _tree->access<int>(get_arg_path("configure_beams", 0) / "value") + .add_coerced_subscriber([this](int reg_value){ + this->configure_beams(uint32_t(reg_value)); + }) // No update! + ; + /**** Set up legacy compatible properties ******************************/ // For use with multi_usrp APIs etc. // For legacy prop tree init: @@ -185,18 +220,14 @@ void eiscat_radio_ctrl_impl::set_rx_antenna(const std::string &ant, const size_t UHD_LOG_TRACE("EISCAT", "Setting antenna to 'BF' (which is a no-op)"); return; } - if (ant.size() < 3 or ant.size() > 4 - or (ant.substr(0, 2) != "Rx" - and ant.substr(0, 2) != "RX" - and ant.substr(0, 2) != "BF")) { + if (ant.size() < 3) { throw uhd::value_error(str( boost::format("EISCAT: Invalid antenna selection: %s") % ant )); } - bool use_fir_matrix = (ant.substr(0, 2) == "BF"); - + const std::string ant_mode = ant.substr(0, 2); const size_t antenna_idx = [&ant](){ try { return boost::lexical_cast<size_t>(ant.substr(2)); @@ -208,7 +239,7 @@ void eiscat_radio_ctrl_impl::set_rx_antenna(const std::string &ant, const size_t } }(); - if (use_fir_matrix) { + if (ant_mode == "BF") { UHD_LOG_TRACE("EISCAT", str( boost::format("Setting port %d to only receive on antenna %d via FIR matrix") % port % antenna_idx @@ -236,13 +267,39 @@ void eiscat_radio_ctrl_impl::set_rx_antenna(const std::string &ant, const size_t ); } } - } else { + enable_firs(true); + } else if (ant_mode == "RX" or ant_mode == "Rx") { set_arg<int>("choose_beams", 6); UHD_LOG_TRACE("EISCAT", str( boost::format("Setting port %d to only receive on antenna %d directly") % port % antenna_idx )); - sr_write(SR_ANTENNA_SELECT_BASE + port, antenna_idx); + enable_firs(false); + } else if (ant_mode == "FI") { + UHD_LOG_TRACE("EISCAT", str( + boost::format("Setting port %d to filter index %d on all antennas.") + % port % antenna_idx + )); + // Note: antenna_idx is not indexing a physical antenna in this scenario. + // TODO: When we have a way to select neighbour contributions, we will + // need to calculate the beam_index as a function of the port *and* if + // we're the left or right USRP + const size_t beam_index = port; + uhd::time_spec_t send_now(0.0); + for (size_t i = 0; i < EISCAT_NUM_ANTENNAS; i++) { + select_filter( + beam_index, + i, + antenna_idx, + send_now + ); + } + enable_firs(true); + } else { + throw uhd::value_error(str( + boost::format("EISCAT: Invalid antenna selection: %s") + % ant + )); } } @@ -325,7 +382,10 @@ bool eiscat_radio_ctrl_impl::check_radio_config() chan_enables |= (1<<enb.first); } } - UHD_LOG_TRACE("EISCAT", str(boost::format("check_radio_config(): Setting channel enables to 0x%02X") % chan_enables)); + UHD_LOG_TRACE("EISCAT", str( + boost::format("check_radio_config(): Setting channel enables to 0x%02X") + % chan_enables + )); sr_write("SR_RX_STREAM_ENABLE", chan_enables); return true; @@ -440,6 +500,58 @@ void eiscat_radio_ctrl_impl::set_antenna_gain( sr_write(SR_ANTENNA_GAIN_BASE + antenna_idx, fixpoint_gain); } +void eiscat_radio_ctrl_impl::configure_beams(uint32_t reg_value) +{ + UHD_LOGGER_TRACE("EISCAT") + << "Selecting " << + ((reg_value & EISCAT_CONTRIB_UPPER) ? "upper" : "lower") << " beams."; + UHD_LOGGER_TRACE("EISCAT") + << ((reg_value & EISCAT_SKIP_NEIGHBOURS) ? "Disabling" : "Enabling") + << " neighbour contributions."; + UHD_LOGGER_TRACE("EISCAT") + << ((reg_value & EISCAT_BYPASS_MATRIX) ? "Disabling" : "Enabling") + << " FIR matrix."; + UHD_LOGGER_TRACE("EISCAT") + << ((reg_value & EISCAT_OUTPUT_COUNTER) ? "Enabling" : "Disabling") + << " counter."; + UHD_LOG_TRACE("EISCAT", str( + boost::format("Setting SR_BEAMS_TO_NEIGHBOR to 0x%08X.") + % reg_value + )); + sr_write("SR_BEAMS_TO_NEIGHBOR", reg_value); +} + +void eiscat_radio_ctrl_impl::set_beam_selection(int beam_selection) +{ + const uint32_t old_value = user_reg_read32(RB_CHOOSE_BEAMS); + const uint32_t new_value = + (old_value & (~uint32_t(EISCAT_CONTRIB_UPPER|EISCAT_SKIP_NEIGHBOURS))) + | (uint32_t(beam_selection) + & uint32_t(EISCAT_CONTRIB_UPPER|EISCAT_SKIP_NEIGHBOURS)) + ; + configure_beams(new_value); +} + +void eiscat_radio_ctrl_impl::enable_firs(bool enable) +{ + const uint32_t old_value = user_reg_read32(RB_CHOOSE_BEAMS); + const uint32_t new_value = enable ? + (old_value & ~EISCAT_BYPASS_MATRIX) + : old_value | EISCAT_BYPASS_MATRIX + ; + configure_beams(new_value); +} + +void eiscat_radio_ctrl_impl::enable_counter(bool enable) +{ + const uint32_t old_value = user_reg_read32(RB_CHOOSE_BEAMS); + const uint32_t new_value = enable ? + old_value | EISCAT_OUTPUT_COUNTER + : (old_value & ~EISCAT_OUTPUT_COUNTER) + ; + configure_beams(new_value); +} + /**************************************************************************** * Registry ***************************************************************************/ diff --git a/host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.hpp b/host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.hpp index cb45208ce..46b9410cd 100644 --- a/host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.hpp +++ b/host/lib/usrp/dboard/eiscat/eiscat_radio_ctrl_impl.hpp @@ -54,11 +54,17 @@ public: * Valid antenna values: * - BF: This is the default. Will apply the beamforming matrix in whatever * state it currently is. - * - Rx0...Rx15: Will mux the antenna signal 0...15 straight to this - * channel. Note that this will disable the FIR matrix entirely. + * - RX0...RX15: Will mux the antenna signal 0...15 straight to this + * channel. Note that this will disable the FIR matrix entirely, and will + * also disable contributions from other USRPs. * - BF0...BF15: Will configure the FIR filter matrix such that only the - * contributions from antenna 0...15 are passed to this channel. - * + * contributions from antenna 0...15 are passed to this channel. This + * should produce the same signal as RX0..RX15, reduced by 12 dB (because + * the FIR matri needs to account for bit growth from adding 16 channels) + * - FI$idx: Here, $idx is a number (the filter index, hence the name). + * This will apply filter index $idx to all input channels. Useful for + * testing actual beamforming applications, when the same signal is + * applied to all inputs. * * \throws uhd::value_error if the antenna value was not valid */ @@ -125,6 +131,30 @@ private: const double normalized_gain ); + /*! Directly writes a value to the beam configuration register. + */ + void configure_beams(uint32_t reg_value); + + /*! Controls selection of beams coming from the FIR matrix. + * + * The following values are allowed: + * - 0: We stream the lower 5 beams, plus the neighbours contribution + * - 1: We stream the upper 5 beams, plus the neighbours contribution + * - 2: We stream the lower 5 beams, without the neighbours contribution + * - 3: We stream the upper 5 beams, without the neighbours contribution + */ + void set_beam_selection(int beam_selection); + + /*! Controls if we're using the FIR matrix + * + * If this is false, the beam selection is irrelevant. + */ + void enable_firs(bool enable); + + /*! Enables counter instead of JESD core output + */ + void enable_counter(bool enable); + /*! The number of channels this block outputs * * This is *not* the number of antennas, but the number of streams a single |