// 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) #ifndef MAKE_INSTANCE_DWA200296_HPP # define MAKE_INSTANCE_DWA200296_HPP # include # include # include # include # include # include # include # include namespace boost { namespace python { namespace objects { template struct make_instance_impl { typedef objects::instance instance_t; template static inline PyObject* execute(Arg& x) { BOOST_MPL_ASSERT((mpl::or_, boost::python::detail::is_union >)); PyTypeObject* type = Derived::get_class_object(x); if (type == 0) return python::detail::none(); PyObject* raw_result = type->tp_alloc( type, objects::additional_instance_size::value); if (raw_result != 0) { python::detail::decref_guard protect(raw_result); instance_t* instance = (instance_t*)raw_result; // construct the new C++ object and install the pointer // in the Python object. Holder *holder =Derived::construct(instance->storage.bytes, (PyObject*)instance, x); holder->install(raw_result); // Note the position of the internally-stored Holder, // for the sake of destruction const size_t offset = reinterpret_cast(holder) - reinterpret_cast(instance->storage.bytes) + offsetof(instance_t, storage); Py_SET_SIZE(instance, offset); // Release ownership of the python object protect.cancel(); } return raw_result; } }; template struct make_instance : make_instance_impl > { template static inline PyTypeObject* get_class_object(U&) { return converter::registered::converters.get_class_object(); } static inline Holder* construct(void* storage, PyObject* instance, reference_wrapper x) { size_t allocated = objects::additional_instance_size::value; void* aligned_storage = ::boost::alignment::align(boost::python::detail::alignment_of::value, sizeof(Holder), storage, allocated); return new (aligned_storage) Holder(instance, x); } }; }}} // namespace boost::python::object #endif // MAKE_INSTANCE_DWA200296_HPP