aboutsummaryrefslogtreecommitdiffstats
path: root/mpm/python/pyusrp_periphs
diff options
context:
space:
mode:
authorBrent Stapleton <brent.stapleton@ettus.com>2018-04-17 13:48:29 -0700
committerMartin Braun <martin.braun@ettus.com>2018-04-18 15:34:51 -0700
commit13b45a08c90912e8baa33b618b152cf41293bb9d (patch)
tree71d45d03b9f231fea305d493984d05194a5bb8fc /mpm/python/pyusrp_periphs
parent7dcd16f5a6980cb1183bf1a10812c952d92c4ddb (diff)
downloaduhd-13b45a08c90912e8baa33b618b152cf41293bb9d.tar.gz
uhd-13b45a08c90912e8baa33b618b152cf41293bb9d.tar.bz2
uhd-13b45a08c90912e8baa33b618b152cf41293bb9d.zip
mpm: add device-specific Boost.Python bindings
Separating Boost.Python bindings into device-specific files. N3XX code now lives in n3xx/pyusrp_periphs. Only one src file should be added for pyusrp_periphs.so by CMake.
Diffstat (limited to 'mpm/python/pyusrp_periphs')
-rw-r--r--mpm/python/pyusrp_periphs/converters.hpp94
-rw-r--r--mpm/python/pyusrp_periphs/n3xx/pyusrp_periphs.cpp64
2 files changed, 158 insertions, 0 deletions
diff --git a/mpm/python/pyusrp_periphs/converters.hpp b/mpm/python/pyusrp_periphs/converters.hpp
new file mode 100644
index 000000000..ede9507a9
--- /dev/null
+++ b/mpm/python/pyusrp_periphs/converters.hpp
@@ -0,0 +1,94 @@
+//
+// Copyright 2017 Ettus Research, a National Instruments Company
+//
+// SPDX-License-Identifier: GPL-3.0-or-later
+//
+
+#pragma once
+
+#include <boost/python.hpp>
+#include <boost/python/stl_iterator.hpp>
+#include <vector>
+#include <string>
+
+namespace bp = boost::python;
+
+template<typename MapType>
+struct map_to_python_dict
+{
+ static PyObject* convert(MapType const& input_map)
+ {
+ bp::dict py_dict;
+ for (const auto& element: input_map){
+ py_dict[element.first] = element.second;
+ }
+ return bp::incref(py_dict.ptr());
+ }
+
+};
+
+template<typename Container>
+struct iterable_to_python_list
+{
+ static PyObject* convert(Container const& input)
+ {
+ bp::list py_list;
+ for (const auto& element: input){
+ py_list.append(element);
+ }
+ return bp::incref(py_list.ptr());
+ }
+};
+
+// Converter for std::vector / std::list arguments from python iterables
+struct iterable_converter
+{
+ template <typename Container>
+ iterable_converter&
+ from_python()
+ {
+ bp::converter::registry::push_back(
+ &iterable_converter::convertible,
+ &iterable_converter::construct<Container>,
+ bp::type_id<Container>());
+ return *this;
+ }
+
+ static void* convertible(PyObject* object)
+ {
+ return PyObject_GetIter(object) ? object : NULL;
+ }
+
+ template <typename Container>
+ static void construct(
+ PyObject* object,
+ bp::converter::rvalue_from_python_stage1_data* data)
+ {
+ // Object is a borrowed reference, so create a handle indicting it is
+ // borrowed for proper reference counting.
+ bp::handle<> handle(bp::borrowed(object));
+
+ // Obtain a handle to the memory block that the converter has allocated
+ // for the C++ type.
+ typedef bp::converter::rvalue_from_python_storage<Container>
+ storage_type;
+ void* storage = reinterpret_cast<storage_type*>(data)->storage.bytes;
+
+ typedef bp::stl_input_iterator<typename Container::value_type>
+ iterator;
+
+ // Allocate the C++ type into the converter's memory block, and assign
+ // its handle to the converter's convertible variable. The C++
+ // container is populated by passing the begin and end iterators of
+ // the python object to the container's constructor.
+ new (storage) Container(
+ iterator(bp::object(handle)), // begin
+ iterator()); // end
+ data->convertible = storage;
+ }
+};
+
+void export_converter(){
+ // LIBMPM_BOOST_PREAMBLE("helper")
+ bp::to_python_converter<std::vector< std::string >, iterable_to_python_list<std::vector< std::string > >, false>();
+}
diff --git a/mpm/python/pyusrp_periphs/n3xx/pyusrp_periphs.cpp b/mpm/python/pyusrp_periphs/n3xx/pyusrp_periphs.cpp
new file mode 100644
index 000000000..048ca6a44
--- /dev/null
+++ b/mpm/python/pyusrp_periphs/n3xx/pyusrp_periphs.cpp
@@ -0,0 +1,64 @@
+//
+// Copyright 2017 Ettus Research, a National Instruments Company
+//
+// SPDX-License-Identifier: GPL-3.0-or-later
+//
+
+// include hackery to only include boost python and define the macro here
+#include <boost/python.hpp>
+#define LIBMPM_PYTHON
+#define LIBMPM_BOOST_PREAMBLE(module) \
+ /* Register submodule types */ \
+ namespace bp = boost::python; \
+ bp::object py_module(bp::handle<>(bp::borrowed(PyImport_AddModule("libpyusrp_periphs." module)))); \
+ bp::scope().attr(module) = py_module; \
+ bp::scope io_scope = py_module;
+
+//! RAII-style GIL release method
+//
+// To release the GIL using this method, simply instantiate this class in the
+// scope that needs to release the GIL.
+//
+// Note that using this class assumes that threads have already been
+// initialized. See also https://docs.python.org/3.5/c-api/init.html for more
+// documentation on Python initialization and threads.
+class scoped_gil_release
+{
+public:
+ inline scoped_gil_release()
+ {
+ _thread_state = PyEval_SaveThread();
+ }
+
+ inline ~scoped_gil_release()
+ {
+ PyEval_RestoreThread(_thread_state);
+ _thread_state = nullptr;
+ }
+
+private:
+ PyThreadState* _thread_state;
+};
+
+//#include "types.hpp"
+#include "../converters.hpp"
+#include <mpm/xbar_iface.hpp>
+#include <mpm/types/types_python.hpp>
+#include <mpm/spi/spi_python.hpp>
+#include <mpm/ad937x/ad937x_ctrl.hpp>
+#include <mpm/dboards/magnesium_manager.hpp>
+#include <boost/noncopyable.hpp>
+
+namespace bp = boost::python;
+
+BOOST_PYTHON_MODULE(libpyusrp_periphs)
+{
+ bp::object package = bp::scope();
+ package.attr("__path__") = "libpyusrp_periphs";
+ export_converter();
+ export_types();
+ export_spi();
+ export_mykonos();
+ export_xbar();
+ export_magnesium();
+}