diff options
Diffstat (limited to 'host/lib')
| -rw-r--r-- | host/lib/usrp/cores/rx_dsp_core_3000.cpp | 311 | ||||
| -rw-r--r-- | host/lib/usrp/cores/tx_dsp_core_3000.cpp | 187 | 
2 files changed, 268 insertions, 230 deletions
| diff --git a/host/lib/usrp/cores/rx_dsp_core_3000.cpp b/host/lib/usrp/cores/rx_dsp_core_3000.cpp index 325ff5569..191232541 100644 --- a/host/lib/usrp/cores/rx_dsp_core_3000.cpp +++ b/host/lib/usrp/cores/rx_dsp_core_3000.cpp @@ -5,101 +5,103 @@  // SPDX-License-Identifier: GPL-3.0-or-later  // -#include <uhd/types/dict.hpp>  #include <uhd/exception.hpp> -#include <uhd/utils/math.hpp> +#include <uhd/types/dict.hpp>  #include <uhd/utils/log.hpp> +#include <uhd/utils/math.hpp>  #include <uhd/utils/safe_call.hpp> -#include <uhdlib/usrp/cores/rx_dsp_core_3000.hpp>  #include <uhdlib/usrp/cores/dsp_core_utils.hpp> +#include <uhdlib/usrp/cores/rx_dsp_core_3000.hpp>  #include <boost/assign/list_of.hpp> -#include <boost/thread/thread.hpp> //thread sleep  #include <boost/math/special_functions/round.hpp> +#include <boost/thread/thread.hpp> //thread sleep  #include <algorithm>  #include <cmath> -#define REG_DSP_RX_FREQ       _dsp_base + 0 -#define REG_DSP_RX_SCALE_IQ   _dsp_base + 4 -#define REG_DSP_RX_DECIM      _dsp_base + 8 -#define REG_DSP_RX_MUX        _dsp_base + 12 -#define REG_DSP_RX_COEFFS     _dsp_base + 16 -//FIXME: Add code to support REG_DSP_RX_COEFFS +#define REG_DSP_RX_FREQ _dsp_base + 0 +#define REG_DSP_RX_SCALE_IQ _dsp_base + 4 +#define REG_DSP_RX_DECIM _dsp_base + 8 +#define REG_DSP_RX_MUX _dsp_base + 12 +#define REG_DSP_RX_COEFFS _dsp_base + 16 +// FIXME: Add code to support REG_DSP_RX_COEFFS -#define FLAG_DSP_RX_MUX_SWAP_IQ   (1 << 0) +#define FLAG_DSP_RX_MUX_SWAP_IQ (1 << 0)  #define FLAG_DSP_RX_MUX_REAL_MODE (1 << 1) -#define FLAG_DSP_RX_MUX_INVERT_Q  (1 << 2) -#define FLAG_DSP_RX_MUX_INVERT_I  (1 << 3) +#define FLAG_DSP_RX_MUX_INVERT_Q (1 << 2) +#define FLAG_DSP_RX_MUX_INVERT_I (1 << 3) -template <class T> T ceil_log2(T num){ -    return std::ceil(std::log(num)/std::log(T(2))); +template <class T> +T ceil_log2(T num) +{ +    return std::ceil(std::log(num) / std::log(T(2)));  }  using namespace uhd;  const double rx_dsp_core_3000::DEFAULT_CORDIC_FREQ = 0.0; -const double rx_dsp_core_3000::DEFAULT_RATE = 1e6; +const double rx_dsp_core_3000::DEFAULT_RATE        = 1e6; -rx_dsp_core_3000::~rx_dsp_core_3000(void){ +rx_dsp_core_3000::~rx_dsp_core_3000(void) +{      /* NOP */  } -class rx_dsp_core_3000_impl : public rx_dsp_core_3000{ +class rx_dsp_core_3000_impl : public rx_dsp_core_3000 +{  public: -    rx_dsp_core_3000_impl( -        wb_iface::sptr iface, -        const size_t dsp_base, -        const bool is_b200 -    ): -        _iface(iface), _dsp_base(dsp_base), _is_b200(is_b200) +    rx_dsp_core_3000_impl(wb_iface::sptr iface, const size_t dsp_base, const bool is_b200) +        : _iface(iface), _dsp_base(dsp_base), _is_b200(is_b200)      {          // previously uninitialized - assuming zero for all          _link_rate = _host_extra_scaling = _fxpt_scalar_correction = 0.0; -        //init to something so update method has reasonable defaults +        // init to something so update method has reasonable defaults          _scaling_adjustment = 1.0; -        _dsp_extra_scaling = 1.0; -        _tick_rate = 1.0; -        _dsp_freq_offset = 0.0; +        _dsp_extra_scaling  = 1.0; +        _tick_rate          = 1.0; +        _dsp_freq_offset    = 0.0;      }      ~rx_dsp_core_3000_impl(void)      { -        UHD_SAFE_CALL -        ( -            ;//NOP +        UHD_SAFE_CALL(; // NOP          )      } -    void set_mux(const uhd::usrp::fe_connection_t& fe_conn){ +    void set_mux(const uhd::usrp::fe_connection_t& fe_conn) +    {          uint32_t reg_val = 0;          switch (fe_conn.get_sampling_mode()) { -        case uhd::usrp::fe_connection_t::REAL: -        case uhd::usrp::fe_connection_t::HETERODYNE: -            reg_val = FLAG_DSP_RX_MUX_REAL_MODE; -            break; -        default: -            reg_val = 0; -            break; +            case uhd::usrp::fe_connection_t::REAL: +            case uhd::usrp::fe_connection_t::HETERODYNE: +                reg_val = FLAG_DSP_RX_MUX_REAL_MODE; +                break; +            default: +                reg_val = 0; +                break;          } -        if (fe_conn.is_iq_swapped()) reg_val |= FLAG_DSP_RX_MUX_SWAP_IQ; -        if (fe_conn.is_i_inverted()) reg_val |= FLAG_DSP_RX_MUX_INVERT_I; -        if (fe_conn.is_q_inverted()) reg_val |= FLAG_DSP_RX_MUX_INVERT_Q; +        if (fe_conn.is_iq_swapped()) +            reg_val |= FLAG_DSP_RX_MUX_SWAP_IQ; +        if (fe_conn.is_i_inverted()) +            reg_val |= FLAG_DSP_RX_MUX_INVERT_I; +        if (fe_conn.is_q_inverted()) +            reg_val |= FLAG_DSP_RX_MUX_INVERT_Q;          _iface->poke32(REG_DSP_RX_MUX, reg_val);          if (fe_conn.get_sampling_mode() == uhd::usrp::fe_connection_t::HETERODYNE) { -            //1. Remember the sign of the IF frequency. +            // 1. Remember the sign of the IF frequency.              //   It will be discarded in the next step              int if_freq_sign = boost::math::sign(fe_conn.get_if_freq()); -            //2. Map IF frequency to the range [0, _tick_rate) +            // 2. Map IF frequency to the range [0, _tick_rate)              double if_freq = std::abs(std::fmod(fe_conn.get_if_freq(), _tick_rate)); -            //3. Map IF frequency to the range [-_tick_rate/2, _tick_rate/2) +            // 3. Map IF frequency to the range [-_tick_rate/2, _tick_rate/2)              //   This is the aliased frequency              if (if_freq > (_tick_rate / 2.0)) {                  if_freq -= _tick_rate;              } -            //4. Set DSP offset to spin the signal in the opposite +            // 4. Set DSP offset to spin the signal in the opposite              //   direction as the aliased frequency              _dsp_freq_offset = if_freq * (-if_freq_sign);          } else { @@ -107,166 +109,193 @@ public:          }      } -    void set_tick_rate(const double rate){ +    void set_tick_rate(const double rate) +    {          _tick_rate = rate;          set_freq(_current_freq);      } -    void set_link_rate(const double rate){ +    void set_link_rate(const double rate) +    {          //_link_rate = rate/sizeof(uint32_t); //in samps/s -        _link_rate = rate/sizeof(uint16_t); //in samps/s (allows for 8sc) +        _link_rate = rate / sizeof(uint16_t); // in samps/s (allows for 8sc)      } -    uhd::meta_range_t get_host_rates(void){ +    uhd::meta_range_t get_host_rates(void) +    {          meta_range_t range;          if (!_is_b200) { -            for (int rate = 1024; rate > 512; rate -= 8){ -                range.push_back(range_t(_tick_rate/rate)); +            for (int rate = 1024; rate > 512; rate -= 8) { +                range.push_back(range_t(_tick_rate / rate));              }          } -        for (int rate = 512; rate > 256; rate -= 4){ -            range.push_back(range_t(_tick_rate/rate)); +        for (int rate = 512; rate > 256; rate -= 4) { +            range.push_back(range_t(_tick_rate / rate));          } -        for (int rate = 256; rate > 128; rate -= 2){ -            range.push_back(range_t(_tick_rate/rate)); +        for (int rate = 256; rate > 128; rate -= 2) { +            range.push_back(range_t(_tick_rate / rate));          } -        for (int rate = 128; rate >= int(std::ceil(_tick_rate/_link_rate)); rate -= 1){ -            range.push_back(range_t(_tick_rate/rate)); +        for (int rate = 128; rate >= int(std::ceil(_tick_rate / _link_rate)); rate -= 1) { +            range.push_back(range_t(_tick_rate / rate));          }          return range;      } -    double set_host_rate(const double rate){ -        const size_t decim_rate = boost::math::iround(_tick_rate/this->get_host_rates().clip(rate, true)); +    double set_host_rate(const double rate) +    { +        const size_t decim_rate = +            boost::math::iround(_tick_rate / this->get_host_rates().clip(rate, true));          size_t decim = decim_rate; -        //determine which half-band filters are activated -        int hb0 = 0, hb1 = 0, hb2 = 0, hb_enable=0; -        if (decim % 2 == 0){ +        // determine which half-band filters are activated +        int hb0 = 0, hb1 = 0, hb2 = 0, hb_enable = 0; +        if (decim % 2 == 0) {              hb0 = 1;              decim /= 2;          } -        if (decim % 2 == 0){ +        if (decim % 2 == 0) {              hb1 = 1;              decim /= 2;          } -        //the third half-band is not supported by the B200 -        if (decim % 2 == 0 && !_is_b200){ +        // the third half-band is not supported by the B200 +        if (decim % 2 == 0 && !_is_b200) {              hb2 = 1;              decim /= 2;          }          if (_is_b200) { -            _iface->poke32(REG_DSP_RX_DECIM, (hb0 << 9) /*small HB */ | (hb1 << 8) /*large HB*/ | (decim & 0xff)); +            _iface->poke32(REG_DSP_RX_DECIM, +                (hb0 << 9) /*small HB */ | (hb1 << 8) /*large HB*/ | (decim & 0xff));              if (decim > 1 and hb0 == 0 and hb1 == 0) { -                UHD_LOGGER_WARNING("CORES") << boost::format( -                    "The requested decimation is odd; the user should expect CIC rolloff.\n" -                    "Select an even decimation to ensure that a halfband filter is enabled.\n" -                    "decimation = dsp_rate/samp_rate -> %d = (%f MHz)/(%f MHz)\n" -                ) % decim_rate % (_tick_rate/1e6) % (rate/1e6); +                UHD_LOGGER_WARNING("CORES") +                    << boost::format( +                           "The requested decimation is odd; the user should expect CIC " +                           "rolloff.\n" +                           "Select an even decimation to ensure that a halfband filter " +                           "is enabled.\n" +                           "decimation = dsp_rate/samp_rate -> %d = (%f MHz)/(%f MHz)\n") +                           % decim_rate % (_tick_rate / 1e6) % (rate / 1e6);              }          } else {              // Encode Halfband config for setting register programming.              if (hb2) { // Implies HB1 and HB0 also asserted -                hb_enable=3; +                hb_enable = 3;              } else if (hb1) { // Implies HB0 is also asserted -                hb_enable=2; +                hb_enable = 2;              } else if (hb0) { -                hb_enable=1; +                hb_enable = 1;              } else { -                hb_enable=0; +                hb_enable = 0;              } -            _iface->poke32(REG_DSP_RX_DECIM,  (hb_enable << 8) | (decim & 0xff)); +            _iface->poke32(REG_DSP_RX_DECIM, (hb_enable << 8) | (decim & 0xff));              if (decim > 1 and hb0 == 0 and hb1 == 0 and hb2 == 0) { -                UHD_LOGGER_WARNING("CORES") << boost::format( -                    "The requested decimation is odd; the user should expect passband CIC rolloff.\n" -                    "Select an even decimation to ensure that a halfband filter is enabled.\n" -                    "Decimations factorable by 4 will enable 2 halfbands, those factorable by 8 will enable 3 halfbands.\n" -                    "decimation = dsp_rate/samp_rate -> %d = (%f MHz)/(%f MHz)\n" -                ) % decim_rate % (_tick_rate/1e6) % (rate/1e6); +                UHD_LOGGER_WARNING("CORES") +                    << boost::format( +                           "The requested decimation is odd; the user should expect " +                           "passband CIC rolloff.\n" +                           "Select an even decimation to ensure that a halfband filter " +                           "is enabled.\n" +                           "Decimations factorable by 4 will enable 2 halfbands, those " +                           "factorable by 8 will enable 3 halfbands.\n" +                           "decimation = dsp_rate/samp_rate -> %d = (%f MHz)/(%f MHz)\n") +                           % decim_rate % (_tick_rate / 1e6) % (rate / 1e6);              }          }          // Caclulate algorithmic gain of CIC for a given decimation.          // For Ettus CIC R=decim, M=1, N=4. Gain = (R * M) ^ N          const double rate_pow = std::pow(double(decim & 0xff), 4); -        // Calculate compensation gain values for algorithmic gain of CORDIC and CIC taking into account -        // gain compensation blocks already hardcoded in place in DDC (that provide simple 1/2^n gain compensation). -        // CORDIC algorithmic gain limits asymptotically around 1.647 after many iterations. +        // Calculate compensation gain values for algorithmic gain of CORDIC and CIC +        // taking into account gain compensation blocks already hardcoded in place in DDC +        // (that provide simple 1/2^n gain compensation). CORDIC algorithmic gain limits +        // asymptotically around 1.647 after many iterations.          // -        // The polar rotation of [I,Q] = [1,1] by Pi/8 also yields max magnitude of SQRT(2) (~1.4142) however -        // input to the CORDIC thats outside the unit circle can only be sourced from a saturated RF frontend. -        // To provide additional dynamic range head room accordingly using scale factor applied at egress from DDC would -        // cost us small signal performance, thus we do no provide compensation gain for a saturated front end and allow -        // the signal to clip in the H/W as needed. If we wished to avoid the signal clipping in these circumstances then adjust code to read: -        // _scaling_adjustment = std::pow(2, ceil_log2(rate_pow))/(1.648*rate_pow*1.415); -        _scaling_adjustment = std::pow(2, ceil_log2(rate_pow))/(1.648*rate_pow); +        // The polar rotation of [I,Q] = [1,1] by Pi/8 also yields max magnitude of +        // SQRT(2) (~1.4142) however input to the CORDIC thats outside the unit circle can +        // only be sourced from a saturated RF frontend. To provide additional dynamic +        // range head room accordingly using scale factor applied at egress from DDC would +        // cost us small signal performance, thus we do no provide compensation gain for a +        // saturated front end and allow the signal to clip in the H/W as needed. If we +        // wished to avoid the signal clipping in these circumstances then adjust code to +        // read: _scaling_adjustment = std::pow(2, +        // ceil_log2(rate_pow))/(1.648*rate_pow*1.415); +        _scaling_adjustment = std::pow(2, ceil_log2(rate_pow)) / (1.648 * rate_pow);          this->update_scalar(); -        return _tick_rate/decim_rate; +        return _tick_rate / decim_rate;      } -    // Calculate compensation gain values for algorithmic gain of CORDIC and CIC taking into account -    // gain compensation blocks already hardcoded in place in DDC (that provide simple 1/2^n gain compensation). -    // Further more factor in OTW format which adds further gain factor to weight output samples correctly. -    void update_scalar(void){ -        const double target_scalar = (1 << (_is_b200 ? 16 : 15))*_scaling_adjustment/_dsp_extra_scaling; +    // Calculate compensation gain values for algorithmic gain of CORDIC and CIC taking +    // into account gain compensation blocks already hardcoded in place in DDC (that +    // provide simple 1/2^n gain compensation). Further more factor in OTW format which +    // adds further gain factor to weight output samples correctly. +    void update_scalar(void) +    { +        const double target_scalar = +            (1 << (_is_b200 ? 16 : 15)) * _scaling_adjustment / _dsp_extra_scaling;          const int32_t actual_scalar = boost::math::iround(target_scalar); -        // Calculate the error introduced by using integer representation for the scalar, can be corrected in host later. -        _fxpt_scalar_correction = target_scalar/actual_scalar; -        // Write DDC with scaling correction for CIC and CORDIC that maximizes dynamic range in 32/16/12/8bits. +        // Calculate the error introduced by using integer representation for the scalar, +        // can be corrected in host later. +        _fxpt_scalar_correction = target_scalar / actual_scalar; +        // Write DDC with scaling correction for CIC and CORDIC that maximizes dynamic +        // range in 32/16/12/8bits.          _iface->poke32(REG_DSP_RX_SCALE_IQ, actual_scalar);      } -    double get_scaling_adjustment(void){ -        return _fxpt_scalar_correction*_host_extra_scaling/32767.; +    double get_scaling_adjustment(void) +    { +        return _fxpt_scalar_correction * _host_extra_scaling / 32767.;      } -    double set_freq(const double requested_freq){ +    double set_freq(const double requested_freq) +    {          double actual_freq;          int32_t freq_word; -        get_freq_and_freq_word(requested_freq + _dsp_freq_offset, _tick_rate, actual_freq, freq_word); +        get_freq_and_freq_word( +            requested_freq + _dsp_freq_offset, _tick_rate, actual_freq, freq_word);          _iface->poke32(REG_DSP_RX_FREQ, uint32_t(freq_word));          _current_freq = actual_freq;          return actual_freq;      } -    double get_freq(void){ +    double get_freq(void) +    {          return _current_freq;      } -    uhd::meta_range_t get_freq_range(void){ -        //Too keep the DSP range symmetric about 0, we use abs(_dsp_freq_offset) +    uhd::meta_range_t get_freq_range(void) +    { +        // Too keep the DSP range symmetric about 0, we use abs(_dsp_freq_offset)          const double offset = std::abs<double>(_dsp_freq_offset); -        return uhd::meta_range_t(-(_tick_rate-offset)/2, +(_tick_rate-offset)/2, _tick_rate/std::pow(2.0, 32)); +        return uhd::meta_range_t(-(_tick_rate - offset) / 2, +            +(_tick_rate - offset) / 2, +            _tick_rate / std::pow(2.0, 32));      } -    void setup(const uhd::stream_args_t &stream_args){ - -        if (stream_args.otw_format == "sc16"){ -            _dsp_extra_scaling = 1.0; +    void setup(const uhd::stream_args_t& stream_args) +    { +        if (stream_args.otw_format == "sc16") { +            _dsp_extra_scaling  = 1.0;              _host_extra_scaling = 1.0; -        } -        else if (stream_args.otw_format == "sc8"){ -            double peak = stream_args.args.cast<double>("peak", 1.0); -            peak = std::max(peak, 1.0/256); -            _host_extra_scaling = peak*256; -            _dsp_extra_scaling = peak; -        } -        else if (stream_args.otw_format == "sc12"){ -            double peak = stream_args.args.cast<double>("peak", 1.0); -            peak = std::max(peak, 1.0/16); -            _host_extra_scaling = peak*16; -            _dsp_extra_scaling = peak; -        } -        else if (stream_args.otw_format == "fc32"){ +        } else if (stream_args.otw_format == "sc8") { +            double peak         = stream_args.args.cast<double>("peak", 1.0); +            peak                = std::max(peak, 1.0 / 256); +            _host_extra_scaling = peak * 256; +            _dsp_extra_scaling  = peak; +        } else if (stream_args.otw_format == "sc12") { +            double peak         = stream_args.args.cast<double>("peak", 1.0); +            peak                = std::max(peak, 1.0 / 16); +            _host_extra_scaling = peak * 16; +            _dsp_extra_scaling  = peak; +        } else if (stream_args.otw_format == "fc32") {              _host_extra_scaling = 1.0; -            _dsp_extra_scaling = 1.0; -        } -        else throw uhd::value_error("USRP RX cannot handle requested wire format: " + stream_args.otw_format); +            _dsp_extra_scaling  = 1.0; +        } else +            throw uhd::value_error( +                "USRP RX cannot handle requested wire format: " + stream_args.otw_format);          _host_extra_scaling *= stream_args.args.cast<double>("fullscale", 1.0); @@ -276,33 +305,31 @@ public:      void populate_subtree(property_tree::sptr subtree)      {          subtree->create<meta_range_t>("rate/range") -            .set_publisher(boost::bind(&rx_dsp_core_3000::get_host_rates, this)) -        ; +            .set_publisher(boost::bind(&rx_dsp_core_3000::get_host_rates, this));          subtree->create<double>("rate/value")              .set(DEFAULT_RATE) -            .set_coercer(boost::bind(&rx_dsp_core_3000::set_host_rate, this, _1)) -        ; +            .set_coercer(boost::bind(&rx_dsp_core_3000::set_host_rate, this, _1));          subtree->create<double>("freq/value")              .set(DEFAULT_CORDIC_FREQ)              .set_coercer(boost::bind(&rx_dsp_core_3000::set_freq, this, _1)) -            .set_publisher([this](){ return this->get_freq(); }) -        ; +            .set_publisher([this]() { return this->get_freq(); });          subtree->create<meta_range_t>("freq/range") -            .set_publisher(boost::bind(&rx_dsp_core_3000::get_freq_range, this)) -        ; +            .set_publisher(boost::bind(&rx_dsp_core_3000::get_freq_range, this));      }  private:      wb_iface::sptr _iface;      const size_t _dsp_base; -    const bool _is_b200;    //TODO: Obsolete this when we switch to the new DDC on the B200 +    const bool _is_b200; // TODO: Obsolete this when we switch to the new DDC on the B200      double _tick_rate, _link_rate; -    double _scaling_adjustment, _dsp_extra_scaling, _host_extra_scaling, _fxpt_scalar_correction; +    double _scaling_adjustment, _dsp_extra_scaling, _host_extra_scaling, +        _fxpt_scalar_correction;      double _dsp_freq_offset;      double _current_freq;  }; -rx_dsp_core_3000::sptr rx_dsp_core_3000::make(wb_iface::sptr iface, const size_t dsp_base, const bool is_b200 /* = false */) +rx_dsp_core_3000::sptr rx_dsp_core_3000::make( +    wb_iface::sptr iface, const size_t dsp_base, const bool is_b200 /* = false */)  {      return sptr(new rx_dsp_core_3000_impl(iface, dsp_base, is_b200));  } diff --git a/host/lib/usrp/cores/tx_dsp_core_3000.cpp b/host/lib/usrp/cores/tx_dsp_core_3000.cpp index af0587c7f..029ca5c3d 100644 --- a/host/lib/usrp/cores/tx_dsp_core_3000.cpp +++ b/host/lib/usrp/cores/tx_dsp_core_3000.cpp @@ -5,129 +5,142 @@  // SPDX-License-Identifier: GPL-3.0-or-later  // -#include <uhdlib/usrp/cores/tx_dsp_core_3000.hpp> -#include <uhdlib/usrp/cores/dsp_core_utils.hpp> -#include <uhd/types/dict.hpp>  #include <uhd/exception.hpp> -#include <uhd/utils/math.hpp> +#include <uhd/types/dict.hpp>  #include <uhd/utils/log.hpp> +#include <uhd/utils/math.hpp> +#include <uhdlib/usrp/cores/dsp_core_utils.hpp> +#include <uhdlib/usrp/cores/tx_dsp_core_3000.hpp>  #include <boost/assign/list_of.hpp>  #include <boost/math/special_functions/round.hpp>  #include <boost/thread/thread.hpp> //sleep  #include <algorithm>  #include <cmath> -#define REG_DSP_TX_FREQ          _dsp_base + 0 -#define REG_DSP_TX_SCALE_IQ      _dsp_base + 4 -#define REG_DSP_TX_INTERP        _dsp_base + 8 +#define REG_DSP_TX_FREQ _dsp_base + 0 +#define REG_DSP_TX_SCALE_IQ _dsp_base + 4 +#define REG_DSP_TX_INTERP _dsp_base + 8 -template <class T> T ceil_log2(T num){ -    return std::ceil(std::log(num)/std::log(T(2))); +template <class T> +T ceil_log2(T num) +{ +    return std::ceil(std::log(num) / std::log(T(2)));  }  using namespace uhd;  const double tx_dsp_core_3000::DEFAULT_CORDIC_FREQ = 0.0; -const double tx_dsp_core_3000::DEFAULT_RATE = 1e6; +const double tx_dsp_core_3000::DEFAULT_RATE        = 1e6; -tx_dsp_core_3000::~tx_dsp_core_3000(void){ +tx_dsp_core_3000::~tx_dsp_core_3000(void) +{      /* NOP */  } -class tx_dsp_core_3000_impl : public tx_dsp_core_3000{ +class tx_dsp_core_3000_impl : public tx_dsp_core_3000 +{  public: -    tx_dsp_core_3000_impl( -        wb_iface::sptr iface, -        const size_t dsp_base -    ): -        _iface(iface), _dsp_base(dsp_base) +    tx_dsp_core_3000_impl(wb_iface::sptr iface, const size_t dsp_base) +        : _iface(iface), _dsp_base(dsp_base)      {          // previously uninitialized - assuming zero for all          _link_rate = _host_extra_scaling = _fxpt_scalar_correction = 0.0; -        //init to something so update method has reasonable defaults +        // init to something so update method has reasonable defaults          _scaling_adjustment = 1.0; -        _dsp_extra_scaling = 1.0; +        _dsp_extra_scaling  = 1.0;          this->set_tick_rate(1.0);      } -    void set_tick_rate(const double rate){ +    void set_tick_rate(const double rate) +    {          _tick_rate = rate;          set_freq(_current_freq);      } -    void set_link_rate(const double rate){ +    void set_link_rate(const double rate) +    {          //_link_rate = rate/sizeof(uint32_t); //in samps/s -        _link_rate = rate/sizeof(uint16_t); //in samps/s (allows for 8sc) +        _link_rate = rate / sizeof(uint16_t); // in samps/s (allows for 8sc)      } -    uhd::meta_range_t get_host_rates(void){ +    uhd::meta_range_t get_host_rates(void) +    {          meta_range_t range; -        for (int rate = 512; rate > 256; rate -= 4){ -            range.push_back(range_t(_tick_rate/rate)); +        for (int rate = 512; rate > 256; rate -= 4) { +            range.push_back(range_t(_tick_rate / rate));          } -        for (int rate = 256; rate > 128; rate -= 2){ -            range.push_back(range_t(_tick_rate/rate)); +        for (int rate = 256; rate > 128; rate -= 2) { +            range.push_back(range_t(_tick_rate / rate));          } -        for (int rate = 128; rate >= int(std::ceil(_tick_rate/_link_rate)); rate -= 1){ -            range.push_back(range_t(_tick_rate/rate)); +        for (int rate = 128; rate >= int(std::ceil(_tick_rate / _link_rate)); rate -= 1) { +            range.push_back(range_t(_tick_rate / rate));          }          return range;      } -    double set_host_rate(const double rate){ -        const size_t interp_rate = boost::math::iround(_tick_rate/this->get_host_rates().clip(rate, true)); +    double set_host_rate(const double rate) +    { +        const size_t interp_rate = +            boost::math::iround(_tick_rate / this->get_host_rates().clip(rate, true));          size_t interp = interp_rate; -        //determine which half-band filters are activated +        // determine which half-band filters are activated          int hb0 = 0, hb1 = 0; -        if (interp % 2 == 0){ +        if (interp % 2 == 0) {              hb0 = 1;              interp /= 2;          } -        if (interp % 2 == 0){ +        if (interp % 2 == 0) {              hb1 = 1;              interp /= 2;          }          _iface->poke32(REG_DSP_TX_INTERP, (hb1 << 9) | (hb0 << 8) | (interp & 0xff)); -        if (interp > 1 and hb0 == 0 and hb1 == 0) -        { -            UHD_LOGGER_WARNING("CORES") << boost::format( -                "The requested interpolation is odd; the user should expect CIC rolloff.\n" -                "Select an even interpolation to ensure that a halfband filter is enabled.\n" -                "interpolation = dsp_rate/samp_rate -> %d = (%f MHz)/(%f MHz)\n" -            ) % interp_rate % (_tick_rate/1e6) % (rate/1e6); +        if (interp > 1 and hb0 == 0 and hb1 == 0) { +            UHD_LOGGER_WARNING("CORES") +                << boost::format( +                       "The requested interpolation is odd; the user should expect CIC " +                       "rolloff.\n" +                       "Select an even interpolation to ensure that a halfband filter is " +                       "enabled.\n" +                       "interpolation = dsp_rate/samp_rate -> %d = (%f MHz)/(%f MHz)\n") +                       % interp_rate % (_tick_rate / 1e6) % (rate / 1e6);          }          // Caclulate algorithmic gain of CIC for a given interpolation          // For Ettus CIC R=decim, M=1, N=3. Gain = (R * M) ^ N          const double rate_pow = std::pow(double(interp & 0xff), 3); -        // Calculate compensation gain values for algorithmic gain of CORDIC and CIC taking into account -        // gain compensation blocks already hardcoded in place in DDC (that provide simple 1/2^n gain compensation). -        // CORDIC algorithmic gain limits asymptotically around 1.647 after many iterations. -        _scaling_adjustment = std::pow(2, ceil_log2(rate_pow))/(1.648*rate_pow); +        // Calculate compensation gain values for algorithmic gain of CORDIC and CIC +        // taking into account gain compensation blocks already hardcoded in place in DDC +        // (that provide simple 1/2^n gain compensation). CORDIC algorithmic gain limits +        // asymptotically around 1.647 after many iterations. +        _scaling_adjustment = std::pow(2, ceil_log2(rate_pow)) / (1.648 * rate_pow);          this->update_scalar(); -        return _tick_rate/interp_rate; +        return _tick_rate / interp_rate;      } -  // Calculate compensation gain values for algorithmic gain of CORDIC and CIC taking into account -  // gain compensation blocks already hardcoded in place in DDC (that provide simple 1/2^n gain compensation). -  // Further more factor in OTW format which adds further gain factor to weight output samples correctly. -    void update_scalar(void){ -        const double target_scalar = (1 << 16)*_scaling_adjustment/_dsp_extra_scaling; +    // Calculate compensation gain values for algorithmic gain of CORDIC and CIC taking +    // into account gain compensation blocks already hardcoded in place in DDC (that +    // provide simple 1/2^n gain compensation). Further more factor in OTW format which +    // adds further gain factor to weight output samples correctly. +    void update_scalar(void) +    { +        const double target_scalar = (1 << 16) * _scaling_adjustment / _dsp_extra_scaling;          const int32_t actual_scalar = boost::math::iround(target_scalar); -        _fxpt_scalar_correction = target_scalar/actual_scalar; //should be small +        _fxpt_scalar_correction     = target_scalar / actual_scalar; // should be small          _iface->poke32(REG_DSP_TX_SCALE_IQ, actual_scalar);      } -    double get_scaling_adjustment(void){ -        return _fxpt_scalar_correction*_host_extra_scaling*32767.; +    double get_scaling_adjustment(void) +    { +        return _fxpt_scalar_correction * _host_extra_scaling * 32767.;      } -    double set_freq(const double requested_freq) { +    double set_freq(const double requested_freq) +    {          double actual_freq;          int32_t freq_word;          get_freq_and_freq_word(requested_freq, _tick_rate, actual_freq, freq_word); @@ -136,37 +149,38 @@ public:          return actual_freq;      } -    double get_freq(void){ +    double get_freq(void) +    {          return _current_freq;      } -    uhd::meta_range_t get_freq_range(void){ -        return uhd::meta_range_t(-_tick_rate/2, +_tick_rate/2, _tick_rate/std::pow(2.0, 32)); +    uhd::meta_range_t get_freq_range(void) +    { +        return uhd::meta_range_t( +            -_tick_rate / 2, +_tick_rate / 2, _tick_rate / std::pow(2.0, 32));      } -    void setup(const uhd::stream_args_t &stream_args){ - -        if (stream_args.otw_format == "sc16"){ -            _dsp_extra_scaling = 1.0; +    void setup(const uhd::stream_args_t& stream_args) +    { +        if (stream_args.otw_format == "sc16") { +            _dsp_extra_scaling  = 1.0;              _host_extra_scaling = 1.0; -        } -        else if (stream_args.otw_format == "sc8"){ -            double peak = stream_args.args.cast<double>("peak", 1.0); -            peak = std::max(peak, 1.0/256); -            _host_extra_scaling = 1.0/peak/256; -            _dsp_extra_scaling = 1.0/peak; -        } -        else if (stream_args.otw_format == "sc12"){ -            double peak = stream_args.args.cast<double>("peak", 1.0); -            peak = std::max(peak, 1.0/16); -            _host_extra_scaling = 1.0/peak/16; -            _dsp_extra_scaling = 1.0/peak; -        } -        else if (stream_args.otw_format == "fc32"){ +        } else if (stream_args.otw_format == "sc8") { +            double peak         = stream_args.args.cast<double>("peak", 1.0); +            peak                = std::max(peak, 1.0 / 256); +            _host_extra_scaling = 1.0 / peak / 256; +            _dsp_extra_scaling  = 1.0 / peak; +        } else if (stream_args.otw_format == "sc12") { +            double peak         = stream_args.args.cast<double>("peak", 1.0); +            peak                = std::max(peak, 1.0 / 16); +            _host_extra_scaling = 1.0 / peak / 16; +            _dsp_extra_scaling  = 1.0 / peak; +        } else if (stream_args.otw_format == "fc32") {              _host_extra_scaling = 1.0; -            _dsp_extra_scaling = 1.0; -        } -        else throw uhd::value_error("USRP TX cannot handle requested wire format: " + stream_args.otw_format); +            _dsp_extra_scaling  = 1.0; +        } else +            throw uhd::value_error( +                "USRP TX cannot handle requested wire format: " + stream_args.otw_format);          _host_extra_scaling /= stream_args.args.cast<double>("fullscale", 1.0); @@ -176,27 +190,24 @@ public:      void populate_subtree(property_tree::sptr subtree)      {          subtree->create<meta_range_t>("rate/range") -            .set_publisher(boost::bind(&tx_dsp_core_3000::get_host_rates, this)) -        ; +            .set_publisher(boost::bind(&tx_dsp_core_3000::get_host_rates, this));          subtree->create<double>("rate/value")              .set(DEFAULT_RATE) -            .set_coercer(boost::bind(&tx_dsp_core_3000::set_host_rate, this, _1)) -        ; +            .set_coercer(boost::bind(&tx_dsp_core_3000::set_host_rate, this, _1));          subtree->create<double>("freq/value")              .set(DEFAULT_CORDIC_FREQ)              .set_coercer(boost::bind(&tx_dsp_core_3000::set_freq, this, _1)) -            .set_publisher([this](){ return this->get_freq(); }) -        ; +            .set_publisher([this]() { return this->get_freq(); });          subtree->create<meta_range_t>("freq/range") -            .set_publisher(boost::bind(&tx_dsp_core_3000::get_freq_range, this)) -        ; +            .set_publisher(boost::bind(&tx_dsp_core_3000::get_freq_range, this));      }  private:      wb_iface::sptr _iface;      const size_t _dsp_base;      double _tick_rate, _link_rate; -    double _scaling_adjustment, _dsp_extra_scaling, _host_extra_scaling, _fxpt_scalar_correction; +    double _scaling_adjustment, _dsp_extra_scaling, _host_extra_scaling, +        _fxpt_scalar_correction;      double _current_freq;  }; | 
