aboutsummaryrefslogtreecommitdiffstats
path: root/host/lib
diff options
context:
space:
mode:
Diffstat (limited to 'host/lib')
-rw-r--r--host/lib/rfnoc/CMakeLists.txt1
-rw-r--r--host/lib/rfnoc/window_block_control.cpp143
2 files changed, 144 insertions, 0 deletions
diff --git a/host/lib/rfnoc/CMakeLists.txt b/host/lib/rfnoc/CMakeLists.txt
index 8f74e9e6f..1a45ab1ff 100644
--- a/host/lib/rfnoc/CMakeLists.txt
+++ b/host/lib/rfnoc/CMakeLists.txt
@@ -52,5 +52,6 @@ LIBUHD_APPEND_SOURCES(
${CMAKE_CURRENT_SOURCE_DIR}/radio_control_impl.cpp
${CMAKE_CURRENT_SOURCE_DIR}/split_stream_block_control.cpp
${CMAKE_CURRENT_SOURCE_DIR}/vector_iir_block_control.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/window_block_control.cpp
)
diff --git a/host/lib/rfnoc/window_block_control.cpp b/host/lib/rfnoc/window_block_control.cpp
new file mode 100644
index 000000000..1bd6a1295
--- /dev/null
+++ b/host/lib/rfnoc/window_block_control.cpp
@@ -0,0 +1,143 @@
+//
+// Copyright 2020 Ettus Research, a National Instruments Brand
+//
+// SPDX-License-Identifier: GPL-3.0-or-later
+//
+
+#include <uhd/exception.hpp>
+#include <uhd/rfnoc/defaults.hpp>
+#include <uhd/rfnoc/multichan_register_iface.hpp>
+#include <uhd/rfnoc/property.hpp>
+#include <uhd/rfnoc/registry.hpp>
+#include <uhd/rfnoc/window_block_control.hpp>
+
+using namespace uhd::rfnoc;
+
+const uint32_t window_block_control::REG_WINDOW_BLOCK_SIZE = 1 << 4;
+const uint32_t window_block_control::REG_WINDOW_LEN_OFFSET = 0x00;
+const uint32_t window_block_control::REG_WINDOW_MAX_LEN_OFFSET = 0x04;
+const uint32_t window_block_control::REG_WINDOW_LOAD_COEFF_OFFSET = 0x08;
+const uint32_t window_block_control::REG_WINDOW_LOAD_COEFF_LAST_OFFSET = 0x0C;
+
+// User property names
+const char* const PROP_KEY_MAX_LEN = "max_len";
+
+class window_block_control_impl : public window_block_control
+{
+public:
+ RFNOC_BLOCK_CONSTRUCTOR(window_block_control),
+ _window_reg_iface(*this, 0, REG_WINDOW_BLOCK_SIZE)
+ {
+ UHD_ASSERT_THROW(get_num_input_ports() == get_num_output_ports());
+ _register_props();
+ }
+
+ size_t get_max_num_coefficients(const size_t chan) const
+ {
+ return _max_len.at(chan);
+ }
+
+ void set_coefficients(const std::vector<int16_t>& coeffs, const size_t chan)
+ {
+ if (coeffs.size() > _max_len.at(chan)) {
+ std::string error_msg = "Too many window coefficients specified (max "
+ + std::to_string(_max_len.at(chan)) + ")";
+ throw uhd::value_error(error_msg);
+ }
+
+ _coeffs[chan] = coeffs;
+ _program_coefficients(chan);
+ }
+
+ std::vector<int16_t> get_coefficients(const size_t chan) const
+ {
+ return _coeffs.at(chan);
+ }
+
+private:
+ void _register_props()
+ {
+ const size_t num_chans = get_num_input_ports();
+ _max_len.reserve(num_chans);
+ _coeffs.reserve(num_chans);
+ _prop_max_len.reserve(num_chans);
+ _prop_type_in.reserve(num_chans);
+ _prop_type_out.reserve(num_chans);
+
+ for (size_t chan = 0; chan < num_chans; chan++) {
+ const uint32_t max_len =
+ _window_reg_iface.peek32(REG_WINDOW_MAX_LEN_OFFSET, chan);
+ _max_len.emplace_back(max_len);
+
+ // set a default rectangular window for each channel
+ std::vector<int16_t> rect_coeffs(
+ max_len, std::numeric_limits<int16_t>::max());
+ _coeffs.emplace_back(rect_coeffs);
+ _program_coefficients(chan);
+
+ // register user properties
+ _prop_max_len.emplace_back(property_t<int>{PROP_KEY_MAX_LEN,
+ static_cast<int>(max_len),
+ {res_source_info::USER, chan}});
+ register_property(&_prop_max_len.back());
+ add_property_resolver({&ALWAYS_DIRTY},
+ {&_prop_max_len.back()},
+ [this, chan, max_len]() { _prop_max_len.at(chan).set(max_len); });
+
+ // register edge properties
+ _prop_type_in.emplace_back(property_t<std::string>{
+ PROP_KEY_TYPE, IO_TYPE_SC16, {res_source_info::INPUT_EDGE, chan}});
+ _prop_type_out.emplace_back(property_t<std::string>{
+ PROP_KEY_TYPE, IO_TYPE_SC16, {res_source_info::OUTPUT_EDGE, chan}});
+ register_property(&_prop_type_in.back());
+ register_property(&_prop_type_out.back());
+
+ // add resolvers for type
+ add_property_resolver({&_prop_type_in.back()},
+ {&_prop_type_in.back()},
+ [this, chan]() { _prop_type_in.at(chan).set(IO_TYPE_SC16); });
+ add_property_resolver({&_prop_type_out.back()},
+ {&_prop_type_out.back()},
+ [this, chan]() { _prop_type_out.at(chan).set(IO_TYPE_SC16); });
+ }
+ }
+
+ void _program_coefficients(size_t chan)
+ {
+ // Write coefficients [0..num_coeffs-2]...
+ const size_t num_coeffs = _coeffs.at(chan).size();
+ std::vector<uint32_t> coeffs_addr(num_coeffs - 1, REG_WINDOW_LOAD_COEFF_OFFSET);
+ std::vector<uint32_t> coeffs_minus_last(num_coeffs - 1);
+ std::transform(_coeffs.at(chan).begin(),
+ _coeffs.at(chan).end() - 1,
+ coeffs_minus_last.begin(),
+ [this](int16_t value) -> uint32_t { return static_cast<uint32_t>(value); });
+
+ _window_reg_iface.multi_poke32(coeffs_addr, coeffs_minus_last, chan);
+ // ...and the final coefficient (num_coeffs-1)
+ _window_reg_iface.poke32(REG_WINDOW_LOAD_COEFF_LAST_OFFSET,
+ static_cast<uint32_t>(_coeffs.at(chan).at(num_coeffs - 1)),
+ chan);
+ }
+
+ //! Maximum length of window
+ std::vector<size_t> _max_len;
+
+ //! Current window coefficients
+ std::vector<std::vector<int16_t>> _coeffs;
+
+ /**************************************************************************
+ * Attributes
+ *************************************************************************/
+ std::vector<property_t<std::string>> _prop_type_in;
+ std::vector<property_t<std::string>> _prop_type_out;
+ std::vector<property_t<int>> _prop_max_len;
+
+ /**************************************************************************
+ * Register interface
+ *************************************************************************/
+ multichan_register_iface _window_reg_iface;
+};
+
+UHD_RFNOC_BLOCK_REGISTER_DIRECT(
+ window_block_control, WINDOW_BLOCK, "Window", CLOCK_KEY_GRAPH, "bus_clk")