From b5f6ba518e80fe3f3f64f112d657bc9fc631fe4f Mon Sep 17 00:00:00 2001 From: Martin Braun Date: Fri, 3 Nov 2017 16:56:03 -0700 Subject: mg: Refactor magnesium_radio_ctrl and friends - Spin out AD9371 control into its own class - Split file into multiple compilation units - Fixed many minor code formatting issues --- host/lib/usrp/dboard/magnesium/CMakeLists.txt | 3 + .../dboard/magnesium/magnesium_ad9371_iface.cpp | 113 ++ .../dboard/magnesium/magnesium_ad9371_iface.hpp | 76 ++ .../usrp/dboard/magnesium/magnesium_constants.hpp | 64 ++ .../usrp/dboard/magnesium/magnesium_cpld_ctrl.cpp | 3 + .../dboard/magnesium/magnesium_radio_ctrl_cpld.cpp | 281 +++++ .../dboard/magnesium/magnesium_radio_ctrl_impl.cpp | 1136 ++++---------------- .../dboard/magnesium/magnesium_radio_ctrl_impl.hpp | 126 +-- .../dboard/magnesium/magnesium_radio_ctrl_init.cpp | 487 +++++++++ 9 files changed, 1275 insertions(+), 1014 deletions(-) create mode 100644 host/lib/usrp/dboard/magnesium/magnesium_ad9371_iface.cpp create mode 100644 host/lib/usrp/dboard/magnesium/magnesium_ad9371_iface.hpp create mode 100644 host/lib/usrp/dboard/magnesium/magnesium_constants.hpp create mode 100644 host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_cpld.cpp create mode 100644 host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_init.cpp (limited to 'host/lib/usrp/dboard') diff --git a/host/lib/usrp/dboard/magnesium/CMakeLists.txt b/host/lib/usrp/dboard/magnesium/CMakeLists.txt index a0578d7b0..d1db7d1ea 100644 --- a/host/lib/usrp/dboard/magnesium/CMakeLists.txt +++ b/host/lib/usrp/dboard/magnesium/CMakeLists.txt @@ -18,6 +18,9 @@ IF(ENABLE_MPMD) LIST(APPEND MAGNESIUM_SOURCES ${CMAKE_CURRENT_SOURCE_DIR}/magnesium_radio_ctrl_impl.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/magnesium_radio_ctrl_init.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/magnesium_radio_ctrl_cpld.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/magnesium_ad9371_iface.cpp ${CMAKE_CURRENT_SOURCE_DIR}/magnesium_cpld_ctrl.cpp ) LIBUHD_APPEND_SOURCES(${MAGNESIUM_SOURCES}) diff --git a/host/lib/usrp/dboard/magnesium/magnesium_ad9371_iface.cpp b/host/lib/usrp/dboard/magnesium/magnesium_ad9371_iface.cpp new file mode 100644 index 000000000..2fe31f066 --- /dev/null +++ b/host/lib/usrp/dboard/magnesium/magnesium_ad9371_iface.cpp @@ -0,0 +1,113 @@ +// +// Copyright 2017 Ettus Research, a National Instruments Company +// +// SPDX-License-Identifier: GPL-3.0 +// + +#include "magnesium_ad9371_iface.hpp" +#include +#include + +using namespace uhd; + +namespace { + /*! Return a valid 'which' string for use with AD9371 API calls + * + * These strings take the form of "RX1", "TX2", ... + */ + std::string _get_which( + const direction_t dir, + const size_t chan + ) { + UHD_ASSERT_THROW(dir == RX_DIRECTION or dir == TX_DIRECTION); + UHD_ASSERT_THROW(chan == 0 or chan == 1); + return str(boost::format("%s%d") + % (dir == RX_DIRECTION ? "RX" : "TX") + % (chan+1) + ); + } +} + +/****************************************************************************** + * Structors + *****************************************************************************/ +magnesium_ad9371_iface::magnesium_ad9371_iface( + uhd::rpc_client::sptr rpcc, + const size_t slot_idx +) : _rpcc(rpcc) + , _slot_idx(slot_idx) + , _rpc_prefix((slot_idx == 0) ? "db_0_" : "db_1_") + , _L((slot_idx == 0) ? "AD9371-0" : "AD9371-1") +{ + UHD_LOG_TRACE(_L, + "Initialized controls with RPC prefix " << _rpc_prefix << + " for slot " << _slot_idx); +} + +double magnesium_ad9371_iface::set_frequency( + const double freq, + const size_t chan, + const direction_t dir +) { + // Note: This sets the frequency for both channels (1 and 2). + auto which = _get_which(dir, chan); + UHD_LOG_TRACE(_L, + "Calling " << _rpc_prefix << "set_freq on " + << which << " with " << freq); + auto actual_freq = + _rpcc->request_with_token(_rpc_prefix + "set_freq", + which, freq, false); + UHD_LOG_TRACE(_L, + _rpc_prefix << "set_freq returned " << actual_freq); + return actual_freq; +} + +double magnesium_ad9371_iface::set_gain( + const double gain, + const size_t chan, + const direction_t dir +) { + auto which = _get_which(dir, chan); + UHD_LOG_TRACE(_L, "Calling " << _rpc_prefix << "set_gain on " << which << " with " << gain); + auto retval = _rpcc->request_with_token(_rpc_prefix + "set_gain", which, gain); + UHD_LOG_TRACE(_L, _rpc_prefix << "set_gain returned " << retval); + + return retval; + //return 0.0; +} + + +double magnesium_ad9371_iface::set_bandwidth(const double bandwidth, const size_t chan, const direction_t dir) +{ + // TODO: implement + UHD_LOG_WARNING(_L, "Ignoring attempt to set bandwidth"); + return 0.0; +} + +double magnesium_ad9371_iface::get_frequency( + const size_t chan, + const direction_t dir +) { + auto which = _get_which(dir, chan); + UHD_LOG_TRACE(_L, "calling " << _rpc_prefix << "get_freq on " << which); + auto retval = _rpcc->request_with_token(_rpc_prefix + "get_freq", which); + UHD_LOG_TRACE(_L, _rpc_prefix << "get_freq returned " << retval); + return retval; +} + +double magnesium_ad9371_iface::get_gain(const size_t chan, const direction_t dir) +{ + auto which = _get_which(dir, chan); + UHD_LOG_TRACE(_L, "calling " << _rpc_prefix << "get_gain on " << which); + auto retval = _rpcc->request_with_token(_rpc_prefix + "get_gain", which); + UHD_LOG_TRACE(_L, _rpc_prefix << "get_gain returned " << retval); + return retval; +} + +double magnesium_ad9371_iface::get_bandwidth(const size_t chan, const direction_t dir) +{ + // TODO: implement + UHD_LOG_WARNING(_L, "Ignoring attempt to get bandwidth"); + return 0.0; +} + diff --git a/host/lib/usrp/dboard/magnesium/magnesium_ad9371_iface.hpp b/host/lib/usrp/dboard/magnesium/magnesium_ad9371_iface.hpp new file mode 100644 index 000000000..9529e1519 --- /dev/null +++ b/host/lib/usrp/dboard/magnesium/magnesium_ad9371_iface.hpp @@ -0,0 +1,76 @@ +// +// Copyright 2017 Ettus Research, a National Instruments Company +// +// SPDX-License-Identifier: GPL-3.0 +// + +#ifndef INCLUDED_LIBUHD_RFNOC_MAGNESIUM_AD9371_IFACE_HPP +#define INCLUDED_LIBUHD_RFNOC_MAGNESIUM_AD9371_IFACE_HPP + +#include "../../../utils/rpc.hpp" +#include +#include +#include + +class magnesium_ad9371_iface +{ +public: + using uptr = std::unique_ptr; + + magnesium_ad9371_iface( + uhd::rpc_client::sptr rpcc, + const size_t slot_idx + ); + + double set_frequency( + const double freq, + const size_t chan, + const uhd::direction_t dir + ); + + double get_frequency( + const size_t chan, + const uhd::direction_t dir + ); + + double set_gain( + const double gain, + const size_t chan, + const uhd::direction_t dir + ); + + double get_gain( + const size_t chan, + const uhd::direction_t dir + ); + + double set_bandwidth( + const double bandwidth, + const size_t chan, + const uhd::direction_t dir + ); + + double get_bandwidth( + const size_t chan, + const uhd::direction_t dir + ); + +private: + //! Reference to the RPC client + uhd::rpc_client::sptr _rpcc; + + //! Slot index + const size_t _slot_idx; + + //! Stores the prefix to RPC calls + const std::string _rpc_prefix; + + //! Logger prefix + const std::string _L; + + + +}; + +#endif /* INCLUDED_LIBUHD_RFNOC_MAGNESIUM_AD9371_IFACE_HPP */ +// vim: sw=4 et: diff --git a/host/lib/usrp/dboard/magnesium/magnesium_constants.hpp b/host/lib/usrp/dboard/magnesium/magnesium_constants.hpp new file mode 100644 index 000000000..f66d9ab8e --- /dev/null +++ b/host/lib/usrp/dboard/magnesium/magnesium_constants.hpp @@ -0,0 +1,64 @@ +// +// Copyright 2017 Ettus Research, a National Instruments Company +// +// SPDX-License-Identifier: GPL-3.0 +// + +#ifndef INCLUDED_LIBUHD_MAGNESIUM_CONSTANTS_HPP +#define INCLUDED_LIBUHD_MAGNESIUM_CONSTANTS_HPP + +#include +#include +#include + +static const size_t FPGPIO_MASTER_RADIO = 0; + +static const double MAGNESIUM_TICK_RATE = 125e6; // Hz +static const double MAGNESIUM_RADIO_RATE = 125e6; // Hz +static const double MAGNESIUM_MIN_FREQ = 1e6; // Hz +static const double MAGNESIUM_MAX_FREQ = 6e9; // Hz + +static const double MAGNESIUM_LOWBAND_FREQ = 300e6; + +static const double MAGNESIUM_RX_BAND1_MIN_FREQ = MAGNESIUM_LOWBAND_FREQ; +static const double MAGNESIUM_RX_BAND2_MIN_FREQ = 600e6; +static const double MAGNESIUM_RX_BAND3_MIN_FREQ = 1050e6; +static const double MAGNESIUM_RX_BAND4_MIN_FREQ = 1600e6; +static const double MAGNESIUM_RX_BAND5_MIN_FREQ = 2100e6; +static const double MAGNESIUM_RX_BAND6_MIN_FREQ = 2700e6; + +static const double MAGNESIUM_TX_BAND1_MIN_FREQ = MAGNESIUM_LOWBAND_FREQ; +static const double MAGNESIUM_TX_BAND2_MIN_FREQ = 800e6; +static const double MAGNESIUM_TX_BAND3_MIN_FREQ = 1700e6; +static const double MAGNESIUM_TX_BAND4_MIN_FREQ = 3400e6; + +static const double AD9371_MIN_RX_GAIN = 0.0; // dB +static const double AD9371_MAX_RX_GAIN = 30.0; // dB +static const double AD9371_RX_GAIN_STEP = 0.5; +static const double DSA_MIN_GAIN = 0; // dB +static const double DSA_MAX_GAIN = 31.5; // dB +static const double DSA_GAIN_STEP = 0.5; // db +static const double AD9371_MIN_TX_GAIN = 0.0; // dB +static const double AD9371_MAX_TX_GAIN = 41.95; // dB +static const double AD9371_TX_GAIN_STEP = 0.05; +static const double ALL_RX_MIN_GAIN = 0.0; +static const double ALL_RX_MAX_GAIN = 61.5; +static const double ALL_RX_GAIN_STEP = 0.5; +static const double ALL_TX_MIN_GAIN = 0.0; +static const double ALL_TX_MAX_GAIN = 73.45; +static const double ALL_TX_GAIN_STEP = 0.5; +static const double MAGNESIUM_CENTER_FREQ = 2.5e9; // Hz +static const std::vector MAGNESIUM_RX_ANTENNAS = { + "TX/RX", "RX2" +}; +static const char* MAGNESIUM_DEFAULT_RX_ANTENNA = "RX2"; +static const char* MAGNESIUM_DEFAULT_TX_ANTENNA = "TX/RX"; +static const double MAGNESIUM_DEFAULT_BANDWIDTH = 40e6; // Hz TODO: fix +// Note: MAGNESIUM_NUM_CHANS is independent of the number of chans per +// RFNoC block. TODO: When we go to one radio per dboard, this comment can +// be deleted. +static const size_t MAGNESIUM_NUM_CHANS = 2; +static const double MAGNESIUM_RX_IF_FREQ = 2.44e9; +static const double MAGNESIUM_TX_IF_FREQ = 1.95e9; + +#endif /* INCLUDED_LIBUHD_MAGNESIUM_CONSTANTS_HPP */ diff --git a/host/lib/usrp/dboard/magnesium/magnesium_cpld_ctrl.cpp b/host/lib/usrp/dboard/magnesium/magnesium_cpld_ctrl.cpp index 66fa02360..dac8ca5c1 100644 --- a/host/lib/usrp/dboard/magnesium/magnesium_cpld_ctrl.cpp +++ b/host/lib/usrp/dboard/magnesium/magnesium_cpld_ctrl.cpp @@ -5,6 +5,7 @@ // #include "magnesium_cpld_ctrl.hpp" +#include "magnesium_constants.hpp" #include #include #include @@ -12,6 +13,8 @@ namespace { //! Address of the CPLD scratch register const uint8_t CPLD_REGS_SCRATCH = 0x0040; + + //! Address of the CPLD reset register const uint8_t CPLD_REGS_RESET = 0x0041; } diff --git a/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_cpld.cpp b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_cpld.cpp new file mode 100644 index 000000000..fa0c7cba5 --- /dev/null +++ b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_cpld.cpp @@ -0,0 +1,281 @@ +// +// Copyright 2017 Ettus Research, a National Instruments Company +// +// SPDX-License-Identifier: GPL-3.0 +// + +#include "magnesium_radio_ctrl_impl.hpp" +#include "magnesium_cpld_ctrl.hpp" +#include "magnesium_constants.hpp" +#include + +/* + * Magnesium Rev C frequency bands: + * + * RX IF frequency is 2.4418 GHz. Have 80 MHz of bandwidth for loband. + * TX IF frequency is 1.8-2.1 GHz (1.95 GHz is best). + * + * For RX: + * Band SW2-AB SW3-ABC SW4-ABC SW5-ABCD SW6-ABC SW7-AB SW8-AB MIX + * WB RF1 01 OFF 111 NA --- NA ---- RF3 001 RF2 01 RF2 01 0 + * LB RF2 10 RF5 100 NA --- RF3 0010 RF1 100 RF1 10 RF1 10 1 + * 440-530 RF2 10 RF2 001 NA --- RF1 1000 RF1 100 RF2 01 RF2 01 0 + * 650-1000 RF2 10 RF6 101 NA --- RF4 0001 RF1 100 RF2 01 RF2 01 0 + * 1100-1575 RF2 10 RF4 011 NA --- RF2 0100 RF1 100 RF2 01 RF2 01 0 + * 1600-2250 RF2 10 RF3 010 RF2 010 NA ---- RF2 010 RF2 01 RF2 01 0 + * 2100-2850 RF2 10 RF1 000 RF1 100 NA ---- RF2 010 RF2 01 RF2 01 0 + * 2700+ RF3 11 OFF 111 RF3 001 NA ---- RF2 010 RF2 01 RF2 01 0 + * + * For TX: + * Band SW5-AB SW4-AB SW3-X SW2-ABCD SW1-AB SWTRX-AB MIX + * WB RF1 10 RF2 01 RF1 0 NA ---- SHD 00 RF4 11 0 + * LB RF2 01 RF1 10 RF2 1 RF3 0010 RF3 11 RF1 00 1 + * <800 RF1 10 RF2 01 RF2 1 RF3 0010 RF3 11 RF1 00 0 + * 800-1700 RF1 10 RF2 01 RF2 1 RF2 0100 RF2 10 RF1 00 0 + * 1700-3400 RF1 10 RF2 01 RF2 1 RF1 1000 RF1 01 RF1 00 0 + * 3400-6400 RF1 10 RF2 01 RF2 1 RF4 0001 SHD 00 RF2 10 0 + */ + +using namespace uhd; +using namespace uhd::usrp; +using namespace uhd::rfnoc; + +void magnesium_radio_ctrl_impl::_update_atr_switches( + const magnesium_cpld_ctrl::chan_sel_t chan, + const direction_t dir, + const std::string &ant +){ + magnesium_cpld_ctrl::rx_sw1_t rx_sw1 = magnesium_cpld_ctrl::RX_SW1_RX2INPUT; + magnesium_cpld_ctrl::sw_trx_t sw_trx = _sw_trx[chan]; + + bool trx_led = false, rx2_led = true; + //bool tx_pa_enb = true, tx_amp_enb = true, tx_myk_en=true; + if (ant == "TX/RX" and dir == RX_DIRECTION) { + rx_sw1 = magnesium_cpld_ctrl::RX_SW1_TRXSWITCHOUTPUT; + sw_trx = magnesium_cpld_ctrl::SW_TRX_RXCHANNELPATH; + trx_led = true; + rx2_led = false; + } + UHD_LOG_TRACE(unique_id(), "Update all atr related switches for " << dir << " " << ant ); + if (dir == RX_DIRECTION){ + _cpld->set_rx_atr_bits( + chan, + magnesium_cpld_ctrl::ON, + rx_sw1, + trx_led, + rx2_led, + true, + true, + true, + true + ); + _cpld->set_tx_atr_bits( + chan, + magnesium_cpld_ctrl::IDLE, + false, + sw_trx, + false, + false, + false + ); + _cpld->set_rx_atr_bits( + chan, + magnesium_cpld_ctrl::IDLE, + rx_sw1, + false, + false, + false, + false, + false, + true + ); + } + if (dir == TX_DIRECTION){ + _cpld->set_tx_atr_bits( + chan, + magnesium_cpld_ctrl::ON, + true, + sw_trx, + true, + true, + true + ); + _cpld->set_rx_atr_bits( + chan, + magnesium_cpld_ctrl::IDLE, + rx_sw1, + false, + false, + false, + false, + false, + true + ); + }; +} + +void magnesium_radio_ctrl_impl::_update_rx_freq_switches( + const double freq, + const size_t chan +) { + UHD_LOG_TRACE(unique_id(), + "Update all RX freq related switches. f=" << freq << " Hz, " + "chan=" << chan); + + // Set filters based on frequency. Note: We always switch both channels + if (freq < MAGNESIUM_RX_BAND1_MIN_FREQ) { + _cpld->set_rx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, + magnesium_cpld_ctrl::RX_SW3_FILTER0490LPMHZ, + magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, + magnesium_cpld_ctrl::RX_SW5_FILTER0490LPMHZFROM, + magnesium_cpld_ctrl::RX_SW6_LOWERFILTERBANKFROMSWITCH5, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_LOBAND, + true + ); + } else if (freq < MAGNESIUM_RX_BAND2_MIN_FREQ) { + _cpld->set_rx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, + magnesium_cpld_ctrl::RX_SW3_FILTER0440X0530MHZ, + magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, + magnesium_cpld_ctrl::RX_SW5_FILTER0440X0530MHZFROM, + magnesium_cpld_ctrl::RX_SW6_LOWERFILTERBANKFROMSWITCH5, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false + ); + } else if (freq < MAGNESIUM_RX_BAND3_MIN_FREQ) { + _cpld->set_rx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, + magnesium_cpld_ctrl::RX_SW3_FILTER0650X1000MHZ, + magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, + magnesium_cpld_ctrl::RX_SW5_FILTER0650X1000MHZFROM, + magnesium_cpld_ctrl::RX_SW6_LOWERFILTERBANKFROMSWITCH5, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false + ); + } else if (freq < MAGNESIUM_RX_BAND4_MIN_FREQ) { + _cpld->set_rx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, + magnesium_cpld_ctrl::RX_SW3_FILTER1100X1575MHZ, + magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, + magnesium_cpld_ctrl::RX_SW5_FILTER1100X1575MHZFROM, + magnesium_cpld_ctrl::RX_SW6_LOWERFILTERBANKFROMSWITCH5, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false + ); + } else if (freq < MAGNESIUM_RX_BAND5_MIN_FREQ) { + _cpld->set_rx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, + magnesium_cpld_ctrl::RX_SW3_FILTER1600X2250MHZ, + magnesium_cpld_ctrl::RX_SW4_FILTER1600X2250MHZFROM, + magnesium_cpld_ctrl::RX_SW5_FILTER0440X0530MHZFROM, + magnesium_cpld_ctrl::RX_SW6_UPPERFILTERBANKFROMSWITCH4, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false + ); + } else if (freq < MAGNESIUM_RX_BAND6_MIN_FREQ) { + _cpld->set_rx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, + magnesium_cpld_ctrl::RX_SW3_FILTER2100X2850MHZ, + magnesium_cpld_ctrl::RX_SW4_FILTER2100X2850MHZFROM, + magnesium_cpld_ctrl::RX_SW5_FILTER0440X0530MHZFROM, + magnesium_cpld_ctrl::RX_SW6_UPPERFILTERBANKFROMSWITCH4, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false + ); + } else { + _cpld->set_rx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::RX_SW2_UPPERFILTERBANKTOSWITCH4, + magnesium_cpld_ctrl::RX_SW3_SHUTDOWNSW3, + magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, + magnesium_cpld_ctrl::RX_SW5_FILTER0440X0530MHZFROM, + magnesium_cpld_ctrl::RX_SW6_UPPERFILTERBANKFROMSWITCH4, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false + ); + } +} + +void magnesium_radio_ctrl_impl::_update_tx_freq_switches( + const double freq, + const size_t chan +){ + UHD_LOG_TRACE(unique_id(), + "Update all TX freq related switches. f=" << freq << " Hz, " + "chan=" << chan); + magnesium_cpld_ctrl::chan_sel_t chan_sel = + _master ? magnesium_cpld_ctrl::CHAN1 : magnesium_cpld_ctrl::CHAN2; + + // Set filters based on frequency + if (freq < MAGNESIUM_TX_BAND1_MIN_FREQ) { + _cpld->set_tx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1, + magnesium_cpld_ctrl::TX_SW1_FROMTXFILTERLP0800MHZ, + magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP0800MHZ, + magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_LOBAND, + true, + magnesium_cpld_ctrl::ON + ); + _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1; + } else if (freq < MAGNESIUM_TX_BAND2_MIN_FREQ) { + _cpld->set_tx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1, + magnesium_cpld_ctrl::TX_SW1_FROMTXFILTERLP0800MHZ, + magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP0800MHZ, + magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false, + magnesium_cpld_ctrl::ON + ); + _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1; + } else if (freq < MAGNESIUM_TX_BAND3_MIN_FREQ) { + _cpld->set_tx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1, + magnesium_cpld_ctrl::TX_SW1_FROMTXFILTERLP1700MHZ, + magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP1700MHZ, + magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false, + magnesium_cpld_ctrl::ON + ); + _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1; + } else if (freq < MAGNESIUM_TX_BAND4_MIN_FREQ) { + _cpld->set_tx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1, + magnesium_cpld_ctrl::TX_SW1_FROMTXFILTERLP3400MHZ, + magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP3400MHZ, + magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false, + magnesium_cpld_ctrl::ON + ); + _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1; + } else { + _cpld->set_tx_switches( + magnesium_cpld_ctrl::BOTH, + magnesium_cpld_ctrl::SW_TRX_FROMTXUPPERFILTERBANKLP6400MHZ, + magnesium_cpld_ctrl::TX_SW1_SHUTDOWNTXSW1, + magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP6400MHZ, + magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, + magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, + false, + magnesium_cpld_ctrl::ON + ); + _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMTXUPPERFILTERBANKLP6400MHZ; + } +} + + + diff --git a/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp index fe386aa38..b0157bdef 100644 --- a/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp +++ b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp @@ -5,7 +5,7 @@ // #include "magnesium_radio_ctrl_impl.hpp" -#include "spi_core_3000.hpp" +#include "magnesium_constants.hpp" #include #include #include @@ -23,105 +23,54 @@ using namespace uhd::usrp; using namespace uhd::rfnoc; namespace { - enum slave_select_t { - SEN_CPLD = 1, - SEN_TX_LO = 2, - SEN_RX_LO = 4, - SEN_PHASE_DAC = 8 - }; - - const double MAGNESIUM_TICK_RATE = 125e6; // Hz - const double MAGNESIUM_RADIO_RATE = 125e6; // Hz - const double MAGNESIUM_MIN_FREQ = 1e6; // Hz - const double MAGNESIUM_MAX_FREQ = 6e9; // Hz - const double AD9371_MIN_RX_GAIN = 0.0; // dB - const double AD9371_MAX_RX_GAIN = 30.0; // dB - const double AD9371_RX_GAIN_STEP = 0.5; - const double DSA_MIN_GAIN = 0; // dB - const double DSA_MAX_GAIN = 31.5; // dB - const double DSA_GAIN_STEP = 0.5; // db - const double AD9371_MIN_TX_GAIN = 0.0; // dB - const double AD9371_MAX_TX_GAIN = 41.95; // dB - const double AD9371_TX_GAIN_STEP = 0.05; - const double ALL_RX_MIN_GAIN = 0.0; - const double ALL_RX_MAX_GAIN = 61.5; - const double ALL_RX_GAIN_STEP = 0.5; - const double ALL_TX_MIN_GAIN = 0.0; - const double ALL_TX_MAX_GAIN = 73.45; - const double ALL_TX_GAIN_STEP = 0.5; - const double MAGNESIUM_CENTER_FREQ = 2.5e9; // Hz - const char* MAGNESIUM_DEFAULT_RX_ANTENNA = "RX2"; - const char* MAGNESIUM_DEFAULT_TX_ANTENNA = "TX/RX"; - const double MAGNESIUM_DEFAULT_BANDWIDTH = 40e6; // Hz TODO: fix - const size_t MAGNESIUM_NUM_TX_CHANS = 1; - const size_t MAGNESIUM_NUM_RX_CHANS = 1; - const double MAGNESIUM_RX_IF_FREQ = 2.44e9; - const double MAGNESIUM_TX_IF_FREQ = 1.95e9; - const double MAGNESIUM_LOWBAND_FREQ = 300e6; -/* -Magnesium Rev C frequency bands: - -RX IF frequency is 2.4418 GHz. Have 80 MHz of bandwidth for loband. -TX IF frequency is 1.8-2.1 GHz (1.95 GHz is best). - -For RX: - Band SW2-AB SW3-ABC SW4-ABC SW5-ABCD SW6-ABC SW7-AB SW8-AB MIX - WB RF1 01 OFF 111 NA --- NA ---- RF3 001 RF2 01 RF2 01 0 - LB RF2 10 RF5 100 NA --- RF3 0010 RF1 100 RF1 10 RF1 10 1 - 440-530 RF2 10 RF2 001 NA --- RF1 1000 RF1 100 RF2 01 RF2 01 0 - 650-1000 RF2 10 RF6 101 NA --- RF4 0001 RF1 100 RF2 01 RF2 01 0 - 1100-1575 RF2 10 RF4 011 NA --- RF2 0100 RF1 100 RF2 01 RF2 01 0 - 1600-2250 RF2 10 RF3 010 RF2 010 NA ---- RF2 010 RF2 01 RF2 01 0 - 2100-2850 RF2 10 RF1 000 RF1 100 NA ---- RF2 010 RF2 01 RF2 01 0 - 2700+ RF3 11 OFF 111 RF3 001 NA ---- RF2 010 RF2 01 RF2 01 0 - -For TX: - Band SW5-AB SW4-AB SW3-X SW2-ABCD SW1-AB SWTRX-AB MIX - WB RF1 10 RF2 01 RF1 0 NA ---- SHD 00 RF4 11 0 - LB RF2 01 RF1 10 RF2 1 RF3 0010 RF3 11 RF1 00 1 - <800 RF1 10 RF2 01 RF2 1 RF3 0010 RF3 11 RF1 00 0 - 800-1700 RF1 10 RF2 01 RF2 1 RF2 0100 RF2 10 RF1 00 0 - 1700-3400 RF1 10 RF2 01 RF2 1 RF1 1000 RF1 01 RF1 00 0 - 3400-6400 RF1 10 RF2 01 RF2 1 RF4 0001 SHD 00 RF2 10 0 - -*/ - - - const double MAGNESIUM_RX_BAND1_MIN_FREQ = MAGNESIUM_LOWBAND_FREQ; - const double MAGNESIUM_RX_BAND2_MIN_FREQ = 600e6; - const double MAGNESIUM_RX_BAND3_MIN_FREQ = 1050e6; - const double MAGNESIUM_RX_BAND4_MIN_FREQ = 1600e6; - const double MAGNESIUM_RX_BAND5_MIN_FREQ = 2100e6; - const double MAGNESIUM_RX_BAND6_MIN_FREQ = 2700e6; - - const double MAGNESIUM_TX_BAND1_MIN_FREQ = MAGNESIUM_LOWBAND_FREQ; - const double MAGNESIUM_TX_BAND2_MIN_FREQ = 800e6; - const double MAGNESIUM_TX_BAND3_MIN_FREQ = 1700e6; - const double MAGNESIUM_TX_BAND4_MIN_FREQ = 3400e6; + /************************************************************************** + * ADF4351 Controls + *************************************************************************/ + double _lo_set_frequency( + adf435x_iface::sptr lo_iface, + const double freq, + const double ref_clock_freq, + const bool int_n_mode + ) { + UHD_LOG_TRACE("MG/ADF4351", + "Attempting to tune low band LO to " << freq << + " Hz with ref clock freq " << ref_clock_freq); + lo_iface->set_feedback_select(adf435x_iface::FB_SEL_DIVIDED); + lo_iface->set_reference_freq(ref_clock_freq); + lo_iface->set_prescaler(adf435x_iface::PRESCALER_4_5); + const double actual_freq = lo_iface->set_frequency(freq, int_n_mode); + lo_iface->set_output_power( + adf435x_iface::RF_OUTPUT_A, + adf435x_iface::OUTPUT_POWER_2DBM + ); + lo_iface->set_output_power( + adf435x_iface::RF_OUTPUT_B, + adf435x_iface::OUTPUT_POWER_2DBM + ); + lo_iface->set_charge_pump_current( + adf435x_iface::CHARGE_PUMP_CURRENT_0_31MA); + return actual_freq; + } - const size_t FPGPIO_MASTER_RADIO = 0; + double _lo_enable( + adf435x_iface::sptr lo_iface, + const double lo_freq, + const double ref_clock_freq, + const bool int_n_mode + ) { + const double actual_lo_freq = + _lo_set_frequency(lo_iface, lo_freq, ref_clock_freq, int_n_mode); + lo_iface->set_output_enable(adf435x_iface::RF_OUTPUT_A, true); + lo_iface->set_output_enable(adf435x_iface::RF_OUTPUT_B, true); + lo_iface->commit(); + return actual_lo_freq; + } - /*! Return a valid 'which' string for use with AD9371 API calls - * - * These strings take the form of "RX1", "TX2", ... - */ - std::string _get_which(const direction_t dir,std::string _radio_slot) + void _lo_disable(adf435x_iface::sptr lo_iface) { - UHD_ASSERT_THROW(dir == RX_DIRECTION or dir == TX_DIRECTION); - size_t chan = 0; - if (_radio_slot == "A" or _radio_slot == "C") - { - chan = 0; - } - if (_radio_slot == "B" or _radio_slot == "D") - { - chan = 1; - } - UHD_LOG_WARNING("MAGNESIUM_MYKONOS","board slot to chan map " << _radio_slot << " "<set_output_enable(adf435x_iface::RF_OUTPUT_A, false); + lo_iface->set_output_enable(adf435x_iface::RF_OUTPUT_B, false); + lo_iface->commit(); } } @@ -131,379 +80,23 @@ For TX: *****************************************************************************/ UHD_RFNOC_RADIO_BLOCK_CONSTRUCTOR(magnesium_radio_ctrl) { - UHD_LOG_TRACE("MAGNESIUM", "Entering magnesium_radio_ctrl_impl ctor..."); - UHD_LOG_DEBUG("MAGNESIUM", "Note: Running in one-block-per-channel mode!"); - const char radio_slot_name[4] = {'A','B','C','D'}; + UHD_LOG_TRACE(unique_id(), "Entering magnesium_radio_ctrl_impl ctor..."); + UHD_LOG_DEBUG(unique_id(), "Note: Running in one-block-per-channel mode!"); + const char radio_slot_name[4] = {'A', 'B', 'C', 'D'}; _radio_slot = radio_slot_name[get_block_id().get_block_count()]; - UHD_LOG_TRACE("MAGNESIUM", "Radio slot: " << _radio_slot); - if (_radio_slot == "A" or _radio_slot =="B"){ - _rpc_prefix = "db_0_"; - } - if (_radio_slot == "C" or _radio_slot=="D") - { - _rpc_prefix = "db_1_"; - } - UHD_LOG_WARNING("MAGNESIUM", "Using RPC prefix `" << _rpc_prefix << "'"); + UHD_LOG_TRACE(unique_id(), "Radio slot: " << _radio_slot); + _master = _radio_slot == "A" or _radio_slot == "C"; + UHD_LOG_DEBUG(unique_id(), + "Radio type: " << (_master ? "master" : "slave")); _init_peripherals(); _init_defaults(); - - fs_path gain_mode_path = _root_path.branch_path() - / str(boost::format("Radio_%d") % ((get_block_id().get_block_count()/2)*2)) - / "args/0/gain_mode/value"; - UHD_LOG_DEBUG("GAIN_MODE_STRING","Gain mode path " << gain_mode_path); - std::string gain_mode = _tree->access(gain_mode_path).get(); - UHD_LOG_DEBUG("GAIN_MODE_STRING","Gain mode string" << gain_mode); - //////// REST OF CTOR IS PROP TREE SETUP ////////////////////////////////// - - /**** Set up legacy compatible properties ******************************/ - // For use with multi_usrp APIs etc. - // For legacy prop tree init: - // TODO: determine DB number - const fs_path fe_base = fs_path("dboards") / _radio_slot; - const std::vector dir({ RX_DIRECTION, TX_DIRECTION }); - const std::vector fe({ "rx_frontends", "tx_frontends" }); - const std::vector ant({ "RX" , "TX" }); - const std::vector num_chans({ MAGNESIUM_NUM_RX_CHANS , MAGNESIUM_NUM_TX_CHANS }); - const size_t RX_IDX = 0; - // const size_t TX_IDX = 1; - //this->_dsa_set_gain(0.5,0,RX_DIRECTION); - for (size_t fe_idx = 0; fe_idx < fe.size(); ++fe_idx) - { - const fs_path fe_direction_path = fe_base / fe[fe_idx]; - for (size_t chan = 0; chan < num_chans[fe_idx]; ++chan) - { - const fs_path fe_path = fe_direction_path / chan; - UHD_LOG_TRACE("MAGNESIUM", "Adding FE at " << fe_path); - // Shared TX/RX attributes - _tree->create(fe_path / "name") - .set(str(boost::format("Magnesium %s %d") % ant[fe_idx] % chan)) - ; - _tree->create(fe_path / "connection") - .set("IQ") - ; - { - // TODO: fix antenna name - // Now witness the firepower of this fully armed and operational lambda - auto dir_ = dir[fe_idx]; - auto coerced_lambda = [this, chan, dir_](const std::string &ant) - { - return this->_myk_set_antenna(ant, chan, dir_); - }; - auto publisher_lambda = [this, chan, dir_]() - { - return this->_myk_get_antenna(chan, dir_); - }; - _tree->create(fe_path / "antenna" / "value") - .set(str(boost::format("%s%d") % ant[fe_idx] % (chan + 1))) - .add_coerced_subscriber(coerced_lambda) - .set_publisher(publisher_lambda); - // TODO: fix options - _tree->create>(fe_path / "antenna" / "options") - .set(std::vector(1, str(boost::format("%s%d") % ant[fe_idx] % (chan + 1)))); - } - { - auto dir_ = dir[fe_idx]; - auto coerced_lambda = [this, chan, dir_](const double freq) - { - return this->_myk_set_frequency(freq, chan, dir_); - }; - auto publisher_lambda = [this, chan, dir_]() - { - return this->_myk_get_frequency(chan, dir_); - }; - _tree->create(fe_path / "freq" / "value") - .set(MAGNESIUM_CENTER_FREQ) - .set_coercer(coerced_lambda) - .set_publisher(publisher_lambda); - _tree->create(fe_path / "freq" / "range") - .set(meta_range_t(MAGNESIUM_MIN_FREQ, MAGNESIUM_MAX_FREQ)); - } - { - auto ad9371_min_gain = (fe_idx == RX_IDX) ? AD9371_MIN_RX_GAIN : AD9371_MIN_TX_GAIN; - auto ad9371_max_gain = (fe_idx == RX_IDX) ? AD9371_MAX_RX_GAIN : AD9371_MAX_TX_GAIN; - auto ad9371_gain_step = (fe_idx == RX_IDX) ? AD9371_RX_GAIN_STEP : AD9371_TX_GAIN_STEP; - auto dsa_min_gain = DSA_MIN_GAIN; - auto dsa_max_gain = DSA_MAX_GAIN; - auto dsa_gain_step = DSA_GAIN_STEP; - auto all_min_gain = (fe_idx == RX_IDX) ? ALL_RX_MIN_GAIN : ALL_TX_MIN_GAIN; - auto all_max_gain = (fe_idx == RX_IDX) ? ALL_TX_MAX_GAIN : ALL_TX_MAX_GAIN; - auto all_gain_step = 0.5; - if (gain_mode == "auto"){ - ad9371_min_gain = 0; - ad9371_max_gain = 0; - ad9371_gain_step = 0; - dsa_min_gain = 0; - dsa_max_gain = 0; - dsa_gain_step = 0; - } - if (gain_mode == "manual") - { - all_min_gain = 0 ; - all_max_gain = 0 ; - all_gain_step = 0 ; - - } - auto dir_ = dir[fe_idx]; - //Create gain property for mykonos - auto myk_set_gain_func = [this, chan, dir_](const double gain) - { - return this->_myk_set_gain(gain, chan, dir_); - }; - auto myk_get_gain_func = [this, chan, dir_]() - { - return this->_myk_get_gain(chan, dir_); - }; - - _tree->create(fe_path / "gains" / "ad9371" / "value") - .set(0) - .set_coercer(myk_set_gain_func) - .set_publisher(myk_get_gain_func); - _tree->create(fe_path / "gains" / "ad9371" / "range") - .set(meta_range_t(ad9371_min_gain, ad9371_max_gain, ad9371_gain_step)); - // Create gain property for DSA - auto dsa_set_gain_func = [this, chan, dir_](const double gain) - { - return this->_dsa_set_gain(gain, chan, dir_); - }; - auto dsa_get_gain_func = [this, chan, dir_]() - { - return this->_dsa_get_gain(chan, dir_); - }; - _tree->create(fe_path / "gains" / "dsa" / "value") - .set(0) - .set_coercer(dsa_set_gain_func) - .set_publisher(dsa_get_gain_func); - _tree->create(fe_path / "gains" / "dsa" / "range") - .set(meta_range_t(dsa_min_gain, dsa_max_gain, dsa_gain_step)); - - // Create gain property for all gains - auto set_all_gain_func = [this, chan, dir_](const double gain) - { - return this->_set_all_gain(gain, chan, dir_); - }; - auto get_all_gain_func = [this, chan, dir_]() - { - return this->_get_all_gain(chan, dir_); - }; - _tree->create(fe_path / "gains" / "all" / "value") - .set(0) - .set_coercer(set_all_gain_func) - .set_publisher(get_all_gain_func); - _tree->create(fe_path / "gains" / "all" / "range") - .set(meta_range_t(all_min_gain, all_max_gain, all_gain_step)); - - } - // TODO: set up read/write of bandwidth properties correctly - if (fe_idx == RX_IDX) - { - auto coerced_lambda = [this, chan](const double bw) - { - return this->set_rx_bandwidth(bw, chan); - }; - auto publisher_lambda = [this, chan]() - { - return this->get_rx_bandwidth(chan); - }; - _tree->create(fe_path / "bandwidth" / "value") - .set(MAGNESIUM_DEFAULT_BANDWIDTH) - .set_coercer(coerced_lambda) - .set_publisher(publisher_lambda); - } - else { - _tree->create(fe_path / "bandwidth" / "value") - .set(MAGNESIUM_DEFAULT_BANDWIDTH); - } - _tree->create(fe_path / "bandwidth" / "range") - .set(meta_range_t(MAGNESIUM_DEFAULT_BANDWIDTH, MAGNESIUM_DEFAULT_BANDWIDTH)); - } - } - - // EEPROM paths subject to change FIXME - _tree->create(_root_path / "eeprom").set(eeprom_map_t()); - - // TODO change codec names - _tree->create("rx_codecs" / _radio_slot / "gains"); - _tree->create("tx_codecs" / _radio_slot / "gains"); - _tree->create("rx_codecs" / _radio_slot / "name").set("AD9371 Dual ADC"); - _tree->create("tx_codecs" / _radio_slot / "name").set("AD9371 Dual DAC"); - - // TODO remove this dirty hack - if (not _tree->exists("tick_rate")) - { - _tree->create("tick_rate").set(MAGNESIUM_TICK_RATE); - } + _init_prop_tree(); } magnesium_radio_ctrl_impl::~magnesium_radio_ctrl_impl() { - UHD_LOG_TRACE("MAGNESIUM", "magnesium_radio_ctrl_impl::dtor() "); -} - -/************************************************************************** - * Init Helpers - *************************************************************************/ -void magnesium_radio_ctrl_impl::_init_peripherals() -{ - UHD_LOG_TRACE("MAGNESIUM", "Initializing peripherals..."); - fs_path cpld_path = _root_path.branch_path() - / str(boost::format("Radio_%d") % ((get_block_id().get_block_count()/2)*2)) - / "cpld"; - fs_path rx_lo_path = _root_path.branch_path() - / str(boost::format("Radio_%d") % ((get_block_id().get_block_count()/2)*2)) - / "rx_lo"; - fs_path tx_lo_path = _root_path.branch_path() - / str(boost::format("Radio_%d") % ((get_block_id().get_block_count()/2)*2)) - / "tx_lo"; - // TODO: When we move back to 2 chans per RFNoC block, this needs to be - // non-conditional, and the else-branch goes away: - if (_radio_slot == "A" or _radio_slot == "C") { - UHD_LOG_TRACE("MAGNESIUM", "Initializing SPI core..."); - _spi = spi_core_3000::make(_get_ctrl(0), - radio_ctrl_impl::regs::sr_addr(radio_ctrl_impl::regs::SPI), - radio_ctrl_impl::regs::RB_SPI); - } else { - UHD_LOG_TRACE("MAGNESIUM", "Not a master radio, no SPI core."); - } - - UHD_LOG_TRACE("MAGNESIUM", "Initializing CPLD..."); - UHD_LOG_TRACE("MAGNESIUM", "CPLD path: " << cpld_path); - if (not _tree->exists(cpld_path)) { - UHD_LOG_TRACE("MAGNESIUM", "Creating new CPLD object..."); - spi_config_t spi_config; - spi_config.use_custom_divider = true; - spi_config.divider = 125; - spi_config.mosi_edge = spi_config_t::EDGE_RISE; - spi_config.miso_edge = spi_config_t::EDGE_FALL; - UHD_LOG_TRACE("MAGNESIUM", "Making CPLD object..."); - _cpld = std::make_shared( - [this, spi_config](const uint32_t transaction){ // Write functor - this->_spi->write_spi( - SEN_CPLD, - spi_config, - transaction, - 24 - ); - }, - [this, spi_config](const uint32_t transaction){ // Read functor - return this->_spi->read_spi( - SEN_CPLD, - spi_config, - transaction, - 24 - ); - } - ); - _tree->create(cpld_path).set(_cpld); - } else { - UHD_LOG_TRACE("MAGNESIUM", "Reusing someone else's CPLD object..."); - _cpld = _tree->access(cpld_path).get(); - } - - // TODO: Same comment as above applies - if (_radio_slot == "A" or _radio_slot == "C") { - UHD_LOG_TRACE("MAGNESIUM", "Initializing TX LO..."); - _tx_lo = adf435x_iface::make_adf4351( - [this](const std::vector transactions){ - for (const uint32_t transaction: transactions) { - this->_spi->write_spi( - SEN_TX_LO, - spi_config_t::EDGE_RISE, - transaction, - 32 - ); - } - } - ); - UHD_LOG_TRACE("MAGNESIUM", "Initializing RX LO..."); - _rx_lo = adf435x_iface::make_adf4351( - [this](const std::vector transactions){ - for (const uint32_t transaction: transactions) { - this->_spi->write_spi( - SEN_RX_LO, - spi_config_t::EDGE_RISE, - transaction, - 32 - ); - } - } - ); - } else { - UHD_LOG_TRACE("MAGNESIUM", "Not a master radio, no LOs."); - } - if (not _tree->exists(rx_lo_path)) { - _tree->create(rx_lo_path).set(_rx_lo); - }else - { - UHD_LOG_TRACE("MAGNESIUM", "Not a master radio. Getting LO from master" ); - _rx_lo = _tree->access(rx_lo_path).get(); - } - if (not _tree->exists(tx_lo_path)) { - _tree->create(tx_lo_path).set(_tx_lo); - }else - { - UHD_LOG_TRACE("MAGNESIUM", "Not a master radio. Getting LO from master" ); - _tx_lo = _tree->access(tx_lo_path).get(); - } - - _gpio.clear(); // Following the as-if rule, this can get optimized out - for (size_t radio_idx = 0; radio_idx < _get_num_radios(); radio_idx++) { - UHD_LOG_TRACE("MAGNESIUM", - "Initializing GPIOs for channel " << radio_idx); - _gpio.emplace_back( - gpio_atr::gpio_atr_3000::make( - _get_ctrl(radio_idx), - regs::sr_addr(regs::GPIO), - regs::RB_DB_GPIO - ) - ); - // DSA and AD9371 gain bits do *not* toggle on ATR modes. If we ever - // connect anything else to this core, we might need to set_atr_mode() - // to MODE_ATR on those bits. For now, all bits simply do what they're - // told, and don't toggle on RX/TX state changes. - _gpio.back()->set_atr_mode( - usrp::gpio_atr::MODE_GPIO, // Disable ATR mode - usrp::gpio_atr::gpio_atr_3000::MASK_SET_ALL - ); - _gpio.back()->set_gpio_ddr( - usrp::gpio_atr::DDR_OUTPUT, // Make all GPIOs outputs - usrp::gpio_atr::gpio_atr_3000::MASK_SET_ALL - ); - } - if (get_block_id().get_block_count() == FPGPIO_MASTER_RADIO) { - UHD_LOG_TRACE(unique_id(), "Initializing front-panel GPIO control...") - _fp_gpio = gpio_atr::gpio_atr_3000::make( - _get_ctrl(0), regs::sr_addr(regs::FP_GPIO), regs::RB_FP_GPIO); - } -} - -void magnesium_radio_ctrl_impl::_init_defaults() -{ - UHD_LOG_TRACE("MAGNESIUM", "Initializing defaults..."); - const size_t num_rx_chans = get_output_ports().size(); - //UHD_ASSERT_THROW(num_rx_chans == MAGNESIUM_NUM_RX_CHANS); - const size_t num_tx_chans = get_input_ports().size(); - //UHD_ASSERT_THROW(num_tx_chans == MAGNESIUM_NUM_TX_CHANS); - - UHD_LOG_TRACE("MAGNESIUM", - "Num TX chans: " << num_tx_chans - << " Num RX chans: " << num_rx_chans); - UHD_LOG_TRACE("MAGNESIUM", - "Setting tick rate to " << MAGNESIUM_TICK_RATE / 1e6 << " MHz"); - radio_ctrl_impl::set_rate(MAGNESIUM_TICK_RATE); - - for (size_t chan = 0; chan < num_rx_chans; chan++) { - radio_ctrl_impl::set_rx_frequency(MAGNESIUM_CENTER_FREQ, chan); - radio_ctrl_impl::set_rx_gain(0, chan); - radio_ctrl_impl::set_rx_antenna(MAGNESIUM_DEFAULT_RX_ANTENNA, chan); - radio_ctrl_impl::set_rx_bandwidth(MAGNESIUM_DEFAULT_BANDWIDTH, chan); - } - - for (size_t chan = 0; chan < num_tx_chans; chan++) { - radio_ctrl_impl::set_tx_frequency(MAGNESIUM_CENTER_FREQ, chan); - radio_ctrl_impl::set_tx_gain(0, chan); - radio_ctrl_impl::set_tx_antenna(MAGNESIUM_DEFAULT_TX_ANTENNA, chan); - } + UHD_LOG_TRACE(unique_id(), "magnesium_radio_ctrl_impl::dtor() "); } @@ -514,7 +107,7 @@ double magnesium_radio_ctrl_impl::set_rate(double rate) { // TODO: implement if (rate != get_rate()) { - UHD_LOG_WARNING("MAGNESIUM", + UHD_LOG_WARNING(unique_id(), "Attempting to set sampling rate to invalid value " << rate); } return get_rate(); @@ -522,43 +115,134 @@ double magnesium_radio_ctrl_impl::set_rate(double rate) void magnesium_radio_ctrl_impl::set_tx_antenna( const std::string &ant, - const size_t chan + const size_t /* chan */ ) { - _myk_set_antenna(ant, chan, TX_DIRECTION); + if (ant != MAGNESIUM_DEFAULT_TX_ANTENNA) { + throw uhd::value_error(str( + boost::format("[%s] Requesting invalid TX antenna value: %s") + % ant + )); + } + // We can't actually set the TX antenna, so let's stop here. } void magnesium_radio_ctrl_impl::set_rx_antenna( const std::string &ant, const size_t chan ) { - _myk_set_antenna(ant, chan, RX_DIRECTION); + UHD_ASSERT_THROW(chan <= MAGNESIUM_NUM_CHANS); + // TODO can we please not hardcode the antenna names here? + // We need a map somewhere, anyway, to store the available options in the + // prop tree. + if (ant != "RX2" and ant != "TX/RX") { + throw uhd::value_error(str( + boost::format("[%s] Requesting invalid RX antenna value: %s") + % ant + )); + } + + // TODO: When we go to 1 block per dboard, this entire if statement can go + // away. + if (not _master) { + // tbi + } + + UHD_LOG_TRACE(unique_id(), + "Setting RX antenna to " << ant << " for chan " << chan); + magnesium_cpld_ctrl::chan_sel_t chan_sel = + _master ? magnesium_cpld_ctrl::CHAN1 : magnesium_cpld_ctrl::CHAN2; + _update_atr_switches(chan_sel, RX_DIRECTION, ant); + + radio_ctrl_impl::set_rx_antenna(ant, chan); } double magnesium_radio_ctrl_impl::set_tx_frequency( const double freq, const size_t chan ) { - return _myk_set_frequency(freq, chan, TX_DIRECTION); + // Note: There is only one LO per tx or TX, so changing frequency will + // affect the adjacent channel in the same direction. We have to make sure + // that getters will always tell the truth! This is true for low and high + // bands. + UHD_LOG_TRACE(unique_id(), + "set_tx_frequency(f=" << freq << ", chan=" << chan << ")"); + _update_tx_freq_switches(freq, chan); + //double ad9371_freq = freq; + double if_freq = 0.0; + auto lo_iface = _tx_lo; + + if (freq < MAGNESIUM_LOWBAND_FREQ) { // Low band + if_freq = MAGNESIUM_TX_IF_FREQ ; + const double lo_freq = if_freq - freq; + const bool int_n_mode = false; // FIXME no hardcode + const double ref_clk_freq = 100e6; // FIXME no hardcode + //const double actual_lo_freq = + _lo_enable(lo_iface, lo_freq, ref_clk_freq, int_n_mode); + //ad9371_freq = actual_lo_freq - freq; + } else { + _lo_disable(lo_iface); + } + + //const double actual_ad9371_freq = + _ad9371->set_frequency(freq, chan, TX_DIRECTION); + radio_ctrl_impl::set_tx_frequency(freq, chan); + return freq; // FIXME calc the actual frequency } double magnesium_radio_ctrl_impl::set_rx_frequency( const double freq, const size_t chan ) { - return _myk_set_frequency(freq, chan, RX_DIRECTION); + // Note: There is only one LO per RX or TX, so changing frequency will + // affect the adjacent channel in the same direction. We have to make sure + // that getters will always tell the truth! This is true for low and high + // bands. + UHD_LOG_TRACE(unique_id(), + "set_rx_frequency(f=" << freq << ", chan=" << chan << ")"); + _update_rx_freq_switches(freq, chan); + //double ad9371_freq = freq; + double if_freq = 0.0; + auto lo_iface = _rx_lo; + + if (freq < MAGNESIUM_LOWBAND_FREQ) { // Low band + if_freq = MAGNESIUM_RX_IF_FREQ ; + const double lo_freq = if_freq - freq; + const bool int_n_mode = false; // FIXME no hardcode + const double ref_clk_freq = 100e6; // FIXME no hardcode + //const double actual_lo_freq = + _lo_enable(lo_iface, lo_freq, ref_clk_freq, int_n_mode); + //ad9371_freq = actual_lo_freq - freq; + } else { + _lo_disable(lo_iface); + } + + //const double actual_ad9371_freq = + _ad9371->set_frequency(freq, chan, RX_DIRECTION); + radio_ctrl_impl::set_rx_frequency(freq, chan); + return freq; // FIXME calc the actual frequency } double magnesium_radio_ctrl_impl::set_rx_bandwidth( const double bandwidth, const size_t chan ) { - return _myk_set_bandwidth(bandwidth, chan, RX_DIRECTION); + radio_ctrl_impl::set_rx_bandwidth(bandwidth, chan); + return _ad9371->set_bandwidth(bandwidth, chan, RX_DIRECTION); +} + +double magnesium_radio_ctrl_impl::set_tx_bandwidth( + const double bandwidth, + const size_t chan +) { + //radio_ctrl_impl::set_rx_bandwidth(bandwidth, chan); + return _ad9371->set_bandwidth(bandwidth, chan, TX_DIRECTION); } double magnesium_radio_ctrl_impl::set_tx_gain( const double gain, const size_t chan ) { + radio_ctrl_impl::set_tx_gain(gain, chan); return _set_all_gain(gain, chan, TX_DIRECTION); } @@ -566,68 +250,26 @@ double magnesium_radio_ctrl_impl::set_rx_gain( const double gain, const size_t chan ) { - + UHD_LOG_TRACE(unique_id(), + "set_rx_gain(gain=" << gain << ", chan=" << chan << ")"); + radio_ctrl_impl::set_rx_gain(gain, chan); return _set_all_gain(gain, chan, RX_DIRECTION); } -std::string magnesium_radio_ctrl_impl::get_tx_antenna( - const size_t chan -) /* const */ { - return _myk_get_antenna(chan, TX_DIRECTION); -} - -std::string magnesium_radio_ctrl_impl::get_rx_antenna( - const size_t chan -) /* const */ { - return _myk_get_antenna(chan, RX_DIRECTION); -} - -double magnesium_radio_ctrl_impl::get_tx_frequency( - const size_t chan -) /* const */ { - return _myk_get_frequency(chan, TX_DIRECTION); -} - -double magnesium_radio_ctrl_impl::get_rx_frequency( - const size_t chan -) /* const */ { - return _myk_get_frequency(chan, RX_DIRECTION); -} - -double magnesium_radio_ctrl_impl::get_tx_gain( - const size_t chan -) /* const */ { - return _get_all_gain(chan, TX_DIRECTION); -} - -double magnesium_radio_ctrl_impl::get_rx_gain( - const size_t chan -) /* const */ { - return _get_all_gain(chan, RX_DIRECTION); -} - -double magnesium_radio_ctrl_impl::get_rx_bandwidth( - const size_t chan -) /* const */ { - return _myk_get_bandwidth(chan, RX_DIRECTION); -} - size_t magnesium_radio_ctrl_impl::get_chan_from_dboard_fe( - const std::string &fe, const direction_t dir + const std::string &fe, const direction_t /* dir */ ) { - // UHD_LOG_TRACE("MAGNESIUM", "get_chan_from_dboard_fe " << fe << " returns " << boost::lexical_cast(fe)); return boost::lexical_cast(fe); } std::string magnesium_radio_ctrl_impl::get_dboard_fe_from_chan( const size_t chan, - const direction_t dir + const direction_t /* dir */ ) { - // UHD_LOG_TRACE("MAGNESIUM", "get_dboard_fe_from_chan " << chan << " returns " << std::to_string(chan)); return std::to_string(chan); } -double magnesium_radio_ctrl_impl::get_output_samp_rate(size_t port) +double magnesium_radio_ctrl_impl::get_output_samp_rate(size_t /* port */) { return MAGNESIUM_RADIO_RATE; } @@ -638,6 +280,13 @@ void magnesium_radio_ctrl_impl::set_rpc_client( ) { _rpcc = rpcc; _block_args = block_args; + UHD_LOG_TRACE(unique_id(), "Instantiating AD9371 control object..."); + _ad9371 = magnesium_ad9371_iface::uptr( + new magnesium_ad9371_iface( + _rpcc, + (_radio_slot == "A" or _radio_slot == "B") ? 0 : 1 + ) + ); // EEPROM paths subject to change FIXME const size_t db_idx = get_block_id().get_block_count(); @@ -656,396 +305,15 @@ void magnesium_radio_ctrl_impl::set_rpc_client( /****************************************************************************** * Helpers *****************************************************************************/ -fs_path magnesium_radio_ctrl_impl::_get_fe_path(size_t chan, direction_t dir) -{ - switch (dir) - { - case TX_DIRECTION: - return fs_path("dboards" / _radio_slot / "tx_frontends" / get_dboard_fe_from_chan(chan, TX_DIRECTION)); - case RX_DIRECTION: - return fs_path("dboards" / _radio_slot / "rx_frontends" / get_dboard_fe_from_chan(chan, RX_DIRECTION)); - default: - UHD_THROW_INVALID_CODE_PATH(); - } -} - -void magnesium_radio_ctrl_impl::_update_atr_switches( - const magnesium_cpld_ctrl::chan_sel_t chan, - const direction_t dir, - const std::string &ant -){ - magnesium_cpld_ctrl::rx_sw1_t rx_sw1 = magnesium_cpld_ctrl::RX_SW1_RX2INPUT; - magnesium_cpld_ctrl::sw_trx_t sw_trx = _sw_trx[chan]; - - bool trx_led = false, rx2_led = true; - //bool tx_pa_enb = true, tx_amp_enb = true, tx_myk_en=true; - if (ant == "TX/RX" && dir== RX_DIRECTION) - { - rx_sw1 = magnesium_cpld_ctrl::RX_SW1_TRXSWITCHOUTPUT; - sw_trx = magnesium_cpld_ctrl::SW_TRX_RXCHANNELPATH; - trx_led = true; - rx2_led = false; - } - UHD_LOG_TRACE("MAGNESIUM", "Update all atr related switches for " << dir << " " << ant ); - if (dir == RX_DIRECTION){ - _cpld->set_rx_atr_bits( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::ON, - rx_sw1, - trx_led, - rx2_led, - true, - true, - true, - true - ); - _cpld->set_tx_atr_bits( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::IDLE, - false, - sw_trx, - false, - false, - false - ); - _cpld->set_rx_atr_bits( - chan, - magnesium_cpld_ctrl::IDLE, - rx_sw1, - false, - false, - false, - false, - false, - true - ); - } - if (dir == TX_DIRECTION){ - _cpld->set_tx_atr_bits( - chan, - magnesium_cpld_ctrl::ON, - true, - sw_trx, - true, - true, - true - ); - _cpld->set_rx_atr_bits( - chan, - magnesium_cpld_ctrl::IDLE, - rx_sw1, - false, - false, - false, - false, - false, - true - ); - }; -} - -void magnesium_radio_ctrl_impl::_update_freq_switches( - const double freq, - const size_t chan, - const direction_t dir -){ - UHD_LOG_TRACE("MAGNESIUM", "Update all freq related switches for " << freq); - magnesium_cpld_ctrl::chan_sel_t chan_sel = (_radio_slot == "A" or _radio_slot == "C")? magnesium_cpld_ctrl::CHAN1 : magnesium_cpld_ctrl::CHAN2; - - // Set filters based on frequency - if (dir == RX_DIRECTION) { - if (freq < MAGNESIUM_RX_BAND1_MIN_FREQ) { - _cpld->set_rx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, - magnesium_cpld_ctrl::RX_SW3_FILTER0490LPMHZ, - magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, - magnesium_cpld_ctrl::RX_SW5_FILTER0490LPMHZFROM, - magnesium_cpld_ctrl::RX_SW6_LOWERFILTERBANKFROMSWITCH5, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_LOBAND, - true - ); - } else if (freq < MAGNESIUM_RX_BAND2_MIN_FREQ) { - _cpld->set_rx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, - magnesium_cpld_ctrl::RX_SW3_FILTER0440X0530MHZ, - magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, - magnesium_cpld_ctrl::RX_SW5_FILTER0440X0530MHZFROM, - magnesium_cpld_ctrl::RX_SW6_LOWERFILTERBANKFROMSWITCH5, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false - ); - } else if (freq < MAGNESIUM_RX_BAND3_MIN_FREQ) { - _cpld->set_rx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, - magnesium_cpld_ctrl::RX_SW3_FILTER0650X1000MHZ, - magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, - magnesium_cpld_ctrl::RX_SW5_FILTER0650X1000MHZFROM, - magnesium_cpld_ctrl::RX_SW6_LOWERFILTERBANKFROMSWITCH5, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false - ); - } else if (freq < MAGNESIUM_RX_BAND4_MIN_FREQ) { - _cpld->set_rx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, - magnesium_cpld_ctrl::RX_SW3_FILTER1100X1575MHZ, - magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, - magnesium_cpld_ctrl::RX_SW5_FILTER1100X1575MHZFROM, - magnesium_cpld_ctrl::RX_SW6_LOWERFILTERBANKFROMSWITCH5, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false - ); - } else if (freq < MAGNESIUM_RX_BAND5_MIN_FREQ) { - _cpld->set_rx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, - magnesium_cpld_ctrl::RX_SW3_FILTER1600X2250MHZ, - magnesium_cpld_ctrl::RX_SW4_FILTER1600X2250MHZFROM, - magnesium_cpld_ctrl::RX_SW5_FILTER0440X0530MHZFROM, - magnesium_cpld_ctrl::RX_SW6_UPPERFILTERBANKFROMSWITCH4, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false - ); - } else if (freq < MAGNESIUM_RX_BAND6_MIN_FREQ) { - _cpld->set_rx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::RX_SW2_LOWERFILTERBANKTOSWITCH3, - magnesium_cpld_ctrl::RX_SW3_FILTER2100X2850MHZ, - magnesium_cpld_ctrl::RX_SW4_FILTER2100X2850MHZFROM, - magnesium_cpld_ctrl::RX_SW5_FILTER0440X0530MHZFROM, - magnesium_cpld_ctrl::RX_SW6_UPPERFILTERBANKFROMSWITCH4, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false - ); - } else { - _cpld->set_rx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::RX_SW2_UPPERFILTERBANKTOSWITCH4, - magnesium_cpld_ctrl::RX_SW3_SHUTDOWNSW3, - magnesium_cpld_ctrl::RX_SW4_FILTER2700HPMHZ, - magnesium_cpld_ctrl::RX_SW5_FILTER0440X0530MHZFROM, - magnesium_cpld_ctrl::RX_SW6_UPPERFILTERBANKFROMSWITCH4, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false - ); - } - } else { - if (freq < MAGNESIUM_TX_BAND1_MIN_FREQ) { - _cpld->set_tx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1, - magnesium_cpld_ctrl::TX_SW1_FROMTXFILTERLP0800MHZ, - magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP0800MHZ, - magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_LOBAND, - true, - magnesium_cpld_ctrl::ON - ); - _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1; - } else if (freq < MAGNESIUM_TX_BAND2_MIN_FREQ) { - _cpld->set_tx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1, - magnesium_cpld_ctrl::TX_SW1_FROMTXFILTERLP0800MHZ, - magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP0800MHZ, - magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false, - magnesium_cpld_ctrl::ON - ); - _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1; - } else if (freq < MAGNESIUM_TX_BAND3_MIN_FREQ) { - _cpld->set_tx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1, - magnesium_cpld_ctrl::TX_SW1_FROMTXFILTERLP1700MHZ, - magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP1700MHZ, - magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false, - magnesium_cpld_ctrl::ON - ); - _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1; - } else if (freq < MAGNESIUM_TX_BAND4_MIN_FREQ) { - _cpld->set_tx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1, - magnesium_cpld_ctrl::TX_SW1_FROMTXFILTERLP3400MHZ, - magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP3400MHZ, - magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false, - magnesium_cpld_ctrl::ON - ); - _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1; - } else { - _cpld->set_tx_switches( - magnesium_cpld_ctrl::BOTH, - magnesium_cpld_ctrl::SW_TRX_FROMTXUPPERFILTERBANKLP6400MHZ, - magnesium_cpld_ctrl::TX_SW1_SHUTDOWNTXSW1, - magnesium_cpld_ctrl::TX_SW2_TOTXFILTERLP6400MHZ, - magnesium_cpld_ctrl::TX_SW3_TOTXFILTERBANKS, - magnesium_cpld_ctrl::LOWBAND_MIXER_PATH_SEL_BYPASS, - false, - magnesium_cpld_ctrl::ON - ); - _sw_trx[chan_sel] = magnesium_cpld_ctrl::SW_TRX_FROMTXUPPERFILTERBANKLP6400MHZ; - } - } - UHD_LOG_INFO("MAGNESIUM", "Update all freq related switches for " << freq <<" finished!."); -} -/****************************************************************************** - * AD9371 Controls - *****************************************************************************/ -double magnesium_radio_ctrl_impl::_myk_set_frequency( - const double freq, - const size_t chan, - const direction_t dir -) { - - - // Note: There is only one LO per RX or TX, so changing frequency will - // affect the adjacent channel in the same direction. We have to make sure - // that getters will always tell the truth! - auto which = _get_which(dir,_radio_slot); - UHD_LOG_TRACE("MAGNESIUM", "requested frequency of " << freq); - _update_freq_switches(freq,chan,dir); - double ad9371_freq = freq; - auto lo_iface = (dir == RX_DIRECTION) ? _rx_lo : _tx_lo; - - if (freq < MAGNESIUM_LOWBAND_FREQ) { // Low band - UHD_LOG_WARNING("LO BAND", "requested frequency of " << freq); - double if_freq = (dir == RX_DIRECTION) ? MAGNESIUM_RX_IF_FREQ - : MAGNESIUM_TX_IF_FREQ; - double lo_freq = if_freq - freq; - _lo_set_frequency(lo_iface, lo_freq, chan); - lo_iface->set_output_enable(adf435x_iface::RF_OUTPUT_A, true); // TODO: Find correct value - lo_iface->set_output_enable(adf435x_iface::RF_OUTPUT_B, true); // TODO: Find correct value - lo_iface->commit(); - ad9371_freq = if_freq; - } else { - UHD_LOG_WARNING("HI BAND", "requested frequency of " << freq); - lo_iface->set_output_enable(adf435x_iface::RF_OUTPUT_A, false); // TODO: Find correct value - lo_iface->set_output_enable(adf435x_iface::RF_OUTPUT_B, false); // TODO: Find correct value - lo_iface->commit(); - } - - UHD_LOG_TRACE("MAGNESIUM", - "Calling " << _rpc_prefix << "set_freq on " << which << " with " << ad9371_freq); - auto retval = _rpcc->request_with_token(_rpc_prefix + "set_freq", which, ad9371_freq, false); - UHD_LOG_TRACE("MAGNESIUM", - _rpc_prefix << "set_freq returned " << retval); - - - return retval; -} - -double magnesium_radio_ctrl_impl::_myk_set_gain( - const double gain, - const size_t chan, - const direction_t dir -) { - auto which = _get_which(dir,_radio_slot); - UHD_LOG_TRACE("MAGNESIUM", "Calling " << _rpc_prefix << "set_gain on " << which << " with " << gain); - auto retval = _rpcc->request_with_token(_rpc_prefix + "set_gain", which, gain); - UHD_LOG_TRACE("MAGNESIUM", _rpc_prefix << "set_gain returned " << retval); - - return retval; - //return 0.0; -} - -void magnesium_radio_ctrl_impl::_myk_set_antenna( - const std::string &ant, - const size_t chan, - const direction_t dir -) { - // TODO: implement - UHD_LOG_WARNING("MAGNESIUM", "Attempting to set antenna " << ant << " " << chan << " " << dir ); - magnesium_cpld_ctrl::chan_sel_t chan_sel = (_radio_slot == "A" or _radio_slot == "C")? magnesium_cpld_ctrl::CHAN1 : magnesium_cpld_ctrl::CHAN2; - _update_atr_switches(chan_sel,dir,ant); -} - -double magnesium_radio_ctrl_impl::_myk_set_bandwidth(const double bandwidth, const size_t chan, const direction_t dir) -{ - // TODO: implement - UHD_LOG_WARNING("MAGNESIUM", "Ignoring attempt to set bandwidth"); - return get_rx_bandwidth(chan); -} - -double magnesium_radio_ctrl_impl::_myk_get_frequency(const size_t chan, const direction_t dir) -{ - auto which = _get_which(dir,_radio_slot); - UHD_LOG_TRACE("MAGNESIUM", "calling " << _rpc_prefix << "get_freq on " << which); - auto retval = _rpcc->request_with_token(_rpc_prefix + "get_freq", which); - UHD_LOG_TRACE("MAGNESIUM", _rpc_prefix << "get_freq returned " << retval); - return retval; -} - -double magnesium_radio_ctrl_impl::_myk_get_gain(const size_t chan, const direction_t dir) -{ - auto which = _get_which(dir,_radio_slot); - UHD_LOG_TRACE("MAGNESIUM", "calling " << _rpc_prefix << "get_gain on " << which); - auto retval = _rpcc->request_with_token(_rpc_prefix + "get_gain", which); - UHD_LOG_TRACE("MAGNESIUM", _rpc_prefix << "get_gain returned " << retval); - return retval; -} - -std::string magnesium_radio_ctrl_impl::_myk_get_antenna(const size_t chan, const direction_t dir) -{ - // TODO: implement - UHD_LOG_WARNING("MAGNESIUM", "Ignoring attempt to get antenna"); - return "RX1"; - // CPLD control? -} - -double magnesium_radio_ctrl_impl::_myk_get_bandwidth(const size_t chan, const direction_t dir) -{ - // TODO: implement - UHD_LOG_WARNING("MAGNESIUM", "Ignoring attempt to get bandwidth"); - return MAGNESIUM_DEFAULT_BANDWIDTH; -} - -/****************************************************************************** - * ADF4351 Controls - *****************************************************************************/ -double magnesium_radio_ctrl_impl::_lo_set_frequency( - adf435x_iface::sptr lo_iface, - const double freq, - const size_t chan -) { - - UHD_LOG_TRACE("MAGNESIUM", "attempting to tune low band LO to " << freq); - lo_iface->set_feedback_select(adf435x_iface::FB_SEL_DIVIDED); - lo_iface->set_reference_freq(100e6); // FIXME: How to get refclk freq? This can change. - lo_iface->set_prescaler(adf435x_iface::PRESCALER_4_5); - - double actual_freq = 0.0; - actual_freq = lo_iface->set_frequency(freq, false); // FIXME: always fractional-n mode - UHD_LOG_TRACE("MAGNESIUM", "actual low band LO is " << actual_freq); - lo_iface->set_output_power(adf435x_iface::RF_OUTPUT_A, adf435x_iface::OUTPUT_POWER_2DBM); // TODO: Find correct value - lo_iface->set_output_power(adf435x_iface::RF_OUTPUT_B, adf435x_iface::OUTPUT_POWER_2DBM); // TODO: Find correct value - lo_iface->set_charge_pump_current(adf435x_iface::CHARGE_PUMP_CURRENT_0_31MA); - - // TODO: Check for PLL lock - //sleep(1); - //auto lock_det = _rpcc->request_with_token(_slot_prefix + "cpld_peek", 0x12); - //UHD_LOG_TRACE("MAGNESIUM", "lock detect is " << lock_det); - - return actual_freq; -} double magnesium_radio_ctrl_impl::_set_all_gain( const double gain, const size_t chan, const direction_t dir -) -{ - UHD_LOG_TRACE("MAGNESIUM", "Setting all gain " << gain); +) { + UHD_LOG_TRACE(unique_id(), "Setting all gain " << gain); // just naively distributed gain here - _myk_set_gain(gain/2,chan,dir); - _dsa_set_gain(gain/2,chan,dir); + _ad9371->set_gain(gain/2, chan, dir); + _dsa_set_gain(gain/2, chan, dir); if(dir == RX_DIRECTION or dir == DX_DIRECTION) { _all_rx_gain = gain; @@ -1057,70 +325,62 @@ double magnesium_radio_ctrl_impl::_set_all_gain( return gain; } + double magnesium_radio_ctrl_impl::_get_all_gain( - const size_t chan, + const size_t /* chan */, const direction_t dir -) -{ - UHD_LOG_TRACE("MAGNESIUM", "Getting all gain "); - if(dir == RX_DIRECTION ) - { +) { + UHD_LOG_TRACE(unique_id(), "Getting all gain "); + if (dir == RX_DIRECTION) { return _all_rx_gain; } - if(dir == TX_DIRECTION) - { - return _all_tx_gain; - } - + return _all_tx_gain; } -double magnesium_radio_ctrl_impl::_dsa_set_gain( + +/****************************************************************************** + * DSA Controls + *****************************************************************************/ +double magnesium_radio_ctrl_impl::_dsa_set_gain( const double gain, const size_t chan, const direction_t dir ) { - uint32_t dsa_val = 63-2*gain; - UHD_LOG_TRACE("MAGNESIUM", "Setting dsa gain " << dsa_val); + UHD_LOG_TRACE(unique_id(), "Setting dsa gain " << dsa_val); _set_dsa_val(chan, dir, dsa_val); - if(dir == RX_DIRECTION or dir == DX_DIRECTION) - { + if (dir == RX_DIRECTION or dir == DX_DIRECTION) { _dsa_rx_gain = gain; } - if(dir == TX_DIRECTION or dir == DX_DIRECTION) - { + if (dir == TX_DIRECTION or dir == DX_DIRECTION) { _dsa_tx_gain = gain; } return gain; } -double magnesium_radio_ctrl_impl::_dsa_get_gain( - const size_t chan, - const direction_t dir - ){ - UHD_LOG_TRACE("MAGNESIUM", "Getting dsa gain "); - if(dir == RX_DIRECTION ) - { - return _dsa_rx_gain; - } - if(dir == TX_DIRECTION) - { - return _dsa_tx_gain; - } +double magnesium_radio_ctrl_impl::_dsa_get_gain( + const size_t /*chan*/, + const direction_t dir +) { + if (dir == RX_DIRECTION) { + return _dsa_rx_gain; + } + return _dsa_tx_gain; } -/****************************************************************************** - * DSA Controls - *****************************************************************************/ + void magnesium_radio_ctrl_impl::_set_dsa_val( const size_t chan, const direction_t dir, - const uint32_t dsa_val + const uint32_t dsa_val ) { - UHD_LOG_TRACE("MAGNESIUM", "Setting dsa value" << dsa_val); if (dir == RX_DIRECTION or dir == DX_DIRECTION){ + UHD_LOG_TRACE(unique_id(), + "Chan: " << chan << " Setting RX DSA value" << dsa_val); _gpio[chan]->set_gpio_out(dsa_val, 0x3F); } if (dir == TX_DIRECTION or dir == DX_DIRECTION){ + UHD_LOG_TRACE(unique_id(), + "Chan: " << chan << " Setting TX DSA value" << dsa_val); _gpio[chan]->set_gpio_out(dsa_val, 0x0FC0); } } diff --git a/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.hpp b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.hpp index e2cfab304..cf9ba765d 100644 --- a/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.hpp +++ b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.hpp @@ -22,6 +22,7 @@ #include "rpc_block_ctrl.hpp" #include "magnesium_cpld_ctrl.hpp" #include "magnesium_cpld_regs.hpp" +#include "magnesium_ad9371_iface.hpp" #include "adf435x.hpp" #include "gpio_atr_3000.hpp" #include @@ -47,6 +48,8 @@ public: /************************************************************************ * API calls ***********************************************************************/ + // Note: We use the cached values in radio_ctrl_impl, so most getters are + // not reimplemented here double set_rate(double rate); void set_tx_antenna(const std::string &ant, const size_t chan); @@ -54,21 +57,12 @@ public: 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); - std::string get_tx_antenna(const size_t chan); - std::string get_rx_antenna(const size_t chan); - - double get_tx_frequency(const size_t chan); - double get_rx_frequency(const size_t chan); - double get_rx_bandwidth(const size_t chan); - - double get_tx_gain(const size_t chan); - double get_rx_gain(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); @@ -83,70 +77,64 @@ private: /************************************************************************** * Helpers *************************************************************************/ - //! Return the path to the dboards property subtree - fs_path _get_fe_path(const size_t chan, const direction_t dir); - //! Initialize all the peripherals connected to this block void _init_peripherals(); //! Set state of this class to sensible defaults void _init_defaults(); - /************************************************************************** - * AD9371 Controls - *************************************************************************/ - double _myk_set_frequency( - const double freq, - const size_t chan, - const direction_t dir + //! Init a subtree for the RF frontends + void _init_frontend_subtree( + uhd::property_tree::sptr subtree, + const size_t fe_chan_idx, + const size_t chan_idx ); - double _myk_set_gain( + //! Initialize property tree + void _init_prop_tree(); + + /************************************************************************** + * Gain Controls (implemented in magnesium_radio_ctrl_gain.cpp) + *************************************************************************/ + double _dsa_set_gain( const double gain, const size_t chan, const direction_t dir ); - void _myk_set_antenna( - const std::string &ant, - const size_t chan, - const direction_t dir - ); - double _myk_set_bandwidth( - const double bandwidth, + double _dsa_get_gain( const size_t chan, const direction_t dir ); - double _myk_get_frequency( - const size_t chan, - const direction_t dir + double _set_all_gain( + const double gain, + const size_t chan, + const direction_t dir ); - double _myk_get_gain( - const size_t chan, - const direction_t dir + double _get_all_gain( + const size_t chan, + const direction_t dir ); - std::string _myk_get_antenna( + void _set_dsa_val( const size_t chan, - const direction_t dir + const direction_t dir, + const uint32_t dsa_val ); - double _myk_get_bandwidth( - const size_t chan, - const direction_t dir + /************************************************************************** + * CPLD Controls (implemented in magnesium_radio_ctrl_cpld.cpp) + *************************************************************************/ + void _update_rx_freq_switches( + const double freq, + const size_t chan ); - double _lo_set_frequency( - adf435x_iface::sptr lo_iface, - const double freq, - const size_t chan - ); - void _update_freq_switches( + void _update_tx_freq_switches( const double freq, - const size_t chan, - const direction_t dir + const size_t chan ); void _update_atr_switches( @@ -155,34 +143,15 @@ private: const std::string &ant ); - double _dsa_set_gain( - const double gain, - const size_t chan, - const direction_t dir - ); - double _dsa_get_gain( - const size_t chan, - const direction_t dir - ); - double _set_all_gain( - const double gain, - const size_t chan, - const direction_t dir - ); - double _get_all_gain( - const size_t chan, - const direction_t dir - ); - void _set_dsa_val( - const size_t chan, - const direction_t dir, - const uint32_t dsa_val - ); + /************************************************************************** + * Private attributes + *************************************************************************/ //! Letter representation of the radio we're currently running std::string _radio_slot; - //! Stores the prefix to RPC calls - std::string _rpc_prefix; + //! If true, this is a master radio. This attribute will go away when we + // move back to 1 radio block per dboard. + bool _master; //! Additional block args; gets set during set_rpc_client() uhd::device_addr_t _block_args; @@ -203,6 +172,9 @@ private: // there's only one CPLD control. std::shared_ptr _cpld; + //! Reference to the AD9371 controls + magnesium_ad9371_iface::uptr _ad9371; + //! 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. @@ -222,10 +194,12 @@ private: //! All gain double _all_rx_gain = 0.0; double _all_tx_gain = 0.0; - //! TRX switch state of 2 channels - std::map _sw_trx = { - {magnesium_cpld_ctrl::CHAN1, magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1}, - {magnesium_cpld_ctrl::CHAN2, magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1} + //! TRX switch state of 2 channels + std::map _sw_trx = { + {magnesium_cpld_ctrl::CHAN1, + magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1}, + {magnesium_cpld_ctrl::CHAN2, + magnesium_cpld_ctrl::SW_TRX_FROMLOWERFILTERBANKTXSW1} }; }; /* class radio_ctrl_impl */ diff --git a/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_init.cpp b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_init.cpp new file mode 100644 index 000000000..ed5cb61c3 --- /dev/null +++ b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_init.cpp @@ -0,0 +1,487 @@ +// +// Copyright 2017 Ettus Research, a National Instruments Company +// +// SPDX-License-Identifier: GPL-3.0 +// + +#include "magnesium_radio_ctrl_impl.hpp" +#include "magnesium_constants.hpp" +#include "spi_core_3000.hpp" +#include +#include +#include +#include + +using namespace uhd; +using namespace uhd::rfnoc; + +namespace { + enum slave_select_t { + SEN_CPLD = 1, + SEN_TX_LO = 2, + SEN_RX_LO = 4, + SEN_PHASE_DAC = 8 + }; +} + +void magnesium_radio_ctrl_impl::_init_peripherals() +{ + UHD_LOG_TRACE(unique_id(), "Initializing peripherals..."); + fs_path cpld_path = _root_path.branch_path() + / str(boost::format("Radio_%d") % ((get_block_id().get_block_count()/2)*2)) + / "cpld"; + fs_path rx_lo_path = _root_path.branch_path() + / str(boost::format("Radio_%d") % ((get_block_id().get_block_count()/2)*2)) + / "rx_lo"; + fs_path tx_lo_path = _root_path.branch_path() + / str(boost::format("Radio_%d") % ((get_block_id().get_block_count()/2)*2)) + / "tx_lo"; + // TODO: When we move back to 2 chans per RFNoC block, this needs to be + // non-conditional, and the else-branch goes away: + if (_master) { + UHD_LOG_TRACE(unique_id(), "Initializing SPI core..."); + _spi = spi_core_3000::make(_get_ctrl(0), + radio_ctrl_impl::regs::sr_addr(radio_ctrl_impl::regs::SPI), + radio_ctrl_impl::regs::RB_SPI); + } else { + UHD_LOG_TRACE(unique_id(), "Not a master radio, no SPI core."); + } + + UHD_LOG_TRACE(unique_id(), "Initializing CPLD..."); + UHD_LOG_TRACE(unique_id(), "CPLD path: " << cpld_path); + if (not _tree->exists(cpld_path)) { + UHD_LOG_TRACE(unique_id(), "Creating new CPLD object..."); + spi_config_t spi_config; + spi_config.use_custom_divider = true; + spi_config.divider = 125; + spi_config.mosi_edge = spi_config_t::EDGE_RISE; + spi_config.miso_edge = spi_config_t::EDGE_FALL; + UHD_LOG_TRACE(unique_id(), "Making CPLD object..."); + _cpld = std::make_shared( + [this, spi_config](const uint32_t transaction){ // Write functor + this->_spi->write_spi( + SEN_CPLD, + spi_config, + transaction, + 24 + ); + }, + [this, spi_config](const uint32_t transaction){ // Read functor + return this->_spi->read_spi( + SEN_CPLD, + spi_config, + transaction, + 24 + ); + } + ); + _tree->create(cpld_path).set(_cpld); + } else { + UHD_LOG_TRACE(unique_id(), "Reusing someone else's CPLD object..."); + _cpld = _tree->access(cpld_path).get(); + } + + // TODO: Same comment as above applies + if (_master) { + UHD_LOG_TRACE(unique_id(), "Initializing TX LO..."); + _tx_lo = adf435x_iface::make_adf4351( + [this](const std::vector transactions){ + for (const uint32_t transaction: transactions) { + this->_spi->write_spi( + SEN_TX_LO, + spi_config_t::EDGE_RISE, + transaction, + 32 + ); + } + } + ); + UHD_LOG_TRACE(unique_id(), "Initializing RX LO..."); + _rx_lo = adf435x_iface::make_adf4351( + [this](const std::vector transactions){ + for (const uint32_t transaction: transactions) { + this->_spi->write_spi( + SEN_RX_LO, + spi_config_t::EDGE_RISE, + transaction, + 32 + ); + } + } + ); + } else { + UHD_LOG_TRACE(unique_id(), "Not a master radio, no LOs."); + } + if (not _tree->exists(rx_lo_path)) { + _tree->create(rx_lo_path).set(_rx_lo); + } else { + UHD_LOG_TRACE(unique_id(), "Not a master radio. Getting LO from master" ); + _rx_lo = _tree->access(rx_lo_path).get(); + } + if (not _tree->exists(tx_lo_path)) { + _tree->create(tx_lo_path).set(_tx_lo); + } else { + UHD_LOG_TRACE(unique_id(), "Not a master radio. Getting LO from master" ); + _tx_lo = _tree->access(tx_lo_path).get(); + } + + _gpio.clear(); // Following the as-if rule, this can get optimized out + for (size_t radio_idx = 0; radio_idx < _get_num_radios(); radio_idx++) { + UHD_LOG_TRACE(unique_id(), + "Initializing GPIOs for channel " << radio_idx); + _gpio.emplace_back( + usrp::gpio_atr::gpio_atr_3000::make( + _get_ctrl(radio_idx), + regs::sr_addr(regs::GPIO), + regs::RB_DB_GPIO + ) + ); + // DSA and AD9371 gain bits do *not* toggle on ATR modes. If we ever + // connect anything else to this core, we might need to set_atr_mode() + // to MODE_ATR on those bits. For now, all bits simply do what they're + // told, and don't toggle on RX/TX state changes. + _gpio.back()->set_atr_mode( + usrp::gpio_atr::MODE_GPIO, // Disable ATR mode + usrp::gpio_atr::gpio_atr_3000::MASK_SET_ALL + ); + _gpio.back()->set_gpio_ddr( + usrp::gpio_atr::DDR_OUTPUT, // Make all GPIOs outputs + usrp::gpio_atr::gpio_atr_3000::MASK_SET_ALL + ); + } + if (get_block_id().get_block_count() == FPGPIO_MASTER_RADIO) { + UHD_LOG_TRACE(unique_id(), "Initializing front-panel GPIO control...") + _fp_gpio = usrp::gpio_atr::gpio_atr_3000::make( + _get_ctrl(0), regs::sr_addr(regs::FP_GPIO), regs::RB_FP_GPIO); + } +} + +void magnesium_radio_ctrl_impl::_init_defaults() +{ + UHD_LOG_TRACE(unique_id(), "Initializing defaults..."); + const size_t num_rx_chans = get_output_ports().size(); + const size_t num_tx_chans = get_input_ports().size(); + + UHD_LOG_TRACE(unique_id(), + "Num TX chans: " << num_tx_chans + << " Num RX chans: " << num_rx_chans); + UHD_LOG_TRACE(unique_id(), + "Setting tick rate to " << MAGNESIUM_TICK_RATE / 1e6 << " MHz"); + radio_ctrl_impl::set_rate(MAGNESIUM_TICK_RATE); + + for (size_t chan = 0; chan < num_rx_chans; chan++) { + radio_ctrl_impl::set_rx_frequency(MAGNESIUM_CENTER_FREQ, chan); + radio_ctrl_impl::set_rx_gain(0, chan); + radio_ctrl_impl::set_rx_antenna(MAGNESIUM_DEFAULT_RX_ANTENNA, chan); + radio_ctrl_impl::set_rx_bandwidth(MAGNESIUM_DEFAULT_BANDWIDTH, chan); + } + + for (size_t chan = 0; chan < num_tx_chans; chan++) { + radio_ctrl_impl::set_tx_frequency(MAGNESIUM_CENTER_FREQ, chan); + radio_ctrl_impl::set_tx_gain(0, chan); + radio_ctrl_impl::set_tx_antenna(MAGNESIUM_DEFAULT_TX_ANTENNA, chan); + } +} + +void magnesium_radio_ctrl_impl::_init_frontend_subtree( + uhd::property_tree::sptr subtree, + const size_t fe_chan_idx, + const size_t chan_idx +) { + const fs_path tx_fe_path = fs_path("tx_frontends") / fe_chan_idx; + const fs_path rx_fe_path = fs_path("rx_frontends") / fe_chan_idx; + UHD_LOG_TRACE(unique_id(), + "Adding non-RFNoC block properties for channel " << chan_idx << + " to prop tree path " << tx_fe_path << " and " << rx_fe_path); + // TX Standard attributes + subtree->create(tx_fe_path / "name") + .set(str(boost::format("Magnesium"))) + ; + subtree->create(tx_fe_path / "connection") + .set("IQ") + ; + // RX Standard attributes + subtree->create(rx_fe_path / "name") + .set(str(boost::format("Magnesium"))) + ; + subtree->create(rx_fe_path / "connection") + .set("IQ") + ; + // TX Antenna + subtree->create(tx_fe_path / "antenna" / "value") + .add_coerced_subscriber([this, chan_idx](const std::string &ant){ + this->set_tx_antenna(ant, chan_idx); + }) + .set_publisher([this, chan_idx](){ + return this->get_tx_antenna(chan_idx); + }) + ; + subtree->create>(tx_fe_path / "antenna" / "options") + .set({MAGNESIUM_DEFAULT_TX_ANTENNA}) + .add_coerced_subscriber([](const std::vector &){ + throw uhd::runtime_error( + "Attempting to update antenna options!"); + }) + ; + // RX Antenna + subtree->create(rx_fe_path / "antenna" / "value") + .add_coerced_subscriber([this, chan_idx](const std::string &ant){ + this->set_rx_antenna(ant, chan_idx); + }) + .set_publisher([this, chan_idx](){ + return this->get_rx_antenna(chan_idx); + }) + ; + subtree->create>(rx_fe_path / "antenna" / "options") + .set(MAGNESIUM_RX_ANTENNAS) + .add_coerced_subscriber([](const std::vector &){ + throw uhd::runtime_error( + "Attempting to update antenna options!"); + }) + ; + // TX frequency + subtree->create(tx_fe_path / "freq" / "value") + .set_coercer([this, chan_idx](const double freq){ + return this->set_tx_frequency(freq, chan_idx); + }) + .set_publisher([this, chan_idx](){ + return this->get_tx_frequency(chan_idx); + }) + ; + subtree->create(tx_fe_path / "freq" / "range") + .set(meta_range_t(MAGNESIUM_MIN_FREQ, MAGNESIUM_MAX_FREQ, 1.0)) + .add_coerced_subscriber([](const meta_range_t &){ + throw uhd::runtime_error( + "Attempting to update freq range!"); + }) + ; + // RX frequency + subtree->create(rx_fe_path / "freq" / "value") + .set_coercer([this, chan_idx](const double freq){ + return this->set_rx_frequency(freq, chan_idx); + }) + .set_publisher([this, chan_idx](){ + return this->get_rx_frequency(chan_idx); + }) + ; + subtree->create(rx_fe_path / "freq" / "range") + .set(meta_range_t(MAGNESIUM_MIN_FREQ, MAGNESIUM_MAX_FREQ, 1.0)) + .add_coerced_subscriber([](const meta_range_t &){ + throw uhd::runtime_error( + "Attempting to update freq range!"); + }) + ; + // TX bandwidth + subtree->create(tx_fe_path / "bandwidth" / "value") + .set_coercer([this, chan_idx](const double bw){ + return this->set_tx_bandwidth(bw, chan_idx); + }) + .set_publisher([this, chan_idx](){ + //return this->get_tx_bandwidth(chan_idx); + return 0.0; // FIXME + }) + ; + subtree->create(tx_fe_path / "bandwidth" / "range") + .set(meta_range_t(0.0, 0.0, 0.0)) // FIXME + .add_coerced_subscriber([](const meta_range_t &){ + throw uhd::runtime_error( + "Attempting to update bandwidth range!"); + }) + ; + // RX bandwidth + subtree->create(rx_fe_path / "bandwidth" / "value") + .set_coercer([this, chan_idx](const double bw){ + return this->set_rx_bandwidth(bw, chan_idx); + }) + .set_publisher([this, chan_idx](){ + return this->get_rx_bandwidth(chan_idx); + }) + ; + subtree->create(rx_fe_path / "bandwidth" / "range") + .set(meta_range_t(0.0, 0.0, 0.0)) // FIXME + .add_coerced_subscriber([](const meta_range_t &){ + throw uhd::runtime_error( + "Attempting to update bandwidth range!"); + }) + ; + // TX gains + subtree->create(tx_fe_path / "gains" / "all" / "value") + .set_coercer([this, chan_idx](const double gain){ + return this->set_tx_gain(gain, chan_idx); + }) + .set_publisher([this, chan_idx](){ + return this->get_tx_gain(chan_idx); + }) + ; + subtree->create(tx_fe_path / "gains" / "all" / "range") + .set(meta_range_t(0.0, 60.0, 1.0)) // FIXME + .add_coerced_subscriber([](const meta_range_t &){ + throw uhd::runtime_error( + "Attempting to update bandwidth range!"); + }) + ; + // RX gains + subtree->create(rx_fe_path / "gains" / "all" / "value") + .set_coercer([this, chan_idx](const double gain){ + UHD_VAR(gain); + return this->set_rx_gain(gain, chan_idx); + }) + .set_publisher([this, chan_idx](){ + return this->get_rx_gain(chan_idx); + }) + ; + subtree->create(rx_fe_path / "gains" / "all" / "range") + .set(meta_range_t(0.0, 60.0, 1.0)) // FIXME + .add_coerced_subscriber([](const meta_range_t &){ + throw uhd::runtime_error( + "Attempting to update bandwidth range!"); + }) + ; + // FIXME separate DSA and Myk gains +} + +void magnesium_radio_ctrl_impl::_init_prop_tree() +{ + fs_path gain_mode_path = + _root_path.branch_path() + / str(boost::format("Radio_%d") % ((get_block_id().get_block_count()/2)*2)) + / "args/0/gain_mode/value"; + UHD_LOG_DEBUG("GAIN_MODE_STRING","Gain mode path " << gain_mode_path); + std::string gain_mode = _tree->access(gain_mode_path).get(); + UHD_LOG_DEBUG("GAIN_MODE_STRING","Gain mode string" << gain_mode); + + /**** Set up legacy compatible properties ******************************/ + // For use with multi_usrp APIs etc. + // For legacy prop tree init: + // TODO: determine DB number + const fs_path fe_base = fs_path("dboards") / _radio_slot; + const std::vector dir({ RX_DIRECTION, TX_DIRECTION }); + const std::vector fe({ "rx_frontends", "tx_frontends" }); + const std::vector ant({ "RX" , "TX" }); + //const size_t RX_IDX = 0; + // const size_t TX_IDX = 1; + //this->_dsa_set_gain(0.5,0,RX_DIRECTION); + for (size_t fe_idx = 0; fe_idx < fe.size(); ++fe_idx) + { + const fs_path fe_direction_path = fe_base / fe[fe_idx]; + for (size_t chan = 0; chan < MAGNESIUM_NUM_CHANS; ++chan) + { + const fs_path fe_path = fe_direction_path / chan; + UHD_LOG_TRACE(unique_id(), "Adding FE at " << fe_path); + // Shared TX/RX attributes + //{ + //auto ad9371_min_gain = (fe_idx == RX_IDX) ? AD9371_MIN_RX_GAIN : AD9371_MIN_TX_GAIN; + //auto ad9371_max_gain = (fe_idx == RX_IDX) ? AD9371_MAX_RX_GAIN : AD9371_MAX_TX_GAIN; + //auto ad9371_gain_step = (fe_idx == RX_IDX) ? AD9371_RX_GAIN_STEP : AD9371_TX_GAIN_STEP; + //auto dsa_min_gain = DSA_MIN_GAIN; + //auto dsa_max_gain = DSA_MAX_GAIN; + //auto dsa_gain_step = DSA_GAIN_STEP; + //auto all_min_gain = (fe_idx == RX_IDX) ? ALL_RX_MIN_GAIN : ALL_TX_MIN_GAIN; + //auto all_max_gain = (fe_idx == RX_IDX) ? ALL_TX_MAX_GAIN : ALL_TX_MAX_GAIN; + //auto all_gain_step = 0.5; + //if (gain_mode == "auto"){ + //ad9371_min_gain = 0; + //ad9371_max_gain = 0; + //ad9371_gain_step = 0; + //dsa_min_gain = 0; + //dsa_max_gain = 0; + //dsa_gain_step = 0; + //} + //if (gain_mode == "manual") + //{ + //all_min_gain = 0 ; + //all_max_gain = 0 ; + //all_gain_step = 0 ; + + //} + //auto dir_ = dir[fe_idx]; + ////Create gain property for mykonos + //auto myk_set_gain_func = [this, chan, dir_](const double gain) + //{ + //return this->_myk_set_gain(gain, chan, dir_); + //}; + //auto myk_get_gain_func = [this, chan, dir_]() + //{ + //return this->_myk_get_gain(chan, dir_); + //}; + + //_tree->create(fe_path / "gains" / "ad9371" / "value") + //.set(0) + //.set_coercer(myk_set_gain_func) + //.set_publisher(myk_get_gain_func); + //_tree->create(fe_path / "gains" / "ad9371" / "range") + //.set(meta_range_t(ad9371_min_gain, ad9371_max_gain, ad9371_gain_step)); + //// Create gain property for DSA + //auto dsa_set_gain_func = [this, chan, dir_](const double gain) + //{ + //return this->_dsa_set_gain(gain, chan, dir_); + //}; + //auto dsa_get_gain_func = [this, chan, dir_]() + //{ + //return this->_dsa_get_gain(chan, dir_); + //}; + //_tree->create(fe_path / "gains" / "dsa" / "value") + //.set(0) + //.set_coercer(dsa_set_gain_func) + //.set_publisher(dsa_get_gain_func); + //_tree->create(fe_path / "gains" / "dsa" / "range") + //.set(meta_range_t(dsa_min_gain, dsa_max_gain, dsa_gain_step)); + + //// Create gain property for all gains + //auto set_all_gain_func = [this, chan, dir_](const double gain) + //{ + //return this->_set_all_gain(gain, chan, dir_); + //}; + //auto get_all_gain_func = [this, chan, dir_]() + //{ + //return this->_get_all_gain(chan, dir_); + //}; + //_tree->create(fe_path / "gains" / "all" / "value") + //.set(0) + //.set_coercer(set_all_gain_func) + //.set_publisher(get_all_gain_func); + //_tree->create(fe_path / "gains" / "all" / "range") + //.set(meta_range_t(all_min_gain, all_max_gain, all_gain_step)); + + //} + } + } + + + // TODO this might be wrong + if (_master) { + this->_init_frontend_subtree(_tree->subtree(fe_base), 0, 0); + std::string slave_slot = (_radio_slot == "A") ? "B" : "D"; + UHD_LOG_TRACE(unique_id(), + "Also registering props for slave radio " << slave_slot); + this->_init_frontend_subtree( + _tree->subtree(fs_path("dboards") / slave_slot), 0, 1); + } + // TODO: When we go to one radio per dboard, the above if statement goes + // away, and instead we have something like this: + /* + *for (chan_idx = 0; chan_idx < MAGNESIUM_NUM_CHANS; chan_idx++) { + * this->_init_frontend_subtree( + * _tree->get_subtree(fe_base), chan_idx, chan_idx); + *} + */ + + + + // EEPROM paths subject to change FIXME + _tree->create(_root_path / "eeprom") + .set(eeprom_map_t()); + + // TODO change codec names + _tree->create("rx_codecs" / _radio_slot / "gains"); + _tree->create("tx_codecs" / _radio_slot / "gains"); + _tree->create("rx_codecs" / _radio_slot / "name").set("AD9371 Dual ADC"); + _tree->create("tx_codecs" / _radio_slot / "name").set("AD9371 Dual DAC"); + + // TODO remove this dirty hack + if (not _tree->exists("tick_rate")) + { + _tree->create("tick_rate").set(MAGNESIUM_TICK_RATE); + } +} + -- cgit v1.2.3