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/converters.hpp | |
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/converters.hpp')
-rw-r--r-- | mpm/python/pyusrp_periphs/converters.hpp | 94 |
1 files changed, 94 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>(); +} |