mirror of
https://github.com/boostorg/python.git
synced 2026-01-21 17:12:22 +00:00
346 lines
8.3 KiB
C++
346 lines
8.3 KiB
C++
// Copyright David Abrahams 2001. Permission to copy, use,
|
|
// modify, sell and distribute this software is granted provided this
|
|
// copyright notice appears in all copies. This software is provided
|
|
// "as is" without express or implied warranty, and with no claim as
|
|
// to its suitability for any purpose.
|
|
|
|
#include "simple_type.hpp"
|
|
#include "complicated.hpp"
|
|
#include <boost/python/converter/wrapper.hpp>
|
|
#include <boost/python/converter/unwrapper.hpp>
|
|
#include <boost/python/detail/config.hpp>
|
|
#include <boost/python/convert.hpp>
|
|
#include <boost/python/module.hpp>
|
|
#include <boost/python/class.hpp>
|
|
#include <boost/python/object/value_holder.hpp>
|
|
#include <boost/python/object/class.hpp>
|
|
#include <boost/python/object/inheritance.hpp>
|
|
#include <boost/python/converter/class.hpp>
|
|
#include <boost/python/make_function.hpp>
|
|
#include <boost/python/errors.hpp>
|
|
#include <boost/mpl/type_list.hpp>
|
|
#include <string.h>
|
|
|
|
// Declare some straightforward extension types
|
|
extern "C" void
|
|
dealloc(PyObject* self)
|
|
{
|
|
PyObject_Del(self);
|
|
}
|
|
|
|
// Noddy is a type we got from one of the Python sample files
|
|
struct NoddyObject : PyObject
|
|
{
|
|
int x;
|
|
};
|
|
|
|
PyTypeObject NoddyType = {
|
|
PyObject_HEAD_INIT(NULL)
|
|
0,
|
|
"Noddy",
|
|
sizeof(NoddyObject),
|
|
0,
|
|
dealloc, /*tp_dealloc*/
|
|
0, /*tp_print*/
|
|
0, /*tp_getattr*/
|
|
0, /*tp_setattr*/
|
|
0, /*tp_compare*/
|
|
0, /*tp_repr*/
|
|
0, /*tp_as_number*/
|
|
0, /*tp_as_sequence*/
|
|
0, /*tp_as_mapping*/
|
|
0, /*tp_hash */
|
|
};
|
|
|
|
// Create a Noddy containing 42
|
|
PyObject* new_noddy()
|
|
{
|
|
NoddyObject* noddy = PyObject_New(NoddyObject, &NoddyType);
|
|
noddy->x = 42;
|
|
return (PyObject*)noddy;
|
|
}
|
|
|
|
// Simple is a wrapper around a struct simple, which just contains a char*
|
|
struct SimpleObject : PyObject
|
|
{
|
|
simple x;
|
|
};
|
|
|
|
PyTypeObject SimpleType = {
|
|
PyObject_HEAD_INIT(NULL)
|
|
0,
|
|
"Simple",
|
|
sizeof(SimpleObject),
|
|
0,
|
|
dealloc, /*tp_dealloc*/
|
|
0, /*tp_print*/
|
|
0, /*tp_getattr*/
|
|
0, /*tp_setattr*/
|
|
0, /*tp_compare*/
|
|
0, /*tp_repr*/
|
|
0, /*tp_as_number*/
|
|
0, /*tp_as_sequence*/
|
|
0, /*tp_as_mapping*/
|
|
0, /*tp_hash */
|
|
};
|
|
|
|
// Create a Simple containing "hello, world"
|
|
PyObject* new_simple()
|
|
{
|
|
SimpleObject* simple = PyObject_New(SimpleObject, &SimpleType);
|
|
simple->x.s = "hello, world";
|
|
return (PyObject*)simple;
|
|
}
|
|
|
|
//
|
|
// Declare some wrappers/unwrappers to test the low-level conversion
|
|
// mechanism. See boost/python/converter/source.hpp,target.hpp for a
|
|
// description of how the type parameters to wrapper<> and unwrapper<>
|
|
// are selected.
|
|
//
|
|
|
|
// Wrap an int by converting it to a Python Int
|
|
struct int_wrapper
|
|
: boost::python::converter::wrapper<int const&>
|
|
{
|
|
PyObject* convert(int const& x) const
|
|
{
|
|
return PyInt_FromLong(x);
|
|
}
|
|
};
|
|
|
|
// Wrap a simple by converting it to a Simple
|
|
struct simple_wrapper
|
|
: boost::python::converter::wrapper<simple const&>
|
|
{
|
|
PyObject* convert(simple const& x) const
|
|
{
|
|
SimpleObject* p = PyObject_New(SimpleObject, &SimpleType);
|
|
p->x = x;
|
|
return p;
|
|
}
|
|
};
|
|
|
|
// wrap a mutable reference to a simple by converting it to a
|
|
// Simple. Normally we wouldn't do it this way, since modifications to
|
|
// the result clearly don't change the original object, but here we're
|
|
// just proving that the mechanism works.
|
|
struct simple_ref_wrapper
|
|
: boost::python::converter::wrapper<simple&>
|
|
{
|
|
PyObject* convert(simple& x) const
|
|
{
|
|
SimpleObject* p = PyObject_New(SimpleObject, &SimpleType);
|
|
p->x = x;
|
|
return p;
|
|
}
|
|
};
|
|
|
|
// extract an int from a Python Int by converting it to an int. Since
|
|
// int is a scalar type, we convert by-value. Since Python Ints are
|
|
// immutable, there's no non-const reference converter.
|
|
struct native_int_unwrapper
|
|
: boost::python::converter::unwrapper<int>
|
|
{
|
|
void* can_convert(PyObject* p) const
|
|
{
|
|
return PyInt_Check(p) ? non_null : 0;
|
|
}
|
|
|
|
int convert(PyObject* p, void*, boost::type<int>) const
|
|
{
|
|
return PyInt_AsLong(p);
|
|
}
|
|
};
|
|
|
|
// Extract an int from a Noddy
|
|
struct noddy_int_unwrapper
|
|
: boost::python::converter::unwrapper<int>
|
|
{
|
|
void* can_convert(PyObject* p) const
|
|
{
|
|
return p->ob_type == &NoddyType ? non_null : 0;
|
|
}
|
|
|
|
int convert(PyObject* p, void*, boost::type<int>) const
|
|
{
|
|
return static_cast<NoddyObject*>(p)->x;
|
|
}
|
|
};
|
|
|
|
// Extract a mutable reference to an int from a Noddy.
|
|
struct noddy_int_ref_unwrapper
|
|
: boost::python::converter::unwrapper<int&>
|
|
{
|
|
void* can_convert(PyObject* p) const
|
|
{
|
|
return p->ob_type == &NoddyType ? non_null : 0;
|
|
}
|
|
|
|
int& convert(PyObject* p, void*, boost::type<int&>) const
|
|
{
|
|
return static_cast<NoddyObject*>(p)->x;
|
|
}
|
|
};
|
|
|
|
// Extract a mutable reference to a simple from a Simple
|
|
struct simple_ref_unwrapper
|
|
: boost::python::converter::unwrapper<simple&>
|
|
{
|
|
void* can_convert(PyObject* p) const
|
|
{
|
|
return p->ob_type == &SimpleType ? non_null : 0;
|
|
}
|
|
|
|
simple& convert(PyObject* p, void*, boost::type<simple&>) const
|
|
{
|
|
return static_cast<SimpleObject*>(p)->x;
|
|
}
|
|
};
|
|
|
|
// Extract a const reference to a simple from a Simple
|
|
struct simple_const_ref_unwrapper
|
|
: boost::python::converter::unwrapper<simple const&>
|
|
{
|
|
void* can_convert(PyObject* p) const
|
|
{
|
|
return p->ob_type == &SimpleType ? non_null : 0;
|
|
}
|
|
|
|
simple const& convert(PyObject* p, void*, boost::type<simple const&>) const
|
|
{
|
|
return static_cast<SimpleObject*>(p)->x;
|
|
}
|
|
};
|
|
|
|
//
|
|
// Some C++ functions to expose to Python
|
|
//
|
|
|
|
// Returns the length of s's held string
|
|
int f(simple const& s)
|
|
{
|
|
return strlen(s.s);
|
|
}
|
|
|
|
// A trivial passthru function for simple objects
|
|
simple const& g(simple const& x)
|
|
{
|
|
return x;
|
|
}
|
|
|
|
struct A
|
|
{
|
|
A() : x(0) {}
|
|
char const* name() { return "A"; }
|
|
int x;
|
|
};
|
|
|
|
struct B : A
|
|
{
|
|
B() : x(1) {}
|
|
char const* name() { return "B"; }
|
|
int x;
|
|
};
|
|
|
|
|
|
struct C : A
|
|
{
|
|
C() : x(2) {}
|
|
char const* name() { return "C"; }
|
|
virtual ~C() {}
|
|
int x;
|
|
};
|
|
|
|
struct D : B, C
|
|
{
|
|
D() : x(3) {}
|
|
char const* name() { return "D"; }
|
|
int x;
|
|
};
|
|
|
|
int take_a(A const& a) { return a.x; }
|
|
int take_b(B const& b) { return b.x; }
|
|
int take_c(C const& c) { return c.x; }
|
|
int take_d(D const& d) { return d.x; }
|
|
|
|
BOOST_PYTHON_MODULE_INIT(m1)
|
|
{
|
|
using boost::python::module;
|
|
using boost::python::class_;
|
|
|
|
// Create the converters; they are self-registering/unregistering.
|
|
static int_wrapper wrap_int;
|
|
static simple_wrapper wrap_simple;
|
|
static native_int_unwrapper unwrap_int1;
|
|
static noddy_int_unwrapper unwrap_int2;
|
|
static noddy_int_ref_unwrapper unwrap_int3;
|
|
static simple_ref_unwrapper unwrap_simple;
|
|
static simple_const_ref_unwrapper unwrap_simple_const_ref;
|
|
static simple_ref_wrapper wrap_simple_ref;
|
|
|
|
module m1("m1");
|
|
|
|
m1
|
|
// Insert the metaclass for all extension classes
|
|
.setattr("xclass", boost::python::objects::class_metatype())
|
|
|
|
// Insert the base class for all extension classes
|
|
.setattr("xinst", boost::python::objects::class_type())
|
|
|
|
.def("new_noddy", new_noddy)
|
|
.def("new_simple", new_simple)
|
|
|
|
// Expose f()
|
|
.def("f", f)
|
|
|
|
// Expose g()
|
|
.def("g", g)
|
|
|
|
.def("take_a", take_a)
|
|
.def("take_b", take_b)
|
|
.def("take_c", take_c)
|
|
.def("take_d", take_d)
|
|
|
|
.add(
|
|
class_<A>("A")
|
|
.def_init()
|
|
.def("name", &A::name)
|
|
)
|
|
|
|
;
|
|
|
|
// sequence points don't ensure that "A" is constructed before "B"
|
|
// or "C" below if we make them part of the same chain
|
|
m1
|
|
.add(
|
|
class_<B,bases<A> >("B")
|
|
.def_init()
|
|
.def("name", &B::name)
|
|
)
|
|
|
|
.add(
|
|
class_<C,bases<A> >("C")
|
|
.def_init()
|
|
.def("name", &C::name)
|
|
)
|
|
;
|
|
|
|
m1
|
|
.add(
|
|
class_<D,bases<B,C> >("D")
|
|
.def_init()
|
|
.def("name", &D::name)
|
|
)
|
|
|
|
.add(
|
|
class_<complicated>("complicated")
|
|
.def_init(args<simple const&,int>())
|
|
.def_init(args<simple const&>())
|
|
.def("get_n", &complicated::get_n)
|
|
)
|
|
;
|
|
}
|
|
|
|
#include "module_tail.cpp"
|