Hi
I have some code that creates an object in a shared_ptr in C++. The object
is then used within python and passed from Python to another C++ function
that tries to keep hold of it. It appears that when the object goes from
C++ to Python it is converted from a shared_ptr into a python object.
Then when it goes back into C++ it is converted from the python object into
a new shared_ptr, however this shared_ptr is unrelated to the initial
holding shared_ptr.
Is it possible to set up the boost-python bindings so that the conversion
from python object to shared_ptr references the original shared_ptr
internals?
Below is the abridged code I have used to show the problem. I have attached
the full reproducible to this email. In this example an object is initially
held in a shared_ptr named s_iniital. It is grabbed from within python
through the getSharedPtr function, then pushed back into C++ through the
putSharedPtr function. Inside putSharedPtr it is copied into the weak_ptr
s_copied. Inspecting the pointers in the debugger shows that the shared_ptr
used in putSharedPtr does not have the same reference counting internals as
s_initial. The final assert will fire because the weak pointer s_copied was
only related to a single strong pointer (the pointer used in putSharedPtr)
and that pointer was destructed once putSharedPtr was finished.
static shared_ptr s_initial;
static weak_ptr s_copied;
class UseSharedPtr
{
public:
shared_ptr getSharedPtr()
{
return s_initial;
}
void putSharedPtr(shared_ptr ptr)
{
s_copied = ptr;
}
};
BOOST_PYTHON_MODULE(test)
{
class_, boost::noncopyable>("Captured",
no_init);
class_("UseSharedPtr", init<>())
.def("getSharedPtr", &UseSharedPtr::getSharedPtr)
.def("putSharedPtr", &UseSharedPtr::putSharedPtr)
;
}
s_initial = make_shared();
const char* chunk = "\
from test import UseSharedPtr \n\
x = UseSharedPtr() \n\
ptr = x.getSharedPtr() \n\
x.putSharedPtr(ptr)\n\
del x \n\
del ptr \n\
";
object result = exec(chunk, mainNamespace, mainNamespace);
assert(s_copied.lock());
thanks,
Phil
--
Philip Dunstan
p...@philipdunstan.com
www.philipdunstan.com
#include
#include
#include
#include
#include
#include
#include
using std::string;
using boost::shared_ptr;
using boost::weak_ptr;
using boost::make_shared;
using namespace boost::python;
class Captured
{
public:
Captured() {}
~Captured() {}
};
static shared_ptr s_initial;
static weak_ptr s_copied;
class UseSharedPtr
{
public:
shared_ptr getSharedPtr()
{
return s_initial;
}
void putSharedPtr(shared_ptr ptr)
{
s_copied = ptr;
}
};
#define PY_MODULE_NAME "test"
BOOST_PYTHON_MODULE(test)
{
class_, boost::noncopyable>("Captured", no_init);
class_("UseSharedPtr", init<>())
.def("getSharedPtr", &UseSharedPtr::getSharedPtr)
.def("putSharedPtr", &UseSharedPtr::putSharedPtr)
;
}
int main()
{
try
{
PyImport_AppendInittab(PY_MODULE_NAME , &inittest);
Py_Initialize();
object mainModule((handle<>(borrowed(PyImport_AddModule("__main__");
object mainNamespace = mainModule.attr("__dict__");
object test_module(handle<>(PyImport_ImportModule(PY_MODULE_NAME)));
mainNamespace[PY_MODULE_NAME] = test_module;
s_initial = make_shared();
const char* chunk = "\
from test import UseSharedPtr \n\
x = UseSharedPtr() \n\
ptr = x.getSharedPtr() \n\
x.putSharedPtr(ptr)\n\
del x \n\
del ptr \n\
";
object result = exec(chunk, mainNamespace, mainNamespace);
assert(s_copied.lock());
}
catch (const error_already_set&)
{
return 1;
}
return 0;
};___
Cplusplus-sig mailing list
Cplusplus-sig@python.org
http://mail.python.org/mailman/listinfo/cplusplus-sig