diff --git a/CHANGELOG.md b/CHANGELOG.md index 8d6f540b992fdde1fe185505b2f4f163e45f494e..9cdd40e080646fd02d4d8a52a10765f36e7fe8dc 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,6 +9,9 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/). ### Added - Added a deprecation call policy shortcut ([#466](https://github.com/stack-of-tasks/eigenpy/pull/466)) +### Fixed +- Fix register_symbolic_link_to_registered_type() for multiple successive registrations ([#471](https://github.com/stack-of-tasks/eigenpy/pull/471)) + ## [3.5.1] - 2024-04-25 ### Fixed diff --git a/include/eigenpy/registration.hpp b/include/eigenpy/registration.hpp index 7644f291695a49f6bb36e17a357a59dd84fd3d00..a2f13fc96f85f4f9b97bbe94e0678dc76689515f 100644 --- a/include/eigenpy/registration.hpp +++ b/include/eigenpy/registration.hpp @@ -45,6 +45,7 @@ inline bool register_symbolic_link_to_registered_type() { const bp::converter::registration* reg = bp::converter::registry::query(info); bp::handle<> class_obj(reg->get_class_object()); + bp::incref(class_obj.get()); bp::scope().attr(reg->get_class_object()->tp_name) = bp::object(class_obj); return true; } @@ -61,6 +62,7 @@ inline bool register_symbolic_link_to_registered_type(const Visitor& visitor) { const bp::converter::registration* reg = bp::converter::registry::query(info); bp::handle<> class_obj(reg->get_class_object()); + bp::incref(class_obj.get()); bp::object object(class_obj); bp::scope().attr(reg->get_class_object()->tp_name) = object; registration_class<T> cl(object); diff --git a/unittest/CMakeLists.txt b/unittest/CMakeLists.txt index d0c216fb56eb7bd230d90e20fd73bba63bf11a5c..72bfb1bb33e19ab97a0476fe265d576059969635 100644 --- a/unittest/CMakeLists.txt +++ b/unittest/CMakeLists.txt @@ -30,6 +30,7 @@ endmacro(ADD_LIB_UNIT_TEST) add_dependencies(build_tests ${PYWRAP}) add_lib_unit_test(matrix) +add_lib_unit_test(multiple_registration) if(BUILD_TESTING_SCIPY) find_scipy() add_lib_unit_test(sparse_matrix) @@ -101,6 +102,8 @@ endif() add_lib_unit_test(bind_virtual_factory) add_python_lib_unit_test("py-matrix" "unittest/python/test_matrix.py") +add_python_lib_unit_test("py-multiple-registration" + "unittest/python/test_multiple_registration.py") add_python_lib_unit_test("py-tensor" "unittest/python/test_tensor.py") add_python_lib_unit_test("py-geometry" "unittest/python/test_geometry.py") diff --git a/unittest/multiple_registration.cpp b/unittest/multiple_registration.cpp new file mode 100644 index 0000000000000000000000000000000000000000..32940d4a14d04e7516a5a42e02f0e0681e4f9d31 --- /dev/null +++ b/unittest/multiple_registration.cpp @@ -0,0 +1,28 @@ +#include "eigenpy/registration.hpp" +#include <cstdio> + +namespace bp = boost::python; + +class X { + public: + X() {} + void operator()() { printf("DOOT\n"); } +}; + +class X_wrapper : public X, bp::wrapper<X> { + public: + static void expose() { + if (!eigenpy::register_symbolic_link_to_registered_type<X>()) { + bp::class_<X>("X", bp::init<>()).def("__call__", &X::operator()); + } + } +}; + +BOOST_PYTHON_MODULE(multiple_registration) { + X_wrapper::expose(); + X_wrapper::expose(); + X_wrapper::expose(); + X_wrapper::expose(); + X_wrapper::expose(); + X_wrapper::expose(); +} diff --git a/unittest/python/test_multiple_registration.py b/unittest/python/test_multiple_registration.py new file mode 100644 index 0000000000000000000000000000000000000000..d414596b2ed73e1b5c5a41f208e733520e6a77b1 --- /dev/null +++ b/unittest/python/test_multiple_registration.py @@ -0,0 +1 @@ +import multiple_registration # noqa