1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366
|
//
// Copyright 2014 Ettus Research
// Copyright 2018 Ettus Research, a National Instruments Company
//
// SPDX-License-Identifier: GPL-3.0-or-later
//
#ifndef INCLUDED_AD9361_DEVICE_H
#define INCLUDED_AD9361_DEVICE_H
#include <ad9361_client.h>
#include <uhd/types/filters.hpp>
#include <uhd/types/sensors.hpp>
#include <uhd/utils/noncopyable.hpp>
#include <complex>
#include <vector>
#include <map>
#include <tuple>
#include <functional>
#include <mutex>
namespace uhd { namespace usrp {
class ad9361_device_t : public uhd::noncopyable
{
public:
enum direction_t { RX, TX };
enum gain_mode_t {GAIN_MODE_MANUAL, GAIN_MODE_SLOW_AGC, GAIN_MODE_FAST_AGC};
enum chain_t { CHAIN_1, CHAIN_2, CHAIN_BOTH };
enum timing_mode_t { TIMING_MODE_1R1T, TIMING_MODE_2R2T };
ad9361_device_t(ad9361_params::sptr client, ad9361_io::sptr io_iface) :
_client_params(client), _io_iface(io_iface),
_bbpll_freq(0.0), _adcclock_freq(0.0), _req_clock_rate(0.0),
_req_coreclk(0.0), _rx_bbf_tunediv(0), _curr_gain_table(0),
_rx1_gain(0.0), _rx2_gain(0.0), _tx1_gain(0.0), _tx2_gain(0.0),
_tfir_factor(0), _rfir_factor(0),
_rx1_agc_mode(GAIN_MODE_MANUAL), _rx2_agc_mode(GAIN_MODE_MANUAL),
_rx1_agc_enable(false), _rx2_agc_enable(false),
_use_dc_offset_tracking(false), _use_iq_balance_tracking(false),
_rx_filters{
{"LPF_TIA", std::make_tuple(
[this](const chain_t){
return this->_get_filter_lp_tia_sec(RX);
},
[this](const chain_t, filter_info_base::sptr filter_info){
this->_set_filter_lp_tia_sec(RX, filter_info);
}
)
},
{"LPF_BB", std::make_tuple(
[this](const chain_t){
return this->_get_filter_lp_bb(RX);
},
[this](const chain_t, filter_info_base::sptr filter_info){
this->_set_filter_lp_bb(RX, filter_info);
}
)
},
{"HB_3", std::make_tuple(
[this](const chain_t){
return this->_get_filter_hb_3(RX);
},
nullptr
)
},
{"DEC_3", std::make_tuple(
[this](const chain_t){
return this->_get_filter_dec_int_3(RX);
},
nullptr
)
},
{"HB_2", std::make_tuple(
[this](const chain_t){
return this->_get_filter_hb_2(RX);
},
nullptr
)
},
{"HB_1", std::make_tuple(
[this](const chain_t){
return this->_get_filter_hb_1(RX);
},
nullptr
)
},
{"FIR_1", std::make_tuple(
[this](const chain_t channel){
return this->_get_filter_fir(RX, channel);
},
[this](const chain_t channel, filter_info_base::sptr filter_info){
this->_set_filter_fir(RX, channel, filter_info);
}
)
}
},
_tx_filters{
{"LPF_SECONDARY", std::make_tuple(
[this](const chain_t){
return this->_get_filter_lp_tia_sec(TX);
},
[this](const chain_t, filter_info_base::sptr filter_info){
this->_set_filter_lp_tia_sec(TX, filter_info);
}
)
},
{"LPF_BB", std::make_tuple(
[this](const chain_t){
return this->_get_filter_lp_bb(TX);
},
[this](const chain_t, filter_info_base::sptr filter_info){
this->_set_filter_lp_bb(TX, filter_info);
}
)
},
{"HB_3", std::make_tuple(
[this](const chain_t){
return this->_get_filter_hb_3(TX);
},
nullptr
)
},
{"INT_3", std::make_tuple(
[this](const chain_t){
return this->_get_filter_dec_int_3(TX);
},
nullptr
)
},
{"HB_2", std::make_tuple(
[this](const chain_t){
return this->_get_filter_hb_2(TX);
},
nullptr
)
},
{"HB_1", std::make_tuple(
[this](const chain_t){
return this->_get_filter_hb_1(TX);
},
nullptr
)
},
{"FIR_1", std::make_tuple(
[this](const chain_t channel){
return this->_get_filter_fir(TX, channel);
},
[this](const chain_t channel, filter_info_base::sptr filter_info){
this->_set_filter_fir(TX, channel, filter_info);
}
)
},
}
{
}
/* Initialize the AD9361 codec. */
void initialize();
/* Set SPI interface */
void set_io_iface(ad9361_io::sptr io_iface);
/* Get the current clock rate. */
double get_clock_rate() const;
/* This function sets the RX / TX rate between AD9361 and the FPGA, and
* thus determines the interpolation / decimation required in the FPGA to
* achieve the user's requested rate.
*/
double set_clock_rate(const double req_rate);
/* Set which of the four TX / RX chains provided by AD9361 are active.
*
* AD9361 provides two sets of chains, Side A and Side B. Each side
* provides one TX antenna, and one RX antenna. The B200 maintains the USRP
* standard of providing one antenna connection that is both TX & RX, and
* one that is RX-only - for each chain. Thus, the possible antenna and
* chain selections are:
*
*/
void set_active_chains(bool tx1, bool tx2, bool rx1, bool rx2);
/* Setup Timing mode depending on active channels.
*
* LVDS interface can have two timing modes - 1R1T and 2R2T
*/
void set_timing_mode(const timing_mode_t timing_mode);
/* Tune the RX or TX frequency.
*
* This is the publicly-accessible tune function. It makes sure the tune
* isn't a redundant request, and if not, passes it on to the class's
* internal tune function.
*
* After tuning, it runs any appropriate calibrations. */
double tune(direction_t direction, const double value);
/* Get the current RX or TX frequency. */
double get_freq(direction_t direction);
/* Set the gain of RX1, RX2, TX1, or TX2.
*
* Note that the 'value' passed to this function is the actual gain value,
* _not_ the gain index. This is the opposite of the eval software's GUI!
* Also note that the RX chains are done in terms of gain, and the TX chains
* are done in terms of attenuation. */
double set_gain(direction_t direction, chain_t chain, const double value);
/* Make AD9361 output its test tone. */
void output_test_tone();
void digital_test_tone(bool enb); // Digital output
/* Turn on/off AD9361's TX port --> RX port loopback. */
void data_port_loopback(const bool loopback_enabled);
/* Read back the internal RSSI measurement data. */
double get_rssi(chain_t chain);
/*! Read the internal temperature sensor
*\param calibrate return raw sensor readings or apply calibration factor.
*\param num_samples number of measurements to average over
*/
double get_average_temperature(const double cal_offset = -30.0, const size_t num_samples = 3);
/* Turn on/off AD9361's RX DC offset correction */
void set_dc_offset_auto(direction_t direction, const bool on);
/* Turn on/off AD9361's RX IQ imbalance correction */
void set_iq_balance_auto(direction_t direction, const bool on);
/* Configure AD9361's AGC module to use either fast or slow AGC mode. */
void set_agc_mode(chain_t chain, gain_mode_t gain_mode);
/* Enable AD9361's AGC gain mode. */
void set_agc(chain_t chain, bool enable);
/* Set bandwidth of AD9361's analog LP filters.
* Bandwidth should be RF bandwidth */
double set_bw_filter(direction_t direction, const double rf_bw);
/*
* Filter API implementation
* */
filter_info_base::sptr get_filter(direction_t direction, chain_t chain, const std::string &name);
void set_filter(direction_t direction, chain_t chain, const std::string &name, filter_info_base::sptr filter);
std::vector<std::string> get_filter_names(direction_t direction);
//Constants
static const double AD9361_MAX_GAIN;
static const double AD9361_MAX_CLOCK_RATE;
static const double AD9361_MIN_CLOCK_RATE;
static const double AD9361_CAL_VALID_WINDOW;
static const double AD9361_MIN_BW;
static const double AD9361_MAX_BW;
static const double DEFAULT_RX_FREQ;
static const double DEFAULT_TX_FREQ;
private: //Methods
void _program_fir_filter(direction_t direction, int num_taps, uint16_t *coeffs);
void _setup_tx_fir(size_t num_taps, int32_t interpolation);
void _setup_rx_fir(size_t num_taps, int32_t decimation);
void _program_fir_filter(direction_t direction, chain_t chain, int num_taps, uint16_t *coeffs);
void _setup_tx_fir(size_t num_taps);
void _setup_rx_fir(size_t num_taps);
void _calibrate_lock_bbpll();
void _calibrate_synth_charge_pumps();
double _calibrate_baseband_rx_analog_filter(double rfbw);
double _calibrate_baseband_tx_analog_filter(double rfbw);
double _calibrate_secondary_tx_filter(double rfbw);
double _calibrate_rx_TIAs(double rfbw);
void _setup_adc();
void _calibrate_baseband_dc_offset();
void _calibrate_rf_dc_offset();
void _calibrate_rx_quadrature();
void _tx_quadrature_cal_routine();
void _calibrate_tx_quadrature();
void _program_mixer_gm_subtable();
void _program_gain_table();
void _setup_gain_control(bool use_agc);
void _setup_synth(direction_t direction, double vcorate);
double _tune_bbvco(const double rate);
void _reprogram_gains();
double _tune_helper(direction_t direction, const double value);
double _setup_rates(const double rate);
double _get_temperature(const double cal_offset, const double timeout = 0.1);
void _configure_bb_dc_tracking();
void _configure_rx_iq_tracking();
void _setup_agc(chain_t chain, gain_mode_t gain_mode);
void _set_fir_taps(direction_t direction, chain_t chain, const std::vector<int16_t>& taps);
std::vector<int16_t> _get_fir_taps(direction_t direction, chain_t chain);
size_t _get_num_fir_taps(direction_t direction);
size_t _get_fir_dec_int(direction_t direction);
filter_info_base::sptr _get_filter_lp_tia_sec(direction_t direction);
filter_info_base::sptr _get_filter_lp_bb(direction_t direction);
filter_info_base::sptr _get_filter_dec_int_3(direction_t direction);
filter_info_base::sptr _get_filter_hb_3(direction_t direction);
filter_info_base::sptr _get_filter_hb_2(direction_t direction);
filter_info_base::sptr _get_filter_hb_1(direction_t direction);
filter_info_base::sptr _get_filter_fir(direction_t direction, chain_t chain);
void _set_filter_fir(direction_t direction, chain_t channel, filter_info_base::sptr filter);
void _set_filter_lp_bb(direction_t direction, filter_info_base::sptr filter);
void _set_filter_lp_tia_sec(direction_t direction, filter_info_base::sptr filter);
private: //Members
struct chip_regs_t
{
chip_regs_t():
vcodivs(0), inputsel(0), rxfilt(0), txfilt(0),
bbpll(0), bbftune_config(0), bbftune_mode(0) {}
uint8_t vcodivs;
uint8_t inputsel;
uint8_t rxfilt;
uint8_t txfilt;
uint8_t bbpll;
uint8_t bbftune_config;
uint8_t bbftune_mode;
};
//Interfaces
ad9361_params::sptr _client_params;
ad9361_io::sptr _io_iface;
//Intermediate state
double _rx_freq, _tx_freq, _req_rx_freq, _req_tx_freq;
double _last_rx_cal_freq, _last_tx_cal_freq;
double _rx_analog_bw, _tx_analog_bw, _rx_bb_lp_bw, _tx_bb_lp_bw;
double _rx_tia_lp_bw, _tx_sec_lp_bw;
//! Current baseband sampling rate (this is the actual rate the device is
// is running at)
double _baseband_bw;
double _bbpll_freq, _adcclock_freq;
//! This was the last clock rate value that was requested.
// It is cached so we don't need to re-set the clock rate
// if another call to set_clock_rate() actually has the same value.
double _req_clock_rate;
double _req_coreclk;
uint16_t _rx_bbf_tunediv;
uint8_t _curr_gain_table;
double _rx1_gain, _rx2_gain, _tx1_gain, _tx2_gain;
int32_t _tfir_factor;
int32_t _rfir_factor;
gain_mode_t _rx1_agc_mode, _rx2_agc_mode;
bool _rx1_agc_enable, _rx2_agc_enable;
//Register soft-copies
chip_regs_t _regs;
//Synchronization
std::recursive_mutex _mutex;
bool _use_dc_offset_tracking;
bool _use_iq_balance_tracking;
// Filter API
using filter_tuple = std::tuple<
std::function<filter_info_base::sptr(const chain_t)>, // getter
std::function<void(chain_t, filter_info_base::sptr)> // setter
>;
std::map<std::string, filter_tuple> _rx_filters;
std::map<std::string, filter_tuple> _tx_filters;
};
}} //namespace
#endif /* INCLUDED_AD9361_DEVICE_H */
|