From d6d94bf60ed75ee237e77d1a46c8ab5c752f1121 Mon Sep 17 00:00:00 2001 From: Ryan Marlow Date: Wed, 6 Dec 2017 09:10:45 -0500 Subject: ddc/duc: enhance ddc/duc with new parameters - NUM_HALFBANDS and CIC_MAX are no longer hard coded and are now readback regs. - both DDC and DUC use same encoding to enable halfbands (increment the number) - removed hacky hack & fixed get_output_rate/get_input_rate to include only valid rates based on NUM_HALFBANDS and CIC_MAX. - added compatibility number readback w/ warning/error messages - Updated images package to include new DDC/DUC (affects X-Series only) - Updated fpga-src submodule pointer --- host/lib/rfnoc/ddc_block_ctrl_impl.cpp | 75 +++++++++++++++++++++++--------- host/lib/rfnoc/duc_block_ctrl_impl.cpp | 78 +++++++++++++++++++++++++--------- 2 files changed, 113 insertions(+), 40 deletions(-) (limited to 'host/lib') diff --git a/host/lib/rfnoc/ddc_block_ctrl_impl.cpp b/host/lib/rfnoc/ddc_block_ctrl_impl.cpp index b0b510cde..ad7312fca 100644 --- a/host/lib/rfnoc/ddc_block_ctrl_impl.cpp +++ b/host/lib/rfnoc/ddc_block_ctrl_impl.cpp @@ -44,11 +44,12 @@ static double lambda_forward_prop(uhd::property_tree::sptr tree, uhd::fs_path pr class ddc_block_ctrl_impl : public ddc_block_ctrl { public: - static const size_t NUM_HALFBANDS = 3; - static const size_t CIC_MAX_DECIM = 255; UHD_RFNOC_BLOCK_CONSTRUCTOR(ddc_block_ctrl) { + check_compat_num(); + _num_halfbands = (size_t) user_reg_read64(RB_REG_NUM_HALFBANDS); + _cic_max_decim = (size_t) user_reg_read64(RB_REG_CIC_MAX_DECIM); // Argument/prop tree hooks for (size_t chan = 0; chan < get_input_ports().size(); chan++) { double default_freq = get_arg("freq", chan); @@ -176,6 +177,14 @@ public: private: + size_t _num_halfbands; + size_t _cic_max_decim; + static const size_t MAJOR_COMP = 1; + static const size_t MINOR_COMP = 0; + static const size_t RB_REG_COMPAT_NUM = 0; + static const size_t RB_REG_NUM_HALFBANDS = 1; + static const size_t RB_REG_CIC_MAX_DECIM = 2; + //! Set the CORDIC frequency shift the signal to \p requested_freq double set_freq(const double requested_freq, const size_t chan) { @@ -198,23 +207,18 @@ private: ); } - // FIXME this misses a whole bunch of valid rates. Anything with CIC decim <= 255 - // is OK. uhd::meta_range_t get_output_rates(void) { uhd::meta_range_t range; const double input_rate = get_arg("input_rate"); - for (int decim = 1024; decim > 512; decim -= 8){ - range.push_back(uhd::range_t(input_rate/decim)); - } - for (int decim = 512; decim > 256; decim -= 4){ - range.push_back(uhd::range_t(input_rate/decim)); - } - for (int decim = 256; decim > 128; decim -= 2){ - range.push_back(uhd::range_t(input_rate/decim)); - } - for (int decim = 128; decim >= 1; decim -= 1){ - range.push_back(uhd::range_t(input_rate/decim)); + for (int hb = _num_halfbands; hb >= 0; hb--) { + const size_t decim_offset = _cic_max_decim<<(hb-1); + for(size_t decim = _cic_max_decim; decim > 0; decim--) { + const size_t hb_cic_decim = decim*(1< decim_offset) { + range.push_back(uhd::range_t(input_rate/hb_cic_decim)); + } + } } return range; } @@ -222,17 +226,17 @@ private: double set_output_rate(const int requested_rate, const size_t chan) { const double input_rate = get_arg("input_rate"); + const size_t decim_rate = boost::math::iround(input_rate/this->get_output_rates().clip(requested_rate, true)); size_t decim = decim_rate; - // The FPGA knows which halfbands to enable for any given value of hb_enable. uint32_t hb_enable = 0; - while ((decim % 2 == 0) and hb_enable < NUM_HALFBANDS) { + while ((decim % 2 == 0) and hb_enable < _num_halfbands) { hb_enable++; decim /= 2; } - UHD_ASSERT_THROW(hb_enable <= NUM_HALFBANDS); - UHD_ASSERT_THROW(decim <= CIC_MAX_DECIM); + UHD_ASSERT_THROW(hb_enable <= _num_halfbands); + UHD_ASSERT_THROW(decim > 0 and decim <= _cic_max_decim); // What we can't cover with halfbands, we do with the CIC sr_write("DECIM_WORD", (hb_enable << 8) | (decim & 0xff), chan); @@ -293,6 +297,39 @@ private: sr_write("SCALE_IQ", actual_scalar, chan); } + //Get cached value of _num_halfbands + size_t get_num_halfbands() const + { + return (size_t) _num_halfbands; + } + + //Get cached value of _cic_max_decim readback + size_t get_cic_max_decim() const + { + return (size_t) _cic_max_decim; + } + + //Check compatibility num, if not current, throw error. + //MAJOR COMPATIBILITY, top 32 bits = 0x1 + //MINOR COMPATIBILITY, lower 32 bits = 0x0 + void check_compat_num() + { + uint64_t compat_num = user_reg_read64(RB_REG_COMPAT_NUM); + uint32_t compat_num_major = compat_num>>32; + uint32_t compat_num_minor = compat_num&0xFFFFFFFF; + if ( compat_num_major > MAJOR_COMP) { + throw uhd::runtime_error(str(boost::format( + "DDC RFNoC block is too new for this software. Please upgrade to a driver that supports hardware revision %d.") + % compat_num_major)); + } else if ( compat_num_major < MAJOR_COMP) { + throw uhd::runtime_error(str(boost::format( + "DDC software is too new for this hardware. Please downgrade to a driver that supports hardware revision %d.") + % compat_num_major)); + } + if (compat_num_minor != MINOR_COMP) { + UHD_LOGGER_WARNING("DDC") << "DDC hardware compatability does not match software, this may have adverse affects on the block's behavior."; + } + } }; UHD_RFNOC_BLOCK_REGISTER(ddc_block_ctrl, "DDC"); diff --git a/host/lib/rfnoc/duc_block_ctrl_impl.cpp b/host/lib/rfnoc/duc_block_ctrl_impl.cpp index d0742a6d0..ca9e88f81 100644 --- a/host/lib/rfnoc/duc_block_ctrl_impl.cpp +++ b/host/lib/rfnoc/duc_block_ctrl_impl.cpp @@ -44,11 +44,12 @@ static double lambda_forward_prop(uhd::property_tree::sptr tree, uhd::fs_path pr class duc_block_ctrl_impl : public duc_block_ctrl { public: - static const size_t NUM_HALFBANDS = 2; - static const size_t CIC_MAX_INTERP = 128; UHD_RFNOC_BLOCK_CONSTRUCTOR(duc_block_ctrl) { + check_compat_num(); + _num_halfbands = (size_t) user_reg_read64(RB_REG_NUM_HALFBANDS); + _cic_max_interp = (size_t) user_reg_read64(RB_REG_CIC_MAX_INTERP); // Argument/prop tree hooks for (size_t chan = 0; chan < get_input_ports().size(); chan++) { double default_freq = get_arg("freq", chan); @@ -159,6 +160,14 @@ public: private: + size_t _num_halfbands; + size_t _cic_max_interp; + static const size_t MAJOR_COMP = 1; + static const size_t MINOR_COMP = 0; + static const size_t RB_REG_COMPAT_NUM = 0; + static const size_t RB_REG_NUM_HALFBANDS = 1; + static const size_t RB_REG_CIC_MAX_INTERP = 2; + //! Set the CORDIC frequency shift the signal to \p requested_freq double set_freq(const double requested_freq, const size_t chan) { @@ -186,14 +195,14 @@ private: { uhd::meta_range_t range; const double output_rate = get_arg("output_rate"); - for (int rate = 512; rate > 256; rate -= 4){ - range.push_back(uhd::range_t(output_rate/rate)); - } - for (int rate = 256; rate > 128; rate -= 2){ - range.push_back(uhd::range_t(output_rate/rate)); - } - for (int rate = 128; rate >= 1; rate -= 1){ - range.push_back(uhd::range_t(output_rate/rate)); + for (int hb = _num_halfbands; hb >= 0; hb--) { + const size_t interp_offset = _cic_max_interp<<(hb-1); + for(size_t interp = _cic_max_interp; interp > 0; interp--) { + const size_t hb_cic_interp = interp*(1< interp_offset) { + range.push_back(uhd::range_t(output_rate/hb_cic_interp)); + } + } } return range; } @@ -205,21 +214,14 @@ private: size_t interp = interp_rate; uint32_t hb_enable = 0; - while ((interp % 2 == 0) and hb_enable < NUM_HALFBANDS) { + while ((interp % 2 == 0) and hb_enable < _num_halfbands) { hb_enable++; interp /= 2; } - UHD_ASSERT_THROW(hb_enable <= NUM_HALFBANDS); - UHD_ASSERT_THROW(interp > 0 and interp <= CIC_MAX_INTERP); - // hacky hack: Unlike the DUC, the DUC actually simply has 2 - // flags to enable either halfband. - uint32_t hb_enable_word = hb_enable; - if (hb_enable == 2) { - hb_enable_word = 3; - } - hb_enable_word <<= 8; + UHD_ASSERT_THROW(hb_enable <= _num_halfbands); + UHD_ASSERT_THROW(interp > 0 and interp <= _cic_max_interp); // What we can't cover with halfbands, we do with the CIC - sr_write("INTERP_WORD", hb_enable_word | (interp & 0xff), chan); + sr_write("INTERP_WORD", (hb_enable << 8) | (interp & 0xff), chan); // Rate change = M/N sr_write("N", 1, chan); @@ -271,6 +273,40 @@ private: // Write DUC with scaling correction for CIC and CORDIC that maximizes dynamic range in 32/16/12/8bits. sr_write("SCALE_IQ", actual_scalar, chan); } + + //Get cached value of _num_halfbands + size_t get_num_halfbands() const + { + return (size_t) _num_halfbands; + } + + //Get cached value of _cic_max_decim readback + size_t get_cic_max_decim() const + { + return (size_t) _cic_max_interp; + } + + //Check compatibility num, if not current, throw error. + //MAJOR COMPATIBILITY, top 32 bits = 0x1 + //MINOR COMPATIBILITY, lower 32 bits = 0x0 + void check_compat_num() + { + uint64_t compat_num = user_reg_read64(RB_REG_COMPAT_NUM); + uint32_t compat_num_major = compat_num>>32; + uint32_t compat_num_minor = compat_num&0xFFFFFFFF; + if ( compat_num_major > MAJOR_COMP) { + throw uhd::runtime_error(str(boost::format( + "DUC RFNoC block is too new for this software. Please upgrade to a driver that supports hardware revision %d.") + % compat_num_major)); + } else if ( compat_num_major < MAJOR_COMP) { + throw uhd::runtime_error(str(boost::format( + "DUC software is too new for this hardware. Please downgrade to a driver that supports hardware revision %d.") + % compat_num_major)); + } + if (compat_num_minor != MINOR_COMP) { + UHD_LOGGER_WARNING("DDC") << "DDC hardware compatability does not match software, this may have adverse affects on the block's behavior."; + } + } }; UHD_RFNOC_BLOCK_REGISTER(duc_block_ctrl, "DUC"); -- cgit v1.2.3