diff options
Diffstat (limited to 'host/lib/usrp/dboard/db_dbsrx.cpp')
-rw-r--r-- | host/lib/usrp/dboard/db_dbsrx.cpp | 533 |
1 files changed, 292 insertions, 241 deletions
diff --git a/host/lib/usrp/dboard/db_dbsrx.cpp b/host/lib/usrp/dboard/db_dbsrx.cpp index 6e4d482be..238dbd95e 100644 --- a/host/lib/usrp/dboard/db_dbsrx.cpp +++ b/host/lib/usrp/dboard/db_dbsrx.cpp @@ -37,24 +37,23 @@ using namespace boost::assign; **********************************************************************/ static const freq_range_t dbsrx_freq_range(0.8e9, 2.4e9); -//Multiplied by 2.0 for conversion to complex bandpass from lowpass -static const freq_range_t dbsrx_bandwidth_range(2.0*4.0e6, 2.0*33.0e6); +// Multiplied by 2.0 for conversion to complex bandpass from lowpass +static const freq_range_t dbsrx_bandwidth_range(2.0 * 4.0e6, 2.0 * 33.0e6); static const freq_range_t dbsrx_pfd_freq_range(0.15e6, 2.01e6); static const std::vector<std::string> dbsrx_antennas = list_of("J3"); -static const uhd::dict<std::string, gain_range_t> dbsrx_gain_ranges = map_list_of - ("GC1", gain_range_t(0, 56, 0.5)) - ("GC2", gain_range_t(0, 24, 1)) -; +static const uhd::dict<std::string, gain_range_t> dbsrx_gain_ranges = + map_list_of("GC1", gain_range_t(0, 56, 0.5))("GC2", gain_range_t(0, 24, 1)); static const double usrp1_gpio_clock_rate_limit = 4e6; /*********************************************************************** * The DBSRX dboard class **********************************************************************/ -class dbsrx : public rx_dboard_base{ +class dbsrx : public rx_dboard_base +{ public: dbsrx(ctor_args_t args); virtual ~dbsrx(void); @@ -65,65 +64,74 @@ private: uhd::dict<std::string, double> _gains; max2118_write_regs_t _max2118_write_regs; max2118_read_regs_t _max2118_read_regs; - uint8_t _max2118_addr(void){ - return (this->get_iface()->get_special_props().mangle_i2c_addrs)? 0x65 : 0x67; + uint8_t _max2118_addr(void) + { + return (this->get_iface()->get_special_props().mangle_i2c_addrs) ? 0x65 : 0x67; }; double set_lo_freq(double target_freq); - double set_gain(double gain, const std::string &name); + double set_gain(double gain, const std::string& name); double set_bandwidth(double bandwidth); - void send_reg(uint8_t start_reg, uint8_t stop_reg){ + void send_reg(uint8_t start_reg, uint8_t stop_reg) + { start_reg = uint8_t(uhd::clip(int(start_reg), 0x0, 0x5)); - stop_reg = uint8_t(uhd::clip(int(stop_reg), 0x0, 0x5)); + stop_reg = uint8_t(uhd::clip(int(stop_reg), 0x0, 0x5)); - for(uint8_t start_addr=start_reg; start_addr <= stop_reg; start_addr += sizeof(uint32_t) - 1){ - int num_bytes = int(stop_reg - start_addr + 1) > int(sizeof(uint32_t)) - 1 ? sizeof(uint32_t) - 1 : stop_reg - start_addr + 1; + for (uint8_t start_addr = start_reg; start_addr <= stop_reg; + start_addr += sizeof(uint32_t) - 1) { + int num_bytes = int(stop_reg - start_addr + 1) > int(sizeof(uint32_t)) - 1 + ? sizeof(uint32_t) - 1 + : stop_reg - start_addr + 1; - //create buffer for register data (+1 for start address) + // create buffer for register data (+1 for start address) byte_vector_t regs_vector(num_bytes + 1); - //first byte is the address of first register + // first byte is the address of first register regs_vector[0] = start_addr; - //get the register data - for(int i=0; i<num_bytes; i++){ - regs_vector[1+i] = _max2118_write_regs.get_reg(start_addr+i); - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX: send reg 0x%02x, value 0x%04x, start_addr = 0x%04x, num_bytes %d" - ) % int(start_addr+i) % int(regs_vector[1+i]) % int(start_addr) % num_bytes ; + // get the register data + for (int i = 0; i < num_bytes; i++) { + regs_vector[1 + i] = _max2118_write_regs.get_reg(start_addr + i); + UHD_LOGGER_TRACE("DBSRX") + << boost::format("DBSRX: send reg 0x%02x, value 0x%04x, start_addr = " + "0x%04x, num_bytes %d") + % int(start_addr + i) % int(regs_vector[1 + i]) + % int(start_addr) % num_bytes; } - //send the data - this->get_iface()->write_i2c( - _max2118_addr(), regs_vector - ); + // send the data + this->get_iface()->write_i2c(_max2118_addr(), regs_vector); } } - void read_reg(uint8_t start_reg, uint8_t stop_reg){ + void read_reg(uint8_t start_reg, uint8_t stop_reg) + { static const uint8_t status_addr = 0x0; - start_reg = uint8_t(uhd::clip(int(start_reg), 0x0, 0x1)); - stop_reg = uint8_t(uhd::clip(int(stop_reg), 0x0, 0x1)); + start_reg = uint8_t(uhd::clip(int(start_reg), 0x0, 0x1)); + stop_reg = uint8_t(uhd::clip(int(stop_reg), 0x0, 0x1)); - for(uint8_t start_addr=start_reg; start_addr <= stop_reg; start_addr += sizeof(uint32_t)){ - int num_bytes = int(stop_reg - start_addr + 1) > int(sizeof(uint32_t)) ? sizeof(uint32_t) : stop_reg - start_addr + 1; + for (uint8_t start_addr = start_reg; start_addr <= stop_reg; + start_addr += sizeof(uint32_t)) { + int num_bytes = int(stop_reg - start_addr + 1) > int(sizeof(uint32_t)) + ? sizeof(uint32_t) + : stop_reg - start_addr + 1; - //create buffer for register data + // create buffer for register data byte_vector_t regs_vector(num_bytes); - //read from i2c - regs_vector = this->get_iface()->read_i2c( - _max2118_addr(), num_bytes - ); + // read from i2c + regs_vector = this->get_iface()->read_i2c(_max2118_addr(), num_bytes); - for(uint8_t i=0; i < num_bytes; i++){ - if (i + start_addr >= status_addr){ + for (uint8_t i = 0; i < num_bytes; i++) { + if (i + start_addr >= status_addr) { _max2118_read_regs.set_reg(i + start_addr, regs_vector[i]); } - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX: read reg 0x%02x, value 0x%04x, start_addr = 0x%04x, num_bytes %d" - ) % int(start_addr+i) % int(regs_vector[i]) % int(start_addr) % num_bytes ; + UHD_LOGGER_TRACE("DBSRX") + << boost::format("DBSRX: read reg 0x%02x, value 0x%04x, start_addr = " + "0x%04x, num_bytes %d") + % int(start_addr + i) % int(regs_vector[i]) % int(start_addr) + % num_bytes; } } } @@ -132,15 +140,14 @@ private: * Get the lock detect status of the LO. * \return sensor for locked */ - sensor_value_t get_locked(void){ + sensor_value_t get_locked(void) + { read_reg(0x0, 0x0); - //mask and return lock detect + // mask and return lock detect bool locked = 5 >= _max2118_read_regs.adc and _max2118_read_regs.adc >= 2; - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX: locked %d" - ) % locked ; + UHD_LOGGER_TRACE("DBSRX") << boost::format("DBSRX: locked %d") % locked; return sensor_value_t("LO", locked, "locked", "unlocked"); } @@ -149,190 +156,214 @@ private: /*********************************************************************** * Register the DBSRX dboard **********************************************************************/ -static dboard_base::sptr make_dbsrx(dboard_base::ctor_args_t args){ +static dboard_base::sptr make_dbsrx(dboard_base::ctor_args_t args) +{ return dboard_base::sptr(new dbsrx(args)); } -UHD_STATIC_BLOCK(reg_dbsrx_dboard){ - //register the factory function for the rx dbid (others version) +UHD_STATIC_BLOCK(reg_dbsrx_dboard) +{ + // register the factory function for the rx dbid (others version) dboard_manager::register_dboard(0x000D, &make_dbsrx, "DBSRX"); - //register the factory function for the rx dbid (USRP1 version) + // register the factory function for the rx dbid (USRP1 version) dboard_manager::register_dboard(0x0002, &make_dbsrx, "DBSRX"); } /*********************************************************************** * Structors **********************************************************************/ -dbsrx::dbsrx(ctor_args_t args) : rx_dboard_base(args){ - //warn user about incorrect DBID on USRP1, requires R193 populated - if (this->get_iface()->get_special_props().soft_clock_divider and this->get_rx_id() == 0x000D) - UHD_LOGGER_WARNING("DBSRX") << boost::format( - "DBSRX: incorrect dbid\n" - "Expected dbid 0x0002 and R193\n" - "found dbid == %d\n" - "Please see the daughterboard app notes" - ) % this->get_rx_id().to_pp_string(); - - //warn user about incorrect DBID on non-USRP1, requires R194 populated - if (not this->get_iface()->get_special_props().soft_clock_divider and this->get_rx_id() == 0x0002) - UHD_LOGGER_WARNING("DBSRX") << boost::format( - "DBSRX: incorrect dbid\n" - "Expected dbid 0x000D and R194\n" - "found dbid == %d\n" - "Please see the daughterboard app notes" - ) % this->get_rx_id().to_pp_string(); - - //send initial register settings +dbsrx::dbsrx(ctor_args_t args) : rx_dboard_base(args) +{ + // warn user about incorrect DBID on USRP1, requires R193 populated + if (this->get_iface()->get_special_props().soft_clock_divider + and this->get_rx_id() == 0x000D) + UHD_LOGGER_WARNING("DBSRX") + << boost::format("DBSRX: incorrect dbid\n" + "Expected dbid 0x0002 and R193\n" + "found dbid == %d\n" + "Please see the daughterboard app notes") + % this->get_rx_id().to_pp_string(); + + // warn user about incorrect DBID on non-USRP1, requires R194 populated + if (not this->get_iface()->get_special_props().soft_clock_divider + and this->get_rx_id() == 0x0002) + UHD_LOGGER_WARNING("DBSRX") + << boost::format("DBSRX: incorrect dbid\n" + "Expected dbid 0x000D and R194\n" + "found dbid == %d\n" + "Please see the daughterboard app notes") + % this->get_rx_id().to_pp_string(); + + // send initial register settings this->send_reg(0x0, 0x5); - //set defaults for LO, gains, and filter bandwidth + // set defaults for LO, gains, and filter bandwidth double codec_rate = this->get_iface()->get_codec_rate(dboard_iface::UNIT_RX); - _bandwidth = 0.8*codec_rate/2.0; // default to anti-alias at different codec_rate + _bandwidth = 0.8 * codec_rate / 2.0; // default to anti-alias at different codec_rate //////////////////////////////////////////////////////////////////// // Register properties //////////////////////////////////////////////////////////////////// - this->get_rx_subtree()->create<std::string>("name") - .set("DBSRX"); - this->get_rx_subtree()->create<sensor_value_t>("sensors/lo_locked") + this->get_rx_subtree()->create<std::string>("name").set("DBSRX"); + this->get_rx_subtree() + ->create<sensor_value_t>("sensors/lo_locked") .set_publisher(std::bind(&dbsrx::get_locked, this)); - for(const std::string &name: dbsrx_gain_ranges.keys()){ - this->get_rx_subtree()->create<double>("gains/"+name+"/value") + for (const std::string& name : dbsrx_gain_ranges.keys()) { + this->get_rx_subtree() + ->create<double>("gains/" + name + "/value") .set_coercer(std::bind(&dbsrx::set_gain, this, std::placeholders::_1, name)) .set(dbsrx_gain_ranges[name].start()); - this->get_rx_subtree()->create<meta_range_t>("gains/"+name+"/range") + this->get_rx_subtree() + ->create<meta_range_t>("gains/" + name + "/range") .set(dbsrx_gain_ranges[name]); } - this->get_rx_subtree()->create<double>("freq/value") + this->get_rx_subtree() + ->create<double>("freq/value") .set_coercer(std::bind(&dbsrx::set_lo_freq, this, std::placeholders::_1)); - this->get_rx_subtree()->create<meta_range_t>("freq/range") - .set(dbsrx_freq_range); - this->get_rx_subtree()->create<std::string>("antenna/value") + this->get_rx_subtree()->create<meta_range_t>("freq/range").set(dbsrx_freq_range); + this->get_rx_subtree() + ->create<std::string>("antenna/value") .set(dbsrx_antennas.at(0)); - this->get_rx_subtree()->create<std::vector<std::string> >("antenna/options") + this->get_rx_subtree() + ->create<std::vector<std::string>>("antenna/options") .set(dbsrx_antennas); - this->get_rx_subtree()->create<std::string>("connection") - .set("IQ"); - this->get_rx_subtree()->create<bool>("enabled") - .set(true); //always enabled - this->get_rx_subtree()->create<bool>("use_lo_offset") - .set(false); - this->get_rx_subtree()->create<double>("bandwidth/value") + this->get_rx_subtree()->create<std::string>("connection").set("IQ"); + this->get_rx_subtree()->create<bool>("enabled").set(true); // always enabled + this->get_rx_subtree()->create<bool>("use_lo_offset").set(false); + this->get_rx_subtree() + ->create<double>("bandwidth/value") .set_coercer(std::bind(&dbsrx::set_bandwidth, this, std::placeholders::_1)); - this->get_rx_subtree()->create<meta_range_t>("bandwidth/range") + this->get_rx_subtree() + ->create<meta_range_t>("bandwidth/range") .set(dbsrx_bandwidth_range); - //enable only the clocks we need + // enable only the clocks we need this->get_iface()->set_clock_enabled(dboard_iface::UNIT_RX, true); - //set the gpio directions and atr controls (identically) + // set the gpio directions and atr controls (identically) this->get_iface()->set_pin_ctrl(dboard_iface::UNIT_RX, 0x0); // All unused in atr - if (this->get_iface()->get_special_props().soft_clock_divider){ - this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_RX, 0x1); // GPIO0 is clock when on USRP1 - } - else{ + if (this->get_iface()->get_special_props().soft_clock_divider) { + this->get_iface()->set_gpio_ddr( + dboard_iface::UNIT_RX, 0x1); // GPIO0 is clock when on USRP1 + } else { this->get_iface()->set_gpio_ddr(dboard_iface::UNIT_RX, 0x0); // All Inputs } - //now its safe to set inital freq and bw - this->get_rx_subtree()->access<double>("freq/value") - .set(dbsrx_freq_range.start()); - this->get_rx_subtree()->access<double>("bandwidth/value") - .set(2.0*_bandwidth); //_bandwidth in lowpass, convert to complex bandpass + // now its safe to set inital freq and bw + this->get_rx_subtree()->access<double>("freq/value").set(dbsrx_freq_range.start()); + this->get_rx_subtree() + ->access<double>("bandwidth/value") + .set(2.0 * _bandwidth); //_bandwidth in lowpass, convert to complex bandpass } -dbsrx::~dbsrx(void){ -} +dbsrx::~dbsrx(void) {} /*********************************************************************** * Tuning **********************************************************************/ -double dbsrx::set_lo_freq(double target_freq){ +double dbsrx::set_lo_freq(double target_freq) +{ target_freq = dbsrx_freq_range.clip(target_freq); - double actual_freq=0.0, pfd_freq=0.0, ref_clock=0.0; - int R=0, N=0, r=0, m=0; + double actual_freq = 0.0, pfd_freq = 0.0, ref_clock = 0.0; + int R = 0, N = 0, r = 0, m = 0; bool update_filter_settings = false; - //choose refclock - std::vector<double> clock_rates = this->get_iface()->get_clock_rates(dboard_iface::UNIT_RX); + // choose refclock + std::vector<double> clock_rates = + this->get_iface()->get_clock_rates(dboard_iface::UNIT_RX); const double max_clock_rate = uhd::sorted(clock_rates).back(); - for(auto ref_clock: uhd::reversed(uhd::sorted(clock_rates))){ - //USRP1 feeds the DBSRX clock from a FPGA GPIO line. - //make sure that this clock does not exceed rate limit. - if (this->get_iface()->get_special_props().soft_clock_divider){ - if (ref_clock > usrp1_gpio_clock_rate_limit) continue; + for (auto ref_clock : uhd::reversed(uhd::sorted(clock_rates))) { + // USRP1 feeds the DBSRX clock from a FPGA GPIO line. + // make sure that this clock does not exceed rate limit. + if (this->get_iface()->get_special_props().soft_clock_divider) { + if (ref_clock > usrp1_gpio_clock_rate_limit) + continue; } - if (ref_clock > 27.0e6) continue; - if (size_t(max_clock_rate/ref_clock)%2 == 1) continue; //reject asymmetric clocks (odd divisors) + if (ref_clock > 27.0e6) + continue; + if (size_t(max_clock_rate / ref_clock) % 2 == 1) + continue; // reject asymmetric clocks (odd divisors) - //choose m_divider such that filter tuning constraint is met + // choose m_divider such that filter tuning constraint is met m = 31; - while ((ref_clock/m < 1e6 or ref_clock/m > 2.5e6) and m > 0){ m--; } + while ((ref_clock / m < 1e6 or ref_clock / m > 2.5e6) and m > 0) { + m--; + } - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX: trying ref_clock %f and m_divider %d" - ) % (ref_clock) % m ; + UHD_LOGGER_TRACE("DBSRX") + << boost::format("DBSRX: trying ref_clock %f and m_divider %d") % (ref_clock) + % m; - if (m >= 32) continue; + if (m >= 32) + continue; - //choose R - for(auto r = 0; r <= 6; r += 1) { - //compute divider from setting - R = 1 << (r+1); - UHD_LOGGER_TRACE("DBSRX") << boost::format("DBSRX R:%d\n") % R ; + // choose R + for (auto r = 0; r <= 6; r += 1) { + // compute divider from setting + R = 1 << (r + 1); + UHD_LOGGER_TRACE("DBSRX") << boost::format("DBSRX R:%d\n") % R; - //compute PFD compare frequency = ref_clock/R + // compute PFD compare frequency = ref_clock/R pfd_freq = ref_clock / R; - //constrain the PFD frequency to specified range - if ((pfd_freq < dbsrx_pfd_freq_range.start()) or (pfd_freq > dbsrx_pfd_freq_range.stop())) continue; + // constrain the PFD frequency to specified range + if ((pfd_freq < dbsrx_pfd_freq_range.start()) + or (pfd_freq > dbsrx_pfd_freq_range.stop())) + continue; - //compute N - N = int(std::floor(target_freq/pfd_freq)); + // compute N + N = int(std::floor(target_freq / pfd_freq)); - //constrain N to specified range - if ((N < 256) or (N > 32768)) continue; + // constrain N to specified range + if ((N < 256) or (N > 32768)) + continue; goto done_loop; } } - done_loop: +done_loop: - //Assert because we failed to find a suitable combination of ref_clock, R and N + // Assert because we failed to find a suitable combination of ref_clock, R and N UHD_ASSERT_THROW(ref_clock <= 27.0e6 and ref_clock >= 0.0); - UHD_ASSERT_THROW(m and ref_clock/m >= 1e6 and ref_clock/m <= 2.5e6); - UHD_ASSERT_THROW((pfd_freq >= dbsrx_pfd_freq_range.start()) and (pfd_freq <= dbsrx_pfd_freq_range.stop())); + UHD_ASSERT_THROW(m and ref_clock / m >= 1e6 and ref_clock / m <= 2.5e6); + UHD_ASSERT_THROW((pfd_freq >= dbsrx_pfd_freq_range.start()) + and (pfd_freq <= dbsrx_pfd_freq_range.stop())); UHD_ASSERT_THROW((N >= 256) and (N <= 32768)); - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX: choose ref_clock (current: %f, new: %f) and m_divider %d" - ) % (this->get_iface()->get_clock_rate(dboard_iface::UNIT_RX)) % ref_clock % m ; + UHD_LOGGER_TRACE("DBSRX") + << boost::format( + "DBSRX: choose ref_clock (current: %f, new: %f) and m_divider %d") + % (this->get_iface()->get_clock_rate(dboard_iface::UNIT_RX)) % ref_clock + % m; - //if ref_clock or m divider changed, we need to update the filter settings - if (ref_clock != this->get_iface()->get_clock_rate(dboard_iface::UNIT_RX) or m != _max2118_write_regs.m_divider) update_filter_settings = true; + // if ref_clock or m divider changed, we need to update the filter settings + if (ref_clock != this->get_iface()->get_clock_rate(dboard_iface::UNIT_RX) + or m != _max2118_write_regs.m_divider) + update_filter_settings = true; - //compute resulting output frequency + // compute resulting output frequency actual_freq = pfd_freq * N; - //apply ref_clock, R, and N settings + // apply ref_clock, R, and N settings this->get_iface()->set_clock_rate(dboard_iface::UNIT_RX, ref_clock); ref_clock = this->get_iface()->get_clock_rate(dboard_iface::UNIT_RX); _max2118_write_regs.m_divider = m; - _max2118_write_regs.r_divider = (max2118_write_regs_t::r_divider_t) r; + _max2118_write_regs.r_divider = (max2118_write_regs_t::r_divider_t)r; _max2118_write_regs.set_n_divider(N); _max2118_write_regs.ade_vco_ade_read = max2118_write_regs_t::ADE_VCO_ADE_READ_ENABLED; - //compute prescaler variables - int scaler = actual_freq > 1125e6 ? 2 : 4; - _max2118_write_regs.div2 = scaler == 4 ? max2118_write_regs_t::DIV2_DIV4 : max2118_write_regs_t::DIV2_DIV2; + // compute prescaler variables + int scaler = actual_freq > 1125e6 ? 2 : 4; + _max2118_write_regs.div2 = scaler == 4 ? max2118_write_regs_t::DIV2_DIV4 + : max2118_write_regs_t::DIV2_DIV2; - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX: scaler %d, actual_freq %f MHz, register bit: %d" - ) % scaler % (actual_freq/1e6) % int(_max2118_write_regs.div2) ; + UHD_LOGGER_TRACE("DBSRX") + << boost::format("DBSRX: scaler %d, actual_freq %f MHz, register bit: %d") + % scaler % (actual_freq / 1e6) % int(_max2118_write_regs.div2); - //compute vco frequency and select vco + // compute vco frequency and select vco double vco_freq = actual_freq * scaler; if (vco_freq < 2433e6) _max2118_write_regs.osc_band = 0; @@ -351,82 +382,94 @@ double dbsrx::set_lo_freq(double target_freq){ else _max2118_write_regs.osc_band = 7; - //send settings over i2c + // send settings over i2c send_reg(0x0, 0x4); - //check vtune for lock condition + // check vtune for lock condition read_reg(0x0, 0x0); UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX: initial guess for vco %d, vtune adc %d" - ) % int(_max2118_write_regs.osc_band) % int(_max2118_read_regs.adc) ; - - //if we are out of lock for chosen vco, change vco - while ((_max2118_read_regs.adc == 0) or (_max2118_read_regs.adc == 7)){ - - //vtune is too low, try lower frequency vco - if (_max2118_read_regs.adc == 0){ - if (_max2118_write_regs.osc_band == 0){ - UHD_LOGGER_WARNING("DBSRX") << boost::format( - "DBSRX: Tuning exceeded vco range, _max2118_write_regs.osc_band == %d\n" - ) % int(_max2118_write_regs.osc_band); - UHD_ASSERT_THROW(_max2118_read_regs.adc != 0); //just to cause a throw + "DBSRX: initial guess for vco %d, vtune adc %d") + % int(_max2118_write_regs.osc_band) + % int(_max2118_read_regs.adc); + + // if we are out of lock for chosen vco, change vco + while ((_max2118_read_regs.adc == 0) or (_max2118_read_regs.adc == 7)) { + // vtune is too low, try lower frequency vco + if (_max2118_read_regs.adc == 0) { + if (_max2118_write_regs.osc_band == 0) { + UHD_LOGGER_WARNING("DBSRX") + << boost::format("DBSRX: Tuning exceeded vco range, " + "_max2118_write_regs.osc_band == %d\n") + % int(_max2118_write_regs.osc_band); + UHD_ASSERT_THROW(_max2118_read_regs.adc != 0); // just to cause a throw } - if (_max2118_write_regs.osc_band <= 0) break; + if (_max2118_write_regs.osc_band <= 0) + break; _max2118_write_regs.osc_band -= 1; } - //vtune is too high, try higher frequency vco - if (_max2118_read_regs.adc == 7){ - if (_max2118_write_regs.osc_band == 7){ - UHD_LOGGER_WARNING("DBSRX") << boost::format( - "DBSRX: Tuning exceeded vco range, _max2118_write_regs.osc_band == %d\n" - ) % int(_max2118_write_regs.osc_band); - UHD_ASSERT_THROW(_max2118_read_regs.adc != 7); //just to cause a throw + // vtune is too high, try higher frequency vco + if (_max2118_read_regs.adc == 7) { + if (_max2118_write_regs.osc_band == 7) { + UHD_LOGGER_WARNING("DBSRX") + << boost::format("DBSRX: Tuning exceeded vco range, " + "_max2118_write_regs.osc_band == %d\n") + % int(_max2118_write_regs.osc_band); + UHD_ASSERT_THROW(_max2118_read_regs.adc != 7); // just to cause a throw } - if (_max2118_write_regs.osc_band >= 7) break; + if (_max2118_write_regs.osc_band >= 7) + break; _max2118_write_regs.osc_band += 1; } - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX: trying vco %d, vtune adc %d" - ) % int(_max2118_write_regs.osc_band) % int(_max2118_read_regs.adc) ; + UHD_LOGGER_TRACE("DBSRX") << boost::format("DBSRX: trying vco %d, vtune adc %d") + % int(_max2118_write_regs.osc_band) + % int(_max2118_read_regs.adc); - //update vco selection and check vtune + // update vco selection and check vtune send_reg(0x2, 0x2); read_reg(0x0, 0x0); - //allow for setup time before checking condition again + // allow for setup time before checking condition again std::this_thread::sleep_for(std::chrono::milliseconds(10)); } - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX: final vco %d, vtune adc %d" - ) % int(_max2118_write_regs.osc_band) % int(_max2118_read_regs.adc) ; + UHD_LOGGER_TRACE("DBSRX") << boost::format("DBSRX: final vco %d, vtune adc %d") + % int(_max2118_write_regs.osc_band) + % int(_max2118_read_regs.adc); - //select charge pump bias current - if (_max2118_read_regs.adc <= 2) _max2118_write_regs.cp_current = max2118_write_regs_t::CP_CURRENT_I_CP_100UA; - else if (_max2118_read_regs.adc >= 5) _max2118_write_regs.cp_current = max2118_write_regs_t::CP_CURRENT_I_CP_400UA; - else _max2118_write_regs.cp_current = max2118_write_regs_t::CP_CURRENT_I_CP_200UA; + // select charge pump bias current + if (_max2118_read_regs.adc <= 2) + _max2118_write_regs.cp_current = max2118_write_regs_t::CP_CURRENT_I_CP_100UA; + else if (_max2118_read_regs.adc >= 5) + _max2118_write_regs.cp_current = max2118_write_regs_t::CP_CURRENT_I_CP_400UA; + else + _max2118_write_regs.cp_current = max2118_write_regs_t::CP_CURRENT_I_CP_200UA; - //update charge pump bias current setting + // update charge pump bias current setting send_reg(0x2, 0x2); - //compute actual tuned frequency - _lo_freq = this->get_iface()->get_clock_rate(dboard_iface::UNIT_RX) / std::pow(2.0,(1 + _max2118_write_regs.r_divider)) * _max2118_write_regs.get_n_divider(); + // compute actual tuned frequency + _lo_freq = this->get_iface()->get_clock_rate(dboard_iface::UNIT_RX) + / std::pow(2.0, (1 + _max2118_write_regs.r_divider)) + * _max2118_write_regs.get_n_divider(); - //debug output of calculated variables + // debug output of calculated variables UHD_LOGGER_TRACE("DBSRX") << boost::format("DBSRX tune:\n") - << boost::format(" VCO=%d, CP=%d, PFD Freq=%fMHz\n") % int(_max2118_write_regs.osc_band) % _max2118_write_regs.cp_current % (pfd_freq/1e6) - << boost::format(" R=%d, N=%f, scaler=%d, div2=%d\n") % R % N % scaler % int(_max2118_write_regs.div2) - << boost::format(" Ref Freq=%fMHz\n") % (ref_clock/1e6) - << boost::format(" Target Freq=%fMHz\n") % (target_freq/1e6) - << boost::format(" Actual Freq=%fMHz\n") % (_lo_freq/1e6) - << boost::format(" VCO Freq=%fMHz\n") % (vco_freq/1e6) - ; - - if (update_filter_settings) set_bandwidth(_bandwidth); + << boost::format(" VCO=%d, CP=%d, PFD Freq=%fMHz\n") + % int(_max2118_write_regs.osc_band) % _max2118_write_regs.cp_current + % (pfd_freq / 1e6) + << boost::format(" R=%d, N=%f, scaler=%d, div2=%d\n") % R % N % scaler + % int(_max2118_write_regs.div2) + << boost::format(" Ref Freq=%fMHz\n") % (ref_clock / 1e6) + << boost::format(" Target Freq=%fMHz\n") % (target_freq / 1e6) + << boost::format(" Actual Freq=%fMHz\n") % (_lo_freq / 1e6) + << boost::format(" VCO Freq=%fMHz\n") % (vco_freq / 1e6); + + if (update_filter_settings) + set_bandwidth(_bandwidth); get_locked(); return _lo_freq; @@ -441,22 +484,22 @@ double dbsrx::set_lo_freq(double target_freq){ * \param gain the requested gain in dB * \return 5 bit the register value */ -static int gain_to_gc2_vga_reg(double &gain){ +static int gain_to_gc2_vga_reg(double& gain) +{ int reg = 0; - gain = dbsrx_gain_ranges["GC2"].clip(gain); + gain = dbsrx_gain_ranges["GC2"].clip(gain); // Half dB steps from 0-5dB, 1dB steps from 5-24dB if (gain < 5) { - reg = boost::math::iround(31.0 - gain/0.5); + reg = boost::math::iround(31.0 - gain / 0.5); gain = double(boost::math::iround(gain) * 0.5); } else { - reg = boost::math::iround(22.0 - (gain - 4.0)); + reg = boost::math::iround(22.0 - (gain - 4.0)); gain = double(boost::math::iround(gain)); } - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX GC2 Gain: %f dB, reg: %d" - ) % gain % reg ; + UHD_LOGGER_TRACE("DBSRX") << boost::format("DBSRX GC2 Gain: %f dB, reg: %d") % gain + % reg; return reg; } @@ -467,38 +510,39 @@ static int gain_to_gc2_vga_reg(double &gain){ * \param gain the requested gain in dB * \return dac voltage value */ -static double gain_to_gc1_rfvga_dac(double &gain){ - //clip the input +static double gain_to_gc1_rfvga_dac(double& gain) +{ + // clip the input gain = dbsrx_gain_ranges["GC1"].clip(gain); - //voltage level constants + // voltage level constants static const double max_volts = 1.2, min_volts = 2.7; - static const double slope = (max_volts-min_volts)/dbsrx_gain_ranges["GC1"].stop(); + static const double slope = (max_volts - min_volts) / dbsrx_gain_ranges["GC1"].stop(); - //calculate the voltage for the aux dac - double dac_volts = gain*slope + min_volts; + // calculate the voltage for the aux dac + double dac_volts = gain * slope + min_volts; - UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX GC1 Gain: %f dB, dac_volts: %f V" - ) % gain % dac_volts ; + UHD_LOGGER_TRACE("DBSRX") << boost::format("DBSRX GC1 Gain: %f dB, dac_volts: %f V") + % gain % dac_volts; - //the actual gain setting - gain = (dac_volts - min_volts)/slope; + // the actual gain setting + gain = (dac_volts - min_volts) / slope; return dac_volts; } -double dbsrx::set_gain(double gain, const std::string &name){ +double dbsrx::set_gain(double gain, const std::string& name) +{ assert_has(dbsrx_gain_ranges.keys(), name, "dbsrx gain name"); - if (name == "GC2"){ + if (name == "GC2") { _max2118_write_regs.gc2 = gain_to_gc2_vga_reg(gain); send_reg(0x5, 0x5); - } - else if(name == "GC1"){ - //write the new voltage to the aux dac - this->get_iface()->write_aux_dac(dboard_iface::UNIT_RX, dboard_iface::AUX_DAC_A, gain_to_gc1_rfvga_dac(gain)); - } - else UHD_THROW_INVALID_CODE_PATH(); + } else if (name == "GC1") { + // write the new voltage to the aux dac + this->get_iface()->write_aux_dac( + dboard_iface::UNIT_RX, dboard_iface::AUX_DAC_A, gain_to_gc1_rfvga_dac(gain)); + } else + UHD_THROW_INVALID_CODE_PATH(); _gains[name] = gain; return gain; @@ -507,29 +551,36 @@ double dbsrx::set_gain(double gain, const std::string &name){ /*********************************************************************** * Bandwidth Handling **********************************************************************/ -double dbsrx::set_bandwidth(double bandwidth){ - //convert complex bandpass to lowpass bandwidth - bandwidth = bandwidth/2.0; +double dbsrx::set_bandwidth(double bandwidth) +{ + // convert complex bandpass to lowpass bandwidth + bandwidth = bandwidth / 2.0; - //clip the input + // clip the input bandwidth = dbsrx_bandwidth_range.clip(bandwidth); double ref_clock = this->get_iface()->get_clock_rate(dboard_iface::UNIT_RX); - //NOTE: _max2118_write_regs.m_divider set in set_lo_freq + // NOTE: _max2118_write_regs.m_divider set in set_lo_freq - //compute f_dac setting - _max2118_write_regs.f_dac = uhd::clip<int>(int((((bandwidth*_max2118_write_regs.m_divider)/ref_clock) - 4)/0.145),0,127); + // compute f_dac setting + _max2118_write_regs.f_dac = uhd::clip<int>( + int((((bandwidth * _max2118_write_regs.m_divider) / ref_clock) - 4) / 0.145), + 0, + 127); - //determine actual bandwidth - _bandwidth = double((ref_clock/(_max2118_write_regs.m_divider))*(4+0.145*_max2118_write_regs.f_dac)); + // determine actual bandwidth + _bandwidth = double((ref_clock / (_max2118_write_regs.m_divider)) + * (4 + 0.145 * _max2118_write_regs.f_dac)); UHD_LOGGER_TRACE("DBSRX") << boost::format( - "DBSRX Filter Bandwidth: %f MHz, m: %d, f_dac: %d\n" - ) % (_bandwidth/1e6) % int(_max2118_write_regs.m_divider) % int(_max2118_write_regs.f_dac) ; + "DBSRX Filter Bandwidth: %f MHz, m: %d, f_dac: %d\n") + % (_bandwidth / 1e6) + % int(_max2118_write_regs.m_divider) + % int(_max2118_write_regs.f_dac); this->send_reg(0x3, 0x4); - //convert lowpass back to complex bandpass bandwidth - return 2.0*_bandwidth; + // convert lowpass back to complex bandpass bandwidth + return 2.0 * _bandwidth; } |