python/test/virtual_functions.cpp
Ralf W. Grosse-Kunstleve e80224b1ad boost/python, libs/python: all changes from trunk merged into branches/release (without any manual modifications)
Commands used:
  svn merge https://svn.boost.org/svn/boost/branches/release/boost/python https://svn.boost.org/svn/boost/trunk/boost/python /net/chevy/raid1/rwgk/boost_release/merge_attempt/boost/boost/python

  svn merge https://svn.boost.org/svn/boost/branches/release/libs/python https://svn.boost.org/svn/boost/trunk/libs/python /net/chevy/raid1/rwgk/boost_release/merge_attempt/boost/libs/python

  svn, version 1.6.4 (r38063)
     compiled Aug 17 2009, 13:31:03


[SVN r55629]
2009-08-17 21:01:18 +00:00

126 lines
2.9 KiB
C++

// Copyright David Abrahams 2002.
// Distributed under the Boost Software License, Version 1.0. (See
// accompanying file LICENSE_1_0.txt or copy at
// http://www.boost.org/LICENSE_1_0.txt)
#include <boost/python/class.hpp>
#include <boost/python/module.hpp>
#include <boost/python/def.hpp>
#include <boost/python/return_internal_reference.hpp>
#include <boost/python/call_method.hpp>
#include <boost/ref.hpp>
#include <boost/utility.hpp>
#define BOOST_ENABLE_ASSERT_HANDLER
#include <boost/assert.hpp>
using namespace boost::python;
struct X
{
explicit X(int x) : x(x), magic(7654321) { ++counter; }
X(X const& rhs) : x(rhs.x), magic(7654321) { ++counter; }
virtual ~X() { BOOST_ASSERT(magic == 7654321); magic = 6666666; x = 9999; --counter; }
void set(int _x) { BOOST_ASSERT(magic == 7654321); this->x = _x; }
int value() const { BOOST_ASSERT(magic == 7654321); return x; }
static int count() { return counter; }
private:
void operator=(X const&);
private:
int x;
long magic;
static int counter;
};
struct Y : X
{
Y(int x) : X(x) {};
};
struct abstract : X
{
abstract(int x) : X(x) {};
int call_f(Y const& y) { return f(y); }
virtual int f(Y const& y) = 0;
abstract& call_g(Y const& y) { return g(y); }
virtual abstract& g(Y const& y) = 0;
};
struct concrete : X
{
concrete(int x) : X(x) {};
int call_f(Y const& y) { return f(y); }
virtual int f(Y const& y) { set(y.value()); return y.value(); }
};
struct abstract_callback : abstract
{
abstract_callback(PyObject* p, int x)
: abstract(x), self(p)
{}
int f(Y const& y)
{
return call_method<int>(self, "f", boost::ref(y));
}
abstract& g(Y const& y)
{
return call_method<abstract&>(self, "g", boost::ref(y));
}
PyObject* self;
};
struct concrete_callback : concrete
{
concrete_callback(PyObject* p, int x)
: concrete(x), self(p)
{}
concrete_callback(PyObject* p, concrete const& x)
: concrete(x), self(p)
{}
int f(Y const& y)
{
return call_method<int>(self, "f", boost::ref(y));
}
int f_impl(Y const& y)
{
return this->concrete::f(y);
}
PyObject* self;
};
int X::counter;
BOOST_PYTHON_MODULE(virtual_functions_ext)
{
class_<concrete, concrete_callback>("concrete", init<int>())
.def("value", &concrete::value)
.def("set", &concrete::set)
.def("call_f", &concrete::call_f)
.def("f", &concrete_callback::f_impl)
;
class_<abstract, boost::noncopyable, abstract_callback
>("abstract", init<int>())
.def("value", &abstract::value)
.def("call_f", &abstract::call_f)
.def("call_g", &abstract::call_g, return_internal_reference<>())
.def("set", &abstract::set)
;
class_<Y>("Y", init<int>())
.def("value", &Y::value)
.def("set", &Y::set)
;
}
#include "module_tail.cpp"