aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib/usrp/dboard
diff options
context:
space:
mode:
Diffstat (limited to 'host/lib/usrp/dboard')
-rw-r--r--host/lib/usrp/dboard/CMakeLists.txt4
-rw-r--r--host/lib/usrp/dboard/db_basic_and_lf.cpp8
-rw-r--r--host/lib/usrp/dboard/db_dbsrx.cpp10
-rw-r--r--host/lib/usrp/dboard/db_dbsrx2.cpp10
-rw-r--r--host/lib/usrp/dboard/db_e3x0.cpp4
-rw-r--r--host/lib/usrp/dboard/db_rfx.cpp50
-rw-r--r--host/lib/usrp/dboard/db_sbx_common.cpp78
-rw-r--r--host/lib/usrp/dboard/db_sbx_common.hpp25
-rw-r--r--host/lib/usrp/dboard/db_sbx_version3.cpp111
-rw-r--r--host/lib/usrp/dboard/db_sbx_version4.cpp122
-rw-r--r--host/lib/usrp/dboard/db_tvrx.cpp6
-rw-r--r--host/lib/usrp/dboard/db_tvrx2.cpp16
-rw-r--r--host/lib/usrp/dboard/db_twinrx.cpp337
-rw-r--r--host/lib/usrp/dboard/db_ubx.cpp175
-rw-r--r--host/lib/usrp/dboard/db_wbx_common.cpp14
-rw-r--r--host/lib/usrp/dboard/db_wbx_common.hpp20
-rw-r--r--host/lib/usrp/dboard/db_wbx_simple.cpp42
-rw-r--r--host/lib/usrp/dboard/db_wbx_version2.cpp149
-rw-r--r--host/lib/usrp/dboard/db_wbx_version3.cpp153
-rw-r--r--host/lib/usrp/dboard/db_wbx_version4.cpp140
-rw-r--r--host/lib/usrp/dboard/db_xcvr2450.cpp46
-rw-r--r--host/lib/usrp/dboard/twinrx/table_to_cpp.py26
-rw-r--r--host/lib/usrp/dboard/twinrx/twinrx_ctrl.cpp643
-rw-r--r--host/lib/usrp/dboard/twinrx/twinrx_ctrl.hpp101
-rw-r--r--host/lib/usrp/dboard/twinrx/twinrx_experts.cpp637
-rw-r--r--host/lib/usrp/dboard/twinrx/twinrx_experts.hpp630
-rw-r--r--host/lib/usrp/dboard/twinrx/twinrx_gain_tables.cpp860
-rw-r--r--host/lib/usrp/dboard/twinrx/twinrx_gain_tables.hpp83
-rw-r--r--host/lib/usrp/dboard/twinrx/twinrx_io.hpp524
29 files changed, 4280 insertions, 744 deletions
diff --git a/host/lib/usrp/dboard/CMakeLists.txt b/host/lib/usrp/dboard/CMakeLists.txt
index 6cebecdbf..183496734 100644
--- a/host/lib/usrp/dboard/CMakeLists.txt
+++ b/host/lib/usrp/dboard/CMakeLists.txt
@@ -39,5 +39,9 @@ LIBUHD_APPEND_SOURCES(
${CMAKE_CURRENT_SOURCE_DIR}/db_dbsrx2.cpp
${CMAKE_CURRENT_SOURCE_DIR}/db_tvrx2.cpp
${CMAKE_CURRENT_SOURCE_DIR}/db_e3x0.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/twinrx/twinrx_ctrl.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/twinrx/twinrx_experts.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/twinrx/twinrx_gain_tables.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/db_twinrx.cpp
)
diff --git a/host/lib/usrp/dboard/db_basic_and_lf.cpp b/host/lib/usrp/dboard/db_basic_and_lf.cpp
index 2b30dab52..941a80ea4 100644
--- a/host/lib/usrp/dboard/db_basic_and_lf.cpp
+++ b/host/lib/usrp/dboard/db_basic_and_lf.cpp
@@ -50,7 +50,7 @@ static const uhd::dict<std::string, double> subdev_bandwidth_scalar = map_list_o
class basic_rx : public rx_dboard_base{
public:
basic_rx(ctor_args_t args, double max_freq);
- ~basic_rx(void);
+ virtual ~basic_rx(void);
private:
double _max_freq;
@@ -59,7 +59,7 @@ private:
class basic_tx : public tx_dboard_base{
public:
basic_tx(ctor_args_t args, double max_freq);
- ~basic_tx(void);
+ virtual ~basic_tx(void);
private:
double _max_freq;
@@ -121,7 +121,7 @@ basic_rx::basic_rx(ctor_args_t args, double max_freq) : rx_dboard_base(args){
this->get_rx_subtree()->create<int>("gains"); //phony property so this dir exists
this->get_rx_subtree()->create<double>("freq/value")
- .publish(&always_zero_freq);
+ .set_publisher(&always_zero_freq);
this->get_rx_subtree()->create<meta_range_t>("freq/range")
.set(freq_range_t(-_max_freq, +_max_freq));
this->get_rx_subtree()->create<std::string>("antenna/value")
@@ -176,7 +176,7 @@ basic_tx::basic_tx(ctor_args_t args, double max_freq) : tx_dboard_base(args){
this->get_tx_subtree()->create<int>("gains"); //phony property so this dir exists
this->get_tx_subtree()->create<double>("freq/value")
- .publish(&always_zero_freq);
+ .set_publisher(&always_zero_freq);
this->get_tx_subtree()->create<meta_range_t>("freq/range")
.set(freq_range_t(-_max_freq, +_max_freq));
this->get_tx_subtree()->create<std::string>("antenna/value")
diff --git a/host/lib/usrp/dboard/db_dbsrx.cpp b/host/lib/usrp/dboard/db_dbsrx.cpp
index 9d04d8e16..6e1846fb8 100644
--- a/host/lib/usrp/dboard/db_dbsrx.cpp
+++ b/host/lib/usrp/dboard/db_dbsrx.cpp
@@ -66,7 +66,7 @@ static const double usrp1_gpio_clock_rate_limit = 4e6;
class dbsrx : public rx_dboard_base{
public:
dbsrx(ctor_args_t args);
- ~dbsrx(void);
+ virtual ~dbsrx(void);
private:
double _lo_freq;
@@ -204,16 +204,16 @@ dbsrx::dbsrx(ctor_args_t args) : rx_dboard_base(args){
this->get_rx_subtree()->create<std::string>("name")
.set("DBSRX");
this->get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&dbsrx::get_locked, this));
+ .set_publisher(boost::bind(&dbsrx::get_locked, this));
BOOST_FOREACH(const std::string &name, dbsrx_gain_ranges.keys()){
this->get_rx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&dbsrx::set_gain, this, _1, name))
+ .set_coercer(boost::bind(&dbsrx::set_gain, this, _1, name))
.set(dbsrx_gain_ranges[name].start());
this->get_rx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(dbsrx_gain_ranges[name]);
}
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&dbsrx::set_lo_freq, this, _1));
+ .set_coercer(boost::bind(&dbsrx::set_lo_freq, this, _1));
this->get_rx_subtree()->create<meta_range_t>("freq/range")
.set(dbsrx_freq_range);
this->get_rx_subtree()->create<std::string>("antenna/value")
@@ -227,7 +227,7 @@ dbsrx::dbsrx(ctor_args_t args) : rx_dboard_base(args){
this->get_rx_subtree()->create<bool>("use_lo_offset")
.set(false);
this->get_rx_subtree()->create<double>("bandwidth/value")
- .coerce(boost::bind(&dbsrx::set_bandwidth, this, _1));
+ .set_coercer(boost::bind(&dbsrx::set_bandwidth, this, _1));
this->get_rx_subtree()->create<meta_range_t>("bandwidth/range")
.set(dbsrx_bandwidth_range);
diff --git a/host/lib/usrp/dboard/db_dbsrx2.cpp b/host/lib/usrp/dboard/db_dbsrx2.cpp
index 1debe3c8f..11d706ed6 100644
--- a/host/lib/usrp/dboard/db_dbsrx2.cpp
+++ b/host/lib/usrp/dboard/db_dbsrx2.cpp
@@ -60,7 +60,7 @@ static const uhd::dict<std::string, gain_range_t> dbsrx2_gain_ranges = map_list_
class dbsrx2 : public rx_dboard_base{
public:
dbsrx2(ctor_args_t args);
- ~dbsrx2(void);
+ virtual ~dbsrx2(void);
private:
double _lo_freq;
@@ -191,16 +191,16 @@ dbsrx2::dbsrx2(ctor_args_t args) : rx_dboard_base(args){
this->get_rx_subtree()->create<std::string>("name")
.set("DBSRX2");
this->get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&dbsrx2::get_locked, this));
+ .set_publisher(boost::bind(&dbsrx2::get_locked, this));
BOOST_FOREACH(const std::string &name, dbsrx2_gain_ranges.keys()){
this->get_rx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&dbsrx2::set_gain, this, _1, name))
+ .set_coercer(boost::bind(&dbsrx2::set_gain, this, _1, name))
.set(dbsrx2_gain_ranges[name].start());
this->get_rx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(dbsrx2_gain_ranges[name]);
}
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&dbsrx2::set_lo_freq, this, _1))
+ .set_coercer(boost::bind(&dbsrx2::set_lo_freq, this, _1))
.set(dbsrx2_freq_range.start());
this->get_rx_subtree()->create<meta_range_t>("freq/range")
.set(dbsrx2_freq_range);
@@ -218,7 +218,7 @@ dbsrx2::dbsrx2(ctor_args_t args) : rx_dboard_base(args){
double codec_rate = this->get_iface()->get_codec_rate(dboard_iface::UNIT_RX);
this->get_rx_subtree()->create<double>("bandwidth/value")
- .coerce(boost::bind(&dbsrx2::set_bandwidth, this, _1))
+ .set_coercer(boost::bind(&dbsrx2::set_bandwidth, this, _1))
.set(2.0*(0.8*codec_rate/2.0)); //bandwidth in lowpass, convert to complex bandpass
//default to anti-alias at different codec_rate
this->get_rx_subtree()->create<meta_range_t>("bandwidth/range")
diff --git a/host/lib/usrp/dboard/db_e3x0.cpp b/host/lib/usrp/dboard/db_e3x0.cpp
index 523927d49..c7cc52d73 100644
--- a/host/lib/usrp/dboard/db_e3x0.cpp
+++ b/host/lib/usrp/dboard/db_e3x0.cpp
@@ -29,7 +29,7 @@ class e310_dboard : public xcvr_dboard_base{
public:
e310_dboard(ctor_args_t args) : xcvr_dboard_base(args) {}
- ~e310_dboard(void) {}
+ virtual ~e310_dboard(void) {}
};
/***********************************************************************
@@ -40,7 +40,7 @@ class e300_dboard : public xcvr_dboard_base{
public:
e300_dboard(ctor_args_t args) : xcvr_dboard_base(args) {}
- ~e300_dboard(void) {}
+ virtual ~e300_dboard(void) {}
};
/***********************************************************************
diff --git a/host/lib/usrp/dboard/db_rfx.cpp b/host/lib/usrp/dboard/db_rfx.cpp
index 1342c913d..dbb1600ec 100644
--- a/host/lib/usrp/dboard/db_rfx.cpp
+++ b/host/lib/usrp/dboard/db_rfx.cpp
@@ -78,7 +78,7 @@ public:
const freq_range_t &freq_range,
bool rx_div2, bool tx_div2
);
- ~rfx_xcvr(void);
+ virtual ~rfx_xcvr(void);
private:
const freq_range_t _freq_range;
@@ -183,20 +183,20 @@ rfx_xcvr::rfx_xcvr(
else this->get_rx_subtree()->create<std::string>("name").set("RFX RX");
this->get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&rfx_xcvr::get_locked, this, dboard_iface::UNIT_RX));
+ .set_publisher(boost::bind(&rfx_xcvr::get_locked, this, dboard_iface::UNIT_RX));
BOOST_FOREACH(const std::string &name, _rx_gain_ranges.keys()){
this->get_rx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&rfx_xcvr::set_rx_gain, this, _1, name))
+ .set_coercer(boost::bind(&rfx_xcvr::set_rx_gain, this, _1, name))
.set(_rx_gain_ranges[name].start());
this->get_rx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(_rx_gain_ranges[name]);
}
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&rfx_xcvr::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
+ .set_coercer(boost::bind(&rfx_xcvr::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
.set((_freq_range.start() + _freq_range.stop())/2.0);
this->get_rx_subtree()->create<meta_range_t>("freq/range").set(_freq_range);
this->get_rx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&rfx_xcvr::set_rx_ant, this, _1))
+ .add_coerced_subscriber(boost::bind(&rfx_xcvr::set_rx_ant, this, _1))
.set("RX2");
this->get_rx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(rfx_rx_antennas);
@@ -219,14 +219,14 @@ rfx_xcvr::rfx_xcvr(
else this->get_tx_subtree()->create<std::string>("name").set("RFX TX");
this->get_tx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&rfx_xcvr::get_locked, this, dboard_iface::UNIT_TX));
+ .set_publisher(boost::bind(&rfx_xcvr::get_locked, this, dboard_iface::UNIT_TX));
this->get_tx_subtree()->create<int>("gains"); //phony property so this dir exists
this->get_tx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&rfx_xcvr::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
+ .set_coercer(boost::bind(&rfx_xcvr::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
.set((_freq_range.start() + _freq_range.stop())/2.0);
this->get_tx_subtree()->create<meta_range_t>("freq/range").set(_freq_range);
this->get_tx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&rfx_xcvr::set_tx_ant, this, _1)).set(rfx_tx_antennas.at(0));
+ .add_coerced_subscriber(boost::bind(&rfx_xcvr::set_tx_ant, this, _1)).set(rfx_tx_antennas.at(0));
this->get_tx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(rfx_tx_antennas);
this->get_tx_subtree()->create<std::string>("connection").set("IQ");
@@ -248,15 +248,15 @@ rfx_xcvr::rfx_xcvr(
this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_RX, output_enables);
//setup the tx atr (this does not change with antenna)
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_IDLE, _power_up | ANT_XX | MIXER_DIS);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_RX_ONLY, _power_up | ANT_RX | MIXER_DIS);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, _power_up | ANT_TX | MIXER_ENB);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, _power_up | ANT_TX | MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_IDLE, _power_up | ANT_XX | MIXER_DIS);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_RX_ONLY, _power_up | ANT_RX | MIXER_DIS);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, _power_up | ANT_TX | MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, _power_up | ANT_TX | MIXER_ENB);
//setup the rx atr (this does not change with antenna)
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_IDLE, _power_up | ANT_XX | MIXER_DIS);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, _power_up | ANT_XX | MIXER_DIS);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, _power_up | ANT_RX2| MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_IDLE, _power_up | ANT_XX | MIXER_DIS);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_TX_ONLY, _power_up | ANT_XX | MIXER_DIS);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_FULL_DUPLEX, _power_up | ANT_RX2| MIXER_ENB);
}
rfx_xcvr::~rfx_xcvr(void){
@@ -272,14 +272,14 @@ void rfx_xcvr::set_rx_ant(const std::string &ant){
//set the rx atr regs that change with antenna setting
if (ant == "CAL") {
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, _power_up | ANT_TXRX | MIXER_ENB);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, _power_up | ANT_TXRX | MIXER_ENB);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_RX_ONLY, _power_up | MIXER_ENB | ANT_TXRX );
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_TX_ONLY, _power_up | ANT_TXRX | MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_FULL_DUPLEX, _power_up | ANT_TXRX | MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_RX_ONLY, _power_up | MIXER_ENB | ANT_TXRX );
}
else {
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, _power_up | ANT_XX | MIXER_DIS);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, _power_up | ANT_RX2| MIXER_ENB);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_RX_ONLY, _power_up | MIXER_ENB |
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_TX_ONLY, _power_up | ANT_XX | MIXER_DIS);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_FULL_DUPLEX, _power_up | ANT_RX2| MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_RX_ONLY, _power_up | MIXER_ENB |
((ant == "TX/RX")? ANT_TXRX : ANT_RX2));
}
@@ -292,12 +292,12 @@ void rfx_xcvr::set_tx_ant(const std::string &ant){
//set the tx atr regs that change with antenna setting
if (ant == "CAL") {
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, _power_up | ANT_RX | MIXER_ENB);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, _power_up | ANT_RX | MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, _power_up | ANT_RX | MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, _power_up | ANT_RX | MIXER_ENB);
}
else {
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, _power_up | ANT_TX | MIXER_ENB);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, _power_up | ANT_TX | MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, _power_up | ANT_TX | MIXER_ENB);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, _power_up | ANT_TX | MIXER_ENB);
}
}
diff --git a/host/lib/usrp/dboard/db_sbx_common.cpp b/host/lib/usrp/dboard/db_sbx_common.cpp
index ce5166c4c..be02cf77a 100644
--- a/host/lib/usrp/dboard/db_sbx_common.cpp
+++ b/host/lib/usrp/dboard/db_sbx_common.cpp
@@ -159,20 +159,20 @@ sbx_xcvr::sbx_xcvr(ctor_args_t args) : xcvr_dboard_base(args){
else this->get_rx_subtree()->create<std::string>("name").set("SBX/CBX RX");
this->get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&sbx_xcvr::get_locked, this, dboard_iface::UNIT_RX));
+ .set_publisher(boost::bind(&sbx_xcvr::get_locked, this, dboard_iface::UNIT_RX));
BOOST_FOREACH(const std::string &name, sbx_rx_gain_ranges.keys()){
this->get_rx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&sbx_xcvr::set_rx_gain, this, _1, name))
+ .set_coercer(boost::bind(&sbx_xcvr::set_rx_gain, this, _1, name))
.set(sbx_rx_gain_ranges[name].start());
this->get_rx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(sbx_rx_gain_ranges[name]);
}
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&sbx_xcvr::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
+ .set_coercer(boost::bind(&sbx_xcvr::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
.set((freq_range.start() + freq_range.stop())/2.0);
this->get_rx_subtree()->create<meta_range_t>("freq/range").set(freq_range);
this->get_rx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&sbx_xcvr::set_rx_ant, this, _1))
+ .add_coerced_subscriber(boost::bind(&sbx_xcvr::set_rx_ant, this, _1))
.set("RX2");
this->get_rx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(sbx_rx_antennas);
@@ -200,20 +200,20 @@ sbx_xcvr::sbx_xcvr(ctor_args_t args) : xcvr_dboard_base(args){
else this->get_tx_subtree()->create<std::string>("name").set("SBX/CBX TX");
this->get_tx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&sbx_xcvr::get_locked, this, dboard_iface::UNIT_TX));
+ .set_publisher(boost::bind(&sbx_xcvr::get_locked, this, dboard_iface::UNIT_TX));
BOOST_FOREACH(const std::string &name, sbx_tx_gain_ranges.keys()){
this->get_tx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&sbx_xcvr::set_tx_gain, this, _1, name))
+ .set_coercer(boost::bind(&sbx_xcvr::set_tx_gain, this, _1, name))
.set(sbx_tx_gain_ranges[name].start());
this->get_tx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(sbx_tx_gain_ranges[name]);
}
this->get_tx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&sbx_xcvr::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
+ .set_coercer(boost::bind(&sbx_xcvr::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
.set((freq_range.start() + freq_range.stop())/2.0);
this->get_tx_subtree()->create<meta_range_t>("freq/range").set(freq_range);
this->get_tx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&sbx_xcvr::set_tx_ant, this, _1))
+ .add_coerced_subscriber(boost::bind(&sbx_xcvr::set_tx_ant, this, _1))
.set(sbx_tx_antennas.at(0));
this->get_tx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(sbx_tx_antennas);
@@ -237,8 +237,8 @@ sbx_xcvr::sbx_xcvr(ctor_args_t args) : xcvr_dboard_base(args){
this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_TX, (TXIO_MASK|TX_LED_IO));
this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_RX, (RXIO_MASK|RX_LED_IO));
- //flash LEDs
- flash_leds();
+ //Initialize ATR registers after direction and pin ctrl configuration
+ update_atr();
UHD_LOGV(often) << boost::format(
"SBX GPIO Direction: RX: 0x%08x, TX: 0x%08x"
@@ -265,39 +265,39 @@ void sbx_xcvr::update_atr(void){
//setup the tx atr (this does not change with antenna)
this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_IDLE, 0 | tx_lo_lpf_en \
+ gpio_atr::ATR_REG_IDLE, 0 | tx_lo_lpf_en \
| tx_ld_led | tx_ant_led | TX_POWER_UP | ANT_XX | TX_MIXER_DIS);
//setup the rx atr (this does not change with antenna)
this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_IDLE, rx_pga0_iobits | rx_lo_lpf_en \
+ gpio_atr::ATR_REG_IDLE, rx_pga0_iobits | rx_lo_lpf_en \
| rx_ld_led | rx_ant_led | RX_POWER_UP | ANT_XX | RX_MIXER_DIS);
//set the RX atr regs that change with antenna setting
this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_RX_ONLY, rx_pga0_iobits | rx_lo_lpf_en \
+ gpio_atr::ATR_REG_RX_ONLY, rx_pga0_iobits | rx_lo_lpf_en \
| rx_ld_led | rx_ant_led | RX_POWER_UP | RX_MIXER_ENB \
| ((_rx_ant != "RX2")? ANT_TXRX : ANT_RX2));
this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_TX_ONLY, rx_pga0_iobits | rx_lo_lpf_en \
+ gpio_atr::ATR_REG_TX_ONLY, rx_pga0_iobits | rx_lo_lpf_en \
| rx_ld_led | rx_ant_led | RX_POWER_UP | RX_MIXER_DIS \
| ((_rx_ant == "CAL")? ANT_TXRX : ANT_RX2));
this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_FULL_DUPLEX, rx_pga0_iobits | rx_lo_lpf_en \
+ gpio_atr::ATR_REG_FULL_DUPLEX, rx_pga0_iobits | rx_lo_lpf_en \
| rx_ld_led | rx_ant_led | RX_POWER_UP | RX_MIXER_ENB \
| ((_rx_ant == "CAL")? ANT_TXRX : ANT_RX2));
//set the TX atr regs that change with antenna setting
this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_RX_ONLY, 0 | tx_lo_lpf_en \
+ gpio_atr::ATR_REG_RX_ONLY, 0 | tx_lo_lpf_en \
| tx_ld_led | tx_ant_led | TX_POWER_UP | TX_MIXER_DIS \
| ((_rx_ant != "RX2")? ANT_RX : ANT_TX));
this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_TX_ONLY, tx_pga0_iobits | tx_lo_lpf_en \
+ gpio_atr::ATR_REG_TX_ONLY, tx_pga0_iobits | tx_lo_lpf_en \
| tx_ld_led | tx_ant_led | TX_POWER_UP | TX_MIXER_ENB \
| ((_tx_ant == "CAL")? ANT_RX : ANT_TX));
this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_FULL_DUPLEX, tx_pga0_iobits | tx_lo_lpf_en \
+ gpio_atr::ATR_REG_FULL_DUPLEX, tx_pga0_iobits | tx_lo_lpf_en \
| tx_ld_led | tx_ant_led | TX_POWER_UP | TX_MIXER_ENB \
| ((_tx_ant == "CAL")? ANT_RX : ANT_TX));
}
@@ -352,45 +352,3 @@ sensor_value_t sbx_xcvr::get_locked(dboard_iface::unit_t unit) {
return sensor_value_t("LO", locked, "locked", "unlocked");
}
-
-
-void sbx_xcvr::flash_leds(void) {
- //Remove LED gpios from ATR control temporarily and set to outputs
- this->get_iface()->set_pin_ctrl(dboard_iface::UNIT_TX, TXIO_MASK);
- this->get_iface()->set_pin_ctrl(dboard_iface::UNIT_RX, RXIO_MASK);
- this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_TX, (TXIO_MASK|RX_LED_IO));
- this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_RX, (RXIO_MASK|RX_LED_IO));
-
- this->get_iface()->set_gpio_out(dboard_iface::UNIT_TX, TX_LED_LD, TX_LED_IO);
- boost::this_thread::sleep(boost::posix_time::milliseconds(100));
-
- this->get_iface()->set_gpio_out(dboard_iface::UNIT_TX, \
- TX_LED_TXRX|TX_LED_LD, TX_LED_IO);
- boost::this_thread::sleep(boost::posix_time::milliseconds(100));
-
- this->get_iface()->set_gpio_out(dboard_iface::UNIT_RX, RX_LED_LD, RX_LED_IO);
- boost::this_thread::sleep(boost::posix_time::milliseconds(100));
-
- this->get_iface()->set_gpio_out(dboard_iface::UNIT_RX, \
- RX_LED_RX1RX2|RX_LED_LD, RX_LED_IO);
- boost::this_thread::sleep(boost::posix_time::milliseconds(100));
-
- this->get_iface()->set_gpio_out(dboard_iface::UNIT_RX, RX_LED_LD, RX_LED_IO);
- boost::this_thread::sleep(boost::posix_time::milliseconds(100));
-
- this->get_iface()->set_gpio_out(dboard_iface::UNIT_RX, 0, RX_LED_IO);
- boost::this_thread::sleep(boost::posix_time::milliseconds(100));
-
- this->get_iface()->set_gpio_out(dboard_iface::UNIT_TX, TX_LED_LD, TX_LED_IO);
- boost::this_thread::sleep(boost::posix_time::milliseconds(100));
-
- this->get_iface()->set_gpio_out(dboard_iface::UNIT_TX, 0, TX_LED_IO);
- boost::this_thread::sleep(boost::posix_time::milliseconds(100));
-
- //Put LED gpios back in ATR control and update atr
- this->get_iface()->set_pin_ctrl(dboard_iface::UNIT_TX, (TXIO_MASK|TX_LED_IO));
- this->get_iface()->set_pin_ctrl(dboard_iface::UNIT_RX, (RXIO_MASK|RX_LED_IO));
- this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_TX, (TXIO_MASK|TX_LED_IO));
- this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_RX, (RXIO_MASK|RX_LED_IO));
-}
-
diff --git a/host/lib/usrp/dboard/db_sbx_common.hpp b/host/lib/usrp/dboard/db_sbx_common.hpp
index 4800bbd83..c0e29f263 100644
--- a/host/lib/usrp/dboard/db_sbx_common.hpp
+++ b/host/lib/usrp/dboard/db_sbx_common.hpp
@@ -16,10 +16,15 @@
//
#include <uhd/types/device_addr.hpp>
-
-#include "adf435x_common.hpp"
+#include "adf435x.hpp"
#include "max287x.hpp"
+// LO Related
+#define ADF435X_CE (1 << 3)
+#define ADF435X_PDBRF (1 << 2)
+#define ADF435X_MUXOUT (1 << 1) // INPUT!!!
+#define LOCKDET_MASK (1 << 0) // INPUT!!!
+
// Common IO Pins
#define LO_LPF_EN (1 << 15)
@@ -36,6 +41,8 @@
#define RX_LED_LD (1 << 6) // LED for RX Lock Detect
#define DIS_POWER_RX (1 << 5) // on UNIT_RX, 0 powers up RX
#define RX_DISABLE (1 << 4) // on UNIT_RX, 1 disables RX Mixer and Baseband
+#define RX_ATTN_SHIFT 8 //lsb of RX Attenuator Control
+#define RX_ATTN_MASK (63 << RX_ATTN_SHIFT) //valid bits of RX Attenuator Control
// TX Attenuator Pins
#define TX_ATTN_SHIFT 8 // lsb of TX Attenuator Control
@@ -184,12 +191,16 @@ protected:
class sbx_version3 : public sbx_versionx {
public:
sbx_version3(sbx_xcvr *_self_sbx_xcvr);
- ~sbx_version3(void);
+ virtual ~sbx_version3(void);
double set_lo_freq(dboard_iface::unit_t unit, double target_freq);
/*! This is the registered instance of the wrapper class, sbx_base. */
sbx_xcvr *self_base;
+ private:
+ adf435x_iface::sptr _txlo;
+ adf435x_iface::sptr _rxlo;
+ void write_lo_regs(dboard_iface::unit_t unit, const std::vector<boost::uint32_t> &regs);
};
/*!
@@ -200,12 +211,16 @@ protected:
class sbx_version4 : public sbx_versionx {
public:
sbx_version4(sbx_xcvr *_self_sbx_xcvr);
- ~sbx_version4(void);
+ virtual ~sbx_version4(void);
double set_lo_freq(dboard_iface::unit_t unit, double target_freq);
/*! This is the registered instance of the wrapper class, sbx_base. */
sbx_xcvr *self_base;
+ private:
+ adf435x_iface::sptr _txlo;
+ adf435x_iface::sptr _rxlo;
+ void write_lo_regs(dboard_iface::unit_t unit, const std::vector<boost::uint32_t> &regs);
};
/*!
@@ -218,7 +233,7 @@ protected:
class cbx : public sbx_versionx {
public:
cbx(sbx_xcvr *_self_sbx_xcvr);
- ~cbx(void);
+ virtual ~cbx(void);
double set_lo_freq(dboard_iface::unit_t unit, double target_freq);
diff --git a/host/lib/usrp/dboard/db_sbx_version3.cpp b/host/lib/usrp/dboard/db_sbx_version3.cpp
index b848097d1..76ad7b04f 100644
--- a/host/lib/usrp/dboard/db_sbx_version3.cpp
+++ b/host/lib/usrp/dboard/db_sbx_version3.cpp
@@ -16,9 +16,7 @@
//
-#include "adf4350_regs.hpp"
#include "db_sbx_common.hpp"
-#include "../common/adf435x_common.hpp"
#include <uhd/types/tune_request.hpp>
#include <boost/algorithm/string.hpp>
@@ -32,12 +30,21 @@ using namespace boost::assign;
sbx_xcvr::sbx_version3::sbx_version3(sbx_xcvr *_self_sbx_xcvr) {
//register the handle to our base SBX class
self_base = _self_sbx_xcvr;
+ _txlo = adf435x_iface::make_adf4350(boost::bind(&sbx_xcvr::sbx_version3::write_lo_regs, this, dboard_iface::UNIT_TX, _1));
+ _rxlo = adf435x_iface::make_adf4350(boost::bind(&sbx_xcvr::sbx_version3::write_lo_regs, this, dboard_iface::UNIT_RX, _1));
}
sbx_xcvr::sbx_version3::~sbx_version3(void){
/* NOP */
}
+void sbx_xcvr::sbx_version3::write_lo_regs(dboard_iface::unit_t unit, const std::vector<boost::uint32_t> &regs)
+{
+ BOOST_FOREACH(boost::uint32_t reg, regs)
+ {
+ self_base->get_iface()->write_spi(unit, spi_config_t::EDGE_RISE, reg, 32);
+ }
+}
/***********************************************************************
* Tuning
@@ -57,95 +64,27 @@ double sbx_xcvr::sbx_version3::set_lo_freq(dboard_iface::unit_t unit, double tar
device_addr_t tune_args = subtree->access<device_addr_t>("tune_args").get();
bool is_int_n = boost::iequals(tune_args.get("mode_n",""), "integer");
- //clip the input
- target_freq = sbx_freq_range.clip(target_freq);
-
- //map prescaler setting to mininmum integer divider (N) values (pg.18 prescaler)
- static const uhd::dict<int, int> prescaler_to_min_int_div = map_list_of
- (0,23) //adf4350_regs_t::PRESCALER_4_5
- (1,75) //adf4350_regs_t::PRESCALER_8_9
- ;
-
- //map rf divider select output dividers to enums
- static const uhd::dict<int, adf4350_regs_t::rf_divider_select_t> rfdivsel_to_enum = map_list_of
- (1, adf4350_regs_t::RF_DIVIDER_SELECT_DIV1)
- (2, adf4350_regs_t::RF_DIVIDER_SELECT_DIV2)
- (4, adf4350_regs_t::RF_DIVIDER_SELECT_DIV4)
- (8, adf4350_regs_t::RF_DIVIDER_SELECT_DIV8)
- (16, adf4350_regs_t::RF_DIVIDER_SELECT_DIV16)
- ;
-
- //use 8/9 prescaler for vco_freq > 3 GHz (pg.18 prescaler)
- adf4350_regs_t::prescaler_t prescaler = target_freq > 3e9 ? adf4350_regs_t::PRESCALER_8_9 : adf4350_regs_t::PRESCALER_4_5;
-
- adf435x_tuning_constraints tuning_constraints;
- tuning_constraints.force_frac0 = is_int_n;
- tuning_constraints.band_sel_freq_max = 100e3;
- tuning_constraints.ref_doubler_threshold = 12.5e6;
- tuning_constraints.int_range = uhd::range_t(prescaler_to_min_int_div[prescaler], 4095); //INT is a 12-bit field
- tuning_constraints.pfd_freq_max = 25e6;
- tuning_constraints.rf_divider_range = uhd::range_t(1, 16);
- tuning_constraints.feedback_after_divider = true;
+ //Select the LO
+ adf435x_iface::sptr& lo_iface = unit == dboard_iface::UNIT_RX ? _rxlo : _txlo;
+ lo_iface->set_feedback_select(adf435x_iface::FB_SEL_DIVIDED);
+ lo_iface->set_reference_freq(self_base->get_iface()->get_clock_rate(unit));
- double actual_freq;
- adf435x_tuning_settings tuning_settings = tune_adf435x_synth(
- target_freq, self_base->get_iface()->get_clock_rate(unit),
- tuning_constraints, actual_freq);
+ //Use 8/9 prescaler for vco_freq > 3 GHz (pg.18 prescaler)
+ lo_iface->set_prescaler(target_freq > 3e9 ? adf435x_iface::PRESCALER_8_9 : adf435x_iface::PRESCALER_4_5);
- //load the register values
- adf4350_regs_t regs;
+ //Configure the LO
+ double actual_freq = 0.0;
+ actual_freq = lo_iface->set_frequency(sbx_freq_range.clip(target_freq), is_int_n);
- if ((unit == dboard_iface::UNIT_TX) and (actual_freq == sbx_tx_lo_2dbm.clip(actual_freq)))
- regs.output_power = adf4350_regs_t::OUTPUT_POWER_2DBM;
- else
- regs.output_power = adf4350_regs_t::OUTPUT_POWER_5DBM;
-
- regs.frac_12_bit = tuning_settings.frac_12_bit;
- regs.int_16_bit = tuning_settings.int_16_bit;
- regs.mod_12_bit = tuning_settings.mod_12_bit;
- regs.clock_divider_12_bit = tuning_settings.clock_divider_12_bit;
- regs.feedback_select = tuning_constraints.feedback_after_divider ?
- adf4350_regs_t::FEEDBACK_SELECT_DIVIDED :
- adf4350_regs_t::FEEDBACK_SELECT_FUNDAMENTAL;
- regs.clock_div_mode = adf4350_regs_t::CLOCK_DIV_MODE_RESYNC_ENABLE;
- regs.prescaler = prescaler;
- regs.r_counter_10_bit = tuning_settings.r_counter_10_bit;
- regs.reference_divide_by_2 = tuning_settings.r_divide_by_2_en ?
- adf4350_regs_t::REFERENCE_DIVIDE_BY_2_ENABLED :
- adf4350_regs_t::REFERENCE_DIVIDE_BY_2_DISABLED;
- regs.reference_doubler = tuning_settings.r_doubler_en ?
- adf4350_regs_t::REFERENCE_DOUBLER_ENABLED :
- adf4350_regs_t::REFERENCE_DOUBLER_DISABLED;
- regs.band_select_clock_div = tuning_settings.band_select_clock_div;
- UHD_ASSERT_THROW(rfdivsel_to_enum.has_key(tuning_settings.rf_divider));
- regs.rf_divider_select = rfdivsel_to_enum[tuning_settings.rf_divider];
- regs.ldf = is_int_n ?
- adf4350_regs_t::LDF_INT_N :
- adf4350_regs_t::LDF_FRAC_N;
-
- //reset the N and R counter
- regs.counter_reset = adf4350_regs_t::COUNTER_RESET_ENABLED;
- self_base->get_iface()->write_spi(unit, spi_config_t::EDGE_RISE, regs.get_reg(2), 32);
- regs.counter_reset = adf4350_regs_t::COUNTER_RESET_DISABLED;
-
- //write the registers
- //correct power-up sequence to write registers (5, 4, 3, 2, 1, 0)
- int addr;
-
- for(addr=5; addr>=0; addr--){
- UHD_LOGV(often) << boost::format(
- "SBX SPI Reg (0x%02x): 0x%08x"
- ) % addr % regs.get_reg(addr) << std::endl;
- self_base->get_iface()->write_spi(
- unit, spi_config_t::EDGE_RISE,
- regs.get_reg(addr), 32
- );
+ if ((unit == dboard_iface::UNIT_TX) and (actual_freq == sbx_tx_lo_2dbm.clip(actual_freq))) {
+ lo_iface->set_output_power(adf435x_iface::OUTPUT_POWER_2DBM);
+ } else {
+ lo_iface->set_output_power(adf435x_iface::OUTPUT_POWER_5DBM);
}
- //return the actual frequency
- UHD_LOGV(often) << boost::format(
- "SBX tune: actual frequency %f MHz"
- ) % (actual_freq/1e6) << std::endl;
+ //Write to hardware
+ lo_iface->commit();
+
return actual_freq;
}
diff --git a/host/lib/usrp/dboard/db_sbx_version4.cpp b/host/lib/usrp/dboard/db_sbx_version4.cpp
index 8f7e747bc..639bce250 100644
--- a/host/lib/usrp/dboard/db_sbx_version4.cpp
+++ b/host/lib/usrp/dboard/db_sbx_version4.cpp
@@ -16,9 +16,7 @@
//
-#include "adf4351_regs.hpp"
#include "db_sbx_common.hpp"
-#include "../common/adf435x_common.hpp"
#include <uhd/types/tune_request.hpp>
#include <boost/algorithm/string.hpp>
@@ -32,6 +30,8 @@ using namespace boost::assign;
sbx_xcvr::sbx_version4::sbx_version4(sbx_xcvr *_self_sbx_xcvr) {
//register the handle to our base SBX class
self_base = _self_sbx_xcvr;
+ _txlo = adf435x_iface::make_adf4351(boost::bind(&sbx_xcvr::sbx_version4::write_lo_regs, this, dboard_iface::UNIT_TX, _1));
+ _rxlo = adf435x_iface::make_adf4351(boost::bind(&sbx_xcvr::sbx_version4::write_lo_regs, this, dboard_iface::UNIT_RX, _1));
}
@@ -39,6 +39,14 @@ sbx_xcvr::sbx_version4::~sbx_version4(void){
/* NOP */
}
+void sbx_xcvr::sbx_version4::write_lo_regs(dboard_iface::unit_t unit, const std::vector<boost::uint32_t> &regs)
+{
+ BOOST_FOREACH(boost::uint32_t reg, regs)
+ {
+ self_base->get_iface()->write_spi(unit, spi_config_t::EDGE_RISE, reg, 32);
+ }
+}
+
/***********************************************************************
* Tuning
@@ -58,99 +66,27 @@ double sbx_xcvr::sbx_version4::set_lo_freq(dboard_iface::unit_t unit, double tar
device_addr_t tune_args = subtree->access<device_addr_t>("tune_args").get();
bool is_int_n = boost::iequals(tune_args.get("mode_n",""), "integer");
- //clip the input
- target_freq = sbx_freq_range.clip(target_freq);
-
- //map prescaler setting to mininmum integer divider (N) values (pg.18 prescaler)
- static const uhd::dict<int, int> prescaler_to_min_int_div = map_list_of
- (0,23) //adf4351_regs_t::PRESCALER_4_5
- (1,75) //adf4351_regs_t::PRESCALER_8_9
- ;
-
- //map rf divider select output dividers to enums
- static const uhd::dict<int, adf4351_regs_t::rf_divider_select_t> rfdivsel_to_enum = map_list_of
- (1, adf4351_regs_t::RF_DIVIDER_SELECT_DIV1)
- (2, adf4351_regs_t::RF_DIVIDER_SELECT_DIV2)
- (4, adf4351_regs_t::RF_DIVIDER_SELECT_DIV4)
- (8, adf4351_regs_t::RF_DIVIDER_SELECT_DIV8)
- (16, adf4351_regs_t::RF_DIVIDER_SELECT_DIV16)
- (32, adf4351_regs_t::RF_DIVIDER_SELECT_DIV32)
- (64, adf4351_regs_t::RF_DIVIDER_SELECT_DIV64)
- ;
-
- //use 8/9 prescaler for vco_freq > 3 GHz (pg.18 prescaler)
- adf4351_regs_t::prescaler_t prescaler = target_freq > 3.6e9 ? adf4351_regs_t::PRESCALER_8_9 : adf4351_regs_t::PRESCALER_4_5;
-
- adf435x_tuning_constraints tuning_constraints;
- tuning_constraints.force_frac0 = is_int_n;
- tuning_constraints.band_sel_freq_max = 100e3;
- tuning_constraints.ref_doubler_threshold = 12.5e6;
- tuning_constraints.int_range = uhd::range_t(prescaler_to_min_int_div[prescaler], 4095); //INT is a 12-bit field
- tuning_constraints.pfd_freq_max = 25e6;
- tuning_constraints.rf_divider_range = uhd::range_t(1, 64);
- tuning_constraints.feedback_after_divider = true;
-
- double actual_freq;
- adf435x_tuning_settings tuning_settings = tune_adf435x_synth(
- target_freq, self_base->get_iface()->get_clock_rate(unit),
- tuning_constraints, actual_freq);
-
- //load the register values
- adf4351_regs_t regs;
-
- if ((unit == dboard_iface::UNIT_TX) and (actual_freq == sbx_tx_lo_2dbm.clip(actual_freq)))
- regs.output_power = adf4351_regs_t::OUTPUT_POWER_2DBM;
- else
- regs.output_power = adf4351_regs_t::OUTPUT_POWER_5DBM;
-
- regs.frac_12_bit = tuning_settings.frac_12_bit;
- regs.int_16_bit = tuning_settings.int_16_bit;
- regs.mod_12_bit = tuning_settings.mod_12_bit;
- regs.clock_divider_12_bit = tuning_settings.clock_divider_12_bit;
- regs.feedback_select = tuning_constraints.feedback_after_divider ?
- adf4351_regs_t::FEEDBACK_SELECT_DIVIDED :
- adf4351_regs_t::FEEDBACK_SELECT_FUNDAMENTAL;
- regs.clock_div_mode = adf4351_regs_t::CLOCK_DIV_MODE_RESYNC_ENABLE;
- regs.prescaler = prescaler;
- regs.r_counter_10_bit = tuning_settings.r_counter_10_bit;
- regs.reference_divide_by_2 = tuning_settings.r_divide_by_2_en ?
- adf4351_regs_t::REFERENCE_DIVIDE_BY_2_ENABLED :
- adf4351_regs_t::REFERENCE_DIVIDE_BY_2_DISABLED;
- regs.reference_doubler = tuning_settings.r_doubler_en ?
- adf4351_regs_t::REFERENCE_DOUBLER_ENABLED :
- adf4351_regs_t::REFERENCE_DOUBLER_DISABLED;
- regs.band_select_clock_div = tuning_settings.band_select_clock_div;
- UHD_ASSERT_THROW(rfdivsel_to_enum.has_key(tuning_settings.rf_divider));
- regs.rf_divider_select = rfdivsel_to_enum[tuning_settings.rf_divider];
- regs.ldf = is_int_n ?
- adf4351_regs_t::LDF_INT_N :
- adf4351_regs_t::LDF_FRAC_N;
-
- //reset the N and R counter
- regs.counter_reset = adf4351_regs_t::COUNTER_RESET_ENABLED;
- self_base->get_iface()->write_spi(unit, spi_config_t::EDGE_RISE, regs.get_reg(2), 32);
- regs.counter_reset = adf4351_regs_t::COUNTER_RESET_DISABLED;
-
- //write the registers
- //correct power-up sequence to write registers (5, 4, 3, 2, 1, 0)
- int addr;
-
- boost::uint16_t rx_id = self_base->get_rx_id().to_uint16();
- std::string board_name = (rx_id == 0x0083) ? "SBX-120" : "SBX";
- for(addr=5; addr>=0; addr--){
- UHD_LOGV(often) << boost::format(
- "%s SPI Reg (0x%02x): 0x%08x"
- ) % board_name.c_str() % addr % regs.get_reg(addr) << std::endl;
- self_base->get_iface()->write_spi(
- unit, spi_config_t::EDGE_RISE,
- regs.get_reg(addr), 32
- );
+ //Select the LO
+ adf435x_iface::sptr& lo_iface = unit == dboard_iface::UNIT_RX ? _rxlo : _txlo;
+ lo_iface->set_feedback_select(adf435x_iface::FB_SEL_DIVIDED);
+ lo_iface->set_reference_freq(self_base->get_iface()->get_clock_rate(unit));
+
+ //Use 8/9 prescaler for vco_freq > 3 GHz (pg.18 prescaler)
+ lo_iface->set_prescaler(target_freq > 3.6e9 ? adf435x_iface::PRESCALER_8_9 : adf435x_iface::PRESCALER_4_5);
+
+ //Configure the LO
+ double actual_freq = 0.0;
+ actual_freq = lo_iface->set_frequency(sbx_freq_range.clip(target_freq), is_int_n);
+
+ if ((unit == dboard_iface::UNIT_TX) and (actual_freq == sbx_tx_lo_2dbm.clip(actual_freq))) {
+ lo_iface->set_output_power(adf435x_iface::OUTPUT_POWER_2DBM);
+ } else {
+ lo_iface->set_output_power(adf435x_iface::OUTPUT_POWER_5DBM);
}
- //return the actual frequency
- UHD_LOGV(often) << boost::format(
- "%s tune: actual frequency %f MHz"
- ) % board_name.c_str() % (actual_freq/1e6) << std::endl;
+ //Write to hardware
+ lo_iface->commit();
+
return actual_freq;
}
diff --git a/host/lib/usrp/dboard/db_tvrx.cpp b/host/lib/usrp/dboard/db_tvrx.cpp
index e9f60f765..0f84cd68a 100644
--- a/host/lib/usrp/dboard/db_tvrx.cpp
+++ b/host/lib/usrp/dboard/db_tvrx.cpp
@@ -134,7 +134,7 @@ static const double reference_freq = 4.0e6;
class tvrx : public rx_dboard_base{
public:
tvrx(ctor_args_t args);
- ~tvrx(void);
+ virtual ~tvrx(void);
private:
uhd::dict<std::string, double> _gains;
@@ -190,12 +190,12 @@ tvrx::tvrx(ctor_args_t args) : rx_dboard_base(args){
this->get_rx_subtree()->create<int>("sensors"); //phony property so this dir exists
BOOST_FOREACH(const std::string &name, get_tvrx_gain_ranges().keys()){
this->get_rx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&tvrx::set_gain, this, _1, name));
+ .set_coercer(boost::bind(&tvrx::set_gain, this, _1, name));
this->get_rx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(get_tvrx_gain_ranges()[name]);
}
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&tvrx::set_freq, this, _1));
+ .set_coercer(boost::bind(&tvrx::set_freq, this, _1));
this->get_rx_subtree()->create<meta_range_t>("freq/range")
.set(tvrx_freq_range);
this->get_rx_subtree()->create<std::string>("antenna/value")
diff --git a/host/lib/usrp/dboard/db_tvrx2.cpp b/host/lib/usrp/dboard/db_tvrx2.cpp
index ccbac0b5d..6f0604f72 100644
--- a/host/lib/usrp/dboard/db_tvrx2.cpp
+++ b/host/lib/usrp/dboard/db_tvrx2.cpp
@@ -751,7 +751,7 @@ static const uhd::dict<std::string, gain_range_t> tvrx2_gain_ranges = map_list_o
class tvrx2 : public rx_dboard_base{
public:
tvrx2(ctor_args_t args);
- ~tvrx2(void);
+ virtual ~tvrx2(void);
private:
double _freq_scalar;
@@ -957,19 +957,19 @@ tvrx2::tvrx2(ctor_args_t args) : rx_dboard_base(args){
this->get_rx_subtree()->create<std::string>("name")
.set("TVRX2");
this->get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&tvrx2::get_locked, this));
+ .set_publisher(boost::bind(&tvrx2::get_locked, this));
this->get_rx_subtree()->create<sensor_value_t>("sensors/rssi")
- .publish(boost::bind(&tvrx2::get_rssi, this));
+ .set_publisher(boost::bind(&tvrx2::get_rssi, this));
this->get_rx_subtree()->create<sensor_value_t>("sensors/temperature")
- .publish(boost::bind(&tvrx2::get_temp, this));
+ .set_publisher(boost::bind(&tvrx2::get_temp, this));
BOOST_FOREACH(const std::string &name, tvrx2_gain_ranges.keys()){
this->get_rx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&tvrx2::set_gain, this, _1, name));
+ .set_coercer(boost::bind(&tvrx2::set_gain, this, _1, name));
this->get_rx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(tvrx2_gain_ranges[name]);
}
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&tvrx2::set_lo_freq, this, _1));
+ .set_coercer(boost::bind(&tvrx2::set_lo_freq, this, _1));
this->get_rx_subtree()->create<meta_range_t>("freq/range")
.set(tvrx2_freq_range);
this->get_rx_subtree()->create<std::string>("antenna/value")
@@ -979,12 +979,12 @@ tvrx2::tvrx2(ctor_args_t args) : rx_dboard_base(args){
this->get_rx_subtree()->create<std::string>("connection")
.set(tvrx2_sd_name_to_conn[get_subdev_name()]);
this->get_rx_subtree()->create<bool>("enabled")
- .coerce(boost::bind(&tvrx2::set_enabled, this, _1))
+ .set_coercer(boost::bind(&tvrx2::set_enabled, this, _1))
.set(_enabled);
this->get_rx_subtree()->create<bool>("use_lo_offset")
.set(false);
this->get_rx_subtree()->create<double>("bandwidth/value")
- .coerce(boost::bind(&tvrx2::set_bandwidth, this, _1))
+ .set_coercer(boost::bind(&tvrx2::set_bandwidth, this, _1))
.set(_bandwidth);
this->get_rx_subtree()->create<meta_range_t>("bandwidth/range")
.set(tvrx2_bandwidth_range);
diff --git a/host/lib/usrp/dboard/db_twinrx.cpp b/host/lib/usrp/dboard/db_twinrx.cpp
new file mode 100644
index 000000000..e960f134f
--- /dev/null
+++ b/host/lib/usrp/dboard/db_twinrx.cpp
@@ -0,0 +1,337 @@
+//
+// Copyright 2014-15 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#include "twinrx/twinrx_experts.hpp"
+#include "twinrx/twinrx_ctrl.hpp"
+#include "twinrx/twinrx_io.hpp"
+#include <expert_factory.hpp>
+#include <uhd/types/device_addr.hpp>
+#include <uhd/types/ranges.hpp>
+#include <uhd/types/sensors.hpp>
+#include <uhd/usrp/dboard_base.hpp>
+#include <uhd/usrp/dboard_manager.hpp>
+#include <uhd/utils/log.hpp>
+#include <uhd/utils/static.hpp>
+#include "dboard_ctor_args.hpp"
+#include <boost/assign/list_of.hpp>
+#include <boost/make_shared.hpp>
+#include <boost/thread.hpp>
+#include <boost/thread/mutex.hpp>
+//#include <fstream> //Needed for _expert->to_dot() below
+
+using namespace uhd;
+using namespace uhd::usrp;
+using namespace uhd::usrp::dboard::twinrx;
+using namespace uhd::experts;
+
+static const dboard_id_t TWINRX_V100_000_ID(0x91);
+
+/*!
+ * twinrx_rcvr_fe is the dbaord class (dboard_base) that
+ * represents each front-end of a TwinRX board. UHD will
+ * create and hold two instances of this class per TwinRX
+ * dboard.
+ *
+ */
+class twinrx_rcvr_fe : public rx_dboard_base
+{
+public:
+ twinrx_rcvr_fe(
+ ctor_args_t args,
+ expert_container::sptr expert,
+ twinrx_ctrl::sptr ctrl
+ ) :
+ rx_dboard_base(args), _expert(expert), _ctrl(ctrl),
+ _ch_name(dboard_ctor_args_t::cast(args).sd_name)
+ {
+ //---------------------------------------------------------
+ // Add user-visible, channel specific properties to front-end tree
+ //---------------------------------------------------------
+
+ //Generic
+ get_rx_subtree()->create<std::string>("name")
+ .set("TwinRX RX" + _ch_name);
+ get_rx_subtree()->create<bool>("use_lo_offset")
+ .set(false);
+ get_rx_subtree()->create<std::string>("connection")
+ .set(_ch_name == "0" ? "II" : "QQ"); //Ch->ADC port mapping
+ static const double BW = 80e6;
+ get_rx_subtree()->create<double>("bandwidth/value")
+ .set(BW);
+ get_rx_subtree()->create<meta_range_t>("bandwidth/range")
+ .set(freq_range_t(BW, BW));
+
+ //Frequency Specific
+ get_rx_subtree()->create<meta_range_t>("freq/range")
+ .set(freq_range_t(10e6, 6.0e9));
+ expert_factory::add_dual_prop_node<double>(_expert, get_rx_subtree(),
+ "freq/value", prepend_ch("freq/desired", _ch_name), prepend_ch("freq/coerced", _ch_name),
+ 1.0e9, AUTO_RESOLVE_ON_READ_WRITE);
+ get_rx_subtree()->create<device_addr_t>("tune_args")
+ .set(device_addr_t());
+
+ static const double DEFAULT_IF_FREQ = 150e6;
+ meta_range_t if_freq_range;
+ if_freq_range.push_back(range_t(-DEFAULT_IF_FREQ-(BW/2), -DEFAULT_IF_FREQ+(BW/2)));
+ if_freq_range.push_back(range_t( DEFAULT_IF_FREQ-(BW/2), DEFAULT_IF_FREQ+(BW/2)));
+ get_rx_subtree()->create<meta_range_t>("if_freq/range")
+ .set(if_freq_range);
+ expert_factory::add_dual_prop_node<double>(_expert, get_rx_subtree(),
+ "if_freq/value", prepend_ch("if_freq/desired", _ch_name), prepend_ch("if_freq/coerced", _ch_name),
+ DEFAULT_IF_FREQ, AUTO_RESOLVE_ON_WRITE);
+
+ //LO Specific
+ get_rx_subtree()->create<meta_range_t>("los/LO1/freq/range")
+ .set(freq_range_t(2.0e9, 6.8e9));
+ expert_factory::add_dual_prop_node<double>(_expert, get_rx_subtree(),
+ "los/LO1/freq/value", prepend_ch("los/LO1/freq/desired", _ch_name), prepend_ch("los/LO1/freq/coerced", _ch_name),
+ 0.0, AUTO_RESOLVE_ON_READ_WRITE);
+ get_rx_subtree()->create<meta_range_t>("los/LO2/freq/range")
+ .set(freq_range_t(1.0e9, 3.0e9));
+ expert_factory::add_dual_prop_node<double>(_expert, get_rx_subtree(),
+ "los/LO2/freq/value", prepend_ch("los/LO2/freq/desired", _ch_name), prepend_ch("los/LO2/freq/coerced", _ch_name),
+ 0.0, AUTO_RESOLVE_ON_READ_WRITE);
+ get_rx_subtree()->create<std::vector<std::string> >("los/all/source/options")
+ .set(boost::assign::list_of("internal")("external")("companion")("disabled"));
+ expert_factory::add_prop_node<std::string>(_expert, get_rx_subtree(),
+ "los/all/source/value", prepend_ch("los/all/source", _ch_name),
+ "internal", AUTO_RESOLVE_ON_WRITE);
+ expert_factory::add_prop_node<bool>(_expert, get_rx_subtree(),
+ "los/all/export", prepend_ch("los/all/export", _ch_name),
+ false, AUTO_RESOLVE_ON_WRITE);
+
+ //Gain Specific
+ get_rx_subtree()->create<meta_range_t>("gains/all/range")
+ .set(gain_range_t(0, 95, double(1.0)));
+ expert_factory::add_prop_node<double>(_expert, get_rx_subtree(),
+ "gains/all/value", prepend_ch("gain", _ch_name),
+ 0.0, AUTO_RESOLVE_ON_WRITE);
+ get_rx_subtree()->create<std::vector<std::string> >("gains/all/profile/options")
+ .set(boost::assign::list_of("low-noise")("low-distortion")("default"));
+ expert_factory::add_prop_node<std::string>(_expert, get_rx_subtree(),
+ "gains/all/profile/value", prepend_ch("gain_profile", _ch_name),
+ "default", AUTO_RESOLVE_ON_WRITE);
+
+ //Antenna Specific
+ get_rx_subtree()->create<std::vector<std::string> >("antenna/options")
+ .set(boost::assign::list_of("RX1")("RX2"));
+ expert_factory::add_prop_node<std::string>(_expert, get_rx_subtree(),
+ "antenna/value", prepend_ch("antenna", _ch_name),
+ (_ch_name == "0" ? "RX1" : "RX2"), AUTO_RESOLVE_ON_WRITE);
+ expert_factory::add_prop_node<bool>(_expert, get_rx_subtree(),
+ "enabled", prepend_ch("enabled", _ch_name),
+ false, AUTO_RESOLVE_ON_WRITE);
+
+ //Readback
+ get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked")
+ .set_publisher(boost::bind(&twinrx_rcvr_fe::get_lo_locked, this));
+
+ //---------------------------------------------------------
+ // Add internal channel-specific data nodes to expert
+ //---------------------------------------------------------
+ expert_factory::add_data_node<lo_inj_side_t>(_expert,
+ prepend_ch("ch/LO1/inj_side", _ch_name), INJ_LOW_SIDE);
+ expert_factory::add_data_node<lo_inj_side_t>(_expert,
+ prepend_ch("ch/LO2/inj_side", _ch_name), INJ_LOW_SIDE);
+ expert_factory::add_data_node<twinrx_ctrl::signal_path_t>(_expert,
+ prepend_ch("ch/signal_path", _ch_name), twinrx_ctrl::PATH_LOWBAND);
+ expert_factory::add_data_node<twinrx_ctrl::preselector_path_t>(_expert,
+ prepend_ch("ch/lb_presel", _ch_name), twinrx_ctrl::PRESEL_PATH1);
+ expert_factory::add_data_node<twinrx_ctrl::preselector_path_t>(_expert,
+ prepend_ch("ch/hb_presel", _ch_name), twinrx_ctrl::PRESEL_PATH1);
+ expert_factory::add_data_node<bool>(_expert,
+ prepend_ch("ch/lb_preamp_presel", _ch_name), false);
+ expert_factory::add_data_node<bool>(_expert,
+ prepend_ch("ant/lb_preamp_presel", _ch_name), false);
+ expert_factory::add_data_node<twinrx_ctrl::preamp_state_t>(_expert,
+ prepend_ch("ch/preamp1", _ch_name), twinrx_ctrl::PREAMP_BYPASS);
+ expert_factory::add_data_node<twinrx_ctrl::preamp_state_t>(_expert,
+ prepend_ch("ant/preamp1", _ch_name), twinrx_ctrl::PREAMP_BYPASS);
+ expert_factory::add_data_node<bool>(_expert,
+ prepend_ch("ch/preamp2", _ch_name), false);
+ expert_factory::add_data_node<bool>(_expert,
+ prepend_ch("ant/preamp2", _ch_name), false);
+ expert_factory::add_data_node<boost::uint8_t>(_expert,
+ prepend_ch("ch/input_atten", _ch_name), 0);
+ expert_factory::add_data_node<boost::uint8_t>(_expert,
+ prepend_ch("ant/input_atten", _ch_name), 0);
+ expert_factory::add_data_node<boost::uint8_t>(_expert,
+ prepend_ch("ch/lb_atten", _ch_name), 0);
+ expert_factory::add_data_node<boost::uint8_t>(_expert,
+ prepend_ch("ch/hb_atten", _ch_name), 0);
+ expert_factory::add_data_node<twinrx_ctrl::lo_source_t>(_expert,
+ prepend_ch("ch/LO1/source", _ch_name), twinrx_ctrl::LO_INTERNAL);
+ expert_factory::add_data_node<twinrx_ctrl::lo_source_t>(_expert,
+ prepend_ch("ch/LO2/source", _ch_name), twinrx_ctrl::LO_INTERNAL);
+ expert_factory::add_data_node<lo_synth_mapping_t>(_expert,
+ prepend_ch("synth/LO1/mapping", _ch_name), MAPPING_NONE);
+ expert_factory::add_data_node<lo_synth_mapping_t>(_expert,
+ prepend_ch("synth/LO2/mapping", _ch_name), MAPPING_NONE);
+
+ }
+
+ virtual ~twinrx_rcvr_fe(void)
+ {
+ }
+
+ sensor_value_t get_lo_locked()
+ {
+ bool locked = true;
+ twinrx_ctrl::channel_t ch = (_ch_name == "0") ? twinrx_ctrl::CH1 : twinrx_ctrl::CH2;
+ locked &= _ctrl->read_lo1_locked(ch);
+ locked &= _ctrl->read_lo2_locked(ch);
+ return sensor_value_t("LO", locked, "locked", "unlocked");
+ }
+
+private:
+ expert_container::sptr _expert;
+ twinrx_ctrl::sptr _ctrl;
+ const std::string _ch_name;
+};
+
+/*!
+ * twinrx_rcvr is the top-level container for each
+ * TwinRX board. UHD will hold one instance of this
+ * class per TwinRX dboard. This class is responsible
+ * for owning all the control classes for the board.
+ *
+ */
+class twinrx_rcvr : public rx_dboard_base
+{
+public:
+ typedef boost::shared_ptr<twinrx_rcvr> sptr;
+
+ twinrx_rcvr(ctor_args_t args) : rx_dboard_base(args)
+ {
+ _db_iface = get_iface();
+ twinrx_gpio::sptr gpio_iface = boost::make_shared<twinrx_gpio>(_db_iface);
+ twinrx_cpld_regmap::sptr cpld_regs = boost::make_shared<twinrx_cpld_regmap>();
+ cpld_regs->initialize(*gpio_iface, false);
+ _ctrl = twinrx_ctrl::make(_db_iface, gpio_iface, cpld_regs);
+ _expert = expert_factory::create_container("twinrx_expert");
+ }
+
+ virtual ~twinrx_rcvr(void)
+ {
+ }
+
+ inline expert_container::sptr get_expert() {
+ return _expert;
+ }
+
+ inline twinrx_ctrl::sptr get_ctrl() {
+ return _ctrl;
+ }
+
+ virtual void initialize()
+ {
+ //---------------------------------------------------------
+ // Add internal channel-agnostic data nodes to expert
+ //---------------------------------------------------------
+ expert_factory::add_data_node<twinrx_ctrl::lo_export_source_t>(_expert,
+ "com/LO1/export_source", twinrx_ctrl::LO_EXPORT_DISABLED);
+ expert_factory::add_data_node<twinrx_ctrl::lo_export_source_t>(_expert,
+ "com/LO2/export_source", twinrx_ctrl::LO_EXPORT_DISABLED);
+ expert_factory::add_data_node<twinrx_ctrl::antenna_mapping_t>(_expert,
+ "com/ant_mapping", twinrx_ctrl::ANTX_NATIVE);
+ expert_factory::add_data_node<twinrx_ctrl::cal_mode_t>(_expert,
+ "com/cal_mode", twinrx_ctrl::CAL_DISABLED);
+ expert_factory::add_data_node<bool>(_expert,
+ "com/synth/LO1/hopping_enabled", false);
+ expert_factory::add_data_node<bool>(_expert,
+ "com/synth/LO2/hopping_enabled", false);
+
+ //---------------------------------------------------------
+ // Add workers to expert
+ //---------------------------------------------------------
+ //Channel (front-end) specific
+ BOOST_FOREACH(const std::string& fe, _fe_names) {
+ expert_factory::add_worker_node<twinrx_freq_path_expert>(_expert, _expert->node_retriever(), fe);
+ expert_factory::add_worker_node<twinrx_freq_coercion_expert>(_expert, _expert->node_retriever(), fe);
+ expert_factory::add_worker_node<twinrx_chan_gain_expert>(_expert, _expert->node_retriever(), fe);
+ expert_factory::add_worker_node<twinrx_nyquist_expert>(_expert, _expert->node_retriever(), fe, _db_iface);
+ }
+
+ //Channel (front-end) agnostic
+ expert_factory::add_worker_node<twinrx_lo_config_expert>(_expert, _expert->node_retriever());
+ expert_factory::add_worker_node<twinrx_lo_mapping_expert>(_expert, _expert->node_retriever(), STAGE_LO1);
+ expert_factory::add_worker_node<twinrx_lo_mapping_expert>(_expert, _expert->node_retriever(), STAGE_LO2);
+ expert_factory::add_worker_node<twinrx_antenna_expert>(_expert, _expert->node_retriever());
+ expert_factory::add_worker_node<twinrx_ant_gain_expert>(_expert, _expert->node_retriever());
+ expert_factory::add_worker_node<twinrx_settings_expert>(_expert, _expert->node_retriever(), _ctrl);
+
+ /*//Expert debug code
+ std::ofstream dot_file("/tmp/twinrx.dot", std::ios::out);
+ dot_file << _expert->to_dot();
+ dot_file.close();
+ */
+
+ _expert->debug_audit();
+ _expert->resolve_all(true);
+ }
+
+ static dboard_base::sptr make_twinrx_fe(dboard_base::ctor_args_t args)
+ {
+ const dboard_ctor_args_t& db_args = dboard_ctor_args_t::cast(args);
+ sptr container = boost::dynamic_pointer_cast<twinrx_rcvr>(db_args.rx_container);
+ if (container) {
+ dboard_base::sptr fe = dboard_base::sptr(
+ new twinrx_rcvr_fe(args, container->get_expert(), container->get_ctrl()));
+ container->add_twinrx_fe(db_args.sd_name);
+ return fe;
+ } else {
+ throw uhd::assertion_error("error creating twinrx frontend");
+ }
+ }
+
+protected:
+ inline void add_twinrx_fe(const std::string& name) {
+ _fe_names.push_back(name);
+ }
+
+private:
+ typedef std::map<std::string, dboard_base::sptr> twinrx_fe_map_t;
+
+ dboard_iface::sptr _db_iface;
+ twinrx_ctrl::sptr _ctrl;
+ std::vector<std::string> _fe_names;
+ expert_container::sptr _expert;
+};
+
+/*!
+ * Initialization Sequence for each TwinRX board:
+ * - make_twinrx_container is called which creates an instance of twinrx_rcvr
+ * - twinrx_rcvr::make_twinrx_fe is called with channel "0" which creates an instance of twinrx_rcvr_fe
+ * - twinrx_rcvr::make_twinrx_fe is called with channel "1" which creates an instance of twinrx_rcvr_fe
+ * - twinrx_rcvr::initialize is called with finishes the init sequence
+ *
+ */
+static dboard_base::sptr make_twinrx_container(dboard_base::ctor_args_t args)
+{
+ return dboard_base::sptr(new twinrx_rcvr(args));
+}
+
+UHD_STATIC_BLOCK(reg_twinrx_dboards)
+{
+ dboard_manager::register_dboard_restricted(
+ TWINRX_V100_000_ID,
+ &twinrx_rcvr::make_twinrx_fe,
+ "TwinRX v1.0",
+ boost::assign::list_of("0")("1"),
+ &make_twinrx_container
+ );
+}
diff --git a/host/lib/usrp/dboard/db_ubx.cpp b/host/lib/usrp/dboard/db_ubx.cpp
index 15a4d17a9..be3bdd17d 100644
--- a/host/lib/usrp/dboard/db_ubx.cpp
+++ b/host/lib/usrp/dboard/db_ubx.cpp
@@ -27,7 +27,9 @@
#include <uhd/usrp/dboard_manager.hpp>
#include <uhd/utils/assert_has.hpp>
#include <uhd/utils/log.hpp>
+#include <uhd/utils/msg.hpp>
#include <uhd/utils/static.hpp>
+#include <uhd/utils/safe_call.hpp>
#include <boost/assign/list_of.hpp>
#include <boost/shared_ptr.hpp>
#include <boost/math/special_functions/round.hpp>
@@ -319,14 +321,14 @@ public:
write_gpio();
// Configure ATR
- _iface->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_IDLE, _tx_gpio_reg.atr_idle);
- _iface->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, _tx_gpio_reg.atr_tx);
- _iface->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_RX_ONLY, _tx_gpio_reg.atr_rx);
- _iface->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, _tx_gpio_reg.atr_full_duplex);
- _iface->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_IDLE, _rx_gpio_reg.atr_idle);
- _iface->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, _rx_gpio_reg.atr_tx);
- _iface->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_RX_ONLY, _rx_gpio_reg.atr_rx);
- _iface->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, _rx_gpio_reg.atr_full_duplex);
+ _iface->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_IDLE, _tx_gpio_reg.atr_idle);
+ _iface->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, _tx_gpio_reg.atr_tx);
+ _iface->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_RX_ONLY, _tx_gpio_reg.atr_rx);
+ _iface->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, _tx_gpio_reg.atr_full_duplex);
+ _iface->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_IDLE, _rx_gpio_reg.atr_idle);
+ _iface->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_TX_ONLY, _rx_gpio_reg.atr_tx);
+ _iface->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_RX_ONLY, _rx_gpio_reg.atr_rx);
+ _iface->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_FULL_DUPLEX, _rx_gpio_reg.atr_full_duplex);
// Engage ATR control (1 is ATR control, 0 is manual control)
_iface->set_pin_ctrl(dboard_iface::UNIT_TX, _tx_gpio_reg.atr_mask);
@@ -385,23 +387,23 @@ public:
get_rx_subtree()->create<std::vector<std::string> >("power_mode/options")
.set(ubx_power_modes);
get_rx_subtree()->create<std::string>("power_mode/value")
- .subscribe(boost::bind(&ubx_xcvr::set_power_mode, this, _1))
+ .add_coerced_subscriber(boost::bind(&ubx_xcvr::set_power_mode, this, _1))
.set("performance");
get_rx_subtree()->create<std::vector<std::string> >("xcvr_mode/options")
.set(ubx_xcvr_modes);
get_rx_subtree()->create<std::string>("xcvr_mode/value")
- .subscribe(boost::bind(&ubx_xcvr::set_xcvr_mode, this, _1))
+ .add_coerced_subscriber(boost::bind(&ubx_xcvr::set_xcvr_mode, this, _1))
.set("FDX");
get_tx_subtree()->create<std::vector<std::string> >("power_mode/options")
.set(ubx_power_modes);
get_tx_subtree()->create<std::string>("power_mode/value")
- .subscribe(boost::bind(&uhd::property<std::string>::set, &get_rx_subtree()->access<std::string>("power_mode/value"), _1))
- .publish(boost::bind(&uhd::property<std::string>::get, &get_rx_subtree()->access<std::string>("power_mode/value")));
+ .add_coerced_subscriber(boost::bind(&uhd::property<std::string>::set, &get_rx_subtree()->access<std::string>("power_mode/value"), _1))
+ .set_publisher(boost::bind(&uhd::property<std::string>::get, &get_rx_subtree()->access<std::string>("power_mode/value")));
get_tx_subtree()->create<std::vector<std::string> >("xcvr_mode/options")
.set(ubx_xcvr_modes);
get_tx_subtree()->create<std::string>("xcvr_mode/value")
- .subscribe(boost::bind(&uhd::property<std::string>::set, &get_rx_subtree()->access<std::string>("xcvr_mode/value"), _1))
- .publish(boost::bind(&uhd::property<std::string>::get, &get_rx_subtree()->access<std::string>("xcvr_mode/value")));
+ .add_coerced_subscriber(boost::bind(&uhd::property<std::string>::set, &get_rx_subtree()->access<std::string>("xcvr_mode/value"), _1))
+ .set_publisher(boost::bind(&uhd::property<std::string>::get, &get_rx_subtree()->access<std::string>("xcvr_mode/value")));
////////////////////////////////////////////////////////////////////
// Register TX properties
@@ -410,20 +412,20 @@ public:
get_tx_subtree()->create<device_addr_t>("tune_args")
.set(device_addr_t());
get_tx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&ubx_xcvr::get_locked, this, "TXLO"));
+ .set_publisher(boost::bind(&ubx_xcvr::get_locked, this, "TXLO"));
get_tx_subtree()->create<double>("gains/PGA0/value")
- .coerce(boost::bind(&ubx_xcvr::set_tx_gain, this, _1)).set(0);
+ .set_coercer(boost::bind(&ubx_xcvr::set_tx_gain, this, _1)).set(0);
get_tx_subtree()->create<meta_range_t>("gains/PGA0/range")
.set(ubx_tx_gain_range);
get_tx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&ubx_xcvr::set_tx_freq, this, _1))
+ .set_coercer(boost::bind(&ubx_xcvr::set_tx_freq, this, _1))
.set(ubx_freq_range.start());
get_tx_subtree()->create<meta_range_t>("freq/range")
.set(ubx_freq_range);
get_tx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(ubx_tx_antennas);
get_tx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&ubx_xcvr::set_tx_ant, this, _1))
+ .add_coerced_subscriber(boost::bind(&ubx_xcvr::set_tx_ant, this, _1))
.set(ubx_tx_antennas.at(0));
get_tx_subtree()->create<std::string>("connection")
.set("QI");
@@ -436,7 +438,7 @@ public:
get_tx_subtree()->create<meta_range_t>("bandwidth/range")
.set(freq_range_t(bw, bw));
get_tx_subtree()->create<int64_t>("sync_delay")
- .subscribe(boost::bind(&ubx_xcvr::set_sync_delay, this, true, _1))
+ .add_coerced_subscriber(boost::bind(&ubx_xcvr::set_sync_delay, this, true, _1))
.set(-8);
////////////////////////////////////////////////////////////////////
@@ -446,21 +448,21 @@ public:
get_rx_subtree()->create<device_addr_t>("tune_args")
.set(device_addr_t());
get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&ubx_xcvr::get_locked, this, "RXLO"));
+ .set_publisher(boost::bind(&ubx_xcvr::get_locked, this, "RXLO"));
get_rx_subtree()->create<double>("gains/PGA0/value")
- .coerce(boost::bind(&ubx_xcvr::set_rx_gain, this, _1))
+ .set_coercer(boost::bind(&ubx_xcvr::set_rx_gain, this, _1))
.set(0);
get_rx_subtree()->create<meta_range_t>("gains/PGA0/range")
.set(ubx_rx_gain_range);
get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&ubx_xcvr::set_rx_freq, this, _1))
+ .set_coercer(boost::bind(&ubx_xcvr::set_rx_freq, this, _1))
.set(ubx_freq_range.start());
get_rx_subtree()->create<meta_range_t>("freq/range")
.set(ubx_freq_range);
get_rx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(ubx_rx_antennas);
get_rx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&ubx_xcvr::set_rx_ant, this, _1)).set("RX2");
+ .add_coerced_subscriber(boost::bind(&ubx_xcvr::set_rx_ant, this, _1)).set("RX2");
get_rx_subtree()->create<std::string>("connection")
.set("IQ");
get_rx_subtree()->create<bool>("enabled")
@@ -472,35 +474,38 @@ public:
get_rx_subtree()->create<meta_range_t>("bandwidth/range")
.set(freq_range_t(bw, bw));
get_rx_subtree()->create<int64_t>("sync_delay")
- .subscribe(boost::bind(&ubx_xcvr::set_sync_delay, this, false, _1))
+ .add_coerced_subscriber(boost::bind(&ubx_xcvr::set_sync_delay, this, false, _1))
.set(-8);
}
- ~ubx_xcvr(void)
+ virtual ~ubx_xcvr(void)
{
- // Shutdown synthesizers
- _txlo1->shutdown();
- _txlo2->shutdown();
- _rxlo1->shutdown();
- _rxlo2->shutdown();
+ UHD_SAFE_CALL
+ (
+ // Shutdown synthesizers
+ _txlo1->shutdown();
+ _txlo2->shutdown();
+ _rxlo1->shutdown();
+ _rxlo2->shutdown();
- // Reset CPLD values
- _cpld_reg.value = 0;
- write_cpld_reg();
+ // Reset CPLD values
+ _cpld_reg.value = 0;
+ write_cpld_reg();
- // Reset GPIO values
- set_gpio_field(TX_GAIN, 0);
- set_gpio_field(CPLD_RST_N, 0);
- set_gpio_field(RX_ANT, 1);
- set_gpio_field(TX_EN_N, 1);
- set_gpio_field(RX_EN_N, 1);
- set_gpio_field(SPI_ADDR, 0x7);
- set_gpio_field(RX_GAIN, 0);
- set_gpio_field(TXLO1_SYNC, 0);
- set_gpio_field(TXLO2_SYNC, 0);
- set_gpio_field(RXLO1_SYNC, 0);
- set_gpio_field(RXLO1_SYNC, 0);
- write_gpio();
+ // Reset GPIO values
+ set_gpio_field(TX_GAIN, 0);
+ set_gpio_field(CPLD_RST_N, 0);
+ set_gpio_field(RX_ANT, 1);
+ set_gpio_field(TX_EN_N, 1);
+ set_gpio_field(RX_EN_N, 1);
+ set_gpio_field(SPI_ADDR, 0x7);
+ set_gpio_field(RX_GAIN, 0);
+ set_gpio_field(TXLO1_SYNC, 0);
+ set_gpio_field(TXLO2_SYNC, 0);
+ set_gpio_field(RXLO1_SYNC, 0);
+ set_gpio_field(RXLO1_SYNC, 0);
+ write_gpio();
+ )
}
private:
@@ -638,23 +643,23 @@ private:
uint16_t mask = lo1_field_info.mask | lo2_field_info.mask;
dboard_iface::unit_t unit = lo1_field_info.unit;
UHD_ASSERT_THROW(lo1_field_info.unit == lo2_field_info.unit);
- _iface->set_atr_reg(unit, dboard_iface::ATR_REG_IDLE, value, mask);
+ _iface->set_atr_reg(unit, gpio_atr::ATR_REG_IDLE, value, mask);
cmd_time += uhd::time_spec_t(1/pfd_freq);
_iface->set_command_time(cmd_time);
- _iface->set_atr_reg(unit, dboard_iface::ATR_REG_TX_ONLY, value, mask);
+ _iface->set_atr_reg(unit, gpio_atr::ATR_REG_TX_ONLY, value, mask);
cmd_time += uhd::time_spec_t(1/pfd_freq);
_iface->set_command_time(cmd_time);
- _iface->set_atr_reg(unit, dboard_iface::ATR_REG_RX_ONLY, value, mask);
+ _iface->set_atr_reg(unit, gpio_atr::ATR_REG_RX_ONLY, value, mask);
cmd_time += uhd::time_spec_t(1/pfd_freq);
_iface->set_command_time(cmd_time);
- _iface->set_atr_reg(unit, dboard_iface::ATR_REG_FULL_DUPLEX, value, mask);
+ _iface->set_atr_reg(unit, gpio_atr::ATR_REG_FULL_DUPLEX, value, mask);
// De-assert SYNC
// Head of line blocking means the command time does not need to be set.
- _iface->set_atr_reg(unit, dboard_iface::ATR_REG_IDLE, 0, mask);
- _iface->set_atr_reg(unit, dboard_iface::ATR_REG_TX_ONLY, 0, mask);
- _iface->set_atr_reg(unit, dboard_iface::ATR_REG_RX_ONLY, 0, mask);
- _iface->set_atr_reg(unit, dboard_iface::ATR_REG_FULL_DUPLEX, 0, mask);
+ _iface->set_atr_reg(unit, gpio_atr::ATR_REG_IDLE, 0, mask);
+ _iface->set_atr_reg(unit, gpio_atr::ATR_REG_TX_ONLY, 0, mask);
+ _iface->set_atr_reg(unit, gpio_atr::ATR_REG_RX_ONLY, 0, mask);
+ _iface->set_atr_reg(unit, gpio_atr::ATR_REG_FULL_DUPLEX, 0, mask);
}
}
@@ -760,6 +765,20 @@ private:
device_addr_t tune_args = subtree->access<device_addr_t>("tune_args").get();
is_int_n = boost::iequals(tune_args.get("mode_n",""), "integer");
UHD_LOGV(rarely) << boost::format("UBX TX: the requested frequency is %f MHz") % (freq/1e6) << std::endl;
+ double target_pfd_freq = _tx_target_pfd_freq;
+ if (is_int_n and tune_args.has_key("int_n_step"))
+ {
+ target_pfd_freq = tune_args.cast<double>("int_n_step", _tx_target_pfd_freq);
+ if (target_pfd_freq > _tx_target_pfd_freq)
+ {
+ UHD_MSG(warning)
+ << boost::format("Requested int_n_step of %f MHz too large, clipping to %f MHz")
+ % (target_pfd_freq/1e6)
+ % (_tx_target_pfd_freq/1e6)
+ << std::endl;
+ target_pfd_freq = _tx_target_pfd_freq;
+ }
+ }
// Clip the frequency to the valid range
freq = ubx_freq_range.clip(freq);
@@ -796,10 +815,10 @@ private:
set_cpld_field(TXLB_SEL, 1);
set_cpld_field(TXHB_SEL, 0);
// Set LO1 to IF of 2100 MHz (offset from RX IF to reduce leakage)
- freq_lo1 = _txlo1->set_frequency(2100*fMHz, ref_freq, _tx_target_pfd_freq, is_int_n);
+ freq_lo1 = _txlo1->set_frequency(2100*fMHz, ref_freq, target_pfd_freq, is_int_n);
_txlo1->set_output_power(max287x_iface::OUTPUT_POWER_5DBM);
// Set LO2 to IF minus desired frequency
- freq_lo2 = _txlo2->set_frequency(freq_lo1 - freq, ref_freq, _tx_target_pfd_freq, is_int_n);
+ freq_lo2 = _txlo2->set_frequency(freq_lo1 - freq, ref_freq, target_pfd_freq, is_int_n);
_txlo2->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq >= (500*fMHz)) && (freq <= (800*fMHz)))
@@ -809,7 +828,7 @@ private:
set_cpld_field(TXLO1_FSEL1, 1);
set_cpld_field(TXLB_SEL, 0);
set_cpld_field(TXHB_SEL, 1);
- freq_lo1 = _txlo1->set_frequency(freq, ref_freq, _tx_target_pfd_freq, is_int_n);
+ freq_lo1 = _txlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_txlo1->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq > (800*fMHz)) && (freq <= (1000*fMHz)))
@@ -819,7 +838,7 @@ private:
set_cpld_field(TXLO1_FSEL1, 1);
set_cpld_field(TXLB_SEL, 0);
set_cpld_field(TXHB_SEL, 1);
- freq_lo1 = _txlo1->set_frequency(freq, ref_freq, _tx_target_pfd_freq, is_int_n);
+ freq_lo1 = _txlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_txlo1->set_output_power(max287x_iface::OUTPUT_POWER_5DBM);
}
else if ((freq > (1000*fMHz)) && (freq <= (2200*fMHz)))
@@ -829,7 +848,7 @@ private:
set_cpld_field(TXLO1_FSEL1, 0);
set_cpld_field(TXLB_SEL, 0);
set_cpld_field(TXHB_SEL, 1);
- freq_lo1 = _txlo1->set_frequency(freq, ref_freq, _tx_target_pfd_freq, is_int_n);
+ freq_lo1 = _txlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_txlo1->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq > (2200*fMHz)) && (freq <= (2500*fMHz)))
@@ -839,7 +858,7 @@ private:
set_cpld_field(TXLO1_FSEL1, 0);
set_cpld_field(TXLB_SEL, 0);
set_cpld_field(TXHB_SEL, 1);
- freq_lo1 = _txlo1->set_frequency(freq, ref_freq, _tx_target_pfd_freq, is_int_n);
+ freq_lo1 = _txlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_txlo1->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq > (2500*fMHz)) && (freq <= (6000*fMHz)))
@@ -849,7 +868,7 @@ private:
set_cpld_field(TXLO1_FSEL1, 0);
set_cpld_field(TXLB_SEL, 0);
set_cpld_field(TXHB_SEL, 1);
- freq_lo1 = _txlo1->set_frequency(freq, ref_freq, _tx_target_pfd_freq, is_int_n);
+ freq_lo1 = _txlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_txlo1->set_output_power(max287x_iface::OUTPUT_POWER_5DBM);
}
@@ -902,6 +921,20 @@ private:
property_tree::sptr subtree = this->get_rx_subtree();
device_addr_t tune_args = subtree->access<device_addr_t>("tune_args").get();
is_int_n = boost::iequals(tune_args.get("mode_n",""), "integer");
+ double target_pfd_freq = _rx_target_pfd_freq;
+ if (is_int_n and tune_args.has_key("int_n_step"))
+ {
+ target_pfd_freq = tune_args.cast<double>("int_n_step", _rx_target_pfd_freq);
+ if (target_pfd_freq > _rx_target_pfd_freq)
+ {
+ UHD_MSG(warning)
+ << boost::format("Requested int_n_step of %f Mhz too large, clipping to %f MHz")
+ % (target_pfd_freq/1e6)
+ % (_rx_target_pfd_freq/1e6)
+ << std::endl;
+ target_pfd_freq = _rx_target_pfd_freq;
+ }
+ }
// Clip the frequency to the valid range
freq = ubx_freq_range.clip(freq);
@@ -940,10 +973,10 @@ private:
set_cpld_field(RXLB_SEL, 1);
set_cpld_field(RXHB_SEL, 0);
// Set LO1 to IF of 2380 MHz (2440 MHz filter center minus 60 MHz offset to minimize LO leakage)
- freq_lo1 = _rxlo1->set_frequency(2380*fMHz, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo1 = _rxlo1->set_frequency(2380*fMHz, ref_freq, target_pfd_freq, is_int_n);
_rxlo1->set_output_power(max287x_iface::OUTPUT_POWER_5DBM);
// Set LO2 to IF minus desired frequency
- freq_lo2 = _rxlo2->set_frequency(freq_lo1 - freq, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo2 = _rxlo2->set_frequency(freq_lo1 - freq, ref_freq, target_pfd_freq, is_int_n);
_rxlo2->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq >= 100*fMHz) && (freq < 500*fMHz))
@@ -956,10 +989,10 @@ private:
set_cpld_field(RXLB_SEL, 1);
set_cpld_field(RXHB_SEL, 0);
// Set LO1 to IF of 2440 (center of filter)
- freq_lo1 = _rxlo1->set_frequency(2440*fMHz, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo1 = _rxlo1->set_frequency(2440*fMHz, ref_freq, target_pfd_freq, is_int_n);
_rxlo1->set_output_power(max287x_iface::OUTPUT_POWER_5DBM);
// Set LO2 to IF minus desired frequency
- freq_lo2 = _rxlo2->set_frequency(freq_lo1 - freq, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo2 = _rxlo2->set_frequency(freq_lo1 - freq, ref_freq, target_pfd_freq, is_int_n);
_rxlo1->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq >= 500*fMHz) && (freq < 800*fMHz))
@@ -971,7 +1004,7 @@ private:
set_cpld_field(RXLO1_FSEL1, 1);
set_cpld_field(RXLB_SEL, 0);
set_cpld_field(RXHB_SEL, 1);
- freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_rxlo1->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq >= 800*fMHz) && (freq < 1000*fMHz))
@@ -983,7 +1016,7 @@ private:
set_cpld_field(RXLO1_FSEL1, 1);
set_cpld_field(RXLB_SEL, 0);
set_cpld_field(RXHB_SEL, 1);
- freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_rxlo1->set_output_power(max287x_iface::OUTPUT_POWER_5DBM);
}
else if ((freq >= 1000*fMHz) && (freq < 1500*fMHz))
@@ -995,7 +1028,7 @@ private:
set_cpld_field(RXLO1_FSEL1, 0);
set_cpld_field(RXLB_SEL, 0);
set_cpld_field(RXHB_SEL, 1);
- freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_rxlo1->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq >= 1500*fMHz) && (freq < 2200*fMHz))
@@ -1007,7 +1040,7 @@ private:
set_cpld_field(RXLO1_FSEL1, 0);
set_cpld_field(RXLB_SEL, 0);
set_cpld_field(RXHB_SEL, 1);
- freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_rxlo1->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq >= 2200*fMHz) && (freq < 2500*fMHz))
@@ -1019,7 +1052,7 @@ private:
set_cpld_field(RXLO1_FSEL1, 0);
set_cpld_field(RXLB_SEL, 0);
set_cpld_field(RXHB_SEL, 1);
- freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_rxlo1->set_output_power(max287x_iface::OUTPUT_POWER_2DBM);
}
else if ((freq >= 2500*fMHz) && (freq <= 6000*fMHz))
@@ -1031,7 +1064,7 @@ private:
set_cpld_field(RXLO1_FSEL1, 0);
set_cpld_field(RXLB_SEL, 0);
set_cpld_field(RXHB_SEL, 1);
- freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, _rx_target_pfd_freq, is_int_n);
+ freq_lo1 = _rxlo1->set_frequency(freq, ref_freq, target_pfd_freq, is_int_n);
_rxlo1->set_output_power(max287x_iface::OUTPUT_POWER_5DBM);
}
diff --git a/host/lib/usrp/dboard/db_wbx_common.cpp b/host/lib/usrp/dboard/db_wbx_common.cpp
index 97357bc90..6539e798a 100644
--- a/host/lib/usrp/dboard/db_wbx_common.cpp
+++ b/host/lib/usrp/dboard/db_wbx_common.cpp
@@ -69,17 +69,17 @@ wbx_base::wbx_base(ctor_args_t args) : xcvr_dboard_base(args){
this->get_rx_subtree()->create<device_addr_t>("tune_args").set(device_addr_t());
this->get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&wbx_base::get_locked, this, dboard_iface::UNIT_RX));
+ .set_publisher(boost::bind(&wbx_base::get_locked, this, dboard_iface::UNIT_RX));
BOOST_FOREACH(const std::string &name, wbx_rx_gain_ranges.keys()){
this->get_rx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&wbx_base::set_rx_gain, this, _1, name))
+ .set_coercer(boost::bind(&wbx_base::set_rx_gain, this, _1, name))
.set(wbx_rx_gain_ranges[name].start());
this->get_rx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(wbx_rx_gain_ranges[name]);
}
this->get_rx_subtree()->create<std::string>("connection").set("IQ");
this->get_rx_subtree()->create<bool>("enabled")
- .subscribe(boost::bind(&wbx_base::set_rx_enabled, this, _1))
+ .add_coerced_subscriber(boost::bind(&wbx_base::set_rx_enabled, this, _1))
.set(true); //start enabled
this->get_rx_subtree()->create<bool>("use_lo_offset").set(false);
@@ -94,7 +94,7 @@ wbx_base::wbx_base(ctor_args_t args) : xcvr_dboard_base(args){
this->get_tx_subtree()->create<device_addr_t>("tune_args").set(device_addr_t());
this->get_tx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&wbx_base::get_locked, this, dboard_iface::UNIT_TX));
+ .set_publisher(boost::bind(&wbx_base::get_locked, this, dboard_iface::UNIT_TX));
this->get_tx_subtree()->create<std::string>("connection").set("IQ");
this->get_tx_subtree()->create<bool>("use_lo_offset").set(false);
@@ -156,3 +156,9 @@ sensor_value_t wbx_base::get_locked(dboard_iface::unit_t unit){
const bool locked = (this->get_iface()->read_gpio(unit) & LOCKDET_MASK) != 0;
return sensor_value_t("LO", locked, "locked", "unlocked");
}
+
+void wbx_base::wbx_versionx::write_lo_regs(dboard_iface::unit_t unit, const std::vector<boost::uint32_t> &regs) {
+ BOOST_FOREACH(boost::uint32_t reg, regs) {
+ self_base->get_iface()->write_spi(unit, spi_config_t::EDGE_RISE, reg, 32);
+ }
+}
diff --git a/host/lib/usrp/dboard/db_wbx_common.hpp b/host/lib/usrp/dboard/db_wbx_common.hpp
index 6a4224048..0e339e4a3 100644
--- a/host/lib/usrp/dboard/db_wbx_common.hpp
+++ b/host/lib/usrp/dboard/db_wbx_common.hpp
@@ -19,8 +19,13 @@
#define INCLUDED_LIBUHD_USRP_DBOARD_DB_WBX_COMMON_HPP
#include <uhd/types/device_addr.hpp>
+#include "adf435x.hpp"
-#include "../common/adf435x_common.hpp"
+// LO Related
+#define ADF435X_CE (1 << 3)
+#define ADF435X_PDBRF (1 << 2)
+#define ADF435X_MUXOUT (1 << 1) // INPUT!!!
+#define LOCKDET_MASK (1 << 0) // INPUT!!!
// TX IO Pins
#define TX_PUP_5V (1 << 7) // enables 5.0V power supply
@@ -40,6 +45,9 @@
#define TX_ATTN_1 (1 << 1)
#define TX_ATTN_MASK (TX_ATTN_16|TX_ATTN_8|TX_ATTN_4|TX_ATTN_2|TX_ATTN_1) // valid bits of TX Attenuator Control
+#define RX_ATTN_SHIFT 8 //lsb of RX Attenuator Control
+#define RX_ATTN_MASK (63 << RX_ATTN_SHIFT) //valid bits of RX Attenuator Control
+
// Mixer functions
#define TX_MIXER_ENB (TXMOD_EN|ADF435X_PDBRF) // for v3, TXMOD_EN tied to ADF435X_PDBRF rather than separate
#define TX_MIXER_DIS 0
@@ -142,6 +150,10 @@ protected:
property_tree::sptr get_tx_subtree(void){
return self_base->get_tx_subtree();
}
+
+ adf435x_iface::sptr _txlo;
+ adf435x_iface::sptr _rxlo;
+ void write_lo_regs(dboard_iface::unit_t unit, const std::vector<boost::uint32_t> &regs);
};
@@ -153,7 +165,7 @@ protected:
class wbx_version2 : public wbx_versionx {
public:
wbx_version2(wbx_base *_self_wbx_base);
- ~wbx_version2(void);
+ virtual ~wbx_version2(void);
double set_tx_gain(double gain, const std::string &name);
void set_tx_enabled(bool enb);
@@ -168,7 +180,7 @@ protected:
class wbx_version3 : public wbx_versionx {
public:
wbx_version3(wbx_base *_self_wbx_base);
- ~wbx_version3(void);
+ virtual ~wbx_version3(void);
double set_tx_gain(double gain, const std::string &name);
void set_tx_enabled(bool enb);
@@ -183,7 +195,7 @@ protected:
class wbx_version4 : public wbx_versionx {
public:
wbx_version4(wbx_base *_self_wbx_base);
- ~wbx_version4(void);
+ virtual ~wbx_version4(void);
double set_tx_gain(double gain, const std::string &name);
void set_tx_enabled(bool enb);
diff --git a/host/lib/usrp/dboard/db_wbx_simple.cpp b/host/lib/usrp/dboard/db_wbx_simple.cpp
index c8f2be155..062e1294b 100644
--- a/host/lib/usrp/dboard/db_wbx_simple.cpp
+++ b/host/lib/usrp/dboard/db_wbx_simple.cpp
@@ -46,7 +46,7 @@ static const std::vector<std::string> wbx_rx_antennas = list_of("TX/RX")("RX2")(
class wbx_simple : public wbx_base{
public:
wbx_simple(ctor_args_t args);
- ~wbx_simple(void);
+ virtual ~wbx_simple(void);
private:
void set_rx_ant(const std::string &ant);
@@ -88,7 +88,7 @@ wbx_simple::wbx_simple(ctor_args_t args) : wbx_base(args){
std::string(str(boost::format("%s+GDB") % this->get_rx_subtree()->access<std::string>("name").get()
)));
this->get_rx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&wbx_simple::set_rx_ant, this, _1))
+ .add_coerced_subscriber(boost::bind(&wbx_simple::set_rx_ant, this, _1))
.set("RX2");
this->get_rx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(wbx_rx_antennas);
@@ -100,7 +100,7 @@ wbx_simple::wbx_simple(ctor_args_t args) : wbx_base(args){
std::string(str(boost::format("%s+GDB") % this->get_tx_subtree()->access<std::string>("name").get()
)));
this->get_tx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&wbx_simple::set_tx_ant, this, _1))
+ .add_coerced_subscriber(boost::bind(&wbx_simple::set_tx_ant, this, _1))
.set(wbx_tx_antennas.at(0));
this->get_tx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(wbx_tx_antennas);
@@ -112,14 +112,14 @@ wbx_simple::wbx_simple(ctor_args_t args) : wbx_base(args){
this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_RX, ANTSW_IO, ANTSW_IO);
//setup ATR for the antenna switches (constant)
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_IDLE, ANT_RX, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_RX_ONLY, ANT_RX, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, ANT_TX, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, ANT_TX, ANTSW_IO);
-
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_IDLE, ANT_TXRX, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, ANT_RX2, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, ANT_RX2, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_IDLE, ANT_RX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_RX_ONLY, ANT_RX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, ANT_TX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, ANT_TX, ANTSW_IO);
+
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_IDLE, ANT_TXRX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_TX_ONLY, ANT_RX2, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_FULL_DUPLEX, ANT_RX2, ANTSW_IO);
}
wbx_simple::~wbx_simple(void){
@@ -138,14 +138,14 @@ void wbx_simple::set_rx_ant(const std::string &ant){
//write the new antenna setting to atr regs
if (_rx_ant == "CAL") {
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, ANT_TXRX, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, ANT_TXRX, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_RX_ONLY, ANT_TXRX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_TX_ONLY, ANT_TXRX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_FULL_DUPLEX, ANT_TXRX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_RX_ONLY, ANT_TXRX, ANTSW_IO);
}
else {
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, ANT_RX2, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, ANT_RX2, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_RX_ONLY, ((_rx_ant == "TX/RX")? ANT_TXRX : ANT_RX2), ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_TX_ONLY, ANT_RX2, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_FULL_DUPLEX, ANT_RX2, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_RX_ONLY, ((_rx_ant == "TX/RX")? ANT_TXRX : ANT_RX2), ANTSW_IO);
}
}
@@ -154,11 +154,11 @@ void wbx_simple::set_tx_ant(const std::string &ant){
//write the new antenna setting to atr regs
if (ant == "CAL") {
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, ANT_RX, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, ANT_RX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, ANT_RX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, ANT_RX, ANTSW_IO);
}
else {
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, ANT_TX, ANTSW_IO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, ANT_TX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, ANT_TX, ANTSW_IO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, ANT_TX, ANTSW_IO);
}
}
diff --git a/host/lib/usrp/dboard/db_wbx_version2.cpp b/host/lib/usrp/dboard/db_wbx_version2.cpp
index 03c8f37e9..18dc383b7 100644
--- a/host/lib/usrp/dboard/db_wbx_version2.cpp
+++ b/host/lib/usrp/dboard/db_wbx_version2.cpp
@@ -16,8 +16,6 @@
//
#include "db_wbx_common.hpp"
-#include "adf4350_regs.hpp"
-#include "../common/adf435x_common.hpp"
#include <uhd/types/tune_request.hpp>
#include <uhd/utils/log.hpp>
#include <uhd/types/dict.hpp>
@@ -77,13 +75,15 @@ static double tx_pga0_gain_to_dac_volts(double &gain){
wbx_base::wbx_version2::wbx_version2(wbx_base *_self_wbx_base) {
//register our handle on the primary wbx_base instance
self_base = _self_wbx_base;
+ _txlo = adf435x_iface::make_adf4350(boost::bind(&wbx_base::wbx_versionx::write_lo_regs, this, dboard_iface::UNIT_TX, _1));
+ _rxlo = adf435x_iface::make_adf4350(boost::bind(&wbx_base::wbx_versionx::write_lo_regs, this, dboard_iface::UNIT_RX, _1));
////////////////////////////////////////////////////////////////////
// Register RX properties
////////////////////////////////////////////////////////////////////
this->get_rx_subtree()->create<std::string>("name").set("WBXv2 RX");
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&wbx_base::wbx_version2::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
+ .set_coercer(boost::bind(&wbx_base::wbx_version2::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
.set((wbx_v2_freq_range.start() + wbx_v2_freq_range.stop())/2.0);
this->get_rx_subtree()->create<meta_range_t>("freq/range").set(wbx_v2_freq_range);
@@ -93,17 +93,17 @@ wbx_base::wbx_version2::wbx_version2(wbx_base *_self_wbx_base) {
this->get_tx_subtree()->create<std::string>("name").set("WBXv2 TX");
BOOST_FOREACH(const std::string &name, wbx_v2_tx_gain_ranges.keys()){
self_base->get_tx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&wbx_base::wbx_version2::set_tx_gain, this, _1, name))
+ .set_coercer(boost::bind(&wbx_base::wbx_version2::set_tx_gain, this, _1, name))
.set(wbx_v2_tx_gain_ranges[name].start());
self_base->get_tx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(wbx_v2_tx_gain_ranges[name]);
}
this->get_tx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&wbx_base::wbx_version2::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
+ .set_coercer(boost::bind(&wbx_base::wbx_version2::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
.set((wbx_v2_freq_range.start() + wbx_v2_freq_range.stop())/2.0);
this->get_tx_subtree()->create<meta_range_t>("freq/range").set(wbx_v2_freq_range);
this->get_tx_subtree()->create<bool>("enabled")
- .subscribe(boost::bind(&wbx_base::wbx_version2::set_tx_enabled, this, _1))
+ .add_coerced_subscriber(boost::bind(&wbx_base::wbx_version2::set_tx_enabled, this, _1))
.set(true); //start enabled
//set attenuator control bits
@@ -117,15 +117,15 @@ wbx_base::wbx_version2::wbx_version2(wbx_base *_self_wbx_base) {
self_base->get_iface()->set_gpio_ddr(dboard_iface::UNIT_RX, RX_PUP_5V|RX_PUP_3V|ADF435X_CE|RXBB_PDB|ADF435X_PDBRF|RX_ATTN_MASK);
//setup ATR for the mixer enables (always enabled to prevent phase slip between bursts)
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_IDLE, v2_tx_mod, TX_MIXER_DIS | v2_tx_mod);
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_RX_ONLY, v2_tx_mod, TX_MIXER_DIS | v2_tx_mod);
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, v2_tx_mod, TX_MIXER_DIS | v2_tx_mod);
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, v2_tx_mod, TX_MIXER_DIS | v2_tx_mod);
-
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_IDLE, RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_RX_ONLY, RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_IDLE, v2_tx_mod, TX_MIXER_DIS | v2_tx_mod);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_RX_ONLY, v2_tx_mod, TX_MIXER_DIS | v2_tx_mod);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, v2_tx_mod, TX_MIXER_DIS | v2_tx_mod);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, v2_tx_mod, TX_MIXER_DIS | v2_tx_mod);
+
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_IDLE, RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_TX_ONLY, RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_RX_ONLY, RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_FULL_DUPLEX, RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
}
wbx_base::wbx_version2::~wbx_version2(void){
@@ -178,111 +178,42 @@ double wbx_base::wbx_version2::set_lo_freq(dboard_iface::unit_t unit, double tar
: self_base->get_tx_subtree();
device_addr_t tune_args = subtree->access<device_addr_t>("tune_args").get();
bool is_int_n = boost::iequals(tune_args.get("mode_n",""), "integer");
+ double reference_freq = self_base->get_iface()->get_clock_rate(unit);
- //map prescaler setting to mininmum integer divider (N) values (pg.18 prescaler)
- static const uhd::dict<int, int> prescaler_to_min_int_div = map_list_of
- (0,23) //adf4350_regs_t::PRESCALER_4_5
- (1,75) //adf4350_regs_t::PRESCALER_8_9
- ;
-
- //map rf divider select output dividers to enums
- static const uhd::dict<int, adf4350_regs_t::rf_divider_select_t> rfdivsel_to_enum = map_list_of
- (1, adf4350_regs_t::RF_DIVIDER_SELECT_DIV1)
- (2, adf4350_regs_t::RF_DIVIDER_SELECT_DIV2)
- (4, adf4350_regs_t::RF_DIVIDER_SELECT_DIV4)
- (8, adf4350_regs_t::RF_DIVIDER_SELECT_DIV8)
- (16, adf4350_regs_t::RF_DIVIDER_SELECT_DIV16)
- ;
+ //Select the LO
+ adf435x_iface::sptr& lo_iface = unit == dboard_iface::UNIT_RX ? _rxlo : _txlo;
+ lo_iface->set_reference_freq(reference_freq);
- double reference_freq = self_base->get_iface()->get_clock_rate(unit);
//The mixer has a divide-by-2 stage on the LO port so the synthesizer
- //frequency must 2x the target frequency. This introduces a 180 degree
- //phase ambiguity
+ //frequency must 2x the target frequency
double synth_target_freq = target_freq * 2;
- adf4350_regs_t::prescaler_t prescaler =
- synth_target_freq > 3e9 ? adf4350_regs_t::PRESCALER_8_9 : adf4350_regs_t::PRESCALER_4_5;
-
- adf435x_tuning_constraints tuning_constraints;
- tuning_constraints.force_frac0 = is_int_n;
- tuning_constraints.band_sel_freq_max = 100e3;
- tuning_constraints.ref_doubler_threshold = 12.5e6;
- tuning_constraints.int_range = uhd::range_t(prescaler_to_min_int_div[prescaler], 4095);
- tuning_constraints.pfd_freq_max = 25e6;
- tuning_constraints.rf_divider_range = uhd::range_t(1, 16);
- //The feedback of the divided frequency must be disabled whenever the target frequency
- //divided by the minimum PFD frequency cannot meet the minimum integer divider (N) value.
- //If it is disabled, additional phase ambiguity will be introduced. With a minimum PFD
- //frequency of 10 MHz, synthesizer frequencies below 230 MHz (LO frequencies below 115 MHz)
- //will have too much ambiguity to synchronize.
- tuning_constraints.feedback_after_divider =
- (int(synth_target_freq / 10e6) >= prescaler_to_min_int_div[prescaler]);
-
- double synth_actual_freq = 0;
- adf435x_tuning_settings tuning_settings = tune_adf435x_synth(
- synth_target_freq, reference_freq, tuning_constraints, synth_actual_freq);
+ //Use 8/9 prescaler for vco_freq > 3 GHz (pg.18 prescaler)
+ lo_iface->set_prescaler(synth_target_freq > 3e9 ?
+ adf435x_iface::PRESCALER_8_9 : adf435x_iface::PRESCALER_4_5);
+
+ //When divider resync is enabled, a 180 deg phase error is introduced when syncing
+ //multiple WBX boards. Switching to fundamental mode works arounds this issue.
+ //TODO: Document why the following has to be true
+ lo_iface->set_feedback_select((target_freq > reference_freq) ?
+ adf435x_iface::FB_SEL_DIVIDED : adf435x_iface::FB_SEL_FUNDAMENTAL);
+
+ double synth_actual_freq = lo_iface->set_frequency(synth_target_freq, is_int_n);
//The mixer has a divide-by-2 stage on the LO port so the synthesizer
//actual_freq must /2 the synth_actual_freq
double actual_freq = synth_actual_freq / 2;
- //load the register values
- adf4350_regs_t regs;
-
- if (unit == dboard_iface::UNIT_RX)
- regs.output_power = (actual_freq == wbx_rx_lo_5dbm.clip(actual_freq)) ? adf4350_regs_t::OUTPUT_POWER_5DBM
- : adf4350_regs_t::OUTPUT_POWER_2DBM;
- else
- regs.output_power = (actual_freq == wbx_tx_lo_5dbm.clip(actual_freq)) ? adf4350_regs_t::OUTPUT_POWER_5DBM
- : adf4350_regs_t::OUTPUT_POWER_M1DBM;
-
- regs.frac_12_bit = tuning_settings.frac_12_bit;
- regs.int_16_bit = tuning_settings.int_16_bit;
- regs.mod_12_bit = tuning_settings.mod_12_bit;
- regs.clock_divider_12_bit = tuning_settings.clock_divider_12_bit;
- regs.feedback_select = tuning_constraints.feedback_after_divider ?
- adf4350_regs_t::FEEDBACK_SELECT_DIVIDED :
- adf4350_regs_t::FEEDBACK_SELECT_FUNDAMENTAL;
- regs.clock_div_mode = tuning_constraints.feedback_after_divider ?
- adf4350_regs_t::CLOCK_DIV_MODE_RESYNC_ENABLE :
- adf4350_regs_t::CLOCK_DIV_MODE_FAST_LOCK;
- regs.prescaler = prescaler;
- regs.r_counter_10_bit = tuning_settings.r_counter_10_bit;
- regs.reference_divide_by_2 = tuning_settings.r_divide_by_2_en ?
- adf4350_regs_t::REFERENCE_DIVIDE_BY_2_ENABLED :
- adf4350_regs_t::REFERENCE_DIVIDE_BY_2_DISABLED;
- regs.reference_doubler = tuning_settings.r_doubler_en ?
- adf4350_regs_t::REFERENCE_DOUBLER_ENABLED :
- adf4350_regs_t::REFERENCE_DOUBLER_DISABLED;
- regs.band_select_clock_div = tuning_settings.band_select_clock_div;
- UHD_ASSERT_THROW(rfdivsel_to_enum.has_key(tuning_settings.rf_divider));
- regs.rf_divider_select = rfdivsel_to_enum[tuning_settings.rf_divider];
- regs.ldf = is_int_n ?
- adf4350_regs_t::LDF_INT_N :
- adf4350_regs_t::LDF_FRAC_N;
-
- //reset the N and R counter
- regs.counter_reset = adf4350_regs_t::COUNTER_RESET_ENABLED;
- self_base->get_iface()->write_spi(unit, spi_config_t::EDGE_RISE, regs.get_reg(2), 32);
- regs.counter_reset = adf4350_regs_t::COUNTER_RESET_DISABLED;
-
- //write the registers
- //correct power-up sequence to write registers (5, 4, 3, 2, 1, 0)
- int addr;
-
- for(addr=5; addr>=0; addr--){
- UHD_LOGV(often) << boost::format(
- "WBX SPI Reg (0x%02x): 0x%08x"
- ) % addr % regs.get_reg(addr) << std::endl;
- self_base->get_iface()->write_spi(
- unit, spi_config_t::EDGE_RISE,
- regs.get_reg(addr), 32
- );
+ if (unit == dboard_iface::UNIT_RX) {
+ lo_iface->set_output_power((actual_freq == wbx_rx_lo_5dbm.clip(actual_freq)) ?
+ adf435x_iface::OUTPUT_POWER_5DBM : adf435x_iface::OUTPUT_POWER_2DBM);
+ } else {
+ lo_iface->set_output_power((actual_freq == wbx_tx_lo_5dbm.clip(actual_freq)) ?
+ adf435x_iface::OUTPUT_POWER_5DBM : adf435x_iface::OUTPUT_POWER_M1DBM);
}
- //return the actual frequency
- UHD_LOGV(often) << boost::format(
- "WBX tune: actual frequency %f MHz"
- ) % (actual_freq/1e6) << std::endl;
+ //Write to hardware
+ lo_iface->commit();
+
return actual_freq;
}
diff --git a/host/lib/usrp/dboard/db_wbx_version3.cpp b/host/lib/usrp/dboard/db_wbx_version3.cpp
index a559a7b14..2add8d25d 100644
--- a/host/lib/usrp/dboard/db_wbx_version3.cpp
+++ b/host/lib/usrp/dboard/db_wbx_version3.cpp
@@ -16,8 +16,6 @@
//
#include "db_wbx_common.hpp"
-#include "adf4350_regs.hpp"
-#include "../common/adf435x_common.hpp"
#include <uhd/utils/log.hpp>
#include <uhd/types/dict.hpp>
#include <uhd/types/ranges.hpp>
@@ -62,7 +60,7 @@ static int tx_pga0_gain_to_iobits(double &gain){
(attn_code & 8 ? 0 : TX_ATTN_8) |
(attn_code & 4 ? 0 : TX_ATTN_4) |
(attn_code & 2 ? 0 : TX_ATTN_2) |
- (attn_code & 1 ? 0 : TX_ATTN_1)
+ (attn_code & 1 ? 0 : TX_ATTN_1)
) & TX_ATTN_MASK;
UHD_LOGV(often) << boost::format(
@@ -82,13 +80,15 @@ static int tx_pga0_gain_to_iobits(double &gain){
wbx_base::wbx_version3::wbx_version3(wbx_base *_self_wbx_base) {
//register our handle on the primary wbx_base instance
self_base = _self_wbx_base;
+ _txlo = adf435x_iface::make_adf4350(boost::bind(&wbx_base::wbx_versionx::write_lo_regs, this, dboard_iface::UNIT_TX, _1));
+ _rxlo = adf435x_iface::make_adf4350(boost::bind(&wbx_base::wbx_versionx::write_lo_regs, this, dboard_iface::UNIT_RX, _1));
////////////////////////////////////////////////////////////////////
// Register RX properties
////////////////////////////////////////////////////////////////////
this->get_rx_subtree()->create<std::string>("name").set("WBXv3 RX");
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&wbx_base::wbx_version3::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
+ .set_coercer(boost::bind(&wbx_base::wbx_version3::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
.set((wbx_v3_freq_range.start() + wbx_v3_freq_range.stop())/2.0);
this->get_rx_subtree()->create<meta_range_t>("freq/range").set(wbx_v3_freq_range);
@@ -98,17 +98,17 @@ wbx_base::wbx_version3::wbx_version3(wbx_base *_self_wbx_base) {
this->get_tx_subtree()->create<std::string>("name").set("WBXv3 TX");
BOOST_FOREACH(const std::string &name, wbx_v3_tx_gain_ranges.keys()){
self_base->get_tx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&wbx_base::wbx_version3::set_tx_gain, this, _1, name))
+ .set_coercer(boost::bind(&wbx_base::wbx_version3::set_tx_gain, this, _1, name))
.set(wbx_v3_tx_gain_ranges[name].start());
self_base->get_tx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(wbx_v3_tx_gain_ranges[name]);
}
this->get_tx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&wbx_base::wbx_version3::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
+ .set_coercer(boost::bind(&wbx_base::wbx_version3::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
.set((wbx_v3_freq_range.start() + wbx_v3_freq_range.stop())/2.0);
this->get_tx_subtree()->create<meta_range_t>("freq/range").set(wbx_v3_freq_range);
this->get_tx_subtree()->create<bool>("enabled")
- .subscribe(boost::bind(&wbx_base::wbx_version3::set_tx_enabled, this, _1))
+ .add_coerced_subscriber(boost::bind(&wbx_base::wbx_version3::set_tx_enabled, this, _1))
.set(true); //start enabled
//set attenuator control bits
@@ -129,29 +129,29 @@ wbx_base::wbx_version3::wbx_version3(wbx_base *_self_wbx_base) {
//slip between bursts). set TX gain iobits to min gain (max attenuation)
//when RX_ONLY or IDLE to suppress LO leakage
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_IDLE, v3_tx_mod, \
+ gpio_atr::ATR_REG_IDLE, v3_tx_mod, \
TX_ATTN_MASK | TX_MIXER_DIS | v3_tx_mod);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_RX_ONLY, v3_tx_mod, \
+ gpio_atr::ATR_REG_RX_ONLY, v3_tx_mod, \
TX_ATTN_MASK | TX_MIXER_DIS | v3_tx_mod);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_TX_ONLY, v3_tx_mod, \
+ gpio_atr::ATR_REG_TX_ONLY, v3_tx_mod, \
TX_ATTN_MASK | TX_MIXER_DIS | v3_tx_mod);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_FULL_DUPLEX, v3_tx_mod, \
+ gpio_atr::ATR_REG_FULL_DUPLEX, v3_tx_mod, \
TX_ATTN_MASK | TX_MIXER_DIS | v3_tx_mod);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_IDLE, \
+ gpio_atr::ATR_REG_IDLE, \
RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_TX_ONLY, \
+ gpio_atr::ATR_REG_TX_ONLY, \
RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_RX_ONLY, \
+ gpio_atr::ATR_REG_RX_ONLY, \
RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_FULL_DUPLEX, \
+ gpio_atr::ATR_REG_FULL_DUPLEX, \
RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
}
@@ -181,8 +181,8 @@ double wbx_base::wbx_version3::set_tx_gain(double gain, const std::string &name)
//write the new gain to tx gpio outputs
//Update ATR with gain io_bits, only update for TX_ONLY and FULL_DUPLEX ATR states
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, io_bits, TX_ATTN_MASK);
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, io_bits, TX_ATTN_MASK);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, io_bits, TX_ATTN_MASK);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, io_bits, TX_ATTN_MASK);
}
else UHD_THROW_INVALID_CODE_PATH();
return self_base->_tx_gains[name]; //shadow
@@ -209,111 +209,42 @@ double wbx_base::wbx_version3::set_lo_freq(dboard_iface::unit_t unit, double tar
: self_base->get_tx_subtree();
device_addr_t tune_args = subtree->access<device_addr_t>("tune_args").get();
bool is_int_n = boost::iequals(tune_args.get("mode_n",""), "integer");
+ double reference_freq = self_base->get_iface()->get_clock_rate(unit);
- //map prescaler setting to mininmum integer divider (N) values (pg.18 prescaler)
- static const uhd::dict<int, int> prescaler_to_min_int_div = map_list_of
- (0,23) //adf4350_regs_t::PRESCALER_4_5
- (1,75) //adf4350_regs_t::PRESCALER_8_9
- ;
-
- //map rf divider select output dividers to enums
- static const uhd::dict<int, adf4350_regs_t::rf_divider_select_t> rfdivsel_to_enum = map_list_of
- (1, adf4350_regs_t::RF_DIVIDER_SELECT_DIV1)
- (2, adf4350_regs_t::RF_DIVIDER_SELECT_DIV2)
- (4, adf4350_regs_t::RF_DIVIDER_SELECT_DIV4)
- (8, adf4350_regs_t::RF_DIVIDER_SELECT_DIV8)
- (16, adf4350_regs_t::RF_DIVIDER_SELECT_DIV16)
- ;
+ //Select the LO
+ adf435x_iface::sptr& lo_iface = unit == dboard_iface::UNIT_RX ? _rxlo : _txlo;
+ lo_iface->set_reference_freq(reference_freq);
- double reference_freq = self_base->get_iface()->get_clock_rate(unit);
//The mixer has a divide-by-2 stage on the LO port so the synthesizer
- //frequency must 2x the target frequency. This introduces a 180 degree
- //phase ambiguity
+ //frequency must 2x the target frequency
double synth_target_freq = target_freq * 2;
- adf4350_regs_t::prescaler_t prescaler =
- synth_target_freq > 3e9 ? adf4350_regs_t::PRESCALER_8_9 : adf4350_regs_t::PRESCALER_4_5;
-
- adf435x_tuning_constraints tuning_constraints;
- tuning_constraints.force_frac0 = is_int_n;
- tuning_constraints.band_sel_freq_max = 100e3;
- tuning_constraints.ref_doubler_threshold = 12.5e6;
- tuning_constraints.int_range = uhd::range_t(prescaler_to_min_int_div[prescaler], 4095);
- tuning_constraints.pfd_freq_max = 25e6;
- tuning_constraints.rf_divider_range = uhd::range_t(1, 16);
- //The feedback of the divided frequency must be disabled whenever the target frequency
- //divided by the minimum PFD frequency cannot meet the minimum integer divider (N) value.
- //If it is disabled, additional phase ambiguity will be introduced. With a minimum PFD
- //frequency of 10 MHz, synthesizer frequencies below 230 MHz (LO frequencies below 115 MHz)
- //will have too much ambiguity to synchronize.
- tuning_constraints.feedback_after_divider =
- (int(synth_target_freq / 10e6) >= prescaler_to_min_int_div[prescaler]);
-
- double synth_actual_freq = 0;
- adf435x_tuning_settings tuning_settings = tune_adf435x_synth(
- synth_target_freq, reference_freq, tuning_constraints, synth_actual_freq);
+ //Use 8/9 prescaler for vco_freq > 3 GHz (pg.18 prescaler)
+ lo_iface->set_prescaler(synth_target_freq > 3e9 ?
+ adf435x_iface::PRESCALER_8_9 : adf435x_iface::PRESCALER_4_5);
+
+ //When divider resync is enabled, a 180 deg phase error is introduced when syncing
+ //multiple WBX boards. Switching to fundamental mode works arounds this issue.
+ //TODO: Document why the following has to be true
+ lo_iface->set_feedback_select((target_freq > reference_freq) ?
+ adf435x_iface::FB_SEL_DIVIDED : adf435x_iface::FB_SEL_FUNDAMENTAL);
+
+ double synth_actual_freq = lo_iface->set_frequency(synth_target_freq, is_int_n);
//The mixer has a divide-by-2 stage on the LO port so the synthesizer
//actual_freq must /2 the synth_actual_freq
double actual_freq = synth_actual_freq / 2;
- //load the register values
- adf4350_regs_t regs;
-
- if (unit == dboard_iface::UNIT_RX)
- regs.output_power = (actual_freq == wbx_rx_lo_5dbm.clip(actual_freq)) ? adf4350_regs_t::OUTPUT_POWER_5DBM
- : adf4350_regs_t::OUTPUT_POWER_2DBM;
- else
- regs.output_power = (actual_freq == wbx_tx_lo_5dbm.clip(actual_freq)) ? adf4350_regs_t::OUTPUT_POWER_5DBM
- : adf4350_regs_t::OUTPUT_POWER_M1DBM;
-
- regs.frac_12_bit = tuning_settings.frac_12_bit;
- regs.int_16_bit = tuning_settings.int_16_bit;
- regs.mod_12_bit = tuning_settings.mod_12_bit;
- regs.clock_divider_12_bit = tuning_settings.clock_divider_12_bit;
- regs.feedback_select = tuning_constraints.feedback_after_divider ?
- adf4350_regs_t::FEEDBACK_SELECT_DIVIDED :
- adf4350_regs_t::FEEDBACK_SELECT_FUNDAMENTAL;
- regs.clock_div_mode = tuning_constraints.feedback_after_divider ?
- adf4350_regs_t::CLOCK_DIV_MODE_RESYNC_ENABLE :
- adf4350_regs_t::CLOCK_DIV_MODE_FAST_LOCK;
- regs.prescaler = prescaler;
- regs.r_counter_10_bit = tuning_settings.r_counter_10_bit;
- regs.reference_divide_by_2 = tuning_settings.r_divide_by_2_en ?
- adf4350_regs_t::REFERENCE_DIVIDE_BY_2_ENABLED :
- adf4350_regs_t::REFERENCE_DIVIDE_BY_2_DISABLED;
- regs.reference_doubler = tuning_settings.r_doubler_en ?
- adf4350_regs_t::REFERENCE_DOUBLER_ENABLED :
- adf4350_regs_t::REFERENCE_DOUBLER_DISABLED;
- regs.band_select_clock_div = tuning_settings.band_select_clock_div;
- UHD_ASSERT_THROW(rfdivsel_to_enum.has_key(tuning_settings.rf_divider));
- regs.rf_divider_select = rfdivsel_to_enum[tuning_settings.rf_divider];
- regs.ldf = is_int_n ?
- adf4350_regs_t::LDF_INT_N :
- adf4350_regs_t::LDF_FRAC_N;
-
- //reset the N and R counter
- regs.counter_reset = adf4350_regs_t::COUNTER_RESET_ENABLED;
- self_base->get_iface()->write_spi(unit, spi_config_t::EDGE_RISE, regs.get_reg(2), 32);
- regs.counter_reset = adf4350_regs_t::COUNTER_RESET_DISABLED;
-
- //write the registers
- //correct power-up sequence to write registers (5, 4, 3, 2, 1, 0)
- int addr;
-
- for(addr=5; addr>=0; addr--){
- UHD_LOGV(often) << boost::format(
- "WBX SPI Reg (0x%02x): 0x%08x"
- ) % addr % regs.get_reg(addr) << std::endl;
- self_base->get_iface()->write_spi(
- unit, spi_config_t::EDGE_RISE,
- regs.get_reg(addr), 32
- );
+ if (unit == dboard_iface::UNIT_RX) {
+ lo_iface->set_output_power((actual_freq == wbx_rx_lo_5dbm.clip(actual_freq)) ?
+ adf435x_iface::OUTPUT_POWER_5DBM : adf435x_iface::OUTPUT_POWER_2DBM);
+ } else {
+ lo_iface->set_output_power((actual_freq == wbx_tx_lo_5dbm.clip(actual_freq)) ?
+ adf435x_iface::OUTPUT_POWER_5DBM : adf435x_iface::OUTPUT_POWER_M1DBM);
}
- //return the actual frequency
- UHD_LOGV(often) << boost::format(
- "WBX tune: actual frequency %f MHz"
- ) % (actual_freq/1e6) << std::endl;
+ //Write to hardware
+ lo_iface->commit();
+
return actual_freq;
}
diff --git a/host/lib/usrp/dboard/db_wbx_version4.cpp b/host/lib/usrp/dboard/db_wbx_version4.cpp
index 81cdaefac..dc351af1d 100644
--- a/host/lib/usrp/dboard/db_wbx_version4.cpp
+++ b/host/lib/usrp/dboard/db_wbx_version4.cpp
@@ -16,8 +16,6 @@
//
#include "db_wbx_common.hpp"
-#include "adf4351_regs.hpp"
-#include "../common/adf435x_common.hpp"
#include <uhd/utils/log.hpp>
#include <uhd/types/dict.hpp>
#include <uhd/types/ranges.hpp>
@@ -83,6 +81,8 @@ static int tx_pga0_gain_to_iobits(double &gain){
wbx_base::wbx_version4::wbx_version4(wbx_base *_self_wbx_base) {
//register our handle on the primary wbx_base instance
self_base = _self_wbx_base;
+ _txlo = adf435x_iface::make_adf4351(boost::bind(&wbx_base::wbx_versionx::write_lo_regs, this, dboard_iface::UNIT_TX, _1));
+ _rxlo = adf435x_iface::make_adf4351(boost::bind(&wbx_base::wbx_versionx::write_lo_regs, this, dboard_iface::UNIT_RX, _1));
////////////////////////////////////////////////////////////////////
// Register RX properties
@@ -92,7 +92,7 @@ wbx_base::wbx_version4::wbx_version4(wbx_base *_self_wbx_base) {
if(rx_id == 0x0063) this->get_rx_subtree()->create<std::string>("name").set("WBXv4 RX");
else if(rx_id == 0x0081) this->get_rx_subtree()->create<std::string>("name").set("WBX-120 RX");
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&wbx_base::wbx_version4::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
+ .set_coercer(boost::bind(&wbx_base::wbx_version4::set_lo_freq, this, dboard_iface::UNIT_RX, _1))
.set((wbx_v4_freq_range.start() + wbx_v4_freq_range.stop())/2.0);
this->get_rx_subtree()->create<meta_range_t>("freq/range").set(wbx_v4_freq_range);
@@ -105,17 +105,17 @@ wbx_base::wbx_version4::wbx_version4(wbx_base *_self_wbx_base) {
else if(rx_id == 0x0081) this->get_tx_subtree()->create<std::string>("name").set("WBX-120 TX");
BOOST_FOREACH(const std::string &name, wbx_v4_tx_gain_ranges.keys()){
self_base->get_tx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&wbx_base::wbx_version4::set_tx_gain, this, _1, name))
+ .set_coercer(boost::bind(&wbx_base::wbx_version4::set_tx_gain, this, _1, name))
.set(wbx_v4_tx_gain_ranges[name].start());
self_base->get_tx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(wbx_v4_tx_gain_ranges[name]);
}
this->get_tx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&wbx_base::wbx_version4::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
+ .set_coercer(boost::bind(&wbx_base::wbx_version4::set_lo_freq, this, dboard_iface::UNIT_TX, _1))
.set((wbx_v4_freq_range.start() + wbx_v4_freq_range.stop())/2.0);
this->get_tx_subtree()->create<meta_range_t>("freq/range").set(wbx_v4_freq_range);
this->get_tx_subtree()->create<bool>("enabled")
- .subscribe(boost::bind(&wbx_base::wbx_version4::set_tx_enabled, this, _1))
+ .add_coerced_subscriber(boost::bind(&wbx_base::wbx_version4::set_tx_enabled, this, _1))
.set(true); //start enabled
//set attenuator control bits
@@ -136,29 +136,29 @@ wbx_base::wbx_version4::wbx_version4(wbx_base *_self_wbx_base) {
//between bursts) set TX gain iobits to min gain (max attenuation) when
//RX_ONLY or IDLE to suppress LO leakage
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_IDLE, v4_tx_mod, \
+ gpio_atr::ATR_REG_IDLE, v4_tx_mod, \
TX_ATTN_MASK | TX_MIXER_DIS | v4_tx_mod);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_RX_ONLY, v4_tx_mod, \
+ gpio_atr::ATR_REG_RX_ONLY, v4_tx_mod, \
TX_ATTN_MASK | TX_MIXER_DIS | v4_tx_mod);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_TX_ONLY, v4_tx_mod, \
+ gpio_atr::ATR_REG_TX_ONLY, v4_tx_mod, \
TX_ATTN_MASK | TX_MIXER_DIS | v4_tx_mod);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, \
- dboard_iface::ATR_REG_FULL_DUPLEX, v4_tx_mod, \
+ gpio_atr::ATR_REG_FULL_DUPLEX, v4_tx_mod, \
TX_ATTN_MASK | TX_MIXER_DIS | v4_tx_mod);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_IDLE, \
+ gpio_atr::ATR_REG_IDLE, \
RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_TX_ONLY, \
+ gpio_atr::ATR_REG_TX_ONLY, \
RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_RX_ONLY, \
+ gpio_atr::ATR_REG_RX_ONLY, \
RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, \
- dboard_iface::ATR_REG_FULL_DUPLEX, \
+ gpio_atr::ATR_REG_FULL_DUPLEX, \
RX_MIXER_ENB, RX_MIXER_DIS | RX_MIXER_ENB);
}
@@ -188,8 +188,8 @@ double wbx_base::wbx_version4::set_tx_gain(double gain, const std::string &name)
//write the new gain to tx gpio outputs
//Update ATR with gain io_bits, only update for TX_ONLY and FULL_DUPLEX ATR states
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, io_bits, TX_ATTN_MASK);
- self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, io_bits, TX_ATTN_MASK);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, io_bits, TX_ATTN_MASK);
+ self_base->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, io_bits, TX_ATTN_MASK);
}
else UHD_THROW_INVALID_CODE_PATH();
@@ -217,116 +217,46 @@ double wbx_base::wbx_version4::set_lo_freq(dboard_iface::unit_t unit, double tar
: self_base->get_tx_subtree();
device_addr_t tune_args = subtree->access<device_addr_t>("tune_args").get();
bool is_int_n = boost::iequals(tune_args.get("mode_n",""), "integer");
+ double reference_freq = self_base->get_iface()->get_clock_rate(unit);
- //map prescaler setting to mininmum integer divider (N) values (pg.18 prescaler)
- static const uhd::dict<int, int> prescaler_to_min_int_div = map_list_of
- (adf4351_regs_t::PRESCALER_4_5, 23)
- (adf4351_regs_t::PRESCALER_8_9, 75)
- ;
-
- //map rf divider select output dividers to enums
- static const uhd::dict<int, adf4351_regs_t::rf_divider_select_t> rfdivsel_to_enum = map_list_of
- (1, adf4351_regs_t::RF_DIVIDER_SELECT_DIV1)
- (2, adf4351_regs_t::RF_DIVIDER_SELECT_DIV2)
- (4, adf4351_regs_t::RF_DIVIDER_SELECT_DIV4)
- (8, adf4351_regs_t::RF_DIVIDER_SELECT_DIV8)
- (16, adf4351_regs_t::RF_DIVIDER_SELECT_DIV16)
- (32, adf4351_regs_t::RF_DIVIDER_SELECT_DIV32)
- (64, adf4351_regs_t::RF_DIVIDER_SELECT_DIV64)
- ;
+ //Select the LO
+ adf435x_iface::sptr& lo_iface = unit == dboard_iface::UNIT_RX ? _rxlo : _txlo;
+ lo_iface->set_reference_freq(reference_freq);
- double reference_freq = self_base->get_iface()->get_clock_rate(unit);
//The mixer has a divide-by-2 stage on the LO port so the synthesizer
//frequency must 2x the target frequency. This introduces a 180 degree phase
//ambiguity when trying to synchronize the phase of multiple boards.
double synth_target_freq = target_freq * 2;
- adf4351_regs_t::prescaler_t prescaler =
- synth_target_freq > 3.6e9 ? adf4351_regs_t::PRESCALER_8_9 : adf4351_regs_t::PRESCALER_4_5;
+ //Use 8/9 prescaler for vco_freq > 3 GHz (pg.18 prescaler)
+ lo_iface->set_prescaler(synth_target_freq > 3.6e9 ?
+ adf435x_iface::PRESCALER_8_9 : adf435x_iface::PRESCALER_4_5);
- adf435x_tuning_constraints tuning_constraints;
- tuning_constraints.force_frac0 = is_int_n;
- tuning_constraints.band_sel_freq_max = 100e3;
- tuning_constraints.ref_doubler_threshold = 12.5e6;
- tuning_constraints.int_range = uhd::range_t(prescaler_to_min_int_div[prescaler], 4095);
- tuning_constraints.pfd_freq_max = 25e6;
- tuning_constraints.rf_divider_range = uhd::range_t(1, 64);
//The feedback of the divided frequency must be disabled whenever the target frequency
//divided by the minimum PFD frequency cannot meet the minimum integer divider (N) value.
//If it is disabled, additional phase ambiguity will be introduced. With a minimum PFD
//frequency of 10 MHz, synthesizer frequencies below 230 MHz (LO frequencies below 115 MHz)
//will have too much ambiguity to synchronize.
- tuning_constraints.feedback_after_divider =
- (int(synth_target_freq / 10e6) >= prescaler_to_min_int_div[prescaler]);
+ lo_iface->set_feedback_select(
+ (int(synth_target_freq / 10e6) >= lo_iface->get_int_range().start() ?
+ adf435x_iface::FB_SEL_DIVIDED : adf435x_iface::FB_SEL_FUNDAMENTAL));
- double synth_actual_freq = 0;
- adf435x_tuning_settings tuning_settings = tune_adf435x_synth(
- synth_target_freq, reference_freq, tuning_constraints, synth_actual_freq);
+ double synth_actual_freq = lo_iface->set_frequency(synth_target_freq, is_int_n);
//The mixer has a divide-by-2 stage on the LO port so the synthesizer
//actual_freq must /2 the synth_actual_freq
double actual_freq = synth_actual_freq / 2;
- //load the register values
- adf4351_regs_t regs;
-
- if (unit == dboard_iface::UNIT_RX)
- regs.output_power = (actual_freq == wbx_rx_lo_5dbm.clip(actual_freq)) ? adf4351_regs_t::OUTPUT_POWER_5DBM
- : adf4351_regs_t::OUTPUT_POWER_2DBM;
- else
- regs.output_power = (actual_freq == wbx_tx_lo_5dbm.clip(actual_freq)) ? adf4351_regs_t::OUTPUT_POWER_5DBM
- : adf4351_regs_t::OUTPUT_POWER_M1DBM;
-
- regs.frac_12_bit = tuning_settings.frac_12_bit;
- regs.int_16_bit = tuning_settings.int_16_bit;
- regs.mod_12_bit = tuning_settings.mod_12_bit;
- regs.clock_divider_12_bit = tuning_settings.clock_divider_12_bit;
- regs.feedback_select = tuning_constraints.feedback_after_divider ?
- adf4351_regs_t::FEEDBACK_SELECT_DIVIDED :
- adf4351_regs_t::FEEDBACK_SELECT_FUNDAMENTAL;
- regs.clock_div_mode = tuning_constraints.feedback_after_divider ?
- adf4351_regs_t::CLOCK_DIV_MODE_RESYNC_ENABLE :
- adf4351_regs_t::CLOCK_DIV_MODE_FAST_LOCK;
- regs.prescaler = prescaler;
- regs.r_counter_10_bit = tuning_settings.r_counter_10_bit;
- regs.reference_divide_by_2 = tuning_settings.r_divide_by_2_en ?
- adf4351_regs_t::REFERENCE_DIVIDE_BY_2_ENABLED :
- adf4351_regs_t::REFERENCE_DIVIDE_BY_2_DISABLED;
- regs.reference_doubler = tuning_settings.r_doubler_en ?
- adf4351_regs_t::REFERENCE_DOUBLER_ENABLED :
- adf4351_regs_t::REFERENCE_DOUBLER_DISABLED;
- regs.band_select_clock_div = tuning_settings.band_select_clock_div;
- UHD_ASSERT_THROW(rfdivsel_to_enum.has_key(tuning_settings.rf_divider));
- regs.rf_divider_select = rfdivsel_to_enum[tuning_settings.rf_divider];
- regs.ldf = is_int_n ?
- adf4351_regs_t::LDF_INT_N :
- adf4351_regs_t::LDF_FRAC_N;
-
- //reset the N and R counter
- regs.counter_reset = adf4351_regs_t::COUNTER_RESET_ENABLED;
- self_base->get_iface()->write_spi(unit, spi_config_t::EDGE_RISE, regs.get_reg(2), 32);
- regs.counter_reset = adf4351_regs_t::COUNTER_RESET_DISABLED;
-
- //write the registers
- //correct power-up sequence to write registers (5, 4, 3, 2, 1, 0)
- int addr;
-
- boost::uint16_t rx_id = self_base->get_rx_id().to_uint16();
- std::string board_name = (rx_id == 0x0081) ? "WBX-120" : "WBX";
- for(addr=5; addr>=0; addr--){
- UHD_LOGV(often) << boost::format(
- "%s SPI Reg (0x%02x): 0x%08x"
- ) % board_name.c_str() % addr % regs.get_reg(addr) << std::endl;
- self_base->get_iface()->write_spi(
- unit, spi_config_t::EDGE_RISE,
- regs.get_reg(addr), 32
- );
+ if (unit == dboard_iface::UNIT_RX) {
+ lo_iface->set_output_power((actual_freq == wbx_rx_lo_5dbm.clip(actual_freq)) ?
+ adf435x_iface::OUTPUT_POWER_5DBM : adf435x_iface::OUTPUT_POWER_2DBM);
+ } else {
+ lo_iface->set_output_power((actual_freq == wbx_tx_lo_5dbm.clip(actual_freq)) ?
+ adf435x_iface::OUTPUT_POWER_5DBM : adf435x_iface::OUTPUT_POWER_M1DBM);
}
- //return the actual frequency
- UHD_LOGV(often) << boost::format(
- "%s tune: actual frequency %f MHz"
- ) % board_name.c_str() % (actual_freq/1e6) << std::endl;
+ //Write to hardware
+ lo_iface->commit();
return actual_freq;
}
diff --git a/host/lib/usrp/dboard/db_xcvr2450.cpp b/host/lib/usrp/dboard/db_xcvr2450.cpp
index 50c67991a..4a3f69f69 100644
--- a/host/lib/usrp/dboard/db_xcvr2450.cpp
+++ b/host/lib/usrp/dboard/db_xcvr2450.cpp
@@ -112,7 +112,7 @@ static const uhd::dict<std::string, gain_range_t> xcvr_rx_gain_ranges = map_list
class xcvr2450 : public xcvr_dboard_base{
public:
xcvr2450(ctor_args_t args);
- ~xcvr2450(void);
+ virtual ~xcvr2450(void);
private:
double _lo_freq;
@@ -231,23 +231,23 @@ xcvr2450::xcvr2450(ctor_args_t args) : xcvr_dboard_base(args){
this->get_rx_subtree()->create<std::string>("name")
.set("XCVR2450 RX");
this->get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&xcvr2450::get_locked, this));
+ .set_publisher(boost::bind(&xcvr2450::get_locked, this));
this->get_rx_subtree()->create<sensor_value_t>("sensors/rssi")
- .publish(boost::bind(&xcvr2450::get_rssi, this));
+ .set_publisher(boost::bind(&xcvr2450::get_rssi, this));
BOOST_FOREACH(const std::string &name, xcvr_rx_gain_ranges.keys()){
this->get_rx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&xcvr2450::set_rx_gain, this, _1, name))
+ .set_coercer(boost::bind(&xcvr2450::set_rx_gain, this, _1, name))
.set(xcvr_rx_gain_ranges[name].start());
this->get_rx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(xcvr_rx_gain_ranges[name]);
}
this->get_rx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&xcvr2450::set_lo_freq, this, _1))
+ .set_coercer(boost::bind(&xcvr2450::set_lo_freq, this, _1))
.set(double(2.45e9));
this->get_rx_subtree()->create<meta_range_t>("freq/range")
.set(xcvr_freq_range);
this->get_rx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&xcvr2450::set_rx_ant, this, _1))
+ .add_coerced_subscriber(boost::bind(&xcvr2450::set_rx_ant, this, _1))
.set(xcvr_antennas.at(0));
this->get_rx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(xcvr_antennas);
@@ -258,7 +258,7 @@ xcvr2450::xcvr2450(ctor_args_t args) : xcvr_dboard_base(args){
this->get_rx_subtree()->create<bool>("use_lo_offset")
.set(false);
this->get_rx_subtree()->create<double>("bandwidth/value")
- .coerce(boost::bind(&xcvr2450::set_rx_bandwidth, this, _1)) //complex bandpass bandwidth
+ .set_coercer(boost::bind(&xcvr2450::set_rx_bandwidth, this, _1)) //complex bandpass bandwidth
.set(2.0*_rx_bandwidth); //_rx_bandwidth in lowpass, convert to complex bandpass
this->get_rx_subtree()->create<meta_range_t>("bandwidth/range")
.set(xcvr_rx_bandwidth_range);
@@ -269,21 +269,21 @@ xcvr2450::xcvr2450(ctor_args_t args) : xcvr_dboard_base(args){
this->get_tx_subtree()->create<std::string>("name")
.set("XCVR2450 TX");
this->get_tx_subtree()->create<sensor_value_t>("sensors/lo_locked")
- .publish(boost::bind(&xcvr2450::get_locked, this));
+ .set_publisher(boost::bind(&xcvr2450::get_locked, this));
BOOST_FOREACH(const std::string &name, xcvr_tx_gain_ranges.keys()){
this->get_tx_subtree()->create<double>("gains/"+name+"/value")
- .coerce(boost::bind(&xcvr2450::set_tx_gain, this, _1, name))
+ .set_coercer(boost::bind(&xcvr2450::set_tx_gain, this, _1, name))
.set(xcvr_tx_gain_ranges[name].start());
this->get_tx_subtree()->create<meta_range_t>("gains/"+name+"/range")
.set(xcvr_tx_gain_ranges[name]);
}
this->get_tx_subtree()->create<double>("freq/value")
- .coerce(boost::bind(&xcvr2450::set_lo_freq, this, _1))
+ .set_coercer(boost::bind(&xcvr2450::set_lo_freq, this, _1))
.set(double(2.45e9));
this->get_tx_subtree()->create<meta_range_t>("freq/range")
.set(xcvr_freq_range);
this->get_tx_subtree()->create<std::string>("antenna/value")
- .subscribe(boost::bind(&xcvr2450::set_tx_ant, this, _1))
+ .add_coerced_subscriber(boost::bind(&xcvr2450::set_tx_ant, this, _1))
.set(xcvr_antennas.at(1));
this->get_tx_subtree()->create<std::vector<std::string> >("antenna/options")
.set(xcvr_antennas);
@@ -294,7 +294,7 @@ xcvr2450::xcvr2450(ctor_args_t args) : xcvr_dboard_base(args){
this->get_tx_subtree()->create<bool>("use_lo_offset")
.set(false);
this->get_tx_subtree()->create<double>("bandwidth/value")
- .coerce(boost::bind(&xcvr2450::set_tx_bandwidth, this, _1)) //complex bandpass bandwidth
+ .set_coercer(boost::bind(&xcvr2450::set_tx_bandwidth, this, _1)) //complex bandpass bandwidth
.set(2.0*_tx_bandwidth); //_tx_bandwidth in lowpass, convert to complex bandpass
this->get_tx_subtree()->create<meta_range_t>("bandwidth/range")
.set(xcvr_tx_bandwidth_range);
@@ -315,12 +315,12 @@ xcvr2450::~xcvr2450(void){
void xcvr2450::spi_reset(void){
//spi reset mode: global enable = off, tx and rx enable = on
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_IDLE, TX_ENB_TXIO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_IDLE, RX_ENB_RXIO | POWER_DOWN_RXIO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_IDLE, TX_ENB_TXIO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_IDLE, RX_ENB_RXIO | POWER_DOWN_RXIO);
boost::this_thread::sleep(boost::posix_time::milliseconds(10));
//take it back out of spi reset mode and wait a bit
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_IDLE, RX_DIS_RXIO | POWER_UP_RXIO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_IDLE, RX_DIS_RXIO | POWER_UP_RXIO);
boost::this_thread::sleep(boost::posix_time::milliseconds(10));
}
@@ -337,16 +337,16 @@ void xcvr2450::update_atr(void){
int ad9515div = (_ad9515div == 3)? AD9515DIV_3_TXIO : AD9515DIV_2_TXIO;
//set the tx registers
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_IDLE, band_sel | ad9515div | TX_DIS_TXIO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_RX_ONLY, band_sel | ad9515div | TX_DIS_TXIO | rx_ant_sel);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_TX_ONLY, band_sel | ad9515div | TX_ENB_TXIO | tx_ant_sel);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, dboard_iface::ATR_REG_FULL_DUPLEX, band_sel | ad9515div | TX_ENB_TXIO | xx_ant_sel);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_IDLE, band_sel | ad9515div | TX_DIS_TXIO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_RX_ONLY, band_sel | ad9515div | TX_DIS_TXIO | rx_ant_sel);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_TX_ONLY, band_sel | ad9515div | TX_ENB_TXIO | tx_ant_sel);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_TX, gpio_atr::ATR_REG_FULL_DUPLEX, band_sel | ad9515div | TX_ENB_TXIO | xx_ant_sel);
//set the rx registers
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_IDLE, POWER_UP_RXIO | RX_DIS_RXIO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_RX_ONLY, POWER_UP_RXIO | RX_ENB_RXIO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, POWER_UP_RXIO | RX_DIS_RXIO);
- this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, POWER_UP_RXIO | RX_DIS_RXIO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_IDLE, POWER_UP_RXIO | RX_DIS_RXIO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_RX_ONLY, POWER_UP_RXIO | RX_ENB_RXIO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_TX_ONLY, POWER_UP_RXIO | RX_DIS_RXIO);
+ this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, gpio_atr::ATR_REG_FULL_DUPLEX, POWER_UP_RXIO | RX_DIS_RXIO);
}
/***********************************************************************
diff --git a/host/lib/usrp/dboard/twinrx/table_to_cpp.py b/host/lib/usrp/dboard/twinrx/table_to_cpp.py
new file mode 100644
index 000000000..ea0e3bf66
--- /dev/null
+++ b/host/lib/usrp/dboard/twinrx/table_to_cpp.py
@@ -0,0 +1,26 @@
+#!/usr/bin/python
+
+import sys
+
+if len(sys.argv) != 3:
+ print 'Usage: ' + sys.argv[0] + ' <table filename> <table name>'
+ sys.exit(1)
+
+with open(sys.argv[1], 'rb') as f:
+ print ('static const std::vector<twinrx_gain_config_t> ' +
+ sys.argv[2] + ' = boost::assign::list_of')
+ i = -1
+ for line in f.readlines():
+ if (i != -1):
+ row = line.strip().split(',')
+ print (' ( twinrx_gain_config_t( %s, %6.1f, %s, %s, %s, %s ) )' % (
+ str(i).rjust(5),
+ float(row[0].rjust(6)),
+ row[1].rjust(6), row[2].rjust(6),
+ ('true' if int(row[3]) == 1 else 'false').rjust(5),
+ ('true' if int(row[4]) == 1 else 'false').rjust(5)))
+ else:
+ print (' // %s, %s, %s, %s, %s, %s' % (
+ 'Index', ' Gain', 'Atten1', 'Atten2', ' Amp1', ' Amp2'))
+ i += 1
+ print ';'
diff --git a/host/lib/usrp/dboard/twinrx/twinrx_ctrl.cpp b/host/lib/usrp/dboard/twinrx/twinrx_ctrl.cpp
new file mode 100644
index 000000000..172992f0a
--- /dev/null
+++ b/host/lib/usrp/dboard/twinrx/twinrx_ctrl.cpp
@@ -0,0 +1,643 @@
+//
+// Copyright 2015 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#include <boost/thread/thread.hpp>
+#include <uhd/utils/math.hpp>
+#include <uhd/utils/safe_call.hpp>
+#include "twinrx_ctrl.hpp"
+#include "adf435x.hpp"
+#include "adf5355.hpp"
+
+using namespace uhd;
+using namespace usrp;
+using namespace dboard::twinrx;
+typedef twinrx_cpld_regmap rm;
+
+static boost::uint32_t bool2bin(bool x) { return x ? 1 : 0; }
+
+static const double TWINRX_DESIRED_REFERENCE_FREQ = 50e6;
+
+class twinrx_ctrl_impl : public twinrx_ctrl {
+public:
+ twinrx_ctrl_impl(
+ dboard_iface::sptr db_iface,
+ twinrx_gpio::sptr gpio_iface,
+ twinrx_cpld_regmap::sptr cpld_regmap
+ ) : _db_iface(db_iface), _gpio_iface(gpio_iface), _cpld_regs(cpld_regmap)
+ {
+
+ //Turn on switcher and wait for power good
+ _gpio_iface->set_field(twinrx_gpio::FIELD_SWPS_EN, 1);
+ size_t timeout_ms = 100;
+ while (_gpio_iface->get_field(twinrx_gpio::FIELD_SWPS_PWR_GOOD) == 0) {
+ boost::this_thread::sleep(boost::posix_time::microsec(1000));
+ if (--timeout_ms == 0) {
+ throw uhd::runtime_error("power supply failure");
+ }
+ }
+ //Initialize synthesizer objects
+ _lo1_iface[size_t(CH1)] = adf5355_iface::make(
+ boost::bind(&twinrx_ctrl_impl::_write_lo_spi, this, dboard_iface::UNIT_TX, _1));
+ _lo1_iface[size_t(CH2)] = adf5355_iface::make(
+ boost::bind(&twinrx_ctrl_impl::_write_lo_spi, this, dboard_iface::UNIT_TX, _1));
+
+ _lo2_iface[size_t(CH1)] = adf435x_iface::make_adf4351(
+ boost::bind(&twinrx_ctrl_impl::_write_lo_spi, this, dboard_iface::UNIT_RX, _1));
+ _lo2_iface[size_t(CH2)] = adf435x_iface::make_adf4351(
+ boost::bind(&twinrx_ctrl_impl::_write_lo_spi, this, dboard_iface::UNIT_RX, _1));
+
+ // Assert synthesizer chip enables
+ _gpio_iface->set_field(twinrx_gpio::FIELD_LO1_CE_CH1, 1);
+ _gpio_iface->set_field(twinrx_gpio::FIELD_LO1_CE_CH2, 1);
+ _gpio_iface->set_field(twinrx_gpio::FIELD_LO2_CE_CH1, 1);
+ _gpio_iface->set_field(twinrx_gpio::FIELD_LO2_CE_CH2, 1);
+
+ //Initialize default state
+ set_chan_enabled(BOTH, false, false);
+ set_preamp1(BOTH, PREAMP_BYPASS, false);
+ set_preamp2(BOTH, false, false);
+ set_lb_preamp_preselector(BOTH, false, false);
+ set_signal_path(BOTH, PATH_LOWBAND, false);
+ set_lb_preselector(BOTH, PRESEL_PATH3, false);
+ set_hb_preselector(BOTH, PRESEL_PATH1, false);
+ set_input_atten(BOTH, 31, false);
+ set_lb_atten(BOTH, 31, false);
+ set_hb_atten(BOTH, 31, false);
+ set_lo1_source(BOTH, LO_INTERNAL, false);
+ set_lo2_source(BOTH, LO_INTERNAL, false);
+ set_lo1_export_source(LO_EXPORT_DISABLED, false);
+ set_lo2_export_source(LO_EXPORT_DISABLED, false);
+ set_antenna_mapping(ANTX_NATIVE, false);
+ set_crossover_cal_mode(CAL_DISABLED, false);
+ commit();
+
+ //Initialize clocks and LO
+ bool found_rate = false;
+ BOOST_FOREACH(double rate, _db_iface->get_clock_rates(dboard_iface::UNIT_TX)) {
+ found_rate |= uhd::math::frequencies_are_equal(rate, TWINRX_DESIRED_REFERENCE_FREQ);
+ }
+ BOOST_FOREACH(double rate, _db_iface->get_clock_rates(dboard_iface::UNIT_RX)) {
+ found_rate |= uhd::math::frequencies_are_equal(rate, TWINRX_DESIRED_REFERENCE_FREQ);
+ }
+ if (not found_rate) {
+ throw uhd::runtime_error("TwinRX not supported on this motherboard");
+ }
+ _db_iface->set_clock_rate(dboard_iface::UNIT_TX, TWINRX_DESIRED_REFERENCE_FREQ);
+ _db_iface->set_clock_rate(dboard_iface::UNIT_RX, TWINRX_DESIRED_REFERENCE_FREQ);
+
+ _db_iface->set_clock_enabled(dboard_iface::UNIT_TX, true);
+ _db_iface->set_clock_enabled(dboard_iface::UNIT_RX, true);
+ for (size_t i = 0; i < NUM_CHANS; i++) {
+ _config_lo1_route(i==0?LO_CONFIG_CH1:LO_CONFIG_CH2);
+ _config_lo2_route(i==0?LO_CONFIG_CH1:LO_CONFIG_CH2);
+ _lo1_iface[i]->set_output_power(adf5355_iface::OUTPUT_POWER_5DBM);
+ _lo1_iface[i]->set_reference_freq(TWINRX_DESIRED_REFERENCE_FREQ);
+ _lo1_iface[i]->set_muxout_mode(adf5355_iface::MUXOUT_DLD);
+ _lo1_iface[i]->set_frequency(3e9, 1.0e3);
+ _lo2_iface[i]->set_feedback_select(adf435x_iface::FB_SEL_DIVIDED);
+ _lo2_iface[i]->set_output_power(adf435x_iface::OUTPUT_POWER_5DBM);
+ _lo2_iface[i]->set_reference_freq(TWINRX_DESIRED_REFERENCE_FREQ);
+ _lo2_iface[i]->set_muxout_mode(adf435x_iface::MUXOUT_DLD);
+ _lo1_iface[i]->commit();
+ _lo2_iface[i]->commit();
+ }
+ _config_lo1_route(LO_CONFIG_NONE);
+ _config_lo2_route(LO_CONFIG_NONE);
+ }
+
+ ~twinrx_ctrl_impl()
+ {
+ UHD_SAFE_CALL(
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ _gpio_iface->set_field(twinrx_gpio::FIELD_SWPS_EN, 0);
+ )
+ }
+
+ void commit()
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ _commit();
+ }
+
+ void set_chan_enabled(channel_t ch, bool enabled, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf1_reg1.set(rm::rf1_reg1_t::AMP_LO1_EN_CH1, bool2bin(enabled));
+ _cpld_regs->if0_reg3.set(rm::if0_reg3_t::IF1_IF2_EN_CH1, bool2bin(enabled));
+ _cpld_regs->if0_reg0.set(rm::if0_reg0_t::AMP_LO2_EN_CH1, bool2bin(enabled));
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf1_reg5.set(rm::rf1_reg5_t::AMP_LO1_EN_CH2, bool2bin(enabled));
+ _cpld_regs->if0_reg4.set(rm::if0_reg4_t::IF1_IF2_EN_CH2, bool2bin(enabled));
+ _cpld_regs->if0_reg0.set(rm::if0_reg0_t::AMP_LO2_EN_CH2, bool2bin(enabled));
+ }
+ if (commit) _commit();
+ }
+
+ void set_preamp1(channel_t ch, preamp_state_t value, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf0_reg1.set(rm::rf0_reg1_t::SWPA1_CTL_CH1, bool2bin(value==PREAMP_HIGHBAND));
+ _cpld_regs->rf2_reg2.set(rm::rf2_reg2_t::SWPA2_CTRL_CH1, bool2bin(value==PREAMP_BYPASS));
+ _cpld_regs->rf0_reg1.set(rm::rf0_reg1_t::HB_PREAMP_EN_CH1, bool2bin(value==PREAMP_HIGHBAND));
+ _cpld_regs->rf0_reg1.set(rm::rf0_reg1_t::LB_PREAMP_EN_CH1, bool2bin(value==PREAMP_LOWBAND));
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf0_reg7.set(rm::rf0_reg7_t::SWPA1_CTRL_CH2, bool2bin(value==PREAMP_HIGHBAND));
+ _cpld_regs->rf2_reg5.set(rm::rf2_reg5_t::SWPA2_CTRL_CH2, bool2bin(value==PREAMP_BYPASS));
+ _cpld_regs->rf0_reg5.set(rm::rf0_reg5_t::HB_PREAMP_EN_CH2, bool2bin(value==PREAMP_HIGHBAND));
+ _cpld_regs->rf2_reg6.set(rm::rf2_reg6_t::LB_PREAMP_EN_CH2, bool2bin(value==PREAMP_LOWBAND));
+ }
+ if (commit) _commit();
+ }
+
+ void set_preamp2(channel_t ch, bool enabled, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf2_reg7.set(rm::rf2_reg7_t::SWPA4_CTRL_CH1, bool2bin(not enabled));
+ _cpld_regs->rf2_reg3.set(rm::rf2_reg3_t::PREAMP2_EN_CH1, bool2bin(enabled));
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf0_reg6.set(rm::rf0_reg6_t::SWPA4_CTRL_CH2, bool2bin(not enabled));
+ _cpld_regs->rf1_reg6.set(rm::rf1_reg6_t::PREAMP2_EN_CH2, bool2bin(enabled));
+ }
+ if (commit) _commit();
+ }
+
+ void set_lb_preamp_preselector(channel_t ch, bool enabled, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf0_reg7.set(rm::rf0_reg7_t::SWPA3_CTRL_CH1, bool2bin(not enabled));
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf0_reg1.set(rm::rf0_reg1_t::SWPA3_CTRL_CH2, bool2bin(not enabled));
+ }
+ if (commit) _commit();
+ }
+
+ void set_signal_path(channel_t ch, signal_path_t path, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf2_reg2.set(rm::rf2_reg2_t::SW11_CTRL_CH1, bool2bin(path==PATH_LOWBAND));
+ _cpld_regs->rf1_reg2.set(rm::rf1_reg2_t::SW12_CTRL_CH1, bool2bin(path==PATH_LOWBAND));
+ _cpld_regs->rf1_reg6.set(rm::rf1_reg6_t::HB_PRESEL_PGA_EN_CH1, bool2bin(path==PATH_HIGHBAND));
+ _cpld_regs->rf0_reg2.set(rm::rf0_reg2_t::SW6_CTRL_CH1, bool2bin(path==PATH_LOWBAND));
+ _cpld_regs->if0_reg3.set(rm::if0_reg3_t::SW13_CTRL_CH1, bool2bin(path==PATH_LOWBAND));
+ _cpld_regs->if0_reg2.set(rm::if0_reg2_t::AMP_LB_IF1_EN_CH1, bool2bin(path==PATH_LOWBAND));
+ _cpld_regs->if0_reg0.set(rm::if0_reg0_t::AMP_HB_IF1_EN_CH1, bool2bin(path==PATH_HIGHBAND));
+ _cpld_regs->rf1_reg2.set(rm::rf1_reg2_t::AMP_HB_EN_CH1, bool2bin(path==PATH_HIGHBAND));
+ _cpld_regs->rf2_reg2.set(rm::rf2_reg2_t::AMP_LB_EN_CH1, bool2bin(path==PATH_LOWBAND));
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf2_reg7.set(rm::rf2_reg7_t::SW11_CTRL_CH2, bool2bin(path==PATH_LOWBAND));
+ _cpld_regs->rf1_reg7.set(rm::rf1_reg7_t::SW12_CTRL_CH2, bool2bin(path==PATH_LOWBAND));
+ _cpld_regs->rf1_reg2.set(rm::rf1_reg2_t::HB_PRESEL_PGA_EN_CH2, bool2bin(path==PATH_HIGHBAND));
+ _cpld_regs->rf0_reg6.set(rm::rf0_reg6_t::SW6_CTRL_CH2, bool2bin(path==PATH_HIGHBAND));
+ _cpld_regs->if0_reg6.set(rm::if0_reg6_t::SW13_CTRL_CH2, bool2bin(path==PATH_HIGHBAND));
+ _cpld_regs->if0_reg2.set(rm::if0_reg2_t::AMP_LB_IF1_EN_CH2, bool2bin(path==PATH_LOWBAND));
+ _cpld_regs->if0_reg6.set(rm::if0_reg6_t::AMP_HB_IF1_EN_CH2, bool2bin(path==PATH_HIGHBAND));
+ _cpld_regs->rf1_reg7.set(rm::rf1_reg7_t::AMP_HB_EN_CH2, bool2bin(path==PATH_HIGHBAND));
+ _cpld_regs->rf2_reg7.set(rm::rf2_reg7_t::AMP_LB_EN_CH2, bool2bin(path==PATH_LOWBAND));
+ }
+ if (commit) _commit();
+ }
+
+ void set_lb_preselector(channel_t ch, preselector_path_t path, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ boost::uint32_t sw7val = 0, sw8val = 0;
+ switch (path) {
+ case PRESEL_PATH1: sw7val = 3; sw8val = 1; break;
+ case PRESEL_PATH2: sw7val = 2; sw8val = 0; break;
+ case PRESEL_PATH3: sw7val = 0; sw8val = 2; break;
+ case PRESEL_PATH4: sw7val = 1; sw8val = 3; break;
+ default: UHD_THROW_INVALID_CODE_PATH();
+ }
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf0_reg3.set(rm::rf0_reg3_t::SW7_CTRL_CH1, sw7val);
+ _cpld_regs->rf2_reg3.set(rm::rf2_reg3_t::SW8_CTRL_CH1, sw8val);
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf0_reg7.set(rm::rf0_reg7_t::SW7_CTRL_CH2, sw7val);
+ _cpld_regs->rf2_reg7.set(rm::rf2_reg7_t::SW8_CTRL_CH2, sw8val);
+ }
+ if (commit) _commit();
+ }
+
+ void set_hb_preselector(channel_t ch, preselector_path_t path, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ boost::uint32_t sw9ch1val = 0, sw10ch1val = 0, sw9ch2val = 0, sw10ch2val = 0;
+ switch (path) {
+ case PRESEL_PATH1: sw9ch1val = 3; sw10ch1val = 0; sw9ch2val = 0; sw10ch2val = 3; break;
+ case PRESEL_PATH2: sw9ch1val = 1; sw10ch1val = 2; sw9ch2val = 1; sw10ch2val = 1; break;
+ case PRESEL_PATH3: sw9ch1val = 2; sw10ch1val = 1; sw9ch2val = 2; sw10ch2val = 2; break;
+ case PRESEL_PATH4: sw9ch1val = 0; sw10ch1val = 3; sw9ch2val = 3; sw10ch2val = 0; break;
+ default: UHD_THROW_INVALID_CODE_PATH();
+ }
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf0_reg5.set(rm::rf0_reg5_t::SW9_CTRL_CH1, sw9ch1val);
+ _cpld_regs->rf1_reg3.set(rm::rf1_reg3_t::SW10_CTRL_CH1, sw10ch1val);
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf0_reg3.set(rm::rf0_reg3_t::SW9_CTRL_CH2, sw9ch2val);
+ _cpld_regs->rf1_reg7.set(rm::rf1_reg7_t::SW10_CTRL_CH2, sw10ch2val);
+ }
+ if (commit) _commit();
+
+ }
+
+ void set_input_atten(channel_t ch, boost::uint8_t atten, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf0_reg0.set(rm::rf0_reg0_t::ATTEN_IN_CH1, atten&0x1F);
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf0_reg4.set(rm::rf0_reg4_t::ATTEN_IN_CH2, atten&0x1F);
+ }
+ if (commit) _commit();
+ }
+
+ void set_lb_atten(channel_t ch, boost::uint8_t atten, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf2_reg0.set(rm::rf2_reg0_t::ATTEN_LB_CH1, atten&0x1F);
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf2_reg4.set(rm::rf2_reg4_t::ATTEN_LB_CH2, atten&0x1F);
+ }
+ if (commit) _commit();
+ }
+
+ void set_hb_atten(channel_t ch, boost::uint8_t atten, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf1_reg0.set(rm::rf1_reg0_t::ATTEN_HB_CH1, atten&0x1F);
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf1_reg4.set(rm::rf1_reg4_t::ATTEN_HB_CH2, atten&0x1F);
+ }
+ if (commit) _commit();
+ }
+
+ void set_lo1_source(channel_t ch, lo_source_t source, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->rf1_reg5.set(rm::rf1_reg5_t::SW14_CTRL_CH2, bool2bin(source!=LO_COMPANION));
+ _cpld_regs->rf1_reg1.set(rm::rf1_reg1_t::SW15_CTRL_CH1, bool2bin(source==LO_EXTERNAL));
+ _cpld_regs->rf1_reg1.set(rm::rf1_reg1_t::SW16_CTRL_CH1, bool2bin(source!=LO_INTERNAL));
+ _lo1_src[size_t(CH1)] = source;
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->rf1_reg1.set(rm::rf1_reg1_t::SW14_CTRL_CH1, bool2bin(source==LO_COMPANION));
+ _cpld_regs->rf1_reg5.set(rm::rf1_reg5_t::SW15_CTRL_CH2, bool2bin(source!=LO_INTERNAL));
+ _cpld_regs->rf1_reg6.set(rm::rf1_reg6_t::SW16_CTRL_CH2, bool2bin(source==LO_INTERNAL));
+ _lo1_src[size_t(CH2)] = source;
+ }
+ if (commit) _commit();
+ }
+
+ void set_lo2_source(channel_t ch, lo_source_t source, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (ch == CH1 or ch == BOTH) {
+ _cpld_regs->if0_reg0.set(rm::if0_reg0_t::SW19_CTRL_CH2, bool2bin(source==LO_COMPANION));
+ _cpld_regs->if0_reg1.set(rm::if0_reg1_t::SW20_CTRL_CH1, bool2bin(source==LO_COMPANION));
+ _cpld_regs->if0_reg4.set(rm::if0_reg4_t::SW21_CTRL_CH1, bool2bin(source==LO_INTERNAL));
+ _lo2_src[size_t(CH1)] = source;
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _cpld_regs->if0_reg4.set(rm::if0_reg4_t::SW19_CTRL_CH1, bool2bin(source==LO_EXTERNAL));
+ _cpld_regs->if0_reg0.set(rm::if0_reg0_t::SW20_CTRL_CH2, bool2bin(source==LO_INTERNAL||source==LO_DISABLED));
+ _cpld_regs->if0_reg4.set(rm::if0_reg4_t::SW21_CTRL_CH2, bool2bin(source==LO_INTERNAL));
+ _lo2_src[size_t(CH2)] = source;
+ }
+ if (commit) _commit();
+ }
+
+ void set_lo1_export_source(lo_export_source_t source, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ //SW22 may conflict with the cal switch but this attr takes priority and we assume
+ //that the cal switch is disabled (by disabling it!)
+ _set_cal_mode(CAL_DISABLED, source);
+ _cpld_regs->rf1_reg3.set(rm::rf1_reg3_t::SW23_CTRL, bool2bin(source!=LO_CH1_SYNTH));
+ _lo1_export = source;
+
+ if (commit) _commit();
+ }
+
+ void set_lo2_export_source(lo_export_source_t source, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ _cpld_regs->if0_reg7.set(rm::if0_reg7_t::SW24_CTRL_CH2, bool2bin(source==LO_CH2_SYNTH));
+ _cpld_regs->if0_reg4.set(rm::if0_reg4_t::SW25_CTRL, bool2bin(source!=LO_CH1_SYNTH));
+ _cpld_regs->if0_reg3.set(rm::if0_reg3_t::SW24_CTRL_CH1, bool2bin(source!=LO_CH1_SYNTH));
+ _lo2_export = source;
+
+ if (commit) _commit();
+ }
+
+ void set_antenna_mapping(antenna_mapping_t mapping, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+
+ enum switch_path_t { CONNECT, TERM, EXPORT, IMPORT, SWAP };
+ switch_path_t path1, path2;
+
+ switch (mapping) {
+ case ANTX_NATIVE:
+ path1 = CONNECT; path2 = CONNECT; break;
+ case ANT1_SHARED:
+ path1 = EXPORT; path2 = IMPORT; break;
+ case ANT2_SHARED:
+ path1 = IMPORT; path2 = EXPORT; break;
+ case ANTX_SWAPPED:
+ path1 = SWAP; path2 = SWAP; break;
+ default:
+ path1 = TERM; path2 = TERM; break;
+ }
+
+ _cpld_regs->rf0_reg5.set(rm::rf0_reg5_t::SW3_CTRL_CH1, bool2bin(path1==EXPORT||path1==SWAP));
+ _cpld_regs->rf0_reg2.set(rm::rf0_reg2_t::SW4_CTRL_CH1, bool2bin(!(path1==IMPORT||path1==SWAP)));
+ _cpld_regs->rf0_reg2.set(rm::rf0_reg2_t::SW5_CTRL_CH1, bool2bin(path1==CONNECT));
+ _cpld_regs->rf0_reg7.set(rm::rf0_reg7_t::SW3_CTRL_CH2, bool2bin(path2==EXPORT||path2==SWAP));
+ _cpld_regs->rf0_reg6.set(rm::rf0_reg6_t::SW4_CTRL_CH2, bool2bin(path2==IMPORT||path2==SWAP));
+ _cpld_regs->rf0_reg6.set(rm::rf0_reg6_t::SW5_CTRL_CH2, bool2bin(path2==CONNECT));
+
+ if (commit) _commit();
+ }
+
+ void set_crossover_cal_mode(cal_mode_t cal_mode, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ if (_lo1_export == LO_CH1_SYNTH && cal_mode == CAL_CH2) {
+ throw uhd::runtime_error("cannot enable cal crossover on CH2 when LO1 in CH1 is exported");
+ }
+ if (_lo1_export == LO_CH2_SYNTH && cal_mode == CAL_CH1) {
+ throw uhd::runtime_error("cannot enable cal crossover on CH1 when LO1 in CH2 is exported");
+ }
+ _set_cal_mode(cal_mode, _lo1_export);
+
+ if (commit) _commit();
+ }
+
+ double set_lo1_synth_freq(channel_t ch, double freq, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ static const double RESOLUTION = 1e3;
+
+ double coerced_freq = 0.0;
+ if (ch == CH1 or ch == BOTH) {
+ coerced_freq = _lo1_iface[size_t(CH1)]->set_frequency(freq, RESOLUTION, false);
+ _lo1_freq[size_t(CH1)] = tune_freq_t(freq);
+ }
+ if (ch == CH2 or ch == BOTH) {
+ coerced_freq = _lo1_iface[size_t(CH2)]->set_frequency(freq, RESOLUTION, false);
+ _lo1_freq[size_t(CH2)] = tune_freq_t(freq);
+ }
+
+ if (commit) _commit();
+ return coerced_freq;
+ }
+
+ double set_lo2_synth_freq(channel_t ch, double freq, bool commit = true)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ static const double PRESCALER_THRESH = 3.6e9;
+
+ double coerced_freq = 0.0;
+ if (ch == CH1 or ch == BOTH) {
+ _lo2_iface[size_t(CH1)]->set_prescaler(freq > PRESCALER_THRESH ?
+ adf435x_iface::PRESCALER_8_9 : adf435x_iface::PRESCALER_4_5);
+ coerced_freq = _lo2_iface[size_t(CH1)]->set_frequency(freq, false, false);
+ _lo2_freq[size_t(CH1)] = tune_freq_t(freq);
+ }
+ if (ch == CH2 or ch == BOTH) {
+ _lo2_iface[size_t(CH2)]->set_prescaler(freq > PRESCALER_THRESH ?
+ adf435x_iface::PRESCALER_8_9 : adf435x_iface::PRESCALER_4_5);
+ coerced_freq = _lo2_iface[size_t(CH2)]->set_frequency(freq, false, false);
+ _lo2_freq[size_t(CH2)] = tune_freq_t(freq);
+ }
+
+ if (commit) _commit();
+ return coerced_freq;
+ }
+
+ bool read_lo1_locked(channel_t ch)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+
+ bool locked = true;
+ if (ch == CH1 or ch == BOTH) {
+ locked = locked && (_gpio_iface->get_field(twinrx_gpio::FIELD_LO1_MUXOUT_CH1) == 1);
+ }
+ if (ch == CH2 or ch == BOTH) {
+ locked = locked && (_gpio_iface->get_field(twinrx_gpio::FIELD_LO1_MUXOUT_CH2) == 1);
+ }
+ return locked;
+ }
+
+ bool read_lo2_locked(channel_t ch)
+ {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+
+ bool locked = true;
+ if (ch == CH1 or ch == BOTH) {
+ locked = locked && (_gpio_iface->get_field(twinrx_gpio::FIELD_LO2_MUXOUT_CH1) == 1);
+ }
+ if (ch == CH2 or ch == BOTH) {
+ locked = locked && (_gpio_iface->get_field(twinrx_gpio::FIELD_LO2_MUXOUT_CH2) == 1);
+ }
+ return locked;
+ }
+
+private: //Functions
+ void _set_cal_mode(cal_mode_t cal_mode, lo_export_source_t lo1_export_src)
+ {
+ _cpld_regs->rf1_reg1.set(rm::rf1_reg1_t::SW17_CTRL_CH1, bool2bin(cal_mode!=CAL_CH1));
+ _cpld_regs->rf1_reg6.set(rm::rf1_reg6_t::SW17_CTRL_CH2, bool2bin(cal_mode!=CAL_CH2));
+ _cpld_regs->rf1_reg5.set(rm::rf1_reg5_t::SW18_CTRL_CH1, bool2bin(cal_mode!=CAL_CH1));
+ _cpld_regs->rf2_reg3.set(rm::rf2_reg3_t::SW18_CTRL_CH2, bool2bin(cal_mode!=CAL_CH2));
+ _cpld_regs->rf1_reg3.set(rm::rf1_reg3_t::SW22_CTRL_CH1, bool2bin((lo1_export_src!=LO_CH1_SYNTH)||(cal_mode==CAL_CH1)));
+ _cpld_regs->rf1_reg7.set(rm::rf1_reg7_t::SW22_CTRL_CH2, bool2bin((lo1_export_src!=LO_CH2_SYNTH)||(cal_mode==CAL_CH2)));
+ }
+
+ void _config_lo1_route(lo_config_route_t source)
+ {
+ //Route SPI LEs through CPLD (will not assert them)
+ _cpld_regs->rf0_reg2.set(rm::rf0_reg2_t::LO1_LE_CH1, bool2bin(source==LO_CONFIG_CH1||source==LO_CONFIG_BOTH));
+ _cpld_regs->rf0_reg2.set(rm::rf0_reg2_t::LO1_LE_CH2, bool2bin(source==LO_CONFIG_CH2||source==LO_CONFIG_BOTH));
+ _cpld_regs->rf0_reg2.flush();
+ }
+
+ void _config_lo2_route(lo_config_route_t source)
+ {
+ //Route SPI LEs through CPLD (will not assert them)
+ _cpld_regs->if0_reg2.set(rm::if0_reg2_t::LO2_LE_CH1, bool2bin(source==LO_CONFIG_CH1||source==LO_CONFIG_BOTH));
+ _cpld_regs->if0_reg2.set(rm::if0_reg2_t::LO2_LE_CH2, bool2bin(source==LO_CONFIG_CH2||source==LO_CONFIG_BOTH));
+ _cpld_regs->if0_reg2.flush();
+ }
+
+ void _write_lo_spi(dboard_iface::unit_t unit, const std::vector<boost::uint32_t> &regs)
+ {
+ BOOST_FOREACH(boost::uint32_t reg, regs) {
+ spi_config_t spi_config = spi_config_t(spi_config_t::EDGE_RISE);
+ spi_config.use_custom_divider = true;
+ spi_config.divider = 67;
+ _db_iface->write_spi(unit, spi_config, reg, 32);
+ }
+ }
+
+ void _commit()
+ {
+ //Commit everything except the LO synthesizers
+ _cpld_regs->flush();
+
+ // Disable unused LO synthesizers
+ _lo1_enable[size_t(CH1)] = _lo1_src[size_t(CH1)] == LO_INTERNAL ||
+ _lo1_src[size_t(CH2)] == LO_COMPANION ||
+ _lo1_export == LO_CH1_SYNTH;
+
+ _lo1_enable[size_t(CH2)] = _lo1_src[size_t(CH2)] == LO_INTERNAL ||
+ _lo1_src[size_t(CH1)] == LO_COMPANION ||
+ _lo1_export == LO_CH2_SYNTH;
+ _lo2_enable[size_t(CH1)] = _lo2_src[size_t(CH1)] == LO_INTERNAL ||
+ _lo2_src[size_t(CH2)] == LO_COMPANION ||
+ _lo2_export == LO_CH1_SYNTH;
+
+ _lo2_enable[size_t(CH2)] = _lo2_src[size_t(CH2)] == LO_INTERNAL ||
+ _lo2_src[size_t(CH1)] == LO_COMPANION ||
+ _lo2_export == LO_CH2_SYNTH;
+
+ _lo1_iface[size_t(CH1)]->set_output_enable(adf5355_iface::RF_OUTPUT_A, _lo1_enable[size_t(CH1)].get());
+ _lo1_iface[size_t(CH2)]->set_output_enable(adf5355_iface::RF_OUTPUT_A, _lo1_enable[size_t(CH2)].get());
+
+ _lo2_iface[size_t(CH1)]->set_output_enable(adf435x_iface::RF_OUTPUT_A, _lo2_enable[size_t(CH1)].get());
+ _lo2_iface[size_t(CH2)]->set_output_enable(adf435x_iface::RF_OUTPUT_A, _lo2_enable[size_t(CH2)].get());
+
+ //Commit LO1 frequency
+ // Commit Channel 1's settings to both channels simultaneously if the frequency is the same.
+ bool simultaneous_commit_lo1 = _lo1_freq[size_t(CH1)].is_dirty() and
+ _lo1_freq[size_t(CH2)].is_dirty() and
+ _lo1_freq[size_t(CH1)].get() == _lo1_freq[size_t(CH2)].get() and
+ _lo1_enable[size_t(CH1)].get() == _lo1_enable[size_t(CH2)].get();
+
+ if (simultaneous_commit_lo1) {
+ _config_lo1_route(LO_CONFIG_BOTH);
+ //Only commit one of the channels. The route LO_CONFIG_BOTH
+ //will ensure that the LEs for both channels are enabled
+ _lo1_iface[size_t(CH1)]->commit();
+ _lo1_freq[size_t(CH1)].mark_clean();
+ _lo1_freq[size_t(CH2)].mark_clean();
+ _lo1_enable[size_t(CH1)].mark_clean();
+ _lo1_enable[size_t(CH2)].mark_clean();
+ _config_lo1_route(LO_CONFIG_NONE);
+ } else {
+ if (_lo1_freq[size_t(CH1)].is_dirty() || _lo1_enable[size_t(CH1)].is_dirty()) {
+ _config_lo1_route(LO_CONFIG_CH1);
+ _lo1_iface[size_t(CH1)]->commit();
+ _lo1_freq[size_t(CH1)].mark_clean();
+ _lo1_enable[size_t(CH1)].mark_clean();
+ _config_lo1_route(LO_CONFIG_NONE);
+ }
+ if (_lo1_freq[size_t(CH2)].is_dirty() || _lo1_enable[size_t(CH2)].is_dirty()) {
+ _config_lo1_route(LO_CONFIG_CH2);
+ _lo1_iface[size_t(CH2)]->commit();
+ _lo1_freq[size_t(CH2)].mark_clean();
+ _lo1_enable[size_t(CH2)].mark_clean();
+ _config_lo1_route(LO_CONFIG_NONE);
+ }
+ }
+
+ //Commit LO2 frequency
+ bool simultaneous_commit_lo2 = _lo2_freq[size_t(CH1)].is_dirty() and
+ _lo2_freq[size_t(CH2)].is_dirty() and
+ _lo2_freq[size_t(CH1)].get() == _lo2_freq[size_t(CH2)].get() and
+ _lo2_enable[size_t(CH1)].get() == _lo2_enable[size_t(CH2)].get();
+
+ if (simultaneous_commit_lo2) {
+ _config_lo2_route(LO_CONFIG_BOTH);
+ //Only commit one of the channels. The route LO_CONFIG_BOTH
+ //will ensure that the LEs for both channels are enabled
+ _lo2_iface[size_t(CH1)]->commit();
+ _lo2_freq[size_t(CH1)].mark_clean();
+ _lo2_freq[size_t(CH2)].mark_clean();
+ _lo2_enable[size_t(CH1)].mark_clean();
+ _lo2_enable[size_t(CH2)].mark_clean();
+ _config_lo2_route(LO_CONFIG_NONE);
+ } else {
+ if (_lo2_freq[size_t(CH1)].is_dirty() || _lo2_enable[size_t(CH1)].is_dirty()) {
+ _config_lo2_route(LO_CONFIG_CH1);
+ _lo2_iface[size_t(CH1)]->commit();
+ _lo2_freq[size_t(CH1)].mark_clean();
+ _lo2_enable[size_t(CH1)].mark_clean();
+ _config_lo2_route(LO_CONFIG_NONE);
+ }
+ if (_lo2_freq[size_t(CH2)].is_dirty() || _lo2_enable[size_t(CH2)].is_dirty()) {
+ _config_lo2_route(LO_CONFIG_CH2);
+ _lo2_iface[size_t(CH2)]->commit();
+ _lo2_freq[size_t(CH2)].mark_clean();
+ _lo2_enable[size_t(CH2)].mark_clean();
+ _config_lo2_route(LO_CONFIG_NONE);
+ }
+ }
+ }
+
+private: //Members
+ static const size_t NUM_CHANS = 2;
+
+ struct tune_freq_t : public uhd::math::fp_compare::fp_compare_delta<double> {
+ tune_freq_t() : uhd::math::fp_compare::fp_compare_delta<double>(
+ 0.0, uhd::math::FREQ_COMPARISON_DELTA_HZ) {}
+
+ tune_freq_t(double freq) : uhd::math::fp_compare::fp_compare_delta<double>(
+ freq, uhd::math::FREQ_COMPARISON_DELTA_HZ) {}
+ };
+
+ boost::mutex _mutex;
+ dboard_iface::sptr _db_iface;
+ twinrx_gpio::sptr _gpio_iface;
+ twinrx_cpld_regmap::sptr _cpld_regs;
+ adf5355_iface::sptr _lo1_iface[NUM_CHANS];
+ adf435x_iface::sptr _lo2_iface[NUM_CHANS];
+ lo_source_t _lo1_src[NUM_CHANS];
+ lo_source_t _lo2_src[NUM_CHANS];
+ dirty_tracked<tune_freq_t> _lo1_freq[NUM_CHANS];
+ dirty_tracked<tune_freq_t> _lo2_freq[NUM_CHANS];
+ dirty_tracked<bool> _lo1_enable[NUM_CHANS];
+ dirty_tracked<bool> _lo2_enable[NUM_CHANS];
+ lo_export_source_t _lo1_export;
+ lo_export_source_t _lo2_export;
+};
+
+twinrx_ctrl::sptr twinrx_ctrl::make(
+ dboard_iface::sptr db_iface,
+ twinrx_gpio::sptr gpio_iface,
+ twinrx_cpld_regmap::sptr cpld_regmap
+) {
+ return sptr(new twinrx_ctrl_impl(db_iface, gpio_iface, cpld_regmap));
+}
diff --git a/host/lib/usrp/dboard/twinrx/twinrx_ctrl.hpp b/host/lib/usrp/dboard/twinrx/twinrx_ctrl.hpp
new file mode 100644
index 000000000..521e27ae9
--- /dev/null
+++ b/host/lib/usrp/dboard/twinrx/twinrx_ctrl.hpp
@@ -0,0 +1,101 @@
+//
+// Copyright 2015 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#ifndef INCLUDED_DBOARD_TWINRX_CTRL_HPP
+#define INCLUDED_DBOARD_TWINRX_CTRL_HPP
+
+#include <boost/noncopyable.hpp>
+#include <uhd/types/wb_iface.hpp>
+#include "twinrx_io.hpp"
+
+namespace uhd { namespace usrp { namespace dboard { namespace twinrx {
+
+class twinrx_ctrl : public boost::noncopyable {
+public:
+ typedef boost::shared_ptr<twinrx_ctrl> sptr;
+
+ static sptr make(
+ dboard_iface::sptr db_iface,
+ twinrx_gpio::sptr gpio_iface,
+ twinrx_cpld_regmap::sptr cpld_regmap);
+
+ virtual ~twinrx_ctrl() {}
+
+ enum channel_t { CH1 = 0, CH2 = 1, BOTH = 2};
+
+ enum preamp_state_t { PREAMP_LOWBAND, PREAMP_HIGHBAND, PREAMP_BYPASS };
+
+ enum signal_path_t { PATH_LOWBAND, PATH_HIGHBAND };
+
+ enum preselector_path_t { PRESEL_PATH1, PRESEL_PATH2, PRESEL_PATH3, PRESEL_PATH4 };
+
+ enum lo_source_t { LO_INTERNAL, LO_EXTERNAL, LO_COMPANION, LO_DISABLED };
+
+ enum lo_export_source_t { LO_CH1_SYNTH, LO_CH2_SYNTH, LO_EXPORT_DISABLED };
+
+ enum antenna_mapping_t { ANTX_NATIVE, ANT1_SHARED, ANT2_SHARED, ANTX_SWAPPED, ANTX_DISABLED };
+
+ enum lo_config_route_t { LO_CONFIG_CH1, LO_CONFIG_CH2, LO_CONFIG_BOTH, LO_CONFIG_NONE };
+
+ enum cal_mode_t { CAL_DISABLED, CAL_CH1, CAL_CH2 };
+
+ virtual void commit() = 0;
+
+ virtual void set_chan_enabled(channel_t ch, bool enabled, bool commit = true) = 0;
+
+ virtual void set_preamp1(channel_t ch, preamp_state_t value, bool commit = true) = 0;
+
+ virtual void set_preamp2(channel_t ch, bool enabled, bool commit = true) = 0;
+
+ virtual void set_lb_preamp_preselector(channel_t ch, bool enabled, bool commit = true) = 0;
+
+ virtual void set_signal_path(channel_t ch, signal_path_t path, bool commit = true) = 0;
+
+ virtual void set_lb_preselector(channel_t ch, preselector_path_t path, bool commit = true) = 0;
+
+ virtual void set_hb_preselector(channel_t ch, preselector_path_t path, bool commit = true) = 0;
+
+ virtual void set_input_atten(channel_t ch, boost::uint8_t atten, bool commit = true) = 0;
+
+ virtual void set_lb_atten(channel_t ch, boost::uint8_t atten, bool commit = true) = 0;
+
+ virtual void set_hb_atten(channel_t ch, boost::uint8_t atten, bool commit = true) = 0;
+
+ virtual void set_lo1_source(channel_t ch, lo_source_t source, bool commit = true) = 0;
+
+ virtual void set_lo2_source(channel_t ch, lo_source_t source, bool commit = true) = 0;
+
+ virtual void set_lo1_export_source(lo_export_source_t source, bool commit = true) = 0;
+
+ virtual void set_lo2_export_source(lo_export_source_t source, bool commit = true) = 0;
+
+ virtual void set_antenna_mapping(antenna_mapping_t mapping, bool commit = true) = 0;
+
+ virtual void set_crossover_cal_mode(cal_mode_t cal_mode, bool commit = true) = 0;
+
+ virtual double set_lo1_synth_freq(channel_t ch, double freq, bool commit = true) = 0;
+
+ virtual double set_lo2_synth_freq(channel_t ch, double freq, bool commit = true) = 0;
+
+ virtual bool read_lo1_locked(channel_t ch) = 0;
+
+ virtual bool read_lo2_locked(channel_t ch) = 0;
+};
+
+}}}} //namespaces
+
+#endif /* INCLUDED_DBOARD_TWINRX_CTRL_HPP */
diff --git a/host/lib/usrp/dboard/twinrx/twinrx_experts.cpp b/host/lib/usrp/dboard/twinrx/twinrx_experts.cpp
new file mode 100644
index 000000000..3d04098e9
--- /dev/null
+++ b/host/lib/usrp/dboard/twinrx/twinrx_experts.cpp
@@ -0,0 +1,637 @@
+//
+// Copyright 2016 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#include "twinrx_experts.hpp"
+#include "twinrx_gain_tables.hpp"
+#include <uhd/utils/math.hpp>
+#include <uhd/exception.hpp>
+#include <uhd/types/dict.hpp>
+#include <uhd/types/ranges.hpp>
+#include <uhd/types/dict.hpp>
+#include <boost/assign/list_of.hpp>
+#include <boost/math/special_functions/round.hpp>
+
+using namespace uhd::experts;
+using namespace uhd::math;
+using namespace uhd::usrp::dboard::twinrx;
+
+/*!---------------------------------------------------------
+ * twinrx_freq_path_expert::resolve
+ * ---------------------------------------------------------
+ */
+void twinrx_freq_path_expert::resolve()
+{
+ //Lowband/highband switch point
+ static const double LB_HB_THRESHOLD_FREQ = 1.8e9;
+ static const double LB_TARGET_IF1_FREQ = 2.345e9;
+ static const double HB_TARGET_IF1_FREQ = 1.25e9;
+ static const double INJ_SIDE_THRESHOLD_FREQ = 5.1e9;
+
+ static const double FIXED_LO1_THRESHOLD_FREQ= 50e6;
+
+ //Preselector filter switch point
+ static const double LB_FILT1_THRESHOLD_FREQ = 0.5e9;
+ static const double LB_FILT2_THRESHOLD_FREQ = 0.8e9;
+ static const double LB_FILT3_THRESHOLD_FREQ = 1.2e9;
+ static const double LB_FILT4_THRESHOLD_FREQ = 1.8e9;
+ static const double HB_FILT1_THRESHOLD_FREQ = 3.0e9;
+ static const double HB_FILT2_THRESHOLD_FREQ = 4.1e9;
+ static const double HB_FILT3_THRESHOLD_FREQ = 5.1e9;
+ static const double HB_FILT4_THRESHOLD_FREQ = 6.0e9;
+
+ static const double LB_PREAMP_PRESEL_THRESHOLD_FREQ = 0.8e9;
+
+ //Misc
+ static const double INST_BANDWIDTH = 80e6;
+ static const double MANUAL_LO_HYSTERESIS_PPM = 1.0;
+
+ static const freq_range_t FREQ_RANGE(10e6, 6e9);
+ rf_freq_abs_t rf_freq(FREQ_RANGE.clip(_rf_freq_d));
+
+ // Choose low-band vs high-band depending on frequency
+ _signal_path = (rf_freq > LB_HB_THRESHOLD_FREQ) ?
+ twinrx_ctrl::PATH_HIGHBAND : twinrx_ctrl::PATH_LOWBAND;
+ if (_signal_path == twinrx_ctrl::PATH_LOWBAND) {
+ // Choose low-band preselector filter
+ if (rf_freq < LB_FILT1_THRESHOLD_FREQ) {
+ _lb_presel = twinrx_ctrl::PRESEL_PATH1;
+ } else if (rf_freq < LB_FILT2_THRESHOLD_FREQ) {
+ _lb_presel = twinrx_ctrl::PRESEL_PATH2;
+ } else if (rf_freq < LB_FILT3_THRESHOLD_FREQ) {
+ _lb_presel = twinrx_ctrl::PRESEL_PATH3;
+ } else if (rf_freq < LB_FILT4_THRESHOLD_FREQ) {
+ _lb_presel = twinrx_ctrl::PRESEL_PATH4;
+ } else {
+ _lb_presel = twinrx_ctrl::PRESEL_PATH4;
+ }
+ } else if (_signal_path == twinrx_ctrl::PATH_HIGHBAND) {
+ // Choose high-band preselector filter
+ if (rf_freq < HB_FILT1_THRESHOLD_FREQ) {
+ _hb_presel = twinrx_ctrl::PRESEL_PATH1;
+ } else if (rf_freq < HB_FILT2_THRESHOLD_FREQ) {
+ _hb_presel = twinrx_ctrl::PRESEL_PATH2;
+ } else if (rf_freq < HB_FILT3_THRESHOLD_FREQ) {
+ _hb_presel = twinrx_ctrl::PRESEL_PATH3;
+ } else if (rf_freq < HB_FILT4_THRESHOLD_FREQ) {
+ _hb_presel = twinrx_ctrl::PRESEL_PATH4;
+ } else {
+ _hb_presel = twinrx_ctrl::PRESEL_PATH4;
+ }
+ } else {
+ UHD_THROW_INVALID_CODE_PATH();
+ }
+
+ //Choose low-band preamp preselector
+ _lb_preamp_presel = (rf_freq > LB_PREAMP_PRESEL_THRESHOLD_FREQ);
+
+ //Choose LO frequencies
+ const double target_if1_freq = (_signal_path == twinrx_ctrl::PATH_HIGHBAND) ?
+ HB_TARGET_IF1_FREQ : LB_TARGET_IF1_FREQ;
+ const double target_if2_freq = _if_freq_d;
+
+ // LO1
+ double lo1_freq_ideal = 0.0, lo2_freq_ideal = 0.0;
+ if (rf_freq <= FIXED_LO1_THRESHOLD_FREQ) {
+ //LO1 Freq static
+ lo1_freq_ideal = target_if1_freq + FIXED_LO1_THRESHOLD_FREQ;
+ } else if (rf_freq <= INJ_SIDE_THRESHOLD_FREQ) {
+ //High-side LO1 Injection
+ lo1_freq_ideal = rf_freq.get() + target_if1_freq;
+ } else {
+ //Low-side LO1 Injection
+ lo1_freq_ideal = rf_freq.get() - target_if1_freq;
+ }
+
+ if (_lo1_freq_d.get_author() == experts::AUTHOR_USER) {
+ if (_lo1_freq_d.is_dirty()) { //Are we here because the LO frequency was set?
+ // The user explicitly requested to set the LO freq so don't touch it!
+ } else {
+ // Something else changed which may cause the LO frequency to update.
+ // Only commit if the frequency is stale. If the user's value is stale
+ // reset the author to expert.
+ if (rf_freq_ppm_t(lo1_freq_ideal, MANUAL_LO_HYSTERESIS_PPM) != _lo1_freq_d.get()) {
+ _lo1_freq_d = lo1_freq_ideal; //Reset author
+ }
+ }
+ } else {
+ // The LO frequency was never set by the user. Let the expert take care of it
+ _lo1_freq_d = lo1_freq_ideal; //Reset author
+ }
+
+ // LO2
+ lo_inj_side_t lo2_inj_side_ideal = _compute_lo2_inj_side(
+ lo1_freq_ideal, target_if1_freq, target_if2_freq, INST_BANDWIDTH);
+ if (lo2_inj_side_ideal == INJ_HIGH_SIDE) {
+ lo2_freq_ideal = target_if1_freq + target_if2_freq;
+ } else {
+ lo2_freq_ideal = target_if1_freq - target_if2_freq;
+ }
+
+ if (_lo2_freq_d.get_author() == experts::AUTHOR_USER) {
+ if (_lo2_freq_d.is_dirty()) { //Are we here because the LO frequency was set?
+ // The user explicitly requested to set the LO freq so don't touch it!
+ } else {
+ // Something else changed which may cause the LO frequency to update.
+ // Only commit if the frequency is stale. If the user's value is stale
+ // reset the author to expert.
+ if (rf_freq_ppm_t(lo2_freq_ideal, MANUAL_LO_HYSTERESIS_PPM) != _lo2_freq_d.get()) {
+ _lo2_freq_d = lo2_freq_ideal; //Reset author
+ }
+ }
+ } else {
+ // The LO frequency was never set by the user. Let the expert take care of it
+ _lo2_freq_d = lo2_freq_ideal; //Reset author
+ }
+
+ // Determine injection side using the final LO frequency
+ _lo1_inj_side = (_lo1_freq_d > rf_freq.get()) ? INJ_HIGH_SIDE : INJ_LOW_SIDE;
+ _lo2_inj_side = (_lo2_freq_d > target_if1_freq) ? INJ_HIGH_SIDE : INJ_LOW_SIDE;
+}
+
+lo_inj_side_t twinrx_freq_path_expert::_compute_lo2_inj_side(
+ double lo1_freq, double if1_freq, double if2_freq, double bandwidth
+) {
+ static const int MAX_SPUR_ORDER = 5;
+ for (int ord = MAX_SPUR_ORDER; ord >= 1; ord--) {
+ // Check high-side injection first
+ if (not _has_mixer_spurs(lo1_freq, if1_freq + if2_freq, if2_freq, bandwidth, ord)) {
+ return INJ_HIGH_SIDE;
+ }
+ // Check low-side injection second
+ if (not _has_mixer_spurs(lo1_freq, if1_freq - if2_freq, if2_freq, bandwidth, ord)) {
+ return INJ_LOW_SIDE;
+ }
+ }
+ // If we reached here, then there are spurs everywhere. Pick high-side as the default
+ return INJ_HIGH_SIDE;
+}
+
+bool twinrx_freq_path_expert::_has_mixer_spurs(
+ double lo1_freq, double lo2_freq, double if2_freq,
+ double bandwidth, int spur_order
+) {
+ // Iterate through all N-th order harmomic combinations
+ // of LOs...
+ for (int lo1h_i = 1; lo1h_i <= spur_order; lo1h_i++) {
+ double lo1harm_freq = lo1_freq * lo1h_i;
+ for (int lo2h_i = 1; lo2h_i <= spur_order; lo2h_i++) {
+ double lo2harm_freq = lo2_freq * lo2h_i;
+ double hdelta = lo1harm_freq - lo2harm_freq;
+ // .. and check if there is a mixer spur in the IF band
+ if (std::abs(hdelta + if2_freq) < bandwidth/2 or
+ std::abs(hdelta - if2_freq) < bandwidth/2) {
+ return true;
+ }
+ }
+ }
+ // No spurs were found after NxN search
+ return false;
+}
+
+/*!---------------------------------------------------------
+ * twinrx_freq_coercion_expert::resolve
+ * ---------------------------------------------------------
+ */
+void twinrx_freq_coercion_expert::resolve()
+{
+ const double actual_if2_freq = _if_freq_d;
+ const double actual_if1_freq = (_lo2_inj_side == INJ_LOW_SIDE) ?
+ (_lo2_freq_c + actual_if2_freq) : (_lo2_freq_c - actual_if2_freq);
+
+ _rf_freq_c = (_lo1_inj_side == INJ_LOW_SIDE) ?
+ (_lo1_freq_c + actual_if1_freq) : (_lo1_freq_c - actual_if1_freq);
+}
+
+/*!---------------------------------------------------------
+ * twinrx_nyquist_expert::resolve
+ * ---------------------------------------------------------
+ */
+void twinrx_nyquist_expert::resolve()
+{
+ double if_freq_sign = 1.0;
+ if (_lo1_inj_side == INJ_HIGH_SIDE) if_freq_sign *= -1.0;
+ if (_lo2_inj_side == INJ_HIGH_SIDE) if_freq_sign *= -1.0;
+ _if_freq_c = _if_freq_d * if_freq_sign;
+
+ _db_iface->set_fe_connection(dboard_iface::UNIT_RX, _channel,
+ usrp::fe_connection_t(_codec_conn, _if_freq_c));
+}
+
+/*!---------------------------------------------------------
+ * twinrx_chan_gain_expert::resolve
+ * ---------------------------------------------------------
+ */
+void twinrx_chan_gain_expert::resolve()
+{
+ if (_gain_profile != "default") {
+ //TODO: Implement me!
+ throw uhd::not_implemented_error("custom gain strategies not implemeted yet");
+ }
+
+ //Lookup table using settings
+ const twinrx_gain_table table = twinrx_gain_table::lookup_table(
+ _signal_path,
+ (_signal_path==twinrx_ctrl::PATH_HIGHBAND) ? _hb_presel : _lb_presel,
+ _gain_profile);
+
+ //Compute minimum gain. The user-specified gain value will be interpreted as
+ //the gain applied on top of the minimum gain state.
+ //If antennas are shared or swapped, the switch has 6dB of loss
+ size_t gain_index = std::min(static_cast<size_t>(boost::math::round(_gain.get())), table.get_num_entries()-1);
+
+ //Translate gain to an index in the gain table
+ const twinrx_gain_config_t& config = table.find_by_index(gain_index);
+
+ _input_atten = config.atten1;
+ if (_signal_path == twinrx_ctrl::PATH_HIGHBAND) {
+ _hb_atten = config.atten2;
+ } else {
+ _lb_atten = config.atten2;
+ }
+
+ // Preamp 1 should use the Highband amp for frequencies above 3 GHz
+ if (_signal_path == twinrx_ctrl::PATH_HIGHBAND && _hb_presel != twinrx_ctrl::PRESEL_PATH1) {
+ _preamp1 = config.amp1 ? twinrx_ctrl::PREAMP_HIGHBAND : twinrx_ctrl::PREAMP_BYPASS;
+ } else {
+ _preamp1 = config.amp1 ? twinrx_ctrl::PREAMP_LOWBAND : twinrx_ctrl::PREAMP_BYPASS;
+ }
+
+ _preamp2 = config.amp2;
+}
+
+/*!---------------------------------------------------------
+ * twinrx_lo_config_expert::resolve
+ * ---------------------------------------------------------
+ */
+void twinrx_lo_config_expert::resolve()
+{
+ static const uhd::dict<std::string, twinrx_ctrl::lo_source_t> src_lookup =
+ boost::assign::map_list_of
+ ("internal", twinrx_ctrl::LO_INTERNAL)
+ ("external", twinrx_ctrl::LO_EXTERNAL)
+ ("companion", twinrx_ctrl::LO_COMPANION)
+ ("disabled", twinrx_ctrl::LO_DISABLED);
+
+ if (src_lookup.has_key(_lo_source_ch0)) {
+ _lo1_src_ch0 = _lo2_src_ch0 = src_lookup[_lo_source_ch0];
+ } else {
+ throw uhd::value_error("Invalid LO source for channel 0.Choose from {internal, external, companion}");
+ }
+ if (src_lookup.has_key(_lo_source_ch1)) {
+ _lo1_src_ch1 = _lo2_src_ch1 = src_lookup[_lo_source_ch1];
+ } else {
+ throw uhd::value_error("Invalid LO source for channel 1.Choose from {internal, external, companion}");
+ }
+
+ twinrx_ctrl::lo_export_source_t export_src = twinrx_ctrl::LO_EXPORT_DISABLED;
+ if (_lo_export_ch0 and (_lo_source_ch0 == "external")) {
+ throw uhd::value_error("Cannot export an external LO for channel 0");
+ }
+ if (_lo_export_ch1 and (_lo_source_ch1 == "external")) {
+ throw uhd::value_error("Cannot export an external LO for channel 1");
+ }
+ if (_lo_export_ch0 and _lo_export_ch1) {
+ throw uhd::value_error("Cannot export LOs for both channels");
+ } else if (_lo_export_ch0) {
+ export_src = (_lo1_src_ch0 == twinrx_ctrl::LO_INTERNAL) ?
+ twinrx_ctrl::LO_CH1_SYNTH : twinrx_ctrl::LO_CH2_SYNTH;
+ } else if (_lo_export_ch1) {
+ export_src = (_lo1_src_ch1 == twinrx_ctrl::LO_INTERNAL) ?
+ twinrx_ctrl::LO_CH2_SYNTH : twinrx_ctrl::LO_CH1_SYNTH;
+ }
+ _lo1_export_src = _lo2_export_src = export_src;
+}
+
+/*!---------------------------------------------------------
+ * twinrx_lo_freq_expert::resolve
+ * ---------------------------------------------------------
+ */
+void twinrx_lo_mapping_expert::resolve()
+{
+ static const size_t CH0_MSK = 0x1;
+ static const size_t CH1_MSK = 0x2;
+
+ // Determine which channels are "driving" each synthesizer
+ // First check for explicit requests i.e. lo_source "internal" or "companion"
+ size_t synth_map[] = {0, 0};
+ if (_lox_src_ch0 == twinrx_ctrl::LO_INTERNAL) {
+ synth_map[0] = synth_map[0] | CH0_MSK;
+ } else if (_lox_src_ch0 == twinrx_ctrl::LO_COMPANION) {
+ synth_map[1] = synth_map[1] | CH0_MSK;
+ }
+ if (_lox_src_ch1 == twinrx_ctrl::LO_INTERNAL) {
+ synth_map[1] = synth_map[1] | CH1_MSK;
+ } else if (_lox_src_ch1 == twinrx_ctrl::LO_COMPANION) {
+ synth_map[0] = synth_map[0] | CH1_MSK;
+ }
+
+ // If a particular channel has its LO source disabled then the other
+ // channel is automatically put in hop mode i.e. the synthesizer that
+ // belongs to the disabled channel can be re-purposed as a redundant LO
+ // to overlap tuning with signal dwell time.
+ bool hopping_enabled = false;
+ if (_lox_src_ch0 == twinrx_ctrl::LO_DISABLED) {
+ if (_lox_src_ch1 == twinrx_ctrl::LO_INTERNAL) {
+ synth_map[0] = synth_map[0] | CH0_MSK;
+ hopping_enabled = true;
+ } else if (_lox_src_ch1 == twinrx_ctrl::LO_COMPANION) {
+ synth_map[1] = synth_map[1] | CH0_MSK;
+ hopping_enabled = true;
+ }
+ }
+ if (_lox_src_ch1 == twinrx_ctrl::LO_DISABLED) {
+ if (_lox_src_ch0 == twinrx_ctrl::LO_INTERNAL) {
+ synth_map[1] = synth_map[1] | CH1_MSK;
+ hopping_enabled = true;
+ } else if (_lox_src_ch0 == twinrx_ctrl::LO_COMPANION) {
+ synth_map[0] = synth_map[0] | CH1_MSK;
+ hopping_enabled = true;
+ }
+ }
+
+ // For each synthesizer come up with the final mapping
+ for (size_t synth = 0; synth < 2; synth++) {
+ experts::data_writer_t<lo_synth_mapping_t>& lox_mapping =
+ (synth == 0) ? _lox_mapping_synth0 : _lox_mapping_synth1;
+ if (synth_map[synth] == (CH0_MSK|CH1_MSK)) {
+ lox_mapping = MAPPING_SHARED;
+ } else if (synth_map[synth] == CH0_MSK) {
+ lox_mapping = MAPPING_CH0;
+ } else if (synth_map[synth] == CH1_MSK) {
+ lox_mapping = MAPPING_CH1;
+ } else {
+ lox_mapping = MAPPING_NONE;
+ }
+ }
+ _lox_hopping_enabled = hopping_enabled;
+}
+
+/*!---------------------------------------------------------
+ * twinrx_antenna_expert::resolve
+ * ---------------------------------------------------------
+ */
+void twinrx_antenna_expert::resolve()
+{
+ static const std::string ANT0 = "RX1", ANT1 = "RX2";
+
+ if (_antenna_ch0 == ANT0 and _antenna_ch1 == ANT1) {
+ _ant_mapping = twinrx_ctrl::ANTX_NATIVE;
+ } else if (_antenna_ch0 == ANT0 and _antenna_ch1 == ANT0) {
+ if (_enabled_ch0 and _enabled_ch1) {
+ _ant_mapping = twinrx_ctrl::ANT1_SHARED;
+ } else if (_enabled_ch0) {
+ _ant_mapping = twinrx_ctrl::ANTX_NATIVE;
+ } else if (_enabled_ch1) {
+ _ant_mapping = twinrx_ctrl::ANTX_SWAPPED;
+ }
+ } else if (_antenna_ch0 == ANT1 and _antenna_ch1 == ANT1) {
+ if (_enabled_ch0 and _enabled_ch1) {
+ _ant_mapping = twinrx_ctrl::ANT2_SHARED;
+ } else if (_enabled_ch0) {
+ _ant_mapping = twinrx_ctrl::ANTX_SWAPPED;
+ } else if (_enabled_ch1) {
+ _ant_mapping = twinrx_ctrl::ANTX_NATIVE;
+ }
+ } else if (_antenna_ch0 == ANT1 and _antenna_ch1 == ANT0) {
+ _ant_mapping = twinrx_ctrl::ANTX_SWAPPED;
+ } else if (_antenna_ch0 != ANT0 and _antenna_ch0 != ANT1) {
+ throw uhd::value_error("Invalid antenna selection " + _antenna_ch0.get() + " for channel 0. Must be " + ANT0 + " or " + ANT1);
+ } else if (_antenna_ch1 != ANT0 and _antenna_ch1 != ANT1) {
+ throw uhd::value_error("Invalid antenna selection " + _antenna_ch1.get() + " for channel 1. Must be " + ANT0 + " or " + ANT1);
+ }
+
+ //TODO: Implement hooks for the calibration switch
+ _cal_mode = twinrx_ctrl::CAL_DISABLED;
+
+ if (_cal_mode == twinrx_ctrl::CAL_CH1 and _lo_export_ch1) {
+ throw uhd::value_error("Cannot calibrate channel 0 and export the LO for channel 1.");
+ } else if (_cal_mode == twinrx_ctrl::CAL_CH2 and _lo_export_ch0) {
+ throw uhd::value_error("Cannot calibrate channel 1 and export the LO for channel 0.");
+ }
+}
+
+/*!---------------------------------------------------------
+ * twinrx_ant_gain_expert::resolve
+ * ---------------------------------------------------------
+ */
+void twinrx_ant_gain_expert::resolve()
+{
+ switch (_ant_mapping) {
+ case twinrx_ctrl::ANTX_NATIVE:
+ _ant0_input_atten = _ch0_input_atten;
+ _ant0_preamp1 = _ch0_preamp1;
+ _ant0_preamp2 = _ch0_preamp2;
+ _ant0_lb_preamp_presel = _ch0_lb_preamp_presel;
+ _ant1_input_atten = _ch1_input_atten;
+ _ant1_preamp1 = _ch1_preamp1;
+ _ant1_preamp2 = _ch1_preamp2;
+ _ant1_lb_preamp_presel = _ch1_lb_preamp_presel;
+ break;
+ case twinrx_ctrl::ANTX_SWAPPED:
+ _ant0_input_atten = _ch1_input_atten;
+ _ant0_preamp1 = _ch1_preamp1;
+ _ant0_preamp2 = _ch1_preamp2;
+ _ant0_lb_preamp_presel = _ch1_lb_preamp_presel;
+ _ant1_input_atten = _ch0_input_atten;
+ _ant1_preamp1 = _ch0_preamp1;
+ _ant1_preamp2 = _ch0_preamp2;
+ _ant1_lb_preamp_presel = _ch0_lb_preamp_presel;
+ break;
+ case twinrx_ctrl::ANT1_SHARED:
+ if ((_ch0_input_atten != _ch1_input_atten) or
+ (_ch0_preamp1 != _ch1_preamp1) or
+ (_ch0_preamp2 != _ch1_preamp2) or
+ (_ch0_lb_preamp_presel != _ch1_lb_preamp_presel))
+ {
+ UHD_MSG(warning) << "incompatible gain settings for antenna sharing. temporarily using Ch0 settings for Ch1.";
+ }
+ _ant0_input_atten = _ch0_input_atten;
+ _ant0_preamp1 = _ch0_preamp1;
+ _ant0_preamp2 = _ch0_preamp2;
+ _ant0_lb_preamp_presel = _ch0_lb_preamp_presel;
+
+ _ant1_input_atten = 0;
+ _ant1_preamp1 = twinrx_ctrl::PREAMP_BYPASS;
+ _ant1_preamp2 = false;
+ _ant1_lb_preamp_presel = false;
+ break;
+ case twinrx_ctrl::ANT2_SHARED:
+ if ((_ch0_input_atten != _ch1_input_atten) or
+ (_ch0_preamp1 != _ch1_preamp1) or
+ (_ch0_preamp2 != _ch1_preamp2) or
+ (_ch0_lb_preamp_presel != _ch1_lb_preamp_presel))
+ {
+ UHD_MSG(warning) << "incompatible gain settings for antenna sharing. temporarily using Ch0 settings for Ch1.";
+ }
+ _ant1_input_atten = _ch0_input_atten;
+ _ant1_preamp1 = _ch0_preamp1;
+ _ant1_preamp2 = _ch0_preamp2;
+ _ant1_lb_preamp_presel = _ch0_lb_preamp_presel;
+
+ _ant0_input_atten = 0;
+ _ant0_preamp1 = twinrx_ctrl::PREAMP_BYPASS;
+ _ant0_preamp2 = false;
+ _ant0_lb_preamp_presel = false;
+ break;
+ default:
+ _ant0_input_atten = 0;
+ _ant0_preamp1 = twinrx_ctrl::PREAMP_BYPASS;
+ _ant0_preamp2 = false;
+ _ant0_lb_preamp_presel = false;
+ _ant1_input_atten = 0;
+ _ant1_preamp1 = twinrx_ctrl::PREAMP_BYPASS;
+ _ant1_preamp2 = false;
+ _ant1_lb_preamp_presel = false;
+ break;
+ }
+}
+
+/*!---------------------------------------------------------
+ * twinrx_settings_expert::resolve
+ * ---------------------------------------------------------
+ */
+const bool twinrx_settings_expert::FORCE_COMMIT = false;
+
+void twinrx_settings_expert::resolve()
+{
+ for (size_t i = 0; i < 2; i++) {
+ ch_settings& ch_set = (i == 1) ? _ch1 : _ch0;
+ twinrx_ctrl::channel_t ch = (i == 1) ? twinrx_ctrl::CH2 : twinrx_ctrl::CH1;
+ _ctrl->set_chan_enabled(ch, ch_set.chan_enabled, FORCE_COMMIT);
+ _ctrl->set_preamp1(ch, ch_set.preamp1, FORCE_COMMIT);
+ _ctrl->set_preamp2(ch, ch_set.preamp2, FORCE_COMMIT);
+ _ctrl->set_lb_preamp_preselector(ch, ch_set.lb_preamp_presel, FORCE_COMMIT);
+ _ctrl->set_signal_path(ch, ch_set.signal_path, FORCE_COMMIT);
+ _ctrl->set_lb_preselector(ch, ch_set.lb_presel, FORCE_COMMIT);
+ _ctrl->set_hb_preselector(ch, ch_set.hb_presel, FORCE_COMMIT);
+ _ctrl->set_input_atten(ch, ch_set.input_atten, FORCE_COMMIT);
+ _ctrl->set_lb_atten(ch, ch_set.lb_atten, FORCE_COMMIT);
+ _ctrl->set_hb_atten(ch, ch_set.hb_atten, FORCE_COMMIT);
+ _ctrl->set_lo1_source(ch, ch_set.lo1_source, FORCE_COMMIT);
+ _ctrl->set_lo2_source(ch, ch_set.lo2_source, FORCE_COMMIT);
+ }
+
+ _resolve_lox_freq(STAGE_LO1,
+ _ch0.lo1_freq_d, _ch1.lo1_freq_d, _ch0.lo1_freq_c, _ch1.lo1_freq_c,
+ _ch0.lo1_source, _ch1.lo1_source, _lo1_synth0_mapping, _lo1_synth1_mapping,
+ _lo1_hopping_enabled);
+ _resolve_lox_freq(STAGE_LO2,
+ _ch0.lo2_freq_d, _ch1.lo2_freq_d, _ch0.lo2_freq_c, _ch1.lo2_freq_c,
+ _ch0.lo2_source, _ch1.lo2_source, _lo2_synth0_mapping, _lo2_synth1_mapping,
+ _lo2_hopping_enabled);
+
+ _ctrl->set_lo1_export_source(_lo1_export_src, FORCE_COMMIT);
+ _ctrl->set_lo2_export_source(_lo2_export_src, FORCE_COMMIT);
+ _ctrl->set_antenna_mapping(_ant_mapping, FORCE_COMMIT);
+ //TODO: Re-enable this when we support this mode
+ //_ctrl->set_crossover_cal_mode(_cal_mode, FORCE_COMMIT);
+
+ _ctrl->commit();
+}
+
+void twinrx_settings_expert::_resolve_lox_freq(
+ lo_stage_t lo_stage,
+ uhd::experts::data_reader_t<double>& ch0_freq_d,
+ uhd::experts::data_reader_t<double>& ch1_freq_d,
+ uhd::experts::data_writer_t<double>& ch0_freq_c,
+ uhd::experts::data_writer_t<double>& ch1_freq_c,
+ twinrx_ctrl::lo_source_t ch0_lo_source,
+ twinrx_ctrl::lo_source_t ch1_lo_source,
+ lo_synth_mapping_t synth0_mapping,
+ lo_synth_mapping_t synth1_mapping,
+ bool hopping_enabled)
+{
+ if (ch0_lo_source == twinrx_ctrl::LO_EXTERNAL) {
+ // If the LO is external then we don't need to program any synthesizers
+ ch0_freq_c = ch0_freq_d;
+ } else {
+ // When in hopping mode, only attempt to write the LO frequency if it is actually
+ // dirty to avoid reconfiguring the LO if it is being "double-buffered". If not
+ // hopping, then always write the frequency because other inputs might require
+ // an LO re-commit
+ const bool freq_update_request = (not hopping_enabled) or ch0_freq_d.is_dirty();
+ if (synth0_mapping == MAPPING_CH0 and freq_update_request) {
+ ch0_freq_c = _set_lox_synth_freq(lo_stage, twinrx_ctrl::CH1, ch0_freq_d);
+ } else if (synth1_mapping == MAPPING_CH0 and freq_update_request) {
+ ch0_freq_c = _set_lox_synth_freq(lo_stage, twinrx_ctrl::CH2, ch0_freq_d);
+ } else if (synth0_mapping == MAPPING_SHARED or synth1_mapping == MAPPING_SHARED) {
+ // If any synthesizer is being shared then we are not in hopping mode
+ if (rf_freq_ppm_t(ch0_freq_d) != ch1_freq_d) {
+ UHD_MSG(warning) <<
+ "Incompatible RF/LO frequencies for LO sharing. Using Ch0 settings for both channels.";
+ }
+ twinrx_ctrl::channel_t ch = (synth0_mapping == MAPPING_SHARED) ? twinrx_ctrl::CH1 : twinrx_ctrl::CH2;
+ ch0_freq_c = _set_lox_synth_freq(lo_stage, ch, ch0_freq_d);
+ ch1_freq_c = ch0_freq_c;
+ }
+ }
+
+ if (ch1_lo_source == twinrx_ctrl::LO_EXTERNAL) {
+ // If the LO is external then we don't need to program any synthesizers
+ ch1_freq_c = ch1_freq_d;
+ } else {
+ // When in hopping mode, only attempt to write the LO frequency if it is actually
+ // dirty to avoid reconfiguring the LO if it is being "double-buffered". If not
+ // hopping, then always write the frequency because other inputs might require
+ // an LO re-commit
+ const bool freq_update_request = (not hopping_enabled) or ch1_freq_d.is_dirty();
+ // As an additional layer of protection from unnecessarily commiting the LO, check
+ // if the frequency has actually changed.
+ if (synth0_mapping == MAPPING_CH1 and freq_update_request) {
+ ch1_freq_c = _set_lox_synth_freq(lo_stage, twinrx_ctrl::CH1, ch1_freq_d);
+ } else if (synth1_mapping == MAPPING_CH1 and freq_update_request) {
+ ch1_freq_c = _set_lox_synth_freq(lo_stage, twinrx_ctrl::CH2, ch1_freq_d);
+ } else if (synth0_mapping == MAPPING_SHARED or synth1_mapping == MAPPING_SHARED) {
+ // If any synthesizer is being shared then we are not in hopping mode
+ if (rf_freq_ppm_t(ch0_freq_d) != ch1_freq_d) {
+ UHD_MSG(warning) <<
+ "Incompatible RF/LO frequencies for LO sharing. Using Ch0 settings for both channels.";
+ }
+ twinrx_ctrl::channel_t ch = (synth0_mapping == MAPPING_SHARED) ? twinrx_ctrl::CH1 : twinrx_ctrl::CH2;
+ ch0_freq_c = _set_lox_synth_freq(lo_stage, ch, ch0_freq_d);
+ ch1_freq_c = ch0_freq_c;
+ }
+ }
+}
+
+double twinrx_settings_expert::_set_lox_synth_freq(lo_stage_t stage, twinrx_ctrl::channel_t ch, double freq)
+{
+ lo_freq_cache_t* freq_cache = NULL;
+ if (stage == STAGE_LO1) {
+ freq_cache = (ch == twinrx_ctrl::CH1) ? &_cached_lo1_synth0_freq : &_cached_lo1_synth1_freq;
+ } else if (stage == STAGE_LO2) {
+ freq_cache = (ch == twinrx_ctrl::CH1) ? &_cached_lo2_synth0_freq : &_cached_lo2_synth1_freq;
+ } else {
+ throw uhd::assertion_error("Invalid LO stage");
+ }
+
+ // Check if the frequency has actually changed before configuring synthesizers
+ double coerced_freq = 0.0;
+ if (freq_cache->desired != freq) {
+ if (stage == STAGE_LO1) {
+ coerced_freq = _ctrl->set_lo1_synth_freq(ch, freq, FORCE_COMMIT);
+ } else {
+ coerced_freq = _ctrl->set_lo2_synth_freq(ch, freq, FORCE_COMMIT);
+ }
+ freq_cache->desired = rf_freq_ppm_t(freq);
+ freq_cache->coerced = coerced_freq;
+ } else {
+ coerced_freq = freq_cache->coerced;
+ }
+ return coerced_freq;
+}
+
diff --git a/host/lib/usrp/dboard/twinrx/twinrx_experts.hpp b/host/lib/usrp/dboard/twinrx/twinrx_experts.hpp
new file mode 100644
index 000000000..f2601a09b
--- /dev/null
+++ b/host/lib/usrp/dboard/twinrx/twinrx_experts.hpp
@@ -0,0 +1,630 @@
+//
+// Copyright 2016 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#ifndef INCLUDED_DBOARD_TWINRX_EXPERTS_HPP
+#define INCLUDED_DBOARD_TWINRX_EXPERTS_HPP
+
+#include "twinrx_ctrl.hpp"
+#include "expert_nodes.hpp"
+#include <uhd/utils/math.hpp>
+
+namespace uhd { namespace usrp { namespace dboard { namespace twinrx {
+
+//---------------------------------------------------------
+// Misc types and definitions
+//---------------------------------------------------------
+
+struct rf_freq_abs_t : public uhd::math::fp_compare::fp_compare_delta<double> {
+ rf_freq_abs_t(double freq = 0.0, double epsilon = 1.0 /* 1Hz epsilon */) :
+ uhd::math::fp_compare::fp_compare_delta<double>(freq, epsilon) {}
+ inline double get() const { return _value; }
+};
+
+struct rf_freq_ppm_t : public rf_freq_abs_t {
+ rf_freq_ppm_t(double freq = 0.0, double epsilon_ppm = 0.1 /* 1PPM epsilon */) :
+ rf_freq_abs_t(freq, 1e-6 * freq * epsilon_ppm) {}
+};
+
+enum lo_stage_t { STAGE_LO1, STAGE_LO2 };
+enum lo_inj_side_t { INJ_LOW_SIDE, INJ_HIGH_SIDE };
+enum lo_synth_mapping_t { MAPPING_NONE, MAPPING_CH0, MAPPING_CH1, MAPPING_SHARED };
+
+static const std::string prepend_ch(std::string name, const std::string& ch) {
+ return ch + "/" + name;
+}
+
+static const std::string lo_stage_str(lo_stage_t stage, bool lower = false) {
+ std::string prefix = lower ? "lo" : "LO";
+ return prefix + ((stage == STAGE_LO1) ? "1" : "2");
+}
+
+/*!---------------------------------------------------------
+ * twinrx_freq_path_expert
+ *
+ * This expert is responsble for translating a user-specified
+ * RF and IF center frequency into TwinRX specific settings
+ * like band, preselector path, LO frequency and injection
+ * sides for both the LO stages.
+ *
+ * One instance of this expert is required for each channel
+ * ---------------------------------------------------------
+ */
+class twinrx_freq_path_expert : public experts::worker_node_t {
+public:
+ twinrx_freq_path_expert(const experts::node_retriever_t& db, std::string ch)
+ : experts::worker_node_t(prepend_ch("twinrx_freq_path_expert", ch)),
+ _rf_freq_d (db, prepend_ch("freq/desired", ch)),
+ _if_freq_d (db, prepend_ch("if_freq/desired", ch)),
+ _signal_path (db, prepend_ch("ch/signal_path", ch)),
+ _lb_presel (db, prepend_ch("ch/lb_presel", ch)),
+ _hb_presel (db, prepend_ch("ch/hb_presel", ch)),
+ _lb_preamp_presel (db, prepend_ch("ch/lb_preamp_presel", ch)),
+ _lo1_freq_d (db, prepend_ch("los/LO1/freq/desired", ch)),
+ _lo2_freq_d (db, prepend_ch("los/LO2/freq/desired", ch)),
+ _lo1_inj_side (db, prepend_ch("ch/LO1/inj_side", ch)),
+ _lo2_inj_side (db, prepend_ch("ch/LO2/inj_side", ch))
+ {
+ bind_accessor(_rf_freq_d);
+ bind_accessor(_if_freq_d);
+ bind_accessor(_signal_path);
+ bind_accessor(_lb_presel);
+ bind_accessor(_hb_presel);
+ bind_accessor(_lb_preamp_presel);
+ bind_accessor(_lo1_freq_d);
+ bind_accessor(_lo2_freq_d);
+ bind_accessor(_lo1_inj_side);
+ bind_accessor(_lo2_inj_side);
+ }
+
+private:
+ virtual void resolve();
+ static lo_inj_side_t _compute_lo2_inj_side(
+ double lo1_freq, double if1_freq, double if2_freq, double bandwidth);
+ static bool _has_mixer_spurs(
+ double lo1_freq, double lo2_freq, double if2_freq,
+ double bandwidth, int spur_order);
+
+ //Inputs
+ experts::data_reader_t<double> _rf_freq_d;
+ experts::data_reader_t<double> _if_freq_d;
+ //Outputs
+ experts::data_writer_t<twinrx_ctrl::signal_path_t> _signal_path;
+ experts::data_writer_t<twinrx_ctrl::preselector_path_t> _lb_presel;
+ experts::data_writer_t<twinrx_ctrl::preselector_path_t> _hb_presel;
+ experts::data_writer_t<bool> _lb_preamp_presel;
+ experts::data_writer_t<double> _lo1_freq_d;
+ experts::data_writer_t<double> _lo2_freq_d;
+ experts::data_writer_t<lo_inj_side_t> _lo1_inj_side;
+ experts::data_writer_t<lo_inj_side_t> _lo2_inj_side;
+};
+
+/*!---------------------------------------------------------
+ * twinrx_lo_config_expert
+ *
+ * This expert is responsible for translating high level
+ * channel-scoped LO source and export settings to low-level
+ * channel-scoped settings. The expert only deals with
+ * the source and export attributes, not frequency.
+ *
+ * One instance of this expert is required for all channels
+ * ---------------------------------------------------------
+ */
+class twinrx_lo_config_expert : public experts::worker_node_t {
+public:
+ twinrx_lo_config_expert(const experts::node_retriever_t& db)
+ : experts::worker_node_t("twinrx_lo_config_expert"),
+ _lo_source_ch0 (db, prepend_ch("los/all/source", "0")),
+ _lo_source_ch1 (db, prepend_ch("los/all/source", "1")),
+ _lo_export_ch0 (db, prepend_ch("los/all/export", "0")),
+ _lo_export_ch1 (db, prepend_ch("los/all/export", "1")),
+ _lo1_src_ch0 (db, prepend_ch("ch/LO1/source", "0")),
+ _lo1_src_ch1 (db, prepend_ch("ch/LO1/source", "1")),
+ _lo2_src_ch0 (db, prepend_ch("ch/LO2/source", "0")),
+ _lo2_src_ch1 (db, prepend_ch("ch/LO2/source", "1")),
+ _lo1_export_src (db, "com/LO1/export_source"),
+ _lo2_export_src (db, "com/LO2/export_source")
+ {
+ bind_accessor(_lo_source_ch0);
+ bind_accessor(_lo_source_ch1);
+ bind_accessor(_lo_export_ch0);
+ bind_accessor(_lo_export_ch1);
+ bind_accessor(_lo1_src_ch0);
+ bind_accessor(_lo1_src_ch1);
+ bind_accessor(_lo2_src_ch0);
+ bind_accessor(_lo2_src_ch1);
+ bind_accessor(_lo1_export_src);
+ bind_accessor(_lo2_export_src);
+ }
+
+private:
+ virtual void resolve();
+
+ //Inputs
+ experts::data_reader_t<std::string> _lo_source_ch0;
+ experts::data_reader_t<std::string> _lo_source_ch1;
+ experts::data_reader_t<bool> _lo_export_ch0;
+ experts::data_reader_t<bool> _lo_export_ch1;
+ //Outputs
+ experts::data_writer_t<twinrx_ctrl::lo_source_t> _lo1_src_ch0;
+ experts::data_writer_t<twinrx_ctrl::lo_source_t> _lo1_src_ch1;
+ experts::data_writer_t<twinrx_ctrl::lo_source_t> _lo2_src_ch0;
+ experts::data_writer_t<twinrx_ctrl::lo_source_t> _lo2_src_ch1;
+ experts::data_writer_t<twinrx_ctrl::lo_export_source_t> _lo1_export_src;
+ experts::data_writer_t<twinrx_ctrl::lo_export_source_t> _lo2_export_src;
+};
+
+/*!---------------------------------------------------------
+ * twinrx_lo_mapping_expert
+ *
+ * This expert is responsible for translating low-level
+ * channel-scoped LO source and export settings to low-level
+ * synthesizer-scoped settings. The expert deals with the
+ * extremely flexible channel->synthesizer mapping and handles
+ * frequency hopping modes.
+ *
+ * One instance of this expert is required for each LO stage
+ * ---------------------------------------------------------
+ */
+class twinrx_lo_mapping_expert : public experts::worker_node_t {
+public:
+ twinrx_lo_mapping_expert(const experts::node_retriever_t& db, lo_stage_t stage)
+ : experts::worker_node_t("twinrx_" + lo_stage_str(stage, true) + "_mapping_expert"),
+ _lox_src_ch0 (db, prepend_ch("ch/" + lo_stage_str(stage) + "/source", "0")),
+ _lox_src_ch1 (db, prepend_ch("ch/" + lo_stage_str(stage) + "/source", "1")),
+ _lox_mapping_synth0 (db, prepend_ch("synth/" + lo_stage_str(stage) + "/mapping", "0")),
+ _lox_mapping_synth1 (db, prepend_ch("synth/" + lo_stage_str(stage) + "/mapping", "1")),
+ _lox_hopping_enabled (db, "com/synth/" + lo_stage_str(stage) + "/hopping_enabled")
+ {
+ bind_accessor(_lox_src_ch0);
+ bind_accessor(_lox_src_ch1);
+ bind_accessor(_lox_mapping_synth0);
+ bind_accessor(_lox_mapping_synth1);
+ bind_accessor(_lox_hopping_enabled);
+ }
+
+private:
+ virtual void resolve();
+
+ //Inputs
+ experts::data_reader_t<twinrx_ctrl::lo_source_t> _lox_src_ch0;
+ experts::data_reader_t<twinrx_ctrl::lo_source_t> _lox_src_ch1;
+ //Outputs
+ experts::data_writer_t<lo_synth_mapping_t> _lox_mapping_synth0;
+ experts::data_writer_t<lo_synth_mapping_t> _lox_mapping_synth1;
+ experts::data_writer_t<bool> _lox_hopping_enabled;
+};
+
+/*!---------------------------------------------------------
+ * twinrx_freq_coercion_expert
+ *
+ * This expert is responsible for calculating the coerced
+ * RF frequency after most settings and modes have been
+ * resolved.
+ *
+ * One instance of this expert is required for each channel
+ * ---------------------------------------------------------
+ */
+class twinrx_freq_coercion_expert : public experts::worker_node_t {
+public:
+ twinrx_freq_coercion_expert(const experts::node_retriever_t& db, std::string ch)
+ : experts::worker_node_t(prepend_ch("twinrx_freq_coercion_expert", ch)),
+ _lo1_freq_c (db, prepend_ch("los/LO1/freq/coerced", ch)),
+ _lo2_freq_c (db, prepend_ch("los/LO2/freq/coerced", ch)),
+ _if_freq_d (db, prepend_ch("if_freq/desired", ch)),
+ _lo1_inj_side (db, prepend_ch("ch/LO1/inj_side", ch)),
+ _lo2_inj_side (db, prepend_ch("ch/LO2/inj_side", ch)),
+ _rf_freq_c (db, prepend_ch("freq/coerced", ch))
+ {
+ bind_accessor(_lo1_freq_c);
+ bind_accessor(_lo2_freq_c);
+ bind_accessor(_if_freq_d);
+ bind_accessor(_lo1_inj_side);
+ bind_accessor(_lo2_inj_side);
+ bind_accessor(_rf_freq_c);
+ }
+
+private:
+ virtual void resolve();
+
+ //Inputs
+ experts::data_reader_t<double> _lo1_freq_c;
+ experts::data_reader_t<double> _lo2_freq_c;
+ experts::data_reader_t<double> _if_freq_d;
+ experts::data_reader_t<lo_inj_side_t> _lo1_inj_side;
+ experts::data_reader_t<lo_inj_side_t> _lo2_inj_side;
+ //Outputs
+ experts::data_writer_t<double> _rf_freq_c;
+};
+
+/*!---------------------------------------------------------
+ * twinrx_nyquist_expert
+ *
+ * This expert is responsible for figuring out the DSP
+ * front-end settings required for each channel
+ *
+ * One instance of this expert is required for each channel
+ * ---------------------------------------------------------
+ */
+class twinrx_nyquist_expert : public experts::worker_node_t {
+public:
+ twinrx_nyquist_expert(const experts::node_retriever_t& db, std::string ch,
+ dboard_iface::sptr db_iface)
+ : experts::worker_node_t(prepend_ch("twinrx_nyquist_expert", ch)),
+ _channel (ch),
+ _codec_conn (ch=="0"?"II":"QQ"), //Ch->ADC Port mapping
+ _if_freq_d (db, prepend_ch("if_freq/desired", ch)),
+ _lo1_inj_side (db, prepend_ch("ch/LO1/inj_side", ch)),
+ _lo2_inj_side (db, prepend_ch("ch/LO2/inj_side", ch)),
+ _if_freq_c (db, prepend_ch("if_freq/coerced", ch)),
+ _db_iface (db_iface)
+ {
+ bind_accessor(_if_freq_d);
+ bind_accessor(_lo1_inj_side);
+ bind_accessor(_lo2_inj_side);
+ bind_accessor(_if_freq_c);
+ }
+
+private:
+ virtual void resolve();
+
+ //Inputs
+ const std::string _channel;
+ const std::string _codec_conn;
+ experts::data_reader_t<double> _if_freq_d;
+ experts::data_reader_t<lo_inj_side_t> _lo1_inj_side;
+ experts::data_reader_t<lo_inj_side_t> _lo2_inj_side;
+ //Outputs
+ experts::data_writer_t<double> _if_freq_c;
+ dboard_iface::sptr _db_iface;
+};
+
+/*!---------------------------------------------------------
+ * twinrx_antenna_expert
+ *
+ * This expert is responsible for translating high-level
+ * antenna selection settings and channel enables to low-level
+ * switch configurations.
+ *
+ * One instance of this expert is required for all channels
+ * ---------------------------------------------------------
+ */
+class twinrx_antenna_expert : public experts::worker_node_t {
+public:
+ twinrx_antenna_expert(const experts::node_retriever_t& db)
+ : experts::worker_node_t("twinrx_antenna_expert"),
+ _antenna_ch0 (db, prepend_ch("antenna", "0")),
+ _antenna_ch1 (db, prepend_ch("antenna", "1")),
+ _enabled_ch0 (db, prepend_ch("enabled", "0")),
+ _enabled_ch1 (db, prepend_ch("enabled", "1")),
+ _lo_export_ch0 (db, prepend_ch("los/all/export", "0")),
+ _lo_export_ch1 (db, prepend_ch("los/all/export", "1")),
+ _ant_mapping (db, "com/ant_mapping"),
+ _cal_mode (db, "com/cal_mode")
+ {
+ bind_accessor(_antenna_ch0);
+ bind_accessor(_antenna_ch1);
+ bind_accessor(_enabled_ch0);
+ bind_accessor(_enabled_ch1);
+ bind_accessor(_lo_export_ch0);
+ bind_accessor(_lo_export_ch1);
+ bind_accessor(_ant_mapping);
+ bind_accessor(_cal_mode);
+ }
+
+private:
+ virtual void resolve();
+
+ //Inputs
+ experts::data_reader_t<std::string> _antenna_ch0;
+ experts::data_reader_t<std::string> _antenna_ch1;
+ experts::data_reader_t<bool> _enabled_ch0;
+ experts::data_reader_t<bool> _enabled_ch1;
+ experts::data_reader_t<bool> _lo_export_ch0;
+ experts::data_reader_t<bool> _lo_export_ch1;
+ //Outputs
+ experts::data_writer_t<twinrx_ctrl::antenna_mapping_t> _ant_mapping;
+ experts::data_writer_t<twinrx_ctrl::cal_mode_t> _cal_mode;
+};
+
+/*!---------------------------------------------------------
+ * twinrx_chan_gain_expert
+ *
+ * This expert is responsible for mapping high-level channel
+ * gain settings to individual attenuator and amp configurations
+ * that are also channel-scoped. This expert will implement
+ * the gain distribution strategy.
+ *
+ * One instance of this expert is required for each channel
+ * ---------------------------------------------------------
+ */
+class twinrx_chan_gain_expert : public experts::worker_node_t {
+public:
+ twinrx_chan_gain_expert(const experts::node_retriever_t& db, std::string ch)
+ : experts::worker_node_t(prepend_ch("twinrx_chan_gain_expert", ch)),
+ _gain (db, prepend_ch("gain", ch)),
+ _gain_profile (db, prepend_ch("gain_profile", ch)),
+ _signal_path (db, prepend_ch("ch/signal_path", ch)),
+ _lb_presel (db, prepend_ch("ch/lb_presel", ch)),
+ _hb_presel (db, prepend_ch("ch/hb_presel", ch)),
+ _ant_mapping (db, "com/ant_mapping"),
+ _input_atten (db, prepend_ch("ch/input_atten", ch)),
+ _lb_atten (db, prepend_ch("ch/lb_atten", ch)),
+ _hb_atten (db, prepend_ch("ch/hb_atten", ch)),
+ _preamp1 (db, prepend_ch("ch/preamp1", ch)),
+ _preamp2 (db, prepend_ch("ch/preamp2", ch))
+ {
+ bind_accessor(_gain);
+ bind_accessor(_gain_profile);
+ bind_accessor(_signal_path);
+ bind_accessor(_lb_presel);
+ bind_accessor(_hb_presel);
+ bind_accessor(_ant_mapping);
+ bind_accessor(_input_atten);
+ bind_accessor(_lb_atten);
+ bind_accessor(_hb_atten);
+ bind_accessor(_preamp1);
+ bind_accessor(_preamp2);
+ }
+
+private:
+ virtual void resolve();
+
+ //Inputs
+ experts::data_reader_t<double> _gain;
+ experts::data_reader_t<std::string> _gain_profile;
+ experts::data_reader_t<twinrx_ctrl::signal_path_t> _signal_path;
+ experts::data_reader_t<twinrx_ctrl::preselector_path_t> _lb_presel;
+ experts::data_reader_t<twinrx_ctrl::preselector_path_t> _hb_presel;
+ experts::data_reader_t<twinrx_ctrl::antenna_mapping_t> _ant_mapping;
+ //Outputs
+ experts::data_writer_t<boost::uint8_t> _input_atten;
+ experts::data_writer_t<boost::uint8_t> _lb_atten;
+ experts::data_writer_t<boost::uint8_t> _hb_atten;
+ experts::data_writer_t<twinrx_ctrl::preamp_state_t> _preamp1;
+ experts::data_writer_t<bool> _preamp2;
+};
+
+/*!---------------------------------------------------------
+ * twinrx_ant_gain_expert
+ *
+ * This expert is responsible for translating between the
+ * channel-scoped low-level gain settings to antenna-scoped
+ * gain settings.
+ *
+ * One instance of this expert is required for all channels
+ * ---------------------------------------------------------
+ */
+class twinrx_ant_gain_expert : public experts::worker_node_t {
+public:
+ twinrx_ant_gain_expert(const experts::node_retriever_t& db)
+ : experts::worker_node_t("twinrx_ant_gain_expert"),
+ _ant_mapping (db, "com/ant_mapping"),
+ _ch0_input_atten (db, prepend_ch("ch/input_atten", "0")),
+ _ch0_preamp1 (db, prepend_ch("ch/preamp1", "0")),
+ _ch0_preamp2 (db, prepend_ch("ch/preamp2", "0")),
+ _ch0_lb_preamp_presel (db, prepend_ch("ch/lb_preamp_presel", "0")),
+ _ch1_input_atten (db, prepend_ch("ch/input_atten", "1")),
+ _ch1_preamp1 (db, prepend_ch("ch/preamp1", "1")),
+ _ch1_preamp2 (db, prepend_ch("ch/preamp2", "1")),
+ _ch1_lb_preamp_presel (db, prepend_ch("ch/lb_preamp_presel", "1")),
+ _ant0_input_atten (db, prepend_ch("ant/input_atten", "0")),
+ _ant0_preamp1 (db, prepend_ch("ant/preamp1", "0")),
+ _ant0_preamp2 (db, prepend_ch("ant/preamp2", "0")),
+ _ant0_lb_preamp_presel(db, prepend_ch("ant/lb_preamp_presel", "0")),
+ _ant1_input_atten (db, prepend_ch("ant/input_atten", "1")),
+ _ant1_preamp1 (db, prepend_ch("ant/preamp1", "1")),
+ _ant1_preamp2 (db, prepend_ch("ant/preamp2", "1")),
+ _ant1_lb_preamp_presel(db, prepend_ch("ant/lb_preamp_presel", "1"))
+ {
+ bind_accessor(_ant_mapping);
+ bind_accessor(_ch0_input_atten);
+ bind_accessor(_ch0_preamp1);
+ bind_accessor(_ch0_preamp2);
+ bind_accessor(_ch0_lb_preamp_presel);
+ bind_accessor(_ch1_input_atten);
+ bind_accessor(_ch1_preamp1);
+ bind_accessor(_ch1_preamp2);
+ bind_accessor(_ch1_lb_preamp_presel);
+ bind_accessor(_ant0_input_atten);
+ bind_accessor(_ant0_preamp1);
+ bind_accessor(_ant0_preamp2);
+ bind_accessor(_ant0_lb_preamp_presel);
+ bind_accessor(_ant1_input_atten);
+ bind_accessor(_ant1_preamp1);
+ bind_accessor(_ant1_preamp2);
+ bind_accessor(_ant1_lb_preamp_presel);
+ }
+
+private:
+ virtual void resolve();
+
+ //Inputs
+ experts::data_reader_t<twinrx_ctrl::antenna_mapping_t> _ant_mapping;
+ experts::data_reader_t<boost::uint8_t> _ch0_input_atten;
+ experts::data_reader_t<twinrx_ctrl::preamp_state_t> _ch0_preamp1;
+ experts::data_reader_t<bool> _ch0_preamp2;
+ experts::data_reader_t<bool> _ch0_lb_preamp_presel;
+ experts::data_reader_t<boost::uint8_t> _ch1_input_atten;
+ experts::data_reader_t<twinrx_ctrl::preamp_state_t> _ch1_preamp1;
+ experts::data_reader_t<bool> _ch1_preamp2;
+ experts::data_reader_t<bool> _ch1_lb_preamp_presel;
+
+ //Outputs
+ experts::data_writer_t<boost::uint8_t> _ant0_input_atten;
+ experts::data_writer_t<twinrx_ctrl::preamp_state_t> _ant0_preamp1;
+ experts::data_writer_t<bool> _ant0_preamp2;
+ experts::data_writer_t<bool> _ant0_lb_preamp_presel;
+ experts::data_writer_t<boost::uint8_t> _ant1_input_atten;
+ experts::data_writer_t<twinrx_ctrl::preamp_state_t> _ant1_preamp1;
+ experts::data_writer_t<bool> _ant1_preamp2;
+ experts::data_writer_t<bool> _ant1_lb_preamp_presel;
+};
+
+/*!---------------------------------------------------------
+ * twinrx_settings_expert
+ *
+ * This expert is responsible for gathering all low-level
+ * settings and writing them to hardware. All LO frequency
+ * settings are cached with a hysteresis. All other settings
+ * are always written to twinrx_ctrl and rely on register
+ * level caching.
+ *
+ * One instance of this expert is required for all channels
+ * ---------------------------------------------------------
+ */
+class twinrx_settings_expert : public experts::worker_node_t {
+public:
+ twinrx_settings_expert(const experts::node_retriever_t& db, twinrx_ctrl::sptr ctrl)
+ : experts::worker_node_t("twinrx_settings_expert"), _ctrl(ctrl),
+ _ch0 (db, "0"),
+ _ch1 (db, "1"),
+ _lo1_synth0_mapping(db, "0/synth/LO1/mapping"),
+ _lo1_synth1_mapping(db, "1/synth/LO1/mapping"),
+ _lo2_synth0_mapping(db, "0/synth/LO2/mapping"),
+ _lo2_synth1_mapping(db, "1/synth/LO2/mapping"),
+ _lo1_hopping_enabled(db, "com/synth/LO1/hopping_enabled"),
+ _lo2_hopping_enabled(db, "com/synth/LO2/hopping_enabled"),
+ _lo1_export_src (db, "com/LO1/export_source"),
+ _lo2_export_src (db, "com/LO2/export_source"),
+ _ant_mapping (db, "com/ant_mapping"),
+ _cal_mode (db, "com/cal_mode")
+ {
+ for (size_t i = 0; i < 2; i++) {
+ ch_settings& ch = (i==1) ? _ch1 : _ch0;
+ bind_accessor(ch.chan_enabled);
+ bind_accessor(ch.preamp1);
+ bind_accessor(ch.preamp2);
+ bind_accessor(ch.lb_preamp_presel);
+ bind_accessor(ch.signal_path);
+ bind_accessor(ch.lb_presel);
+ bind_accessor(ch.hb_presel);
+ bind_accessor(ch.input_atten);
+ bind_accessor(ch.lb_atten);
+ bind_accessor(ch.hb_atten);
+ bind_accessor(ch.lo1_source);
+ bind_accessor(ch.lo2_source);
+ bind_accessor(ch.lo1_freq_d);
+ bind_accessor(ch.lo2_freq_d);
+ bind_accessor(ch.lo1_freq_c);
+ bind_accessor(ch.lo2_freq_c);
+ }
+ bind_accessor(_lo1_synth0_mapping);
+ bind_accessor(_lo1_synth1_mapping);
+ bind_accessor(_lo2_synth0_mapping);
+ bind_accessor(_lo2_synth1_mapping);
+ bind_accessor(_lo1_hopping_enabled);
+ bind_accessor(_lo2_hopping_enabled);
+ bind_accessor(_lo1_export_src);
+ bind_accessor(_lo2_export_src);
+ bind_accessor(_ant_mapping);
+ bind_accessor(_cal_mode);
+ }
+
+private:
+ virtual void resolve();
+ void _resolve_lox_freq(
+ lo_stage_t lo_stage,
+ experts::data_reader_t<double>& ch0_freq_d,
+ experts::data_reader_t<double>& ch1_freq_d,
+ experts::data_writer_t<double>& ch0_freq_c,
+ experts::data_writer_t<double>& ch1_freq_c,
+ twinrx_ctrl::lo_source_t ch0_lo_source,
+ twinrx_ctrl::lo_source_t ch1_lo_source,
+ lo_synth_mapping_t synth0_mapping,
+ lo_synth_mapping_t synth1_mapping,
+ bool hopping_enabled);
+ double _set_lox_synth_freq(lo_stage_t stage, twinrx_ctrl::channel_t ch, double freq);
+
+ class ch_settings {
+ public:
+ ch_settings(const experts::node_retriever_t& db, const std::string& ch) :
+ chan_enabled (db, prepend_ch("enabled", ch)),
+ preamp1 (db, prepend_ch("ant/preamp1", ch)),
+ preamp2 (db, prepend_ch("ant/preamp2", ch)),
+ lb_preamp_presel(db, prepend_ch("ant/lb_preamp_presel", ch)),
+ signal_path (db, prepend_ch("ch/signal_path", ch)),
+ lb_presel (db, prepend_ch("ch/lb_presel", ch)),
+ hb_presel (db, prepend_ch("ch/hb_presel", ch)),
+ input_atten (db, prepend_ch("ant/input_atten", ch)),
+ lb_atten (db, prepend_ch("ch/lb_atten", ch)),
+ hb_atten (db, prepend_ch("ch/hb_atten", ch)),
+ lo1_source (db, prepend_ch("ch/LO1/source", ch)),
+ lo2_source (db, prepend_ch("ch/LO2/source", ch)),
+ lo1_freq_d (db, prepend_ch("los/LO1/freq/desired", ch)),
+ lo2_freq_d (db, prepend_ch("los/LO2/freq/desired", ch)),
+ lo1_freq_c (db, prepend_ch("los/LO1/freq/coerced", ch)),
+ lo2_freq_c (db, prepend_ch("los/LO2/freq/coerced", ch))
+ {}
+
+ //Inputs (channel specific)
+ experts::data_reader_t<bool> chan_enabled;
+ experts::data_reader_t<twinrx_ctrl::preamp_state_t> preamp1;
+ experts::data_reader_t<bool> preamp2;
+ experts::data_reader_t<bool> lb_preamp_presel;
+ experts::data_reader_t<twinrx_ctrl::signal_path_t> signal_path;
+ experts::data_reader_t<twinrx_ctrl::preselector_path_t> lb_presel;
+ experts::data_reader_t<twinrx_ctrl::preselector_path_t> hb_presel;
+ experts::data_reader_t<boost::uint8_t> input_atten;
+ experts::data_reader_t<boost::uint8_t> lb_atten;
+ experts::data_reader_t<boost::uint8_t> hb_atten;
+ experts::data_reader_t<twinrx_ctrl::lo_source_t> lo1_source;
+ experts::data_reader_t<twinrx_ctrl::lo_source_t> lo2_source;
+ experts::data_reader_t<double> lo1_freq_d;
+ experts::data_reader_t<double> lo2_freq_d;
+
+ //Output (channel specific)
+ experts::data_writer_t<double> lo1_freq_c;
+ experts::data_writer_t<double> lo2_freq_c;
+ };
+
+ //External interface
+ twinrx_ctrl::sptr _ctrl;
+
+ //Inputs (channel agnostic)
+ ch_settings _ch0;
+ ch_settings _ch1;
+ experts::data_reader_t<lo_synth_mapping_t> _lo1_synth0_mapping;
+ experts::data_reader_t<lo_synth_mapping_t> _lo1_synth1_mapping;
+ experts::data_reader_t<lo_synth_mapping_t> _lo2_synth0_mapping;
+ experts::data_reader_t<lo_synth_mapping_t> _lo2_synth1_mapping;
+ experts::data_reader_t<bool> _lo1_hopping_enabled;
+ experts::data_reader_t<bool> _lo2_hopping_enabled;
+ experts::data_reader_t<twinrx_ctrl::lo_export_source_t> _lo1_export_src;
+ experts::data_reader_t<twinrx_ctrl::lo_export_source_t> _lo2_export_src;
+ experts::data_reader_t<twinrx_ctrl::antenna_mapping_t> _ant_mapping;
+ experts::data_reader_t<twinrx_ctrl::cal_mode_t> _cal_mode;
+
+ //Outputs (channel agnostic)
+ //None
+
+ //Misc
+ struct lo_freq_cache_t {
+ rf_freq_ppm_t desired;
+ double coerced;
+ };
+ lo_freq_cache_t _cached_lo1_synth0_freq;
+ lo_freq_cache_t _cached_lo2_synth0_freq;
+ lo_freq_cache_t _cached_lo1_synth1_freq;
+ lo_freq_cache_t _cached_lo2_synth1_freq;
+
+ static const bool FORCE_COMMIT;
+};
+
+
+}}}} //namespaces
+
+#endif /* INCLUDED_DBOARD_TWINRX_EXPERTS_HPP */
diff --git a/host/lib/usrp/dboard/twinrx/twinrx_gain_tables.cpp b/host/lib/usrp/dboard/twinrx/twinrx_gain_tables.cpp
new file mode 100644
index 000000000..5cc8b49f3
--- /dev/null
+++ b/host/lib/usrp/dboard/twinrx/twinrx_gain_tables.cpp
@@ -0,0 +1,860 @@
+//
+// Copyright 2016 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#include "twinrx_gain_tables.hpp"
+#include <uhd/exception.hpp>
+#include <boost/assign/list_of.hpp>
+
+using namespace uhd::usrp::dboard::twinrx;
+
+static const std::vector<twinrx_gain_config_t> HIGHBAND1_TABLE = boost::assign::list_of
+ // Index, Gain, Atten1, Atten2, Amp1, Amp2
+ ( twinrx_gain_config_t( 0, -28.3, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 1, -28.3, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 2, -28.3, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 3, -28.3, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 4, -28.3, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 5, -28.3, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 6, -28.3, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 7, -27.3, 31, 30, false, false ) )
+ ( twinrx_gain_config_t( 8, -26.3, 31, 29, false, false ) )
+ ( twinrx_gain_config_t( 9, -25.3, 31, 28, false, false ) )
+ ( twinrx_gain_config_t( 10, -24.3, 31, 27, false, false ) )
+ ( twinrx_gain_config_t( 11, -23.3, 31, 26, false, false ) )
+ ( twinrx_gain_config_t( 12, -22.3, 31, 25, false, false ) )
+ ( twinrx_gain_config_t( 13, -21.3, 31, 24, false, false ) )
+ ( twinrx_gain_config_t( 14, -20.3, 31, 23, false, false ) )
+ ( twinrx_gain_config_t( 15, -19.3, 31, 22, false, false ) )
+ ( twinrx_gain_config_t( 16, -18.3, 31, 21, false, false ) )
+ ( twinrx_gain_config_t( 17, -17.3, 31, 20, false, false ) )
+ ( twinrx_gain_config_t( 18, -16.3, 31, 19, false, false ) )
+ ( twinrx_gain_config_t( 19, -15.3, 31, 18, false, false ) )
+ ( twinrx_gain_config_t( 20, -14.3, 31, 17, false, false ) )
+ ( twinrx_gain_config_t( 21, -13.3, 31, 16, false, false ) )
+ ( twinrx_gain_config_t( 22, -12.3, 31, 15, false, false ) )
+ ( twinrx_gain_config_t( 23, -11.3, 31, 14, false, false ) )
+ ( twinrx_gain_config_t( 24, -10.3, 31, 13, false, false ) )
+ ( twinrx_gain_config_t( 25, -9.3, 31, 12, false, false ) )
+ ( twinrx_gain_config_t( 26, -8.3, 30, 12, false, false ) )
+ ( twinrx_gain_config_t( 27, -7.3, 30, 11, false, false ) )
+ ( twinrx_gain_config_t( 28, -6.3, 29, 11, false, false ) )
+ ( twinrx_gain_config_t( 29, -5.3, 28, 11, false, false ) )
+ ( twinrx_gain_config_t( 30, -4.3, 27, 11, false, false ) )
+ ( twinrx_gain_config_t( 31, -3.3, 27, 10, false, false ) )
+ ( twinrx_gain_config_t( 32, -2.3, 26, 10, false, false ) )
+ ( twinrx_gain_config_t( 33, -1.3, 25, 10, false, false ) )
+ ( twinrx_gain_config_t( 34, -0.3, 24, 10, false, false ) )
+ ( twinrx_gain_config_t( 35, 0.7, 23, 10, false, false ) )
+ ( twinrx_gain_config_t( 36, 1.7, 22, 10, false, false ) )
+ ( twinrx_gain_config_t( 37, 2.7, 21, 10, false, false ) )
+ ( twinrx_gain_config_t( 38, 3.7, 21, 9, false, false ) )
+ ( twinrx_gain_config_t( 39, 4.7, 20, 9, false, false ) )
+ ( twinrx_gain_config_t( 40, 5.7, 19, 9, false, false ) )
+ ( twinrx_gain_config_t( 41, 6.7, 18, 9, false, false ) )
+ ( twinrx_gain_config_t( 42, 7.7, 17, 9, false, false ) )
+ ( twinrx_gain_config_t( 43, 8.7, 16, 9, false, false ) )
+ ( twinrx_gain_config_t( 44, 9.7, 15, 9, false, false ) )
+ ( twinrx_gain_config_t( 45, 10.7, 14, 9, false, false ) )
+ ( twinrx_gain_config_t( 46, 11.7, 13, 9, false, false ) )
+ ( twinrx_gain_config_t( 47, 12.7, 12, 9, false, false ) )
+ ( twinrx_gain_config_t( 48, 13.7, 11, 9, false, false ) )
+ ( twinrx_gain_config_t( 49, 14.7, 10, 9, false, false ) )
+ ( twinrx_gain_config_t( 50, 15.7, 9, 9, false, false ) )
+ ( twinrx_gain_config_t( 51, 16.7, 8, 9, false, false ) )
+ ( twinrx_gain_config_t( 52, 17.7, 7, 9, false, false ) )
+ ( twinrx_gain_config_t( 53, 18.7, 6, 9, false, false ) )
+ ( twinrx_gain_config_t( 54, 19.7, 5, 9, false, false ) )
+ ( twinrx_gain_config_t( 55, 20.7, 4, 9, false, false ) )
+ ( twinrx_gain_config_t( 56, 21.7, 3, 9, false, false ) )
+ ( twinrx_gain_config_t( 57, 22.7, 2, 9, false, false ) )
+ ( twinrx_gain_config_t( 58, 23.7, 1, 9, false, false ) )
+ ( twinrx_gain_config_t( 59, 24.7, 0, 9, false, false ) )
+ ( twinrx_gain_config_t( 60, 25.7, 0, 8, false, false ) )
+ ( twinrx_gain_config_t( 61, 26.7, 0, 7, false, false ) )
+ ( twinrx_gain_config_t( 62, 27.7, 0, 6, false, false ) )
+ ( twinrx_gain_config_t( 63, 28.7, 0, 5, false, false ) )
+ ( twinrx_gain_config_t( 64, 29.7, 0, 4, false, false ) )
+ ( twinrx_gain_config_t( 65, 30.7, 0, 3, false, false ) )
+ ( twinrx_gain_config_t( 66, 31.7, 0, 2, false, false ) )
+ ( twinrx_gain_config_t( 67, 32.7, 0, 1, false, false ) )
+ ( twinrx_gain_config_t( 68, 33.7, 0, 0, false, false ) )
+ ( twinrx_gain_config_t( 69, 33.9, 3, 9, true, false ) )
+ ( twinrx_gain_config_t( 70, 34.9, 2, 9, true, false ) )
+ ( twinrx_gain_config_t( 71, 35.9, 1, 9, true, false ) )
+ ( twinrx_gain_config_t( 72, 36.9, 0, 9, true, false ) )
+ ( twinrx_gain_config_t( 73, 37.9, 0, 8, true, false ) )
+ ( twinrx_gain_config_t( 74, 38.9, 0, 7, true, false ) )
+ ( twinrx_gain_config_t( 75, 39.9, 0, 6, true, false ) )
+ ( twinrx_gain_config_t( 76, 40.9, 0, 5, true, false ) )
+ ( twinrx_gain_config_t( 77, 41.9, 0, 4, true, false ) )
+ ( twinrx_gain_config_t( 78, 42.9, 0, 3, true, false ) )
+ ( twinrx_gain_config_t( 79, 43.9, 0, 2, true, false ) )
+ ( twinrx_gain_config_t( 80, 44.9, 0, 1, true, false ) )
+ ( twinrx_gain_config_t( 81, 45.9, 0, 0, true, false ) )
+ ( twinrx_gain_config_t( 82, 47.3, 1, 10, true, true ) )
+ ( twinrx_gain_config_t( 83, 48.3, 0, 10, true, true ) )
+ ( twinrx_gain_config_t( 84, 49.3, 0, 9, true, true ) )
+ ( twinrx_gain_config_t( 85, 50.3, 0, 8, true, true ) )
+ ( twinrx_gain_config_t( 86, 51.3, 0, 7, true, true ) )
+ ( twinrx_gain_config_t( 87, 52.3, 0, 6, true, true ) )
+ ( twinrx_gain_config_t( 88, 53.3, 0, 5, true, true ) )
+ ( twinrx_gain_config_t( 89, 54.3, 0, 4, true, true ) )
+ ( twinrx_gain_config_t( 90, 55.3, 0, 3, true, true ) )
+ ( twinrx_gain_config_t( 91, 56.3, 0, 2, true, true ) )
+ ( twinrx_gain_config_t( 92, 57.3, 0, 1, true, true ) )
+ ( twinrx_gain_config_t( 93, 58.3, 0, 0, true, true ) )
+;
+
+static const std::vector<twinrx_gain_config_t> HIGHBAND2_TABLE = boost::assign::list_of
+ // Index, Gain, Atten1, Atten2, Amp1, Amp2
+ ( twinrx_gain_config_t( 0, -30.9, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 1, -30.9, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 2, -30.9, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 3, -30.9, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 4, -30.9, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 5, -30.9, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 6, -30.9, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 7, -30.9, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 8, -29.9, 31, 30, false, false ) )
+ ( twinrx_gain_config_t( 9, -28.9, 31, 29, false, false ) )
+ ( twinrx_gain_config_t( 10, -27.9, 31, 28, false, false ) )
+ ( twinrx_gain_config_t( 11, -26.9, 31, 27, false, false ) )
+ ( twinrx_gain_config_t( 12, -25.9, 31, 26, false, false ) )
+ ( twinrx_gain_config_t( 13, -24.9, 31, 25, false, false ) )
+ ( twinrx_gain_config_t( 14, -23.9, 31, 24, false, false ) )
+ ( twinrx_gain_config_t( 15, -22.9, 31, 23, false, false ) )
+ ( twinrx_gain_config_t( 16, -21.9, 31, 22, false, false ) )
+ ( twinrx_gain_config_t( 17, -20.9, 31, 21, false, false ) )
+ ( twinrx_gain_config_t( 18, -19.9, 31, 20, false, false ) )
+ ( twinrx_gain_config_t( 19, -18.9, 31, 19, false, false ) )
+ ( twinrx_gain_config_t( 20, -17.9, 31, 18, false, false ) )
+ ( twinrx_gain_config_t( 21, -16.9, 31, 17, false, false ) )
+ ( twinrx_gain_config_t( 22, -15.9, 31, 16, false, false ) )
+ ( twinrx_gain_config_t( 23, -14.9, 31, 15, false, false ) )
+ ( twinrx_gain_config_t( 24, -13.9, 31, 14, false, false ) )
+ ( twinrx_gain_config_t( 25, -12.9, 31, 13, false, false ) )
+ ( twinrx_gain_config_t( 26, -11.9, 31, 12, false, false ) )
+ ( twinrx_gain_config_t( 27, -10.9, 31, 11, false, false ) )
+ ( twinrx_gain_config_t( 28, -9.9, 30, 11, false, false ) )
+ ( twinrx_gain_config_t( 29, -8.9, 29, 11, false, false ) )
+ ( twinrx_gain_config_t( 30, -7.9, 29, 10, false, false ) )
+ ( twinrx_gain_config_t( 31, -6.9, 28, 10, false, false ) )
+ ( twinrx_gain_config_t( 32, -5.9, 27, 10, false, false ) )
+ ( twinrx_gain_config_t( 33, -4.9, 27, 9, false, false ) )
+ ( twinrx_gain_config_t( 34, -3.9, 26, 9, false, false ) )
+ ( twinrx_gain_config_t( 35, -2.9, 25, 9, false, false ) )
+ ( twinrx_gain_config_t( 36, -1.9, 24, 9, false, false ) )
+ ( twinrx_gain_config_t( 37, -0.9, 23, 9, false, false ) )
+ ( twinrx_gain_config_t( 38, 0.1, 23, 8, false, false ) )
+ ( twinrx_gain_config_t( 39, 1.1, 22, 8, false, false ) )
+ ( twinrx_gain_config_t( 40, 2.1, 21, 8, false, false ) )
+ ( twinrx_gain_config_t( 41, 3.1, 20, 8, false, false ) )
+ ( twinrx_gain_config_t( 42, 4.1, 19, 8, false, false ) )
+ ( twinrx_gain_config_t( 43, 5.1, 18, 8, false, false ) )
+ ( twinrx_gain_config_t( 44, 6.1, 17, 8, false, false ) )
+ ( twinrx_gain_config_t( 45, 7.1, 16, 8, false, false ) )
+ ( twinrx_gain_config_t( 46, 8.1, 15, 8, false, false ) )
+ ( twinrx_gain_config_t( 47, 9.1, 14, 8, false, false ) )
+ ( twinrx_gain_config_t( 48, 10.1, 13, 8, false, false ) )
+ ( twinrx_gain_config_t( 49, 11.1, 12, 8, false, false ) )
+ ( twinrx_gain_config_t( 50, 12.1, 11, 8, false, false ) )
+ ( twinrx_gain_config_t( 51, 13.1, 10, 8, false, false ) )
+ ( twinrx_gain_config_t( 52, 14.1, 9, 8, false, false ) )
+ ( twinrx_gain_config_t( 53, 15.1, 8, 8, false, false ) )
+ ( twinrx_gain_config_t( 54, 16.1, 7, 8, false, false ) )
+ ( twinrx_gain_config_t( 55, 17.1, 6, 8, false, false ) )
+ ( twinrx_gain_config_t( 56, 18.1, 5, 8, false, false ) )
+ ( twinrx_gain_config_t( 57, 19.1, 4, 8, false, false ) )
+ ( twinrx_gain_config_t( 58, 20.1, 3, 8, false, false ) )
+ ( twinrx_gain_config_t( 59, 21.1, 2, 8, false, false ) )
+ ( twinrx_gain_config_t( 60, 22.1, 1, 8, false, false ) )
+ ( twinrx_gain_config_t( 61, 23.1, 0, 8, false, false ) )
+ ( twinrx_gain_config_t( 62, 24.1, 0, 7, false, false ) )
+ ( twinrx_gain_config_t( 63, 25.1, 0, 6, false, false ) )
+ ( twinrx_gain_config_t( 64, 26.1, 0, 5, false, false ) )
+ ( twinrx_gain_config_t( 65, 27.1, 0, 4, false, false ) )
+ ( twinrx_gain_config_t( 66, 28.1, 0, 3, false, false ) )
+ ( twinrx_gain_config_t( 67, 29.1, 0, 2, false, false ) )
+ ( twinrx_gain_config_t( 68, 30.1, 0, 1, false, false ) )
+ ( twinrx_gain_config_t( 69, 31.9, 0, 10, false, true ) )
+ ( twinrx_gain_config_t( 70, 31.9, 0, 10, false, true ) )
+ ( twinrx_gain_config_t( 71, 32.9, 0, 9, false, true ) )
+ ( twinrx_gain_config_t( 72, 33.9, 0, 8, false, true ) )
+ ( twinrx_gain_config_t( 73, 34.9, 0, 7, false, true ) )
+ ( twinrx_gain_config_t( 74, 35.9, 0, 6, false, true ) )
+ ( twinrx_gain_config_t( 75, 36.9, 0, 5, false, true ) )
+ ( twinrx_gain_config_t( 76, 38.6, 0, 6, true, false ) )
+ ( twinrx_gain_config_t( 77, 39.6, 0, 5, true, false ) )
+ ( twinrx_gain_config_t( 78, 40.6, 0, 4, true, false ) )
+ ( twinrx_gain_config_t( 79, 41.6, 0, 3, true, false ) )
+ ( twinrx_gain_config_t( 80, 42.6, 0, 2, true, false ) )
+ ( twinrx_gain_config_t( 81, 43.6, 0, 1, true, false ) )
+ ( twinrx_gain_config_t( 82, 44.4, 2, 9, true, true ) )
+ ( twinrx_gain_config_t( 83, 45.4, 1, 9, true, true ) )
+ ( twinrx_gain_config_t( 84, 46.4, 0, 9, true, true ) )
+ ( twinrx_gain_config_t( 85, 47.4, 0, 8, true, true ) )
+ ( twinrx_gain_config_t( 86, 48.4, 0, 7, true, true ) )
+ ( twinrx_gain_config_t( 87, 49.4, 0, 6, true, true ) )
+ ( twinrx_gain_config_t( 88, 50.4, 0, 5, true, true ) )
+ ( twinrx_gain_config_t( 89, 51.4, 0, 4, true, true ) )
+ ( twinrx_gain_config_t( 90, 52.4, 0, 3, true, true ) )
+ ( twinrx_gain_config_t( 91, 53.4, 0, 2, true, true ) )
+ ( twinrx_gain_config_t( 92, 54.4, 0, 1, true, true ) )
+ ( twinrx_gain_config_t( 93, 55.4, 0, 0, true, true ) )
+;
+
+static const std::vector<twinrx_gain_config_t> HIGHBAND3_TABLE = boost::assign::list_of
+ // Index, Gain, Atten1, Atten2, Amp1, Amp2
+ ( twinrx_gain_config_t( 0, -31.1, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 1, -31.1, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 2, -31.1, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 3, -31.1, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 4, -31.1, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 5, -31.1, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 6, -31.1, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 7, -30.1, 31, 30, false, false ) )
+ ( twinrx_gain_config_t( 8, -29.1, 31, 29, false, false ) )
+ ( twinrx_gain_config_t( 9, -28.1, 31, 28, false, false ) )
+ ( twinrx_gain_config_t( 10, -27.1, 31, 27, false, false ) )
+ ( twinrx_gain_config_t( 11, -26.1, 31, 26, false, false ) )
+ ( twinrx_gain_config_t( 12, -25.1, 31, 25, false, false ) )
+ ( twinrx_gain_config_t( 13, -24.1, 31, 24, false, false ) )
+ ( twinrx_gain_config_t( 14, -23.1, 31, 23, false, false ) )
+ ( twinrx_gain_config_t( 15, -22.1, 31, 22, false, false ) )
+ ( twinrx_gain_config_t( 16, -21.1, 31, 21, false, false ) )
+ ( twinrx_gain_config_t( 17, -20.1, 31, 20, false, false ) )
+ ( twinrx_gain_config_t( 18, -19.1, 31, 19, false, false ) )
+ ( twinrx_gain_config_t( 19, -18.1, 31, 18, false, false ) )
+ ( twinrx_gain_config_t( 20, -17.1, 31, 17, false, false ) )
+ ( twinrx_gain_config_t( 21, -16.1, 31, 16, false, false ) )
+ ( twinrx_gain_config_t( 22, -15.1, 31, 15, false, false ) )
+ ( twinrx_gain_config_t( 23, -14.1, 31, 14, false, false ) )
+ ( twinrx_gain_config_t( 24, -13.1, 31, 13, false, false ) )
+ ( twinrx_gain_config_t( 25, -12.1, 30, 13, false, false ) )
+ ( twinrx_gain_config_t( 26, -11.1, 30, 12, false, false ) )
+ ( twinrx_gain_config_t( 27, -10.1, 29, 12, false, false ) )
+ ( twinrx_gain_config_t( 28, -9.1, 28, 12, false, false ) )
+ ( twinrx_gain_config_t( 29, -8.1, 28, 11, false, false ) )
+ ( twinrx_gain_config_t( 30, -7.1, 27, 11, false, false ) )
+ ( twinrx_gain_config_t( 31, -6.1, 26, 11, false, false ) )
+ ( twinrx_gain_config_t( 32, -5.1, 26, 10, false, false ) )
+ ( twinrx_gain_config_t( 33, -4.1, 25, 10, false, false ) )
+ ( twinrx_gain_config_t( 34, -3.1, 24, 10, false, false ) )
+ ( twinrx_gain_config_t( 35, -2.1, 23, 10, false, false ) )
+ ( twinrx_gain_config_t( 36, -1.1, 22, 10, false, false ) )
+ ( twinrx_gain_config_t( 37, -0.1, 21, 10, false, false ) )
+ ( twinrx_gain_config_t( 38, 0.9, 21, 9, false, false ) )
+ ( twinrx_gain_config_t( 39, 1.9, 20, 9, false, false ) )
+ ( twinrx_gain_config_t( 40, 2.9, 19, 9, false, false ) )
+ ( twinrx_gain_config_t( 41, 3.9, 18, 9, false, false ) )
+ ( twinrx_gain_config_t( 42, 4.9, 17, 9, false, false ) )
+ ( twinrx_gain_config_t( 43, 5.9, 16, 9, false, false ) )
+ ( twinrx_gain_config_t( 44, 6.9, 15, 9, false, false ) )
+ ( twinrx_gain_config_t( 45, 7.9, 14, 9, false, false ) )
+ ( twinrx_gain_config_t( 46, 8.9, 13, 9, false, false ) )
+ ( twinrx_gain_config_t( 47, 9.9, 12, 9, false, false ) )
+ ( twinrx_gain_config_t( 48, 10.9, 11, 9, false, false ) )
+ ( twinrx_gain_config_t( 49, 11.9, 10, 9, false, false ) )
+ ( twinrx_gain_config_t( 50, 12.9, 9, 9, false, false ) )
+ ( twinrx_gain_config_t( 51, 13.9, 8, 9, false, false ) )
+ ( twinrx_gain_config_t( 52, 14.9, 7, 9, false, false ) )
+ ( twinrx_gain_config_t( 53, 15.9, 6, 9, false, false ) )
+ ( twinrx_gain_config_t( 54, 16.9, 5, 9, false, false ) )
+ ( twinrx_gain_config_t( 55, 17.9, 4, 9, false, false ) )
+ ( twinrx_gain_config_t( 56, 18.9, 3, 9, false, false ) )
+ ( twinrx_gain_config_t( 57, 19.9, 2, 9, false, false ) )
+ ( twinrx_gain_config_t( 58, 20.9, 1, 9, false, false ) )
+ ( twinrx_gain_config_t( 59, 21.9, 0, 9, false, false ) )
+ ( twinrx_gain_config_t( 60, 22.9, 0, 8, false, false ) )
+ ( twinrx_gain_config_t( 61, 23.9, 0, 7, false, false ) )
+ ( twinrx_gain_config_t( 62, 24.9, 0, 6, false, false ) )
+ ( twinrx_gain_config_t( 63, 25.9, 0, 5, false, false ) )
+ ( twinrx_gain_config_t( 64, 26.9, 0, 4, false, false ) )
+ ( twinrx_gain_config_t( 65, 27.9, 0, 3, false, false ) )
+ ( twinrx_gain_config_t( 66, 28.9, 0, 2, false, false ) )
+ ( twinrx_gain_config_t( 67, 29.9, 0, 1, false, false ) )
+ ( twinrx_gain_config_t( 68, 31.3, 0, 9, false, true ) )
+ ( twinrx_gain_config_t( 69, 32.3, 0, 8, false, true ) )
+ ( twinrx_gain_config_t( 70, 33.3, 0, 7, false, true ) )
+ ( twinrx_gain_config_t( 71, 34.3, 0, 6, false, true ) )
+ ( twinrx_gain_config_t( 72, 35.3, 0, 5, false, true ) )
+ ( twinrx_gain_config_t( 73, 36.3, 0, 4, false, true ) )
+ ( twinrx_gain_config_t( 74, 37.3, 0, 3, false, true ) )
+ ( twinrx_gain_config_t( 75, 37.6, 0, 9, true, false ) )
+ ( twinrx_gain_config_t( 76, 38.6, 0, 8, true, false ) )
+ ( twinrx_gain_config_t( 77, 39.6, 0, 7, true, false ) )
+ ( twinrx_gain_config_t( 78, 40.6, 0, 6, true, false ) )
+ ( twinrx_gain_config_t( 79, 41.6, 0, 5, true, false ) )
+ ( twinrx_gain_config_t( 80, 42.6, 0, 4, true, false ) )
+ ( twinrx_gain_config_t( 81, 43.6, 0, 3, true, false ) )
+ ( twinrx_gain_config_t( 82, 44.6, 0, 2, true, false ) )
+ ( twinrx_gain_config_t( 83, 45.6, 0, 1, true, false ) )
+ ( twinrx_gain_config_t( 84, 47.0, 0, 9, true, true ) )
+ ( twinrx_gain_config_t( 85, 48.0, 0, 8, true, true ) )
+ ( twinrx_gain_config_t( 86, 49.0, 0, 7, true, true ) )
+ ( twinrx_gain_config_t( 87, 50.0, 0, 6, true, true ) )
+ ( twinrx_gain_config_t( 88, 51.0, 0, 5, true, true ) )
+ ( twinrx_gain_config_t( 89, 52.0, 0, 4, true, true ) )
+ ( twinrx_gain_config_t( 90, 53.0, 0, 3, true, true ) )
+ ( twinrx_gain_config_t( 91, 54.0, 0, 2, true, true ) )
+ ( twinrx_gain_config_t( 92, 55.0, 0, 1, true, true ) )
+ ( twinrx_gain_config_t( 93, 56.0, 0, 0, true, true ) )
+;
+
+static const std::vector<twinrx_gain_config_t> HIGHBAND4_TABLE = boost::assign::list_of
+ // Index, Gain, Atten1, Atten2, Amp1, Amp2
+ ( twinrx_gain_config_t( 0, -37.2, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 1, -37.2, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 2, -37.2, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 3, -37.2, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 4, -37.2, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 5, -37.2, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 6, -37.2, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 7, -37.2, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 8, -37.2, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 9, -36.2, 31, 30, false, false ) )
+ ( twinrx_gain_config_t( 10, -35.2, 31, 29, false, false ) )
+ ( twinrx_gain_config_t( 11, -34.2, 31, 28, false, false ) )
+ ( twinrx_gain_config_t( 12, -33.2, 31, 27, false, false ) )
+ ( twinrx_gain_config_t( 13, -32.2, 31, 26, false, false ) )
+ ( twinrx_gain_config_t( 14, -31.2, 31, 25, false, false ) )
+ ( twinrx_gain_config_t( 15, -30.2, 31, 24, false, false ) )
+ ( twinrx_gain_config_t( 16, -29.2, 31, 23, false, false ) )
+ ( twinrx_gain_config_t( 17, -28.2, 31, 22, false, false ) )
+ ( twinrx_gain_config_t( 18, -27.2, 31, 21, false, false ) )
+ ( twinrx_gain_config_t( 19, -26.2, 31, 20, false, false ) )
+ ( twinrx_gain_config_t( 20, -25.2, 31, 19, false, false ) )
+ ( twinrx_gain_config_t( 21, -24.2, 31, 18, false, false ) )
+ ( twinrx_gain_config_t( 22, -23.2, 31, 17, false, false ) )
+ ( twinrx_gain_config_t( 23, -22.2, 31, 16, false, false ) )
+ ( twinrx_gain_config_t( 24, -21.2, 31, 15, false, false ) )
+ ( twinrx_gain_config_t( 25, -20.2, 31, 14, false, false ) )
+ ( twinrx_gain_config_t( 26, -19.2, 31, 13, false, false ) )
+ ( twinrx_gain_config_t( 27, -18.2, 31, 12, false, false ) )
+ ( twinrx_gain_config_t( 28, -17.2, 31, 11, false, false ) )
+ ( twinrx_gain_config_t( 29, -16.2, 31, 10, false, false ) )
+ ( twinrx_gain_config_t( 30, -15.2, 30, 10, false, false ) )
+ ( twinrx_gain_config_t( 31, -14.2, 30, 9, false, false ) )
+ ( twinrx_gain_config_t( 32, -13.2, 29, 9, false, false ) )
+ ( twinrx_gain_config_t( 33, -12.2, 28, 9, false, false ) )
+ ( twinrx_gain_config_t( 34, -11.2, 28, 8, false, false ) )
+ ( twinrx_gain_config_t( 35, -10.2, 27, 8, false, false ) )
+ ( twinrx_gain_config_t( 36, -9.2, 27, 7, false, false ) )
+ ( twinrx_gain_config_t( 37, -8.2, 26, 7, false, false ) )
+ ( twinrx_gain_config_t( 38, -7.2, 25, 7, false, false ) )
+ ( twinrx_gain_config_t( 39, -6.2, 24, 7, false, false ) )
+ ( twinrx_gain_config_t( 40, -5.2, 24, 6, false, false ) )
+ ( twinrx_gain_config_t( 41, -4.2, 23, 6, false, false ) )
+ ( twinrx_gain_config_t( 42, -3.2, 22, 6, false, false ) )
+ ( twinrx_gain_config_t( 43, -2.2, 21, 6, false, false ) )
+ ( twinrx_gain_config_t( 44, -1.2, 20, 6, false, false ) )
+ ( twinrx_gain_config_t( 45, -0.2, 19, 6, false, false ) )
+ ( twinrx_gain_config_t( 46, 0.8, 18, 6, false, false ) )
+ ( twinrx_gain_config_t( 47, 1.8, 17, 6, false, false ) )
+ ( twinrx_gain_config_t( 48, 2.8, 16, 6, false, false ) )
+ ( twinrx_gain_config_t( 49, 3.8, 16, 5, false, false ) )
+ ( twinrx_gain_config_t( 50, 4.8, 15, 5, false, false ) )
+ ( twinrx_gain_config_t( 51, 5.8, 14, 5, false, false ) )
+ ( twinrx_gain_config_t( 52, 6.8, 13, 5, false, false ) )
+ ( twinrx_gain_config_t( 53, 7.8, 12, 5, false, false ) )
+ ( twinrx_gain_config_t( 54, 8.8, 11, 5, false, false ) )
+ ( twinrx_gain_config_t( 55, 9.8, 10, 5, false, false ) )
+ ( twinrx_gain_config_t( 56, 10.8, 9, 5, false, false ) )
+ ( twinrx_gain_config_t( 57, 11.8, 8, 5, false, false ) )
+ ( twinrx_gain_config_t( 58, 12.8, 7, 5, false, false ) )
+ ( twinrx_gain_config_t( 59, 13.8, 6, 5, false, false ) )
+ ( twinrx_gain_config_t( 60, 14.8, 5, 5, false, false ) )
+ ( twinrx_gain_config_t( 61, 15.8, 4, 5, false, false ) )
+ ( twinrx_gain_config_t( 62, 16.8, 3, 5, false, false ) )
+ ( twinrx_gain_config_t( 63, 17.8, 2, 5, false, false ) )
+ ( twinrx_gain_config_t( 64, 18.8, 1, 5, false, false ) )
+ ( twinrx_gain_config_t( 65, 19.8, 0, 5, false, false ) )
+ ( twinrx_gain_config_t( 66, 20.8, 0, 4, false, false ) )
+ ( twinrx_gain_config_t( 67, 21.8, 0, 3, false, false ) )
+ ( twinrx_gain_config_t( 68, 22.8, 0, 2, false, false ) )
+ ( twinrx_gain_config_t( 69, 23.8, 0, 1, false, false ) )
+ ( twinrx_gain_config_t( 70, 24.8, 0, 0, false, false ) )
+ ( twinrx_gain_config_t( 71, 26.1, 0, 6, false, true ) )
+ ( twinrx_gain_config_t( 72, 26.1, 0, 6, false, true ) )
+ ( twinrx_gain_config_t( 73, 27.1, 0, 5, false, true ) )
+ ( twinrx_gain_config_t( 74, 28.1, 0, 4, false, true ) )
+ ( twinrx_gain_config_t( 75, 29.1, 0, 3, false, true ) )
+ ( twinrx_gain_config_t( 76, 30.1, 0, 2, false, true ) )
+ ( twinrx_gain_config_t( 77, 31.1, 0, 1, false, true ) )
+ ( twinrx_gain_config_t( 78, 32.1, 0, 0, false, true ) )
+ ( twinrx_gain_config_t( 79, 33.3, 0, 7, true, false ) )
+ ( twinrx_gain_config_t( 80, 34.3, 0, 6, true, false ) )
+ ( twinrx_gain_config_t( 81, 35.3, 0, 5, true, false ) )
+ ( twinrx_gain_config_t( 82, 36.3, 0, 4, true, false ) )
+ ( twinrx_gain_config_t( 83, 37.3, 0, 3, true, false ) )
+ ( twinrx_gain_config_t( 84, 38.3, 0, 2, true, false ) )
+ ( twinrx_gain_config_t( 85, 39.3, 0, 1, true, false ) )
+ ( twinrx_gain_config_t( 86, 40.3, 0, 0, true, false ) )
+ ( twinrx_gain_config_t( 87, 41.6, 0, 6, true, true ) )
+ ( twinrx_gain_config_t( 88, 42.6, 0, 5, true, true ) )
+ ( twinrx_gain_config_t( 89, 43.6, 0, 4, true, true ) )
+ ( twinrx_gain_config_t( 90, 44.6, 0, 3, true, true ) )
+ ( twinrx_gain_config_t( 91, 45.6, 0, 2, true, true ) )
+ ( twinrx_gain_config_t( 92, 46.6, 0, 1, true, true ) )
+ ( twinrx_gain_config_t( 93, 47.6, 0, 0, true, true ) )
+;
+
+static const std::vector<twinrx_gain_config_t> LOWBAND1_TABLE = boost::assign::list_of
+ // Index, Gain, Atten1, Atten2, Amp1, Amp2
+ ( twinrx_gain_config_t( 0, -31.1, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 1, -30.1, 31, 30, false, false ) )
+ ( twinrx_gain_config_t( 2, -29.1, 31, 29, false, false ) )
+ ( twinrx_gain_config_t( 3, -28.1, 31, 28, false, false ) )
+ ( twinrx_gain_config_t( 4, -27.1, 31, 27, false, false ) )
+ ( twinrx_gain_config_t( 5, -26.1, 31, 26, false, false ) )
+ ( twinrx_gain_config_t( 6, -25.1, 31, 25, false, false ) )
+ ( twinrx_gain_config_t( 7, -24.1, 31, 24, false, false ) )
+ ( twinrx_gain_config_t( 8, -23.1, 31, 23, false, false ) )
+ ( twinrx_gain_config_t( 9, -22.1, 31, 22, false, false ) )
+ ( twinrx_gain_config_t( 10, -21.1, 31, 21, false, false ) )
+ ( twinrx_gain_config_t( 11, -20.1, 31, 20, false, false ) )
+ ( twinrx_gain_config_t( 12, -19.1, 31, 19, false, false ) )
+ ( twinrx_gain_config_t( 13, -18.1, 31, 18, false, false ) )
+ ( twinrx_gain_config_t( 14, -17.1, 31, 17, false, false ) )
+ ( twinrx_gain_config_t( 15, -16.1, 31, 16, false, false ) )
+ ( twinrx_gain_config_t( 16, -15.1, 31, 15, false, false ) )
+ ( twinrx_gain_config_t( 17, -14.1, 31, 14, false, false ) )
+ ( twinrx_gain_config_t( 18, -13.1, 31, 13, false, false ) )
+ ( twinrx_gain_config_t( 19, -12.1, 31, 12, false, false ) )
+ ( twinrx_gain_config_t( 20, -11.1, 31, 11, false, false ) )
+ ( twinrx_gain_config_t( 21, -10.1, 31, 10, false, false ) )
+ ( twinrx_gain_config_t( 22, -9.1, 31, 9, false, false ) )
+ ( twinrx_gain_config_t( 23, -8.1, 31, 8, false, false ) )
+ ( twinrx_gain_config_t( 24, -7.1, 31, 7, false, false ) )
+ ( twinrx_gain_config_t( 25, -6.1, 31, 6, false, false ) )
+ ( twinrx_gain_config_t( 26, -5.1, 31, 5, false, false ) )
+ ( twinrx_gain_config_t( 27, -4.1, 31, 4, false, false ) )
+ ( twinrx_gain_config_t( 28, -3.1, 31, 3, false, false ) )
+ ( twinrx_gain_config_t( 29, -2.1, 31, 2, false, false ) )
+ ( twinrx_gain_config_t( 30, -1.1, 31, 1, false, false ) )
+ ( twinrx_gain_config_t( 31, -0.1, 31, 0, false, false ) )
+ ( twinrx_gain_config_t( 32, 0.9, 30, 0, false, false ) )
+ ( twinrx_gain_config_t( 33, 1.9, 29, 0, false, false ) )
+ ( twinrx_gain_config_t( 34, 2.9, 28, 0, false, false ) )
+ ( twinrx_gain_config_t( 35, 3.9, 27, 0, false, false ) )
+ ( twinrx_gain_config_t( 36, 4.9, 26, 0, false, false ) )
+ ( twinrx_gain_config_t( 37, 5.9, 25, 0, false, false ) )
+ ( twinrx_gain_config_t( 38, 6.9, 24, 0, false, false ) )
+ ( twinrx_gain_config_t( 39, 7.9, 23, 0, false, false ) )
+ ( twinrx_gain_config_t( 40, 8.9, 22, 0, false, false ) )
+ ( twinrx_gain_config_t( 41, 9.9, 21, 0, false, false ) )
+ ( twinrx_gain_config_t( 42, 10.9, 20, 0, false, false ) )
+ ( twinrx_gain_config_t( 43, 11.9, 19, 0, false, false ) )
+ ( twinrx_gain_config_t( 44, 12.9, 18, 0, false, false ) )
+ ( twinrx_gain_config_t( 45, 13.9, 17, 0, false, false ) )
+ ( twinrx_gain_config_t( 46, 14.9, 16, 0, false, false ) )
+ ( twinrx_gain_config_t( 47, 15.9, 15, 0, false, false ) )
+ ( twinrx_gain_config_t( 48, 16.9, 14, 0, false, false ) )
+ ( twinrx_gain_config_t( 49, 17.9, 13, 0, false, false ) )
+ ( twinrx_gain_config_t( 50, 18.9, 12, 0, false, false ) )
+ ( twinrx_gain_config_t( 51, 19.9, 11, 0, false, false ) )
+ ( twinrx_gain_config_t( 52, 20.9, 10, 0, false, false ) )
+ ( twinrx_gain_config_t( 53, 21.9, 9, 0, false, false ) )
+ ( twinrx_gain_config_t( 54, 22.9, 8, 0, false, false ) )
+ ( twinrx_gain_config_t( 55, 23.9, 7, 0, false, false ) )
+ ( twinrx_gain_config_t( 56, 24.9, 6, 0, false, false ) )
+ ( twinrx_gain_config_t( 57, 25.9, 5, 0, false, false ) )
+ ( twinrx_gain_config_t( 58, 26.9, 4, 0, false, false ) )
+ ( twinrx_gain_config_t( 59, 27.9, 3, 0, false, false ) )
+ ( twinrx_gain_config_t( 60, 28.9, 2, 0, false, false ) )
+ ( twinrx_gain_config_t( 61, 29.9, 1, 0, false, false ) )
+ ( twinrx_gain_config_t( 62, 30.9, 0, 0, false, false ) )
+ ( twinrx_gain_config_t( 63, 31.2, 4, 11, false, true ) )
+ ( twinrx_gain_config_t( 64, 32.2, 3, 11, false, true ) )
+ ( twinrx_gain_config_t( 65, 33.2, 2, 11, false, true ) )
+ ( twinrx_gain_config_t( 66, 34.2, 1, 11, false, true ) )
+ ( twinrx_gain_config_t( 67, 35.2, 0, 11, false, true ) )
+ ( twinrx_gain_config_t( 68, 36.2, 10, 0, true, false ) )
+ ( twinrx_gain_config_t( 69, 37.2, 9, 0, true, false ) )
+ ( twinrx_gain_config_t( 70, 38.2, 8, 0, true, false ) )
+ ( twinrx_gain_config_t( 71, 39.2, 7, 0, true, false ) )
+ ( twinrx_gain_config_t( 72, 40.2, 6, 0, true, false ) )
+ ( twinrx_gain_config_t( 73, 41.2, 5, 0, true, false ) )
+ ( twinrx_gain_config_t( 74, 42.2, 4, 0, true, false ) )
+ ( twinrx_gain_config_t( 75, 43.2, 3, 0, true, false ) )
+ ( twinrx_gain_config_t( 76, 44.2, 2, 0, true, false ) )
+ ( twinrx_gain_config_t( 77, 45.2, 1, 0, true, false ) )
+ ( twinrx_gain_config_t( 78, 46.2, 0, 0, true, false ) )
+ ( twinrx_gain_config_t( 79, 47.5, 4, 10, true, true ) )
+ ( twinrx_gain_config_t( 80, 48.5, 3, 10, true, true ) )
+ ( twinrx_gain_config_t( 81, 49.5, 3, 9, true, true ) )
+ ( twinrx_gain_config_t( 82, 50.5, 2, 9, true, true ) )
+ ( twinrx_gain_config_t( 83, 51.5, 1, 9, true, true ) )
+ ( twinrx_gain_config_t( 84, 52.5, 1, 8, true, true ) )
+ ( twinrx_gain_config_t( 85, 53.5, 0, 8, true, true ) )
+ ( twinrx_gain_config_t( 86, 54.5, 0, 7, true, true ) )
+ ( twinrx_gain_config_t( 87, 55.5, 0, 6, true, true ) )
+ ( twinrx_gain_config_t( 88, 56.5, 0, 5, true, true ) )
+ ( twinrx_gain_config_t( 89, 57.5, 0, 4, true, true ) )
+ ( twinrx_gain_config_t( 90, 58.5, 0, 3, true, true ) )
+ ( twinrx_gain_config_t( 91, 59.5, 0, 2, true, true ) )
+ ( twinrx_gain_config_t( 92, 60.5, 0, 1, true, true ) )
+ ( twinrx_gain_config_t( 93, 61.5, 0, 0, true, true ) )
+;
+
+static const std::vector<twinrx_gain_config_t> LOWBAND2_TABLE = boost::assign::list_of
+ // Index, Gain, Atten1, Atten2, Amp1, Amp2
+ ( twinrx_gain_config_t( 0, -33.4, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 1, -33.4, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 2, -32.4, 31, 30, false, false ) )
+ ( twinrx_gain_config_t( 3, -31.4, 31, 29, false, false ) )
+ ( twinrx_gain_config_t( 4, -30.4, 31, 28, false, false ) )
+ ( twinrx_gain_config_t( 5, -29.4, 31, 27, false, false ) )
+ ( twinrx_gain_config_t( 6, -28.4, 31, 26, false, false ) )
+ ( twinrx_gain_config_t( 7, -27.4, 31, 25, false, false ) )
+ ( twinrx_gain_config_t( 8, -26.4, 31, 24, false, false ) )
+ ( twinrx_gain_config_t( 9, -25.4, 31, 23, false, false ) )
+ ( twinrx_gain_config_t( 10, -24.4, 31, 22, false, false ) )
+ ( twinrx_gain_config_t( 11, -23.4, 31, 21, false, false ) )
+ ( twinrx_gain_config_t( 12, -22.4, 31, 20, false, false ) )
+ ( twinrx_gain_config_t( 13, -21.4, 31, 19, false, false ) )
+ ( twinrx_gain_config_t( 14, -20.4, 31, 18, false, false ) )
+ ( twinrx_gain_config_t( 15, -19.4, 31, 17, false, false ) )
+ ( twinrx_gain_config_t( 16, -18.4, 31, 16, false, false ) )
+ ( twinrx_gain_config_t( 17, -17.4, 31, 15, false, false ) )
+ ( twinrx_gain_config_t( 18, -16.4, 31, 14, false, false ) )
+ ( twinrx_gain_config_t( 19, -15.4, 31, 13, false, false ) )
+ ( twinrx_gain_config_t( 20, -14.4, 31, 12, false, false ) )
+ ( twinrx_gain_config_t( 21, -13.4, 31, 11, false, false ) )
+ ( twinrx_gain_config_t( 22, -12.4, 31, 10, false, false ) )
+ ( twinrx_gain_config_t( 23, -11.4, 31, 9, false, false ) )
+ ( twinrx_gain_config_t( 24, -10.4, 31, 8, false, false ) )
+ ( twinrx_gain_config_t( 25, -9.4, 31, 7, false, false ) )
+ ( twinrx_gain_config_t( 26, -8.4, 31, 6, false, false ) )
+ ( twinrx_gain_config_t( 27, -7.4, 31, 5, false, false ) )
+ ( twinrx_gain_config_t( 28, -6.4, 31, 4, false, false ) )
+ ( twinrx_gain_config_t( 29, -5.4, 31, 3, false, false ) )
+ ( twinrx_gain_config_t( 30, -4.4, 31, 2, false, false ) )
+ ( twinrx_gain_config_t( 31, -3.4, 31, 1, false, false ) )
+ ( twinrx_gain_config_t( 32, -2.4, 31, 0, false, false ) )
+ ( twinrx_gain_config_t( 33, -1.4, 30, 0, false, false ) )
+ ( twinrx_gain_config_t( 34, -0.4, 29, 0, false, false ) )
+ ( twinrx_gain_config_t( 35, 0.6, 28, 0, false, false ) )
+ ( twinrx_gain_config_t( 36, 1.6, 27, 0, false, false ) )
+ ( twinrx_gain_config_t( 37, 2.6, 26, 0, false, false ) )
+ ( twinrx_gain_config_t( 38, 3.6, 25, 0, false, false ) )
+ ( twinrx_gain_config_t( 39, 4.6, 24, 0, false, false ) )
+ ( twinrx_gain_config_t( 40, 5.6, 23, 0, false, false ) )
+ ( twinrx_gain_config_t( 41, 6.6, 22, 0, false, false ) )
+ ( twinrx_gain_config_t( 42, 7.6, 21, 0, false, false ) )
+ ( twinrx_gain_config_t( 43, 8.6, 20, 0, false, false ) )
+ ( twinrx_gain_config_t( 44, 9.6, 19, 0, false, false ) )
+ ( twinrx_gain_config_t( 45, 10.6, 18, 0, false, false ) )
+ ( twinrx_gain_config_t( 46, 11.6, 17, 0, false, false ) )
+ ( twinrx_gain_config_t( 47, 12.6, 16, 0, false, false ) )
+ ( twinrx_gain_config_t( 48, 13.6, 15, 0, false, false ) )
+ ( twinrx_gain_config_t( 49, 14.6, 14, 0, false, false ) )
+ ( twinrx_gain_config_t( 50, 15.6, 13, 0, false, false ) )
+ ( twinrx_gain_config_t( 51, 16.6, 12, 0, false, false ) )
+ ( twinrx_gain_config_t( 52, 17.6, 11, 0, false, false ) )
+ ( twinrx_gain_config_t( 53, 18.6, 10, 0, false, false ) )
+ ( twinrx_gain_config_t( 54, 19.6, 9, 0, false, false ) )
+ ( twinrx_gain_config_t( 55, 20.6, 8, 0, false, false ) )
+ ( twinrx_gain_config_t( 56, 21.6, 7, 0, false, false ) )
+ ( twinrx_gain_config_t( 57, 22.6, 6, 0, false, false ) )
+ ( twinrx_gain_config_t( 58, 23.6, 5, 0, false, false ) )
+ ( twinrx_gain_config_t( 59, 24.6, 4, 0, false, false ) )
+ ( twinrx_gain_config_t( 60, 25.6, 3, 0, false, false ) )
+ ( twinrx_gain_config_t( 61, 26.6, 2, 0, false, false ) )
+ ( twinrx_gain_config_t( 62, 27.6, 1, 0, false, false ) )
+ ( twinrx_gain_config_t( 63, 28.6, 0, 0, false, false ) )
+ ( twinrx_gain_config_t( 64, 29.7, 5, 9, false, true ) )
+ ( twinrx_gain_config_t( 65, 30.7, 4, 9, false, true ) )
+ ( twinrx_gain_config_t( 66, 31.7, 3, 9, false, true ) )
+ ( twinrx_gain_config_t( 67, 32.7, 2, 9, false, true ) )
+ ( twinrx_gain_config_t( 68, 33.7, 1, 9, false, true ) )
+ ( twinrx_gain_config_t( 69, 34.7, 0, 9, false, true ) )
+ ( twinrx_gain_config_t( 70, 35.7, 0, 8, false, true ) )
+ ( twinrx_gain_config_t( 71, 36.7, 7, 0, true, false ) )
+ ( twinrx_gain_config_t( 72, 37.7, 6, 0, true, false ) )
+ ( twinrx_gain_config_t( 73, 38.7, 5, 0, true, false ) )
+ ( twinrx_gain_config_t( 74, 39.7, 4, 0, true, false ) )
+ ( twinrx_gain_config_t( 75, 40.7, 3, 0, true, false ) )
+ ( twinrx_gain_config_t( 76, 41.7, 2, 0, true, false ) )
+ ( twinrx_gain_config_t( 77, 42.7, 1, 0, true, false ) )
+ ( twinrx_gain_config_t( 78, 43.7, 0, 0, true, false ) )
+ ( twinrx_gain_config_t( 79, 44.8, 6, 8, true, true ) )
+ ( twinrx_gain_config_t( 80, 45.8, 5, 8, true, true ) )
+ ( twinrx_gain_config_t( 81, 46.8, 4, 8, true, true ) )
+ ( twinrx_gain_config_t( 82, 47.8, 4, 7, true, true ) )
+ ( twinrx_gain_config_t( 83, 48.8, 3, 7, true, true ) )
+ ( twinrx_gain_config_t( 84, 49.8, 2, 7, true, true ) )
+ ( twinrx_gain_config_t( 85, 50.8, 1, 7, true, true ) )
+ ( twinrx_gain_config_t( 86, 51.8, 1, 6, true, true ) )
+ ( twinrx_gain_config_t( 87, 52.8, 0, 6, true, true ) )
+ ( twinrx_gain_config_t( 88, 53.8, 0, 5, true, true ) )
+ ( twinrx_gain_config_t( 89, 54.8, 0, 4, true, true ) )
+ ( twinrx_gain_config_t( 90, 55.8, 0, 3, true, true ) )
+ ( twinrx_gain_config_t( 91, 56.8, 0, 2, true, true ) )
+ ( twinrx_gain_config_t( 92, 57.8, 0, 1, true, true ) )
+ ( twinrx_gain_config_t( 93, 58.8, 0, 0, true, true ) )
+;
+
+static const std::vector<twinrx_gain_config_t> LOWBAND3_TABLE = boost::assign::list_of
+ // Index, Gain, Atten1, Atten2, Amp1, Amp2
+ ( twinrx_gain_config_t( 0, -34.0, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 1, -34.0, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 2, -34.0, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 3, -33.0, 31, 30, false, false ) )
+ ( twinrx_gain_config_t( 4, -32.0, 31, 29, false, false ) )
+ ( twinrx_gain_config_t( 5, -31.0, 31, 28, false, false ) )
+ ( twinrx_gain_config_t( 6, -30.0, 31, 27, false, false ) )
+ ( twinrx_gain_config_t( 7, -29.0, 31, 26, false, false ) )
+ ( twinrx_gain_config_t( 8, -28.0, 31, 25, false, false ) )
+ ( twinrx_gain_config_t( 9, -27.0, 31, 24, false, false ) )
+ ( twinrx_gain_config_t( 10, -26.0, 31, 23, false, false ) )
+ ( twinrx_gain_config_t( 11, -25.0, 31, 22, false, false ) )
+ ( twinrx_gain_config_t( 12, -24.0, 31, 21, false, false ) )
+ ( twinrx_gain_config_t( 13, -23.0, 31, 20, false, false ) )
+ ( twinrx_gain_config_t( 14, -22.0, 31, 19, false, false ) )
+ ( twinrx_gain_config_t( 15, -21.0, 31, 18, false, false ) )
+ ( twinrx_gain_config_t( 16, -20.0, 31, 17, false, false ) )
+ ( twinrx_gain_config_t( 17, -19.0, 31, 16, false, false ) )
+ ( twinrx_gain_config_t( 18, -18.0, 31, 15, false, false ) )
+ ( twinrx_gain_config_t( 19, -17.0, 31, 14, false, false ) )
+ ( twinrx_gain_config_t( 20, -16.0, 31, 13, false, false ) )
+ ( twinrx_gain_config_t( 21, -15.0, 31, 12, false, false ) )
+ ( twinrx_gain_config_t( 22, -14.0, 31, 11, false, false ) )
+ ( twinrx_gain_config_t( 23, -13.0, 31, 10, false, false ) )
+ ( twinrx_gain_config_t( 24, -12.0, 31, 9, false, false ) )
+ ( twinrx_gain_config_t( 25, -11.0, 31, 8, false, false ) )
+ ( twinrx_gain_config_t( 26, -10.0, 31, 7, false, false ) )
+ ( twinrx_gain_config_t( 27, -9.0, 31, 6, false, false ) )
+ ( twinrx_gain_config_t( 28, -8.0, 31, 5, false, false ) )
+ ( twinrx_gain_config_t( 29, -7.0, 31, 4, false, false ) )
+ ( twinrx_gain_config_t( 30, -6.0, 31, 3, false, false ) )
+ ( twinrx_gain_config_t( 31, -5.0, 31, 2, false, false ) )
+ ( twinrx_gain_config_t( 32, -4.0, 31, 1, false, false ) )
+ ( twinrx_gain_config_t( 33, -3.0, 31, 0, false, false ) )
+ ( twinrx_gain_config_t( 34, -2.0, 30, 0, false, false ) )
+ ( twinrx_gain_config_t( 35, -1.0, 29, 0, false, false ) )
+ ( twinrx_gain_config_t( 36, -0.0, 28, 0, false, false ) )
+ ( twinrx_gain_config_t( 37, 1.0, 27, 0, false, false ) )
+ ( twinrx_gain_config_t( 38, 2.0, 26, 0, false, false ) )
+ ( twinrx_gain_config_t( 39, 3.0, 25, 0, false, false ) )
+ ( twinrx_gain_config_t( 40, 4.0, 24, 0, false, false ) )
+ ( twinrx_gain_config_t( 41, 5.0, 23, 0, false, false ) )
+ ( twinrx_gain_config_t( 42, 6.0, 22, 0, false, false ) )
+ ( twinrx_gain_config_t( 43, 7.0, 21, 0, false, false ) )
+ ( twinrx_gain_config_t( 44, 8.0, 20, 0, false, false ) )
+ ( twinrx_gain_config_t( 45, 9.0, 19, 0, false, false ) )
+ ( twinrx_gain_config_t( 46, 10.0, 18, 0, false, false ) )
+ ( twinrx_gain_config_t( 47, 11.0, 17, 0, false, false ) )
+ ( twinrx_gain_config_t( 48, 12.0, 16, 0, false, false ) )
+ ( twinrx_gain_config_t( 49, 13.0, 15, 0, false, false ) )
+ ( twinrx_gain_config_t( 50, 14.0, 14, 0, false, false ) )
+ ( twinrx_gain_config_t( 51, 15.0, 13, 0, false, false ) )
+ ( twinrx_gain_config_t( 52, 16.0, 12, 0, false, false ) )
+ ( twinrx_gain_config_t( 53, 17.0, 11, 0, false, false ) )
+ ( twinrx_gain_config_t( 54, 18.0, 10, 0, false, false ) )
+ ( twinrx_gain_config_t( 55, 19.0, 9, 0, false, false ) )
+ ( twinrx_gain_config_t( 56, 20.0, 8, 0, false, false ) )
+ ( twinrx_gain_config_t( 57, 21.0, 7, 0, false, false ) )
+ ( twinrx_gain_config_t( 58, 22.0, 6, 0, false, false ) )
+ ( twinrx_gain_config_t( 59, 23.0, 5, 0, false, false ) )
+ ( twinrx_gain_config_t( 60, 24.0, 4, 0, false, false ) )
+ ( twinrx_gain_config_t( 61, 25.0, 3, 0, false, false ) )
+ ( twinrx_gain_config_t( 62, 26.0, 2, 0, false, false ) )
+ ( twinrx_gain_config_t( 63, 27.0, 1, 0, false, false ) )
+ ( twinrx_gain_config_t( 64, 28.0, 0, 0, false, false ) )
+ ( twinrx_gain_config_t( 65, 29.5, 5, 8, false, true ) )
+ ( twinrx_gain_config_t( 66, 30.5, 4, 8, false, true ) )
+ ( twinrx_gain_config_t( 67, 31.5, 3, 8, false, true ) )
+ ( twinrx_gain_config_t( 68, 32.5, 2, 8, false, true ) )
+ ( twinrx_gain_config_t( 69, 33.5, 1, 8, false, true ) )
+ ( twinrx_gain_config_t( 70, 34.5, 0, 8, false, true ) )
+ ( twinrx_gain_config_t( 71, 34.5, 0, 8, false, true ) )
+ ( twinrx_gain_config_t( 72, 36.5, 6, 0, true, false ) )
+ ( twinrx_gain_config_t( 73, 36.5, 6, 0, true, false ) )
+ ( twinrx_gain_config_t( 74, 37.5, 5, 0, true, false ) )
+ ( twinrx_gain_config_t( 75, 38.5, 4, 0, true, false ) )
+ ( twinrx_gain_config_t( 76, 39.5, 3, 0, true, false ) )
+ ( twinrx_gain_config_t( 77, 40.5, 2, 0, true, false ) )
+ ( twinrx_gain_config_t( 78, 41.5, 1, 0, true, false ) )
+ ( twinrx_gain_config_t( 79, 42.5, 0, 0, true, false ) )
+ ( twinrx_gain_config_t( 80, 44.0, 6, 7, true, true ) )
+ ( twinrx_gain_config_t( 81, 45.0, 5, 7, true, true ) )
+ ( twinrx_gain_config_t( 82, 46.0, 4, 7, true, true ) )
+ ( twinrx_gain_config_t( 83, 47.0, 3, 7, true, true ) )
+ ( twinrx_gain_config_t( 84, 48.0, 3, 6, true, true ) )
+ ( twinrx_gain_config_t( 85, 49.0, 2, 6, true, true ) )
+ ( twinrx_gain_config_t( 86, 50.0, 1, 6, true, true ) )
+ ( twinrx_gain_config_t( 87, 51.0, 0, 6, true, true ) )
+ ( twinrx_gain_config_t( 88, 52.0, 0, 5, true, true ) )
+ ( twinrx_gain_config_t( 89, 53.0, 0, 4, true, true ) )
+ ( twinrx_gain_config_t( 90, 54.0, 0, 3, true, true ) )
+ ( twinrx_gain_config_t( 91, 55.0, 0, 2, true, true ) )
+ ( twinrx_gain_config_t( 92, 56.0, 0, 1, true, true ) )
+ ( twinrx_gain_config_t( 93, 57.0, 0, 0, true, true ) )
+;
+
+static const std::vector<twinrx_gain_config_t> LOWBAND4_TABLE = boost::assign::list_of
+ // Index, Gain, Atten1, Atten2, Amp1, Amp2
+ ( twinrx_gain_config_t( 0, -32.8, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 1, -32.8, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 2, -32.8, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 3, -32.8, 31, 31, false, false ) )
+ ( twinrx_gain_config_t( 4, -31.8, 31, 30, false, false ) )
+ ( twinrx_gain_config_t( 5, -30.8, 31, 29, false, false ) )
+ ( twinrx_gain_config_t( 6, -29.8, 31, 28, false, false ) )
+ ( twinrx_gain_config_t( 7, -28.8, 31, 27, false, false ) )
+ ( twinrx_gain_config_t( 8, -27.8, 31, 26, false, false ) )
+ ( twinrx_gain_config_t( 9, -26.8, 31, 25, false, false ) )
+ ( twinrx_gain_config_t( 10, -25.8, 31, 24, false, false ) )
+ ( twinrx_gain_config_t( 11, -24.8, 31, 23, false, false ) )
+ ( twinrx_gain_config_t( 12, -23.8, 31, 22, false, false ) )
+ ( twinrx_gain_config_t( 13, -22.8, 31, 21, false, false ) )
+ ( twinrx_gain_config_t( 14, -21.8, 31, 20, false, false ) )
+ ( twinrx_gain_config_t( 15, -20.8, 31, 19, false, false ) )
+ ( twinrx_gain_config_t( 16, -19.8, 31, 18, false, false ) )
+ ( twinrx_gain_config_t( 17, -18.8, 31, 17, false, false ) )
+ ( twinrx_gain_config_t( 18, -17.8, 31, 16, false, false ) )
+ ( twinrx_gain_config_t( 19, -16.8, 31, 15, false, false ) )
+ ( twinrx_gain_config_t( 20, -15.8, 31, 14, false, false ) )
+ ( twinrx_gain_config_t( 21, -14.8, 31, 13, false, false ) )
+ ( twinrx_gain_config_t( 22, -13.8, 31, 12, false, false ) )
+ ( twinrx_gain_config_t( 23, -12.8, 31, 11, false, false ) )
+ ( twinrx_gain_config_t( 24, -11.8, 31, 10, false, false ) )
+ ( twinrx_gain_config_t( 25, -10.8, 31, 9, false, false ) )
+ ( twinrx_gain_config_t( 26, -9.8, 31, 8, false, false ) )
+ ( twinrx_gain_config_t( 27, -8.8, 31, 7, false, false ) )
+ ( twinrx_gain_config_t( 28, -7.8, 31, 6, false, false ) )
+ ( twinrx_gain_config_t( 29, -6.8, 31, 5, false, false ) )
+ ( twinrx_gain_config_t( 30, -5.8, 31, 4, false, false ) )
+ ( twinrx_gain_config_t( 31, -4.8, 31, 3, false, false ) )
+ ( twinrx_gain_config_t( 32, -3.8, 31, 2, false, false ) )
+ ( twinrx_gain_config_t( 33, -2.8, 31, 1, false, false ) )
+ ( twinrx_gain_config_t( 34, -1.8, 31, 0, false, false ) )
+ ( twinrx_gain_config_t( 35, -0.8, 30, 0, false, false ) )
+ ( twinrx_gain_config_t( 36, 0.2, 29, 0, false, false ) )
+ ( twinrx_gain_config_t( 37, 1.2, 28, 0, false, false ) )
+ ( twinrx_gain_config_t( 38, 2.2, 27, 0, false, false ) )
+ ( twinrx_gain_config_t( 39, 3.2, 26, 0, false, false ) )
+ ( twinrx_gain_config_t( 40, 4.2, 25, 0, false, false ) )
+ ( twinrx_gain_config_t( 41, 5.2, 24, 0, false, false ) )
+ ( twinrx_gain_config_t( 42, 6.2, 23, 0, false, false ) )
+ ( twinrx_gain_config_t( 43, 7.2, 22, 0, false, false ) )
+ ( twinrx_gain_config_t( 44, 8.2, 21, 0, false, false ) )
+ ( twinrx_gain_config_t( 45, 9.2, 20, 0, false, false ) )
+ ( twinrx_gain_config_t( 46, 10.2, 19, 0, false, false ) )
+ ( twinrx_gain_config_t( 47, 11.2, 18, 0, false, false ) )
+ ( twinrx_gain_config_t( 48, 12.2, 17, 0, false, false ) )
+ ( twinrx_gain_config_t( 49, 13.2, 16, 0, false, false ) )
+ ( twinrx_gain_config_t( 50, 14.2, 15, 0, false, false ) )
+ ( twinrx_gain_config_t( 51, 15.2, 14, 0, false, false ) )
+ ( twinrx_gain_config_t( 52, 16.2, 13, 0, false, false ) )
+ ( twinrx_gain_config_t( 53, 17.2, 12, 0, false, false ) )
+ ( twinrx_gain_config_t( 54, 18.2, 11, 0, false, false ) )
+ ( twinrx_gain_config_t( 55, 19.2, 10, 0, false, false ) )
+ ( twinrx_gain_config_t( 56, 20.2, 9, 0, false, false ) )
+ ( twinrx_gain_config_t( 57, 21.2, 8, 0, false, false ) )
+ ( twinrx_gain_config_t( 58, 22.2, 7, 0, false, false ) )
+ ( twinrx_gain_config_t( 59, 23.2, 6, 0, false, false ) )
+ ( twinrx_gain_config_t( 60, 24.2, 5, 0, false, false ) )
+ ( twinrx_gain_config_t( 61, 25.2, 4, 0, false, false ) )
+ ( twinrx_gain_config_t( 62, 26.2, 3, 0, false, false ) )
+ ( twinrx_gain_config_t( 63, 27.2, 2, 0, false, false ) )
+ ( twinrx_gain_config_t( 64, 28.2, 1, 0, false, false ) )
+ ( twinrx_gain_config_t( 65, 29.2, 0, 0, false, false ) )
+ ( twinrx_gain_config_t( 66, 30.4, 4, 9, false, true ) )
+ ( twinrx_gain_config_t( 67, 31.4, 3, 9, false, true ) )
+ ( twinrx_gain_config_t( 68, 32.4, 2, 9, false, true ) )
+ ( twinrx_gain_config_t( 69, 33.4, 1, 9, false, true ) )
+ ( twinrx_gain_config_t( 70, 34.4, 0, 9, false, true ) )
+ ( twinrx_gain_config_t( 71, 35.4, 8, 0, true, false ) )
+ ( twinrx_gain_config_t( 72, 36.4, 7, 0, true, false ) )
+ ( twinrx_gain_config_t( 73, 37.4, 6, 0, true, false ) )
+ ( twinrx_gain_config_t( 74, 38.4, 5, 0, true, false ) )
+ ( twinrx_gain_config_t( 75, 39.4, 4, 0, true, false ) )
+ ( twinrx_gain_config_t( 76, 40.4, 3, 0, true, false ) )
+ ( twinrx_gain_config_t( 77, 41.4, 2, 0, true, false ) )
+ ( twinrx_gain_config_t( 78, 42.4, 1, 0, true, false ) )
+ ( twinrx_gain_config_t( 79, 43.4, 0, 0, true, false ) )
+ ( twinrx_gain_config_t( 80, 44.6, 4, 9, true, true ) )
+ ( twinrx_gain_config_t( 81, 45.6, 4, 8, true, true ) )
+ ( twinrx_gain_config_t( 82, 46.6, 3, 8, true, true ) )
+ ( twinrx_gain_config_t( 83, 47.6, 2, 8, true, true ) )
+ ( twinrx_gain_config_t( 84, 48.6, 1, 8, true, true ) )
+ ( twinrx_gain_config_t( 85, 49.6, 1, 7, true, true ) )
+ ( twinrx_gain_config_t( 86, 50.6, 0, 7, true, true ) )
+ ( twinrx_gain_config_t( 87, 51.6, 0, 6, true, true ) )
+ ( twinrx_gain_config_t( 88, 52.6, 0, 5, true, true ) )
+ ( twinrx_gain_config_t( 89, 53.6, 0, 4, true, true ) )
+ ( twinrx_gain_config_t( 90, 54.6, 0, 3, true, true ) )
+ ( twinrx_gain_config_t( 91, 55.6, 0, 2, true, true ) )
+ ( twinrx_gain_config_t( 92, 56.6, 0, 1, true, true ) )
+ ( twinrx_gain_config_t( 93, 57.6, 0, 0, true, true ) )
+;
+
+const twinrx_gain_table twinrx_gain_table::lookup_table
+(
+ twinrx_ctrl::signal_path_t signal_path,
+ twinrx_ctrl::preselector_path_t preselector_path,
+ std::string
+) {
+
+ if (signal_path == twinrx_ctrl::PATH_HIGHBAND) {
+ switch (preselector_path) {
+ case twinrx_ctrl::PRESEL_PATH1:
+ return twinrx_gain_table(HIGHBAND1_TABLE);
+ case twinrx_ctrl::PRESEL_PATH2:
+ return twinrx_gain_table(HIGHBAND2_TABLE);
+ case twinrx_ctrl::PRESEL_PATH3:
+ return twinrx_gain_table(HIGHBAND3_TABLE);
+ case twinrx_ctrl::PRESEL_PATH4:
+ return twinrx_gain_table(HIGHBAND4_TABLE);
+ }
+ } else {
+ switch (preselector_path) {
+ case twinrx_ctrl::PRESEL_PATH1:
+ return twinrx_gain_table(LOWBAND1_TABLE);
+ case twinrx_ctrl::PRESEL_PATH2:
+ return twinrx_gain_table(LOWBAND2_TABLE);
+ case twinrx_ctrl::PRESEL_PATH3:
+ return twinrx_gain_table(LOWBAND3_TABLE);
+ case twinrx_ctrl::PRESEL_PATH4:
+ return twinrx_gain_table(LOWBAND4_TABLE);
+ }
+ }
+ throw runtime_error("NO GAIN TABLE SELECTED");
+ return twinrx_gain_table(HIGHBAND1_TABLE);
+}
+
+const twinrx_gain_config_t& twinrx_gain_table::find_by_index(size_t index) const {
+ if (index >= get_num_entries()) throw uhd::value_error("invalid gain table index");
+ return _tbl.at(index);
+}
+
+uhd::gain_range_t twinrx_gain_table::get_gain_range() const {
+ double max = std::numeric_limits<double>::min();
+ double min = std::numeric_limits<double>::max();
+ for (size_t i = 0; i < get_num_entries(); i++) {
+ const twinrx_gain_config_t& config = find_by_index(i);
+ if (config.sys_gain > max) {
+ max = config.sys_gain;
+ }
+ if (config.sys_gain < min) {
+ min = config.sys_gain;
+ }
+ }
+ return uhd::gain_range_t(min, max, 1.0);
+}
diff --git a/host/lib/usrp/dboard/twinrx/twinrx_gain_tables.hpp b/host/lib/usrp/dboard/twinrx/twinrx_gain_tables.hpp
new file mode 100644
index 000000000..0148965da
--- /dev/null
+++ b/host/lib/usrp/dboard/twinrx/twinrx_gain_tables.hpp
@@ -0,0 +1,83 @@
+//
+// Copyright 2016 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#ifndef INCLUDED_DBOARD_TWINRX_GAIN_TABLES_HPP
+#define INCLUDED_DBOARD_TWINRX_GAIN_TABLES_HPP
+
+#include <uhd/config.hpp>
+#include <boost/cstdint.hpp>
+#include <uhd/types/ranges.hpp>
+#include "twinrx_ctrl.hpp"
+
+namespace uhd { namespace usrp { namespace dboard { namespace twinrx {
+
+class twinrx_gain_config_t {
+public:
+ twinrx_gain_config_t(
+ size_t index_, double sys_gain_,
+ boost::uint8_t atten1_, boost::uint8_t atten2_,
+ bool amp1_, bool amp2_
+ ): index(index_), sys_gain(sys_gain_), atten1(atten1_), atten2(atten2_),
+ amp1(amp1_), amp2(amp2_)
+ {}
+
+ twinrx_gain_config_t& operator=(const twinrx_gain_config_t& src) {
+ if (this != &src) {
+ this->index = src.index;
+ this->sys_gain = src.sys_gain;
+ this->atten1 = src.atten1;
+ this->atten2 = src.atten2;
+ this->amp1 = src.amp1;
+ this->amp2 = src.amp2;
+ }
+ return *this;
+ }
+
+ size_t index;
+ double sys_gain;
+ boost::uint8_t atten1;
+ boost::uint8_t atten2;
+ bool amp1;
+ bool amp2;
+};
+
+class twinrx_gain_table {
+public:
+ static const twinrx_gain_table lookup_table(
+ twinrx_ctrl::signal_path_t signal_path,
+ twinrx_ctrl::preselector_path_t presel_path,
+ std::string profile);
+
+ twinrx_gain_table(const std::vector<twinrx_gain_config_t>& tbl)
+ : _tbl(tbl) {}
+
+ const twinrx_gain_config_t& find_by_index(size_t index) const;
+
+ inline size_t get_num_entries() const {
+ return _tbl.size();
+ }
+
+ uhd::gain_range_t get_gain_range() const;
+
+private:
+ const std::vector<twinrx_gain_config_t>& _tbl;
+};
+
+
+}}}} //namespaces
+
+#endif /* INCLUDED_DBOARD_TWINRX_GAIN_TABLES_HPP */
diff --git a/host/lib/usrp/dboard/twinrx/twinrx_io.hpp b/host/lib/usrp/dboard/twinrx/twinrx_io.hpp
new file mode 100644
index 000000000..8b504b549
--- /dev/null
+++ b/host/lib/usrp/dboard/twinrx/twinrx_io.hpp
@@ -0,0 +1,524 @@
+//
+// Copyright 2015 Ettus Research LLC
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as published by
+// the Free Software Foundation, either version 3 of the License, or
+// (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see <http://www.gnu.org/licenses/>.
+//
+
+#ifndef INCLUDED_DBOARD_TWINRX_IO_HPP
+#define INCLUDED_DBOARD_TWINRX_IO_HPP
+
+#include <uhd/types/wb_iface.hpp>
+#include <uhd/usrp/dboard_base.hpp>
+#include <uhd/utils/soft_register.hpp>
+#include <boost/thread.hpp>
+#include "gpio_atr_3000.hpp"
+
+namespace uhd { namespace usrp { namespace dboard { namespace twinrx {
+
+static const boost::uint32_t SET_ALL_BITS = 0xFFFFFFFF;
+
+namespace cpld {
+static wb_iface::wb_addr_type addr(boost::uint8_t cpld_num, boost::uint8_t cpld_addr) {
+ //Decode CPLD addressing for the following bitmap:
+ // {CPLD1_EN, CPLD2_EN, CPLD3_EN, CPLD4_EN, CPLD_ADDR[2:0]}
+ boost::uint8_t addr = 0;
+ switch (cpld_num) {
+ case 1: addr = 0x8 << 3; break;
+ case 2: addr = 0x4 << 3; break;
+ case 3: addr = 0x2 << 3; break;
+ case 4: addr = 0x1 << 3; break;
+ default: UHD_THROW_INVALID_CODE_PATH();
+ }
+ return static_cast<wb_iface::wb_addr_type>(addr | (cpld_addr & 0x7));
+}
+
+static boost::uint32_t get_reg(wb_iface::wb_addr_type addr) {
+ return static_cast<boost::uint32_t>(addr) & 0x7;
+}
+}
+
+class twinrx_gpio : public timed_wb_iface {
+public:
+ typedef boost::shared_ptr<twinrx_gpio> sptr;
+
+ //----------------------------------------------
+ //Public GPIO fields
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO2_CE_CH1, /*width*/ 1, /*shift*/ 0); //GPIO[0] OUT
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO2_CE_CH2, /*width*/ 1, /*shift*/ 1); //GPIO[1] OUT
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO2_MUXOUT_CH1, /*width*/ 1, /*shift*/ 2); //GPIO[2] IN
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO2_MUXOUT_CH2, /*width*/ 1, /*shift*/ 3); //GPIO[3] IN
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO2_LD_CH1, /*width*/ 1, /*shift*/ 4); //GPIO[4] IN
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO2_LD_CH2, /*width*/ 1, /*shift*/ 5); //GPIO[5] IN
+ // NO CONNECT //GPIO[8:6]
+ // PRIVATE //GPIO[15:9]
+ // NO CONNECT //GPIO[16]
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO1_CE_CH1, /*width*/ 1, /*shift*/ 17); //GPIO[17] OUT
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO1_CE_CH2, /*width*/ 1, /*shift*/ 18); //GPIO[18] OUT
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO1_MUXOUT_CH1, /*width*/ 1, /*shift*/ 19); //GPIO[19] IN
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_LO1_MUXOUT_CH2, /*width*/ 1, /*shift*/ 20); //GPIO[20] IN
+ // NO CONNECT //GPIO[21:23]
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_SWPS_CLK, /*width*/ 1, /*shift*/ 24); //GPIO[24] IN
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_SWPS_PWR_GOOD, /*width*/ 1, /*shift*/ 25); //GPIO[25] IN
+ UHD_DEFINE_SOFT_REG_FIELD(FIELD_SWPS_EN, /*width*/ 1, /*shift*/ 26); //GPIO[26] OUT
+ // PRIVATE //GPIO[27:31]
+ //----------------------------------------------
+
+ twinrx_gpio(dboard_iface::sptr iface) : _db_iface(iface) {
+ _db_iface->set_gpio_ddr(dboard_iface::UNIT_BOTH, GPIO_OUTPUT_MASK, SET_ALL_BITS);
+ _db_iface->set_pin_ctrl(dboard_iface::UNIT_BOTH, GPIO_PINCTRL_MASK, SET_ALL_BITS);
+ _db_iface->set_gpio_out(dboard_iface::UNIT_BOTH, 0, ~GPIO_PINCTRL_MASK);
+ }
+
+ ~twinrx_gpio() {
+ _db_iface->set_gpio_ddr(dboard_iface::UNIT_BOTH, ~GPIO_OUTPUT_MASK, SET_ALL_BITS);
+ }
+
+ void set_field(const uhd::soft_reg_field_t field, const boost::uint32_t value) {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ using namespace soft_reg_field;
+
+ _db_iface->set_gpio_out(dboard_iface::UNIT_BOTH,
+ (value << shift(field)),
+ mask<boost::uint32_t>(field));
+ }
+
+ boost::uint32_t get_field(const uhd::soft_reg_field_t field) {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ using namespace soft_reg_field;
+ return (_db_iface->read_gpio(dboard_iface::UNIT_BOTH) & mask<boost::uint32_t>(field)) >> shift(field);
+ }
+
+ // CPLD register write-only interface
+ void poke32(const wb_addr_type addr, const boost::uint32_t data) {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ using namespace soft_reg_field;
+
+ //Step 1: Write the reg offset and data to the GPIO bus and de-assert all enables
+ _db_iface->set_gpio_out(dboard_iface::UNIT_BOTH,
+ (cpld::get_reg(addr) << shift(CPLD_FULL_ADDR)) | (data << shift(CPLD_DATA)),
+ mask<boost::uint32_t>(CPLD_FULL_ADDR)|mask<boost::uint32_t>(CPLD_DATA));
+ //Sleep for 166ns to ensure that we don't toggle the enables to quickly
+ //nanosleep is not really accurate in userland and it is also not very
+ //cross-platform. So just sleep for the minimum amount of time in us.
+ boost::this_thread::sleep(boost::posix_time::microseconds(0));
+ //Step 2: Write the reg offset and data, and assert the necessary enable
+ _db_iface->set_gpio_out(dboard_iface::UNIT_BOTH,
+ (static_cast<boost::uint32_t>(addr) << shift(CPLD_FULL_ADDR)) | (data << shift(CPLD_DATA)),
+ mask<boost::uint32_t>(CPLD_FULL_ADDR)|mask<boost::uint32_t>(CPLD_DATA));
+ }
+
+ // Timed command interface
+ inline time_spec_t get_time() {
+ return _db_iface->get_command_time();
+ }
+
+ void set_time(const time_spec_t& t) {
+ boost::lock_guard<boost::mutex> lock(_mutex);
+ _db_iface->set_command_time(t);
+ }
+
+private: //Members/definitions
+ static const boost::uint32_t GPIO_OUTPUT_MASK = 0xFC06FE03;
+ static const boost::uint32_t GPIO_PINCTRL_MASK = 0x00000000;
+
+ //Private GPIO fields
+ UHD_DEFINE_SOFT_REG_FIELD(CPLD_FULL_ADDR, /*width*/ 7, /*shift*/ 9); //GPIO[15:9]
+ UHD_DEFINE_SOFT_REG_FIELD(CPLD_DATA, /*width*/ 5, /*shift*/ 27); //GPIO[31:27]
+
+ //Members
+ dboard_iface::sptr _db_iface;
+ boost::mutex _mutex;
+};
+
+class twinrx_cpld_regmap : public uhd::soft_regmap_t {
+public:
+ typedef boost::shared_ptr<twinrx_cpld_regmap> sptr;
+
+ //----------------------------------------------
+ // IF CCA: CPLD 1
+ //----------------------------------------------
+ class if0_reg0_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_HB_IF1_EN_CH1, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_LO2_EN_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_LO2_EN_CH2, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(SW19_CTRL_CH2, /*width*/ 1, /*shift*/ 3);
+ UHD_DEFINE_SOFT_REG_FIELD(SW20_CTRL_CH2, /*width*/ 1, /*shift*/ 4);
+
+ if0_reg0_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 1, /*reg*/ 0), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } if0_reg0;
+
+ class if0_reg1_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW20_CTRL_CH1, /*width*/ 1, /*shift*/ 2);
+
+ if0_reg1_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 1, /*reg*/ 1), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } if0_reg1;
+
+ class if0_reg2_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_LB_IF1_EN_CH2, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_LB_IF1_EN_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(LO2_LE_CH1, /*width*/ 1, /*shift*/ 3);
+ UHD_DEFINE_SOFT_REG_FIELD(LO2_LE_CH2, /*width*/ 1, /*shift*/ 4);
+
+ if0_reg2_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 1, /*reg*/ 2), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } if0_reg2;
+
+ class if0_reg3_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW24_CTRL_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW13_CTRL_CH1, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(IF1_IF2_EN_CH1, /*width*/ 1, /*shift*/ 3);
+
+ if0_reg3_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 1, /*reg*/ 3), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } if0_reg3;
+
+ class if0_reg4_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW21_CTRL_CH2, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW25_CTRL, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(IF1_IF2_EN_CH2, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(SW19_CTRL_CH1, /*width*/ 1, /*shift*/ 3);
+ UHD_DEFINE_SOFT_REG_FIELD(SW21_CTRL_CH1, /*width*/ 1, /*shift*/ 4);
+
+ if0_reg4_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 1, /*reg*/ 4), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } if0_reg4;
+
+ class if0_reg6_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_HB_IF1_EN_CH2, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW13_CTRL_CH2, /*width*/ 1, /*shift*/ 2);
+
+ if0_reg6_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 1, /*reg*/ 6), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } if0_reg6;
+
+ class if0_reg7_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW24_CTRL_CH2, /*width*/ 1, /*shift*/ 0);
+
+ if0_reg7_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 1, /*reg*/ 7), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } if0_reg7;
+
+ //----------------------------------------------
+ // RF CCA: CPLD 2
+ //----------------------------------------------
+ class rf0_reg0_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(ATTEN_IN_CH1, /*width*/ 5, /*shift*/ 0);
+
+ rf0_reg0_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 2, /*reg*/ 0), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf0_reg0;
+
+ class rf0_reg1_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SWPA1_CTL_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(HB_PREAMP_EN_CH1, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(LB_PREAMP_EN_CH1, /*width*/ 1, /*shift*/ 3);
+ UHD_DEFINE_SOFT_REG_FIELD(SWPA3_CTRL_CH2, /*width*/ 1, /*shift*/ 4);
+
+ rf0_reg1_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 2, /*reg*/ 1), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf0_reg1;
+
+ class rf0_reg2_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW6_CTRL_CH1, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW5_CTRL_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW4_CTRL_CH1, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(LO1_LE_CH1, /*width*/ 1, /*shift*/ 3);
+ UHD_DEFINE_SOFT_REG_FIELD(LO1_LE_CH2, /*width*/ 1, /*shift*/ 4);
+
+ rf0_reg2_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 2, /*reg*/ 2), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf0_reg2;
+
+ class rf0_reg3_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW9_CTRL_CH2, /*width*/ 2, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW7_CTRL_CH1, /*width*/ 2, /*shift*/ 2);
+
+ rf0_reg3_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 2, /*reg*/ 3), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf0_reg3;
+
+ class rf0_reg4_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(ATTEN_IN_CH2, /*width*/ 5, /*shift*/ 0);
+
+ rf0_reg4_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 2, /*reg*/ 4), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf0_reg4;
+
+ class rf0_reg5_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW9_CTRL_CH1, /*width*/ 2, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(HB_PREAMP_EN_CH2, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(SW3_CTRL_CH1, /*width*/ 1, /*shift*/ 4);
+
+ rf0_reg5_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 2, /*reg*/ 5), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf0_reg5;
+
+ class rf0_reg6_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW6_CTRL_CH2, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW5_CTRL_CH2, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW4_CTRL_CH2, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(SWPA4_CTRL_CH2, /*width*/ 1, /*shift*/ 4);
+
+ rf0_reg6_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 2, /*reg*/ 6), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf0_reg6;
+
+ class rf0_reg7_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SWPA1_CTRL_CH2, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SWPA3_CTRL_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW3_CTRL_CH2, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(SW7_CTRL_CH2, /*width*/ 2, /*shift*/ 3);
+
+ rf0_reg7_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 2, /*reg*/ 7), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf0_reg7;
+
+ //----------------------------------------------
+ // RF CCA: CPLD 3
+ //----------------------------------------------
+ class rf1_reg0_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(ATTEN_HB_CH1, /*width*/ 5, /*shift*/ 0);
+
+ rf1_reg0_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 3, /*reg*/ 0), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf1_reg0;
+
+ class rf1_reg1_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW17_CTRL_CH1, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_LO1_EN_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW16_CTRL_CH1, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(SW15_CTRL_CH1, /*width*/ 1, /*shift*/ 3);
+ UHD_DEFINE_SOFT_REG_FIELD(SW14_CTRL_CH1, /*width*/ 1, /*shift*/ 4);
+
+ rf1_reg1_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 3, /*reg*/ 1), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf1_reg1;
+
+ class rf1_reg2_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW12_CTRL_CH1, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_HB_EN_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(HB_PRESEL_PGA_EN_CH2, /*width*/ 1, /*shift*/ 2);
+
+ rf1_reg2_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 3, /*reg*/ 2), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf1_reg2;
+
+ class rf1_reg3_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW23_CTRL, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW22_CTRL_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW10_CTRL_CH1, /*width*/ 2, /*shift*/ 2);
+
+ rf1_reg3_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 3, /*reg*/ 3), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf1_reg3;
+
+ class rf1_reg4_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(ATTEN_HB_CH2, /*width*/ 5, /*shift*/ 0);
+
+ rf1_reg4_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 3, /*reg*/ 4), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf1_reg4;
+
+ class rf1_reg5_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_LO1_EN_CH2, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW15_CTRL_CH2, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW14_CTRL_CH2, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(SW18_CTRL_CH1, /*width*/ 1, /*shift*/ 4);
+
+ rf1_reg5_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 3, /*reg*/ 5), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf1_reg5;
+
+ class rf1_reg6_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(HB_PRESEL_PGA_EN_CH1, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW17_CTRL_CH2, /*width*/ 1, /*shift*/ 2);
+ UHD_DEFINE_SOFT_REG_FIELD(SW16_CTRL_CH2, /*width*/ 1, /*shift*/ 3);
+ UHD_DEFINE_SOFT_REG_FIELD(PREAMP2_EN_CH2, /*width*/ 1, /*shift*/ 4);
+
+ rf1_reg6_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 3, /*reg*/ 6), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf1_reg6;
+
+ class rf1_reg7_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW22_CTRL_CH2, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW10_CTRL_CH2, /*width*/ 2, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW12_CTRL_CH2, /*width*/ 1, /*shift*/ 3);
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_HB_EN_CH2, /*width*/ 1, /*shift*/ 4);
+
+ rf1_reg7_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 3, /*reg*/ 7), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf1_reg7;
+
+ //----------------------------------------------
+ // RF CCA: CPLD 4
+ //----------------------------------------------
+ class rf2_reg0_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(ATTEN_LB_CH1, /*width*/ 5, /*shift*/ 0);
+
+ rf2_reg0_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 4, /*reg*/ 0), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf2_reg0;
+
+ class rf2_reg2_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SW11_CTRL_CH1, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_LB_EN_CH1, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SWPA2_CTRL_CH1, /*width*/ 1, /*shift*/ 2);
+
+ rf2_reg2_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 4, /*reg*/ 2), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf2_reg2;
+
+ class rf2_reg3_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(PREAMP2_EN_CH1, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW18_CTRL_CH2, /*width*/ 1, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW8_CTRL_CH1, /*width*/ 2, /*shift*/ 2);
+
+ rf2_reg3_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 4, /*reg*/ 3), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf2_reg3;
+
+ class rf2_reg4_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(ATTEN_LB_CH2, /*width*/ 5, /*shift*/ 0);
+
+ rf2_reg4_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 4, /*reg*/ 4), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf2_reg4;
+
+ class rf2_reg5_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SWPA2_CTRL_CH2, /*width*/ 1, /*shift*/ 0);
+
+ rf2_reg5_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 4, /*reg*/ 5), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf2_reg5;
+
+ class rf2_reg6_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(LB_PREAMP_EN_CH2, /*width*/ 1, /*shift*/ 0);
+
+ rf2_reg6_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 4, /*reg*/ 6), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf2_reg6;
+
+ class rf2_reg7_t : public uhd::soft_reg32_wo_t {
+ public:
+ UHD_DEFINE_SOFT_REG_FIELD(SWPA4_CTRL_CH1, /*width*/ 1, /*shift*/ 0);
+ UHD_DEFINE_SOFT_REG_FIELD(SW8_CTRL_CH2, /*width*/ 2, /*shift*/ 1);
+ UHD_DEFINE_SOFT_REG_FIELD(SW11_CTRL_CH2, /*width*/ 1, /*shift*/ 3);
+ UHD_DEFINE_SOFT_REG_FIELD(AMP_LB_EN_CH2, /*width*/ 1, /*shift*/ 4);
+
+ rf2_reg7_t(): uhd::soft_reg32_wo_t(cpld::addr(/*cpld*/ 4, /*reg*/ 7), OPTIMIZED_FLUSH) {
+ set(REGISTER, 0);
+ }
+ } rf2_reg7;
+
+ twinrx_cpld_regmap() : soft_regmap_t("twinrx_cpld") {
+ // IF CCA: CPLD 1
+ add_to_map(if0_reg0, "if0_reg0");
+ add_to_map(if0_reg1, "if0_reg1");
+ add_to_map(if0_reg2, "if0_reg2");
+ add_to_map(if0_reg3, "if0_reg3");
+ add_to_map(if0_reg4, "if0_reg4");
+ add_to_map(if0_reg6, "if0_reg6");
+ add_to_map(if0_reg7, "if0_reg7");
+ // RF CCA: CPLD 2
+ add_to_map(rf0_reg0, "rf0_reg0");
+ add_to_map(rf0_reg1, "rf0_reg1");
+ add_to_map(rf0_reg2, "rf0_reg2");
+ add_to_map(rf0_reg3, "rf0_reg3");
+ add_to_map(rf0_reg4, "rf0_reg4");
+ add_to_map(rf0_reg5, "rf0_reg5");
+ add_to_map(rf0_reg6, "rf0_reg6");
+ add_to_map(rf0_reg7, "rf0_reg7");
+ // RF CCA: CPLD 3
+ add_to_map(rf1_reg0, "rf1_reg0");
+ add_to_map(rf1_reg1, "rf1_reg1");
+ add_to_map(rf1_reg2, "rf1_reg2");
+ add_to_map(rf1_reg3, "rf1_reg3");
+ add_to_map(rf1_reg4, "rf1_reg4");
+ add_to_map(rf1_reg5, "rf1_reg5");
+ add_to_map(rf1_reg6, "rf1_reg6");
+ add_to_map(rf1_reg7, "rf1_reg7");
+ // RF CCA: CPLD 4
+ add_to_map(rf2_reg0, "rf2_reg0");
+ add_to_map(rf2_reg2, "rf2_reg2");
+ add_to_map(rf2_reg3, "rf2_reg3");
+ add_to_map(rf2_reg4, "rf2_reg4");
+ add_to_map(rf2_reg5, "rf2_reg5");
+ add_to_map(rf2_reg6, "rf2_reg6");
+ add_to_map(rf2_reg7, "rf2_reg7");
+ }
+};
+
+}}}} //namespaces
+
+#endif /* INCLUDED_DBOARD_TWINRX_IO_HPP */