// 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 #include #include #include #include #include #include #define BOOST_ENABLE_ASSERT_HANDLER #include 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(self, "f", boost::ref(y)); } abstract& g(Y const& y) { return call_method(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(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", init()) .def("value", &concrete::value) .def("set", &concrete::set) .def("call_f", &concrete::call_f) .def("f", &concrete_callback::f_impl) ; class_("abstract", init()) .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", init()) .def("value", &Y::value) .def("set", &Y::set) ; } #include "module_tail.cpp"