From 400b00c34338af502c5d786edcfb8512bfbbff8b Mon Sep 17 00:00:00 2001 From: Brent Stapleton Date: Mon, 18 Nov 2019 14:06:39 +0530 Subject: rfnoc: adding RFNoC Python API Adding Python bindings for the RFNoC API. This includes the rfnoc_graph, noc_block_base, and several other supporting classes. Templated functions are not currently supported. For example, `rfnoc_graph::get_block` can only return the basic block controller. --- host/lib/rfnoc/rfnoc_python.hpp | 247 ++++++++++++++++++++++++++++++++++++++++ host/python/CMakeLists.txt | 1 + host/python/__init__.py | 1 + host/python/pyuhd.cpp | 6 + host/python/rfnoc.py | 18 +++ 5 files changed, 273 insertions(+) create mode 100644 host/lib/rfnoc/rfnoc_python.hpp create mode 100644 host/python/rfnoc.py (limited to 'host') diff --git a/host/lib/rfnoc/rfnoc_python.hpp b/host/lib/rfnoc/rfnoc_python.hpp new file mode 100644 index 000000000..32bc8d03a --- /dev/null +++ b/host/lib/rfnoc/rfnoc_python.hpp @@ -0,0 +1,247 @@ +// +// Copyright 2019 Ettus Research, a National Instruments Brand +// +// SPDX-License-Identifier: GPL-3.0-or-later +// + +#ifndef INCLUDED_UHD_RFNOC_PYTHON_HPP +#define INCLUDED_UHD_RFNOC_PYTHON_HPP + +#include "../stream_python.hpp" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + + +using namespace uhd::rfnoc; + +namespace { // anon + +using timekeeper = mb_controller::timekeeper; + +// Make a "trampoline" class to help Pybind11 +class PyTimekeeper : public timekeeper +{ +public: + /* Inherit the constructors */ + using timekeeper::timekeeper; + + /* Trampoline (need one for each virtual function) */ + uint64_t get_ticks_now(void) override + { + PYBIND11_OVERLOAD_PURE(uint64_t, /* Return type */ + timekeeper, /* Parent class */ + get_ticks_now /* Name of function in C++ (must match Python name) */ + /* (No) Argument(s) */ + ); + } + uint64_t get_ticks_last_pps(void) override + { + PYBIND11_OVERLOAD_PURE(uint64_t, timekeeper, get_ticks_last_pps); + } + void set_ticks_now(const uint64_t ticks) override + { + PYBIND11_OVERLOAD_PURE(void, timekeeper, set_ticks_now, ticks); + } + void set_ticks_next_pps(const uint64_t ticks) override + { + PYBIND11_OVERLOAD_PURE(void, timekeeper, set_ticks_next_pps, ticks); + } +}; + +} // namespace + + +void export_rfnoc(py::module& m) +{ + py::class_(m, "block_id") + // Constructors + .def(py::init<>()) + .def(py::init()) + .def(py::init(), + py::arg("device_no"), + py::arg("block_name"), + py::arg("block_ctr") = 0) + + // Methods + .def("__str__", &block_id_t::to_string) + .def("to_string", &block_id_t::to_string) + .def("is_valid_blockname", &block_id_t::is_valid_blockname) + .def("is_valid_block_id", &block_id_t::is_valid_block_id) + .def("match", &block_id_t::match) + .def("get", &block_id_t::get) + .def("get_local", &block_id_t::get_local) + .def("get_tree_root", &block_id_t::get_tree_root) + .def("get_device_no", &block_id_t::get_device_no) + .def("get_block_count", &block_id_t::get_block_count) + .def("get_block_name", &block_id_t::get_block_name) + .def("set", py::overload_cast(&block_id_t::set)) + .def("set", + py::overload_cast( + &block_id_t::set), + py::arg("device_no"), + py::arg("block_name"), + py::arg("block_ctr") = 0) + .def("set_device_no", &block_id_t::set_device_no) + .def("set_block_name", &block_id_t::set_block_name) + .def("set_block_count", &block_id_t::set_block_count) + + // Operators + .def(py::self == py::self) + .def(py::self != py::self) + .def(py::self < py::self) + .def(py::self > py::self) + .def(py::self == std::string()); + + py::enum_(m, "edge") + .value("static", graph_edge_t::STATIC) + .value("dynamic", graph_edge_t::DYNAMIC) + .value("rx_stream", graph_edge_t::RX_STREAM) + .value("tx_stream", graph_edge_t::TX_STREAM) + .export_values(); + + py::class_(m, "graph_edge") + // Constructors + .def(py::init<>()) + .def(py::init()) + + // Properties + .def_readwrite("src_blockid", &graph_edge_t::src_blockid) + .def_readwrite("src_port", &graph_edge_t::src_port) + .def_readwrite("dst_blockid", &graph_edge_t::dst_blockid) + .def_readwrite("dst_port", &graph_edge_t::dst_port) + .def_readwrite("edge", &graph_edge_t::edge) + .def_readwrite( + "property_propagation_active", &graph_edge_t::property_propagation_active) + + // Methods + .def("__str__", &graph_edge_t::to_string) + .def("to_string", &graph_edge_t::to_string); + + py::class_(m, "rfnoc_graph") + .def(py::init(&rfnoc_graph::make)) + + // General RFNoC Graph methods + // TODO: Templated methods?? + .def("find_blocks", + [](rfnoc_graph::sptr& self, const std::string& block_id_hint) { + return self->find_blocks(block_id_hint); + }) + .def("has_block", + [](rfnoc_graph::sptr& self, const block_id_t& block_id) { + return self->has_block(block_id); + }) + .def("get_block", + [](rfnoc_graph::sptr& self, const block_id_t& block_id) { + return self->get_block(block_id); + }) + .def("is_connectable", &rfnoc_graph::is_connectable) + .def("connect", + py::overload_cast( + &rfnoc_graph::connect)) + .def("connect", + py::overload_cast(&rfnoc_graph::connect), + py::arg("streamer"), + py::arg("strm_port"), + py::arg("dst_blk"), + py::arg("dst_blk"), + py::arg("adapter_id") = uhd::transport::NULL_ADAPTER_ID) + .def("connect", + py::overload_cast(&rfnoc_graph::connect), + py::arg("src_blk"), + py::arg("src_blk"), + py::arg("streamer"), + py::arg("strm_port"), + py::arg("adapter_id") = uhd::transport::NULL_ADAPTER_ID) + .def("enumerate_adapters_from_src", &rfnoc_graph::enumerate_adapters_from_src) + .def("enumerate_adapters_to_dst", &rfnoc_graph::enumerate_adapters_to_dst) + .def("enumerate_static_connections", &rfnoc_graph::enumerate_static_connections) + .def("enumerate_active_connections", &rfnoc_graph::enumerate_active_connections) + .def("commit", &rfnoc_graph::commit) + .def("release", &rfnoc_graph::release) + .def("create_rx_streamer", &rfnoc_graph::create_rx_streamer) + .def("create_tx_streamer", &rfnoc_graph::create_tx_streamer) + .def("get_num_mboards", &rfnoc_graph::get_num_mboards) + .def( + "get_mb_controller", &rfnoc_graph::get_mb_controller, py::arg("mb_index") = 0) + .def("synchronize_devices", &rfnoc_graph::synchronize_devices) + .def("get_tree", &rfnoc_graph::get_tree); + + py::class_(m, "mb_controller") + .def("get_num_timekeepers", &mb_controller::get_num_timekeepers) + .def("get_timekeeper", &mb_controller::get_timekeeper) + .def("init", &mb_controller::init) + .def("get_mboard_name", &mb_controller::get_mboard_name) + .def("set_time_source", &mb_controller::set_time_source) + .def("get_time_source", &mb_controller::get_time_source) + .def("get_time_sources", &mb_controller::get_time_sources) + .def("set_clock_source", &mb_controller::set_clock_source) + .def("get_clock_source", &mb_controller::get_clock_source) + .def("get_clock_sources", &mb_controller::get_clock_sources) + .def("set_sync_source", + py::overload_cast( + &mb_controller::set_sync_source)) + .def("set_sync_source", + py::overload_cast(&mb_controller::set_sync_source)) + .def("get_sync_source", &mb_controller::get_sync_source) + .def("get_sync_sources", &mb_controller::get_sync_sources) + .def("set_clock_source_out", &mb_controller::set_clock_source_out) + .def("set_time_source_out", &mb_controller::set_time_source_out) + .def("get_sensor", &mb_controller::get_sensor) + .def("get_sensor_names", &mb_controller::get_sensor_names) + .def("get_eeprom", &mb_controller::get_eeprom) + .def("synchronize", &mb_controller::synchronize) + .def("get_gpio_banks", &mb_controller::get_gpio_banks) + .def("get_gpio_srcs", &mb_controller::get_gpio_srcs) + .def("get_gpio_src", &mb_controller::get_gpio_src) + .def("set_gpio_src", &mb_controller::set_gpio_src); + + py::class_(m, "timekeeper") + // Methods + // FIXME? .def(py::init<>()) + .def("get_time_now", &timekeeper::get_time_now) + .def("get_ticks_now", &timekeeper::get_ticks_now) + .def("get_time_last_pps", &timekeeper::get_time_last_pps) + .def("get_ticks_last_pps", &timekeeper::get_ticks_last_pps) + .def("set_time_now", &timekeeper::set_time_now) + .def("set_ticks_now", &timekeeper::set_ticks_now) + .def("set_time_next_pps", &timekeeper::set_time_next_pps) + .def("set_ticks_next_pps", &timekeeper::set_ticks_next_pps) + .def("get_tick_rate", &timekeeper::get_tick_rate); + + py::class_(m, "noc_block_base") + .def("get_unique_id", &noc_block_base::get_unique_id) + .def("get_num_input_ports", &noc_block_base::get_num_input_ports) + .def("get_num_output_ports", &noc_block_base::get_num_output_ports) + .def("get_noc_id", &noc_block_base::get_noc_id) + .def("get_block_id", &noc_block_base::get_block_id) + .def("get_tick_rate", &noc_block_base::get_tick_rate) + .def("get_mtu", &noc_block_base::get_mtu) + .def("get_block_args", &noc_block_base::get_block_args) + .def("get_tree", [](noc_block_base::sptr& self) { + // Force the non-const `get_tree` + uhd::property_tree::sptr tree = self->get_tree(); + return tree; + }); +} + +#endif /* INCLUDED_UHD_RFNOC_PYTHON_HPP */ diff --git a/host/python/CMakeLists.txt b/host/python/CMakeLists.txt index 5598dda24..2b2a35d2e 100644 --- a/host/python/CMakeLists.txt +++ b/host/python/CMakeLists.txt @@ -52,6 +52,7 @@ set(PYUHD_FILES ${CMAKE_CURRENT_SOURCE_DIR}/types.py ${CMAKE_CURRENT_SOURCE_DIR}/usrp.py ${CMAKE_CURRENT_SOURCE_DIR}/filters.py + ${CMAKE_CURRENT_SOURCE_DIR}/rfnoc.py ) set(SETUP_PY_IN "${CMAKE_CURRENT_SOURCE_DIR}/setup.py.in") diff --git a/host/python/__init__.py b/host/python/__init__.py index 501b599ef..bf37f5215 100644 --- a/host/python/__init__.py +++ b/host/python/__init__.py @@ -10,3 +10,4 @@ UHD Python API module from . import types from . import usrp from . import filters +from . import rfnoc diff --git a/host/python/pyuhd.cpp b/host/python/pyuhd.cpp index e38dc9224..50434b4fc 100644 --- a/host/python/pyuhd.cpp +++ b/host/python/pyuhd.cpp @@ -13,6 +13,8 @@ namespace py = pybind11; +#include "rfnoc/rfnoc_python.hpp" + #include "stream_python.hpp" #include "types/types_python.hpp" @@ -72,5 +74,9 @@ PYBIND11_MODULE(libpyuhd, m) // Register filters submodule auto filters_module = m.def_submodule("filters", "Filter Submodule"); export_filters(filters_module); + + // Register RFNoC submodule + auto rfnoc_module = m.def_submodule("rfnoc", "RFNoC Objects"); + export_rfnoc(rfnoc_module); } diff --git a/host/python/rfnoc.py b/host/python/rfnoc.py new file mode 100644 index 000000000..690b3d841 --- /dev/null +++ b/host/python/rfnoc.py @@ -0,0 +1,18 @@ +# +# Copyright 2019 Ettus Research, a National Instruments Brand +# +# SPDX-License-Identifier: GPL-3.0-or-later +# +""" @package rfnoc +Python UHD module containing the RFNoC API. +""" + +from . import libpyuhd as lib + +BlockID = lib.rfnoc.block_id +Edge = lib.rfnoc.edge +GraphEdge = lib.rfnoc.graph_edge +RfnocGraph = lib.rfnoc.rfnoc_graph +MBController = lib.rfnoc.mb_controller +Timekeeper = lib.rfnoc.timekeeper +NocBlock = lib.rfnoc.noc_block_base -- cgit v1.2.3