aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp')
-rw-r--r--host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp270
1 files changed, 266 insertions, 4 deletions
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 2c0f82aa8..4298df5b0 100644
--- a/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp
+++ b/host/lib/usrp/dboard/magnesium/magnesium_radio_ctrl_impl.cpp
@@ -32,7 +32,7 @@ namespace {
const double MAGNESIUM_TICK_RATE = 125e6; // Hz
const double MAGNESIUM_RADIO_RATE = 125e6; // Hz
- const double MAGNESIUM_MIN_FREQ = 300e6; // Hz
+ const double MAGNESIUM_MIN_FREQ = 1e6; // Hz
const double MAGNESIUM_MAX_FREQ = 6e9; // Hz
const double MAGNESIUM_MIN_RX_GAIN = 0.0; // dB
const double MAGNESIUM_MAX_RX_GAIN = 30.0; // dB
@@ -47,6 +47,49 @@ namespace {
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;
const size_t FPGPIO_MASTER_RADIO = 0;
@@ -58,6 +101,16 @@ namespace {
{
UHD_ASSERT_THROW(dir == RX_DIRECTION or dir == TX_DIRECTION);
UHD_ASSERT_THROW(chan == 0 or chan == 1);
+ 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 << " "<<chan)
return str(boost::format("%s%d")
% (dir == RX_DIRECTION ? "RX" : "TX")
% (chan+1)
@@ -224,7 +277,12 @@ void magnesium_radio_ctrl_impl::_init_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") {
@@ -301,6 +359,20 @@ void magnesium_radio_ctrl_impl::_init_peripherals()
} else {
UHD_LOG_TRACE("MAGNESIUM", "Not a master radio, no LOs.");
}
+ if (not _tree->exists(rx_lo_path)) {
+ _tree->create<adf435x_iface::sptr>(rx_lo_path).set(_rx_lo);
+ }else
+ {
+ UHD_LOG_TRACE("MAGNESIUM", "Not a master radio. Getting LO from master" );
+ _rx_lo = _tree->access<adf435x_iface::sptr>(rx_lo_path).get();
+ }
+ if (not _tree->exists(tx_lo_path)) {
+ _tree->create<adf435x_iface::sptr>(tx_lo_path).set(_tx_lo);
+ }else
+ {
+ UHD_LOG_TRACE("MAGNESIUM", "Not a master radio. Getting LO from master" );
+ _tx_lo = _tree->access<adf435x_iface::sptr>(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++) {
@@ -536,11 +608,172 @@ double magnesium_radio_ctrl_impl::_myk_set_frequency(
// 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, chan);
+ UHD_LOG_TRACE("MAGNESIUM", "requested frequency of " << freq);
+ double ad9371_freq = freq;
+ auto lo_iface = (dir == RX_DIRECTION) ? _rx_lo : _tx_lo;
+
+ if (freq < MAGNESIUM_LOWBAND_FREQ) { // Low band
+ 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 {
+ 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<double>(_rpc_prefix + "set_freq", which, ad9371_freq, false);
+ UHD_LOG_TRACE("MAGNESIUM",
+ _rpc_prefix << "set_freq returned " << retval);
+
+
+ // 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_SW1_RX2INPUT, /* FIXME hard coded */
+ 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_SW1_RX2INPUT, /* FIXME hard coded */
+ 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_SW1_RX2INPUT, /* FIXME hard coded */
+ 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_SW1_RX2INPUT, /* FIXME hard coded */
+ 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_SW1_RX2INPUT, /* FIXME hard coded */
+ 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_SW1_RX2INPUT, /* FIXME hard coded */
+ 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_SW1_RX2INPUT, /* FIXME hard coded */
+ 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) {
+ } else if (freq < MAGNESIUM_TX_BAND2_MIN_FREQ) {
+ } else if (freq < MAGNESIUM_TX_BAND3_MIN_FREQ) {
+ } else if (freq < MAGNESIUM_TX_BAND4_MIN_FREQ) {
+ } else {
+ }
+ }
+ 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 " << freq);
- auto retval = _rpcc->request_with_token<double>(_rpc_prefix + "set_freq", which, freq, false);
+ "Calling " << _rpc_prefix << "set_freq on " << which << " with " << ad9371_freq);
+ auto retval = _rpcc->request_with_token<double>(_rpc_prefix + "set_freq", which, ad9371_freq, false);
UHD_LOG_TRACE("MAGNESIUM",
_rpc_prefix << "set_freq returned " << retval);
+
+
return retval;
}
@@ -606,4 +839,33 @@ double magnesium_radio_ctrl_impl::_myk_get_bandwidth(const size_t chan, const di
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<uint16_t>(_slot_prefix + "cpld_peek", 0x12);
+ //UHD_LOG_TRACE("MAGNESIUM", "lock detect is " << lock_det);
+
+ return actual_freq;
+}
+
UHD_RFNOC_BLOCK_REGISTER(magnesium_radio_ctrl, "MagnesiumRadio");