diff options
author | Brent Stapleton <brent.stapleton@ettus.com> | 2018-04-17 13:48:29 -0700 |
---|---|---|
committer | Martin Braun <martin.braun@ettus.com> | 2018-04-18 15:34:51 -0700 |
commit | 13b45a08c90912e8baa33b618b152cf41293bb9d (patch) | |
tree | 71d45d03b9f231fea305d493984d05194a5bb8fc /mpm/python/pyusrp_periphs | |
parent | 7dcd16f5a6980cb1183bf1a10812c952d92c4ddb (diff) | |
download | uhd-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.hpp | 94 | ||||
-rw-r--r-- | mpm/python/pyusrp_periphs/n3xx/pyusrp_periphs.cpp | 64 |
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(); +} |