[857] | 1 | // Copyright Gottfried Ganßauge 2003.
|
---|
| 2 | // Distributed under the Boost Software License, Version 1.0. (See
|
---|
| 3 | // accompanying file LICENSE_1_0.txt or copy at
|
---|
| 4 | // http://www.boost.org/LICENSE_1_0.txt)
|
---|
| 5 | /*
|
---|
| 6 | * Generic Conversion of opaque C++-pointers to a Python-Wrapper.
|
---|
| 7 | */
|
---|
| 8 | # ifndef OPAQUE_POINTER_CONVERTER_HPP_
|
---|
| 9 | # define OPAQUE_POINTER_CONVERTER_HPP_
|
---|
| 10 |
|
---|
| 11 | # include <boost/python/detail/prefix.hpp>
|
---|
| 12 | # include <boost/python/lvalue_from_pytype.hpp>
|
---|
| 13 | # include <boost/python/to_python_converter.hpp>
|
---|
| 14 | # include <boost/python/detail/dealloc.hpp>
|
---|
| 15 | # include <boost/python/detail/none.hpp>
|
---|
| 16 | # include <boost/type_traits/remove_pointer.hpp>
|
---|
| 17 | # include <boost/type_traits/is_pointer.hpp>
|
---|
| 18 |
|
---|
| 19 | // opaque_pointer_converter --
|
---|
| 20 | //
|
---|
| 21 | // usage: opaque_pointer_converter<Pointer>("name")
|
---|
| 22 | //
|
---|
| 23 | // registers to- and from- python conversions for a type Pointer,
|
---|
| 24 | // and a corresponding Python type called "name".
|
---|
| 25 | //
|
---|
| 26 | // Note:
|
---|
| 27 | // In addition you need to define specializations for type_id
|
---|
| 28 | // on the type pointed to by Pointer using
|
---|
| 29 | // BOOST_PYTHON_OPAQUE_SPECIALIZED_TYPE_ID(Pointee)
|
---|
| 30 | //
|
---|
| 31 | // For an example see libs/python/test/opaque.cpp
|
---|
| 32 | //
|
---|
| 33 | namespace boost { namespace python {
|
---|
| 34 | namespace detail {
|
---|
| 35 | template <class R>
|
---|
| 36 | struct opaque_pointer_converter_requires_a_pointer_type
|
---|
| 37 | # if defined(__GNUC__) && __GNUC__ >= 3 || defined(__EDG__)
|
---|
| 38 | {}
|
---|
| 39 | # endif
|
---|
| 40 | ;
|
---|
| 41 | }
|
---|
| 42 |
|
---|
| 43 | template <class Pointer>
|
---|
| 44 | struct opaque_pointer_converter
|
---|
| 45 | : to_python_converter<
|
---|
| 46 | Pointer, opaque_pointer_converter<Pointer> >
|
---|
| 47 | {
|
---|
| 48 | BOOST_STATIC_CONSTANT(
|
---|
| 49 | bool, ok = is_pointer<Pointer>::value);
|
---|
| 50 |
|
---|
| 51 | typedef typename mpl::if_c<
|
---|
| 52 | ok
|
---|
| 53 | , Pointer
|
---|
| 54 | , detail::opaque_pointer_converter_requires_a_pointer_type<Pointer>
|
---|
| 55 | >::type ptr_type;
|
---|
| 56 |
|
---|
| 57 | private:
|
---|
| 58 | struct instance;
|
---|
| 59 |
|
---|
| 60 | public:
|
---|
| 61 | explicit opaque_pointer_converter(char const* name)
|
---|
| 62 | {
|
---|
| 63 | type_object.tp_name = const_cast<char *> (name);
|
---|
| 64 |
|
---|
| 65 | lvalue_from_pytype<
|
---|
| 66 | opaque_pointer_converter<ptr_type>,
|
---|
| 67 | &opaque_pointer_converter<ptr_type>::type_object
|
---|
| 68 | >();
|
---|
| 69 | }
|
---|
| 70 |
|
---|
| 71 | static PyObject* convert(ptr_type x)
|
---|
| 72 | {
|
---|
| 73 | PyObject *result = 0;
|
---|
| 74 |
|
---|
| 75 | if (x != 0) {
|
---|
| 76 | instance *o = PyObject_New (instance, &type_object);
|
---|
| 77 |
|
---|
| 78 | o->x = x;
|
---|
| 79 | result = &o->base_;
|
---|
| 80 | } else {
|
---|
| 81 | result = detail::none();
|
---|
| 82 | }
|
---|
| 83 |
|
---|
| 84 | return (result);
|
---|
| 85 | }
|
---|
| 86 |
|
---|
| 87 | static typename ::boost::remove_pointer<ptr_type>::type&
|
---|
| 88 | execute(instance &p_)
|
---|
| 89 | {
|
---|
| 90 | return *p_.x;
|
---|
| 91 | }
|
---|
| 92 |
|
---|
| 93 | private:
|
---|
| 94 | static PyTypeObject type_object;
|
---|
| 95 |
|
---|
| 96 | // This is a POD so we can use PyObject_Del on it, for example.
|
---|
| 97 | struct instance
|
---|
| 98 | {
|
---|
| 99 | PyObject base_;
|
---|
| 100 | ptr_type x;
|
---|
| 101 | };
|
---|
| 102 | };
|
---|
| 103 |
|
---|
| 104 | template <class Pointer>
|
---|
| 105 | PyTypeObject opaque_pointer_converter<Pointer>::type_object =
|
---|
| 106 | {
|
---|
| 107 | PyObject_HEAD_INIT(NULL)
|
---|
| 108 | 0,
|
---|
| 109 | 0,
|
---|
| 110 | sizeof(typename opaque_pointer_converter<Pointer>::instance),
|
---|
| 111 | 0,
|
---|
| 112 | ::boost::python::detail::dealloc
|
---|
| 113 | };
|
---|
| 114 | }} // namespace boost::python
|
---|
| 115 | # ifdef BOOST_MSVC
|
---|
| 116 | // MSC works without this workaround, but needs another one ...
|
---|
| 117 | # define BOOST_PYTHON_OPAQUE_SPECIALIZED_TYPE_ID(Pointee) \
|
---|
| 118 | BOOST_BROKEN_COMPILER_TYPE_TRAITS_SPECIALIZATION(Pointee)
|
---|
| 119 | # else
|
---|
| 120 | # define BOOST_PYTHON_OPAQUE_SPECIALIZED_TYPE_ID(Pointee) \
|
---|
| 121 | namespace boost { namespace python { \
|
---|
| 122 | template<> \
|
---|
| 123 | inline type_info type_id<Pointee>(BOOST_PYTHON_EXPLICIT_TT_DEF(Pointee)) \
|
---|
| 124 | { \
|
---|
| 125 | return type_info (typeid (Pointee *)); \
|
---|
| 126 | } \
|
---|
| 127 | template<> \
|
---|
| 128 | inline type_info type_id<const volatile Pointee&>( \
|
---|
| 129 | BOOST_PYTHON_EXPLICIT_TT_DEF(const volatile Pointee&)) \
|
---|
| 130 | { \
|
---|
| 131 | return type_info (typeid (Pointee *)); \
|
---|
| 132 | } \
|
---|
| 133 | }}
|
---|
| 134 | # endif
|
---|
| 135 | # endif // OPAQUE_POINTER_CONVERTER_HPP_
|
---|