//
// Copyright 2010 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 .
//
// TX IO Pins
#define HB_PA_OFF_TXIO (1 << 15) // 5GHz PA, 1 = off, 0 = on
#define LB_PA_OFF_TXIO (1 << 14) // 2.4GHz PA, 1 = off, 0 = on
#define ANTSEL_TX1_RX2_TXIO (1 << 13) // 1 = Ant 1 to TX, Ant 2 to RX
#define ANTSEL_TX2_RX1_TXIO (1 << 12) // 1 = Ant 2 to TX, Ant 1 to RX
#define TX_EN_TXIO (1 << 11) // 1 = TX on, 0 = TX off
#define AD9515DIV_TXIO (1 << 4) // 1 = Div by 3, 0 = Div by 2
#define TXIO_MASK (HB_PA_OFF_TXIO | LB_PA_OFF_TXIO | ANTSEL_TX1_RX2_TXIO | ANTSEL_TX2_RX1_TXIO | TX_EN_TXIO | AD9515DIV_TXIO)
// TX IO Functions
#define HB_PA_TXIO LB_PA_OFF_TXIO
#define LB_PA_TXIO HB_PA_OFF_TXIO
#define TX_ENB_TXIO TX_EN_TXIO
#define TX_DIS_TXIO 0
#define AD9515DIV_3_TXIO AD9515DIV_TXIO
#define AD9515DIV_2_TXIO 0
// RX IO Pins
#define LOCKDET_RXIO (1 << 15) // This is an INPUT!!!
#define EN_RXIO (1 << 14)
#define RX_EN_RXIO (1 << 13) // 1 = RX on, 0 = RX off
#define RX_HP_RXIO (1 << 12) // 0 = Fc set by rx_hpf, 1 = 600 KHz
#define RXIO_MASK (EN_RXIO | RX_EN_RXIO | RX_HP_RXIO)
// RX IO Functions
#define ALL_ENB_RXIO EN_RXIO
#define ALL_DIS_RXIO 0
#define RX_ENB_RXIO RX_EN_RXIO
#define RX_DIS_RXIO 0
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace uhd;
using namespace uhd::usrp;
using namespace boost::assign;
/***********************************************************************
* The XCVR 2450 dboard
**********************************************************************/
static const freq_range_t xcvr_freq_range(2.4e9, 6.0e9);
class xcvr2450 : public xcvr_dboard_base{
public:
xcvr2450(ctor_args_t const& args);
~xcvr2450(void);
void rx_get(const wax::obj &key, wax::obj &val);
void rx_set(const wax::obj &key, const wax::obj &val);
void tx_get(const wax::obj &key, wax::obj &val);
void tx_set(const wax::obj &key, const wax::obj &val);
private:
double _lo_freq;
uhd::dict _tx_gains, _rx_gains;
std::string _tx_ant, _rx_ant;
int _ad9515div;
void set_lo_freq(double target_freq);
void set_tx_ant(const std::string &ant);
void set_rx_ant(const std::string &ant);
void set_tx_gain(float gain, const std::string &name);
void set_rx_gain(float gain, const std::string &name);
void update_atr(void);
};
/***********************************************************************
* Register the XCVR dboard
**********************************************************************/
static dboard_base::sptr make_xcvr2450(dboard_base::ctor_args_t const& args){
return dboard_base::sptr(new xcvr2450(args));
}
UHD_STATIC_BLOCK(reg_xcvr2450_dboard){
//register the factory function for the rx and tx dbids
dboard_manager::register_dboard(0x0060, &make_xcvr2450, "XCVR2450 TX");
dboard_manager::register_dboard(0x0061, &make_xcvr2450, "XCVR2450 RX");
}
/***********************************************************************
* Structors
**********************************************************************/
xcvr2450::xcvr2450(ctor_args_t const& args) : xcvr_dboard_base(args){
//enable only the clocks we need
this->get_iface()->set_clock_enabled(dboard_iface::UNIT_TX, true);
//set the gpio directions
this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_TX, TXIO_MASK);
this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_RX, RXIO_MASK);
//set defaults for LO, gains, antennas
set_lo_freq(2.45e9);
set_rx_ant("J1");
set_tx_ant("J2");
set_rx_gain(0, "RF LNA");
set_rx_gain(0, "BB VGA");
set_tx_gain(0, "VGA");
}
xcvr2450::~xcvr2450(void){
/* NOP */
}
void xcvr2450::update_atr(void){
//calculate tx atr pins
int band_sel = (_lo_freq > 4e9)? HB_PA_TXIO : LB_PA_TXIO;
int tx_ant_sel = (_tx_ant == "J1")? ANTSEL_TX1_RX2_TXIO : ANTSEL_TX2_RX1_TXIO;
int rx_ant_sel = (_rx_ant == "J1")? ANTSEL_TX1_RX2_TXIO : ANTSEL_TX2_RX1_TXIO;
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);
//set the rx registers
this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_IDLE, ALL_ENB_RXIO | RX_DIS_RXIO);
this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_RX_ONLY, ALL_ENB_RXIO | RX_ENB_RXIO);
this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_TX_ONLY, ALL_ENB_RXIO | RX_DIS_RXIO);
this->get_iface()->set_atr_reg(dboard_iface::UNIT_RX, dboard_iface::ATR_REG_FULL_DUPLEX, ALL_ENB_RXIO | RX_ENB_RXIO);
}
/***********************************************************************
* Tuning
**********************************************************************/
void xcvr2450::set_lo_freq(double target_freq){
//TODO
//set _ad9515div
}
/***********************************************************************
* Antenna Handling
**********************************************************************/
static const prop_names_t xcvr_antennas = list_of("J1")("J2");
void xcvr2450::set_tx_ant(const std::string &ant){
assert_has(xcvr_antennas, ant, "xcvr antenna name");
//TODO
}
void xcvr2450::set_rx_ant(const std::string &ant){
assert_has(xcvr_antennas, ant, "xcvr antenna name");
//TODO
}
/***********************************************************************
* Gain Handling
**********************************************************************/
static const uhd::dict xcvr_tx_gain_ranges = map_list_of
("VGA", gain_range_t(0, 30, 0.5))
;
static const uhd::dict xcvr_rx_gain_ranges = map_list_of
("RF LNA", gain_range_t(0, 30.5, 15))
("BB VGA", gain_range_t(0, 62, 2.0))
;
/*!
* Convert a requested gain for the tx vga into the integer register value.
* The gain passed into the function will be set to the actual value.
* \param gain the requested gain in dB
* \return 6 bit the register value
*/
static int gain_to_tx_vga_reg(float &gain){
//calculate the register value
int reg = std::clip(boost::math::iround(gain*60/30.0) + 3, 0, 63);
//calculate the actual gain value
if (reg < 4) gain = 0;
else if (reg < 48) gain = reg/2 - 1;
else gain = reg/2.0 - 1.5;
//return register value
return reg;
}
/*!
* Convert a requested gain for the rx vga into the integer register value.
* The gain passed into the function will be set to the actual value.
* \param gain the requested gain in dB
* \return 5 bit the register value
*/
static int gain_to_rx_bb_vga_reg(float &gain){
int reg = std::clip(boost::math::iround(gain/2.0), 0, 31);
gain = reg*2;
return reg;
}
/*!
* Convert a requested gain for the rx lna into the integer register value.
* The gain passed into the function will be set to the actual value.
* \param gain the requested gain in dB
* \return 2 bit the register value
*/
static int gain_to_rx_rf_lna_reg(float &gain){
int reg = std::clip(boost::math::iround(gain*2/30.5) + 1, 0, 3);
switch(reg){
case 0:
case 1: gain = 0; break;
case 2: gain = 15; break;
case 3: gain = 30.5; break;
}
return reg;
}
void xcvr2450::set_tx_gain(float gain, const std::string &name){
assert_has(xcvr_tx_gain_ranges.keys(), name, "xcvr tx gain name");
//TODO
}
void xcvr2450::set_rx_gain(float gain, const std::string &name){
assert_has(xcvr_rx_gain_ranges.keys(), name, "xcvr rx gain name");
//TODO
}
/***********************************************************************
* RX Get and Set
**********************************************************************/
void xcvr2450::rx_get(const wax::obj &key_, wax::obj &val){
wax::obj key; std::string name;
boost::tie(key, name) = extract_named_prop(key_);
//handle the get request conditioned on the key
switch(key.as()){
case SUBDEV_PROP_NAME:
val = dboard_id::to_string(get_rx_id());
return;
case SUBDEV_PROP_OTHERS:
val = prop_names_t(); //empty
return;
case SUBDEV_PROP_GAIN:
assert_has(_rx_gains.keys(), name, "xcvr rx gain name");
val = _rx_gains[name];
return;
case SUBDEV_PROP_GAIN_RANGE:
assert_has(xcvr_rx_gain_ranges.keys(), name, "xcvr rx gain name");
val = xcvr_rx_gain_ranges[name];
return;
case SUBDEV_PROP_GAIN_NAMES:
val = prop_names_t(xcvr_rx_gain_ranges.keys());
return;
case SUBDEV_PROP_FREQ:
val = _lo_freq;
return;
case SUBDEV_PROP_FREQ_RANGE:
val = xcvr_freq_range;
return;
case SUBDEV_PROP_ANTENNA:
val = _rx_ant;
return;
case SUBDEV_PROP_ANTENNA_NAMES:
val = xcvr_antennas;
return;
case SUBDEV_PROP_QUADRATURE:
val = true;
return;
case SUBDEV_PROP_IQ_SWAPPED:
val = false;
return;
case SUBDEV_PROP_SPECTRUM_INVERTED:
val = false;
return;
case SUBDEV_PROP_USE_LO_OFFSET:
val = false;
return;
}
}
void xcvr2450::rx_set(const wax::obj &key_, const wax::obj &val){
wax::obj key; std::string name;
boost::tie(key, name) = extract_named_prop(key_);
//handle the get request conditioned on the key
switch(key.as()){
case SUBDEV_PROP_FREQ:
this->set_lo_freq(val.as());
return;
case SUBDEV_PROP_GAIN:
this->set_rx_gain(val.as(), name);
return;
case SUBDEV_PROP_ANTENNA:
this->set_rx_ant(val.as());
return;
default: throw std::runtime_error(str(boost::format(
"Error: trying to set read-only property on %s subdev"
) % dboard_id::to_string(get_rx_id())));
}
}
/***********************************************************************
* TX Get and Set
**********************************************************************/
void xcvr2450::tx_get(const wax::obj &key_, wax::obj &val){
wax::obj key; std::string name;
boost::tie(key, name) = extract_named_prop(key_);
//handle the get request conditioned on the key
switch(key.as()){
case SUBDEV_PROP_NAME:
val = dboard_id::to_string(get_tx_id());
return;
case SUBDEV_PROP_OTHERS:
val = prop_names_t(); //empty
return;
case SUBDEV_PROP_GAIN:
assert_has(_tx_gains.keys(), name, "xcvr tx gain name");
val = _tx_gains[name];
return;
case SUBDEV_PROP_GAIN_RANGE:
assert_has(xcvr_tx_gain_ranges.keys(), name, "xcvr tx gain name");
val = xcvr_tx_gain_ranges[name];
return;
case SUBDEV_PROP_GAIN_NAMES:
val = prop_names_t(xcvr_tx_gain_ranges.keys());
return;
case SUBDEV_PROP_FREQ:
val = _lo_freq;
return;
case SUBDEV_PROP_FREQ_RANGE:
val = xcvr_freq_range;
return;
case SUBDEV_PROP_ANTENNA:
val = _tx_ant;
return;
case SUBDEV_PROP_ANTENNA_NAMES:
val = xcvr_antennas;
return;
case SUBDEV_PROP_QUADRATURE:
val = true;
return;
case SUBDEV_PROP_IQ_SWAPPED:
val = true;
return;
case SUBDEV_PROP_SPECTRUM_INVERTED:
val = false;
return;
case SUBDEV_PROP_USE_LO_OFFSET:
val = false;
return;
}
}
void xcvr2450::tx_set(const wax::obj &key_, const wax::obj &val){
wax::obj key; std::string name;
boost::tie(key, name) = extract_named_prop(key_);
//handle the get request conditioned on the key
switch(key.as()){
case SUBDEV_PROP_FREQ:
set_lo_freq(val.as());
return;
case SUBDEV_PROP_GAIN:
this->set_tx_gain(val.as(), name);
return;
case SUBDEV_PROP_ANTENNA:
this->set_tx_ant(val.as());
return;
default: throw std::runtime_error(str(boost::format(
"Error: trying to set read-only property on %s subdev"
) % dboard_id::to_string(get_tx_id())));
}
}