Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 1 | /* |
Dean Moldovan | a0c1ccf | 2016-08-12 13:50:00 +0200 | [diff] [blame] | 2 | tests/test_smart_ptr.cpp -- binding classes with custom reference counting, |
Wenzel Jakob | a576e6a | 2015-07-29 17:51:54 +0200 | [diff] [blame] | 3 | implicit conversions between types |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 4 | |
Wenzel Jakob | 8cb6cb3 | 2016-04-17 20:21:41 +0200 | [diff] [blame] | 5 | Copyright (c) 2016 Wenzel Jakob <wenzel.jakob@epfl.ch> |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 6 | |
| 7 | All rights reserved. Use of this source code is governed by a |
| 8 | BSD-style license that can be found in the LICENSE file. |
| 9 | */ |
| 10 | |
Dean Moldovan | b33475d | 2017-08-23 17:18:57 +0200 | [diff] [blame] | 11 | #if defined(_MSC_VER) && _MSC_VER < 1910 |
| 12 | # pragma warning(disable: 4702) // unreachable code in system header |
| 13 | #endif |
| 14 | |
Dean Moldovan | a0c1ccf | 2016-08-12 13:50:00 +0200 | [diff] [blame] | 15 | #include "pybind11_tests.h" |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 16 | #include "object.h" |
| 17 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 18 | // Make pybind aware of the ref-counted wrapper type (s): |
Wenzel Jakob | 2029171 | 2016-12-15 23:44:23 +0100 | [diff] [blame] | 19 | |
| 20 | // ref<T> is a wrapper for 'Object' which uses intrusive reference counting |
| 21 | // It is always possible to construct a ref<T> from an Object* pointer without |
luzpaz | 4b87461 | 2018-05-06 13:54:10 +0000 | [diff] [blame] | 22 | // possible inconsistencies, hence the 'true' argument at the end. |
Wenzel Jakob | 2029171 | 2016-12-15 23:44:23 +0100 | [diff] [blame] | 23 | PYBIND11_DECLARE_HOLDER_TYPE(T, ref<T>, true); |
Dean Moldovan | ec009a7 | 2017-01-31 17:05:44 +0100 | [diff] [blame] | 24 | // Make pybind11 aware of the non-standard getter member function |
| 25 | namespace pybind11 { namespace detail { |
| 26 | template <typename T> |
| 27 | struct holder_helper<ref<T>> { |
| 28 | static const T *get(const ref<T> &p) { return p.get_ptr(); } |
| 29 | }; |
Henry Schreiner | 8dc31c7 | 2020-09-10 21:16:40 -0400 | [diff] [blame] | 30 | } // namespace detail |
| 31 | } // namespace pybind11 |
Dean Moldovan | ec009a7 | 2017-01-31 17:05:44 +0100 | [diff] [blame] | 32 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 33 | // The following is not required anymore for std::shared_ptr, but it should compile without error: |
| 34 | PYBIND11_DECLARE_HOLDER_TYPE(T, std::shared_ptr<T>); |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 35 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 36 | // This is just a wrapper around unique_ptr, but with extra fields to deliberately bloat up the |
| 37 | // holder size to trigger the non-simple-layout internal instance layout for single inheritance with |
| 38 | // large holder type: |
| 39 | template <typename T> class huge_unique_ptr { |
| 40 | std::unique_ptr<T> ptr; |
| 41 | uint64_t padding[10]; |
| 42 | public: |
| 43 | huge_unique_ptr(T *p) : ptr(p) {}; |
| 44 | T *get() { return ptr.get(); } |
| 45 | }; |
| 46 | PYBIND11_DECLARE_HOLDER_TYPE(T, huge_unique_ptr<T>); |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 47 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 48 | // Simple custom holder that works like unique_ptr |
| 49 | template <typename T> |
| 50 | class custom_unique_ptr { |
| 51 | std::unique_ptr<T> impl; |
| 52 | public: |
| 53 | custom_unique_ptr(T* p) : impl(p) { } |
| 54 | T* get() const { return impl.get(); } |
| 55 | T* release_ptr() { return impl.release(); } |
| 56 | }; |
| 57 | PYBIND11_DECLARE_HOLDER_TYPE(T, custom_unique_ptr<T>); |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 58 | |
Khachajantc Michael | e3cb2a6 | 2018-06-20 18:33:50 +0300 | [diff] [blame] | 59 | // Simple custom holder that works like shared_ptr and has operator& overload |
| 60 | // To obtain address of an instance of this holder pybind should use std::addressof |
| 61 | // Attempt to get address via operator& may leads to segmentation fault |
| 62 | template <typename T> |
| 63 | class shared_ptr_with_addressof_operator { |
| 64 | std::shared_ptr<T> impl; |
| 65 | public: |
| 66 | shared_ptr_with_addressof_operator( ) = default; |
| 67 | shared_ptr_with_addressof_operator(T* p) : impl(p) { } |
| 68 | T* get() const { return impl.get(); } |
| 69 | T** operator&() { throw std::logic_error("Call of overloaded operator& is not expected"); } |
| 70 | }; |
| 71 | PYBIND11_DECLARE_HOLDER_TYPE(T, shared_ptr_with_addressof_operator<T>); |
| 72 | |
| 73 | // Simple custom holder that works like unique_ptr and has operator& overload |
| 74 | // To obtain address of an instance of this holder pybind should use std::addressof |
| 75 | // Attempt to get address via operator& may leads to segmentation fault |
| 76 | template <typename T> |
| 77 | class unique_ptr_with_addressof_operator { |
| 78 | std::unique_ptr<T> impl; |
| 79 | public: |
| 80 | unique_ptr_with_addressof_operator() = default; |
| 81 | unique_ptr_with_addressof_operator(T* p) : impl(p) { } |
| 82 | T* get() const { return impl.get(); } |
| 83 | T* release_ptr() { return impl.release(); } |
| 84 | T** operator&() { throw std::logic_error("Call of overloaded operator& is not expected"); } |
| 85 | }; |
| 86 | PYBIND11_DECLARE_HOLDER_TYPE(T, unique_ptr_with_addressof_operator<T>); |
| 87 | |
Wenzel Jakob | 6e213c9 | 2015-11-24 23:05:58 +0100 | [diff] [blame] | 88 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 89 | TEST_SUBMODULE(smart_ptr, m) { |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 90 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 91 | // test_smart_ptr |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 92 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 93 | // Object implementation in `object.h` |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 94 | py::class_<Object, ref<Object>> obj(m, "Object"); |
| 95 | obj.def("getRefCount", &Object::getRefCount); |
| 96 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 97 | // Custom object with builtin reference counting (see 'object.h' for the implementation) |
| 98 | class MyObject1 : public Object { |
| 99 | public: |
| 100 | MyObject1(int value) : value(value) { print_created(this, toString()); } |
Henry Schreiner | 5dfbe6f | 2020-09-10 22:43:53 -0400 | [diff] [blame] | 101 | std::string toString() const override { return "MyObject1[" + std::to_string(value) + "]"; } |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 102 | protected: |
Henry Schreiner | 5dfbe6f | 2020-09-10 22:43:53 -0400 | [diff] [blame] | 103 | ~MyObject1() override { print_destroyed(this); } |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 104 | private: |
| 105 | int value; |
| 106 | }; |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 107 | py::class_<MyObject1, ref<MyObject1>>(m, "MyObject1", obj) |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 108 | .def(py::init<int>()); |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 109 | py::implicitly_convertible<py::int_, MyObject1>(); |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 110 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 111 | m.def("make_object_1", []() -> Object * { return new MyObject1(1); }); |
| 112 | m.def("make_object_2", []() -> ref<Object> { return new MyObject1(2); }); |
| 113 | m.def("make_myobject1_1", []() -> MyObject1 * { return new MyObject1(4); }); |
| 114 | m.def("make_myobject1_2", []() -> ref<MyObject1> { return new MyObject1(5); }); |
| 115 | m.def("print_object_1", [](const Object *obj) { py::print(obj->toString()); }); |
| 116 | m.def("print_object_2", [](ref<Object> obj) { py::print(obj->toString()); }); |
| 117 | m.def("print_object_3", [](const ref<Object> &obj) { py::print(obj->toString()); }); |
| 118 | m.def("print_object_4", [](const ref<Object> *obj) { py::print((*obj)->toString()); }); |
| 119 | m.def("print_myobject1_1", [](const MyObject1 *obj) { py::print(obj->toString()); }); |
| 120 | m.def("print_myobject1_2", [](ref<MyObject1> obj) { py::print(obj->toString()); }); |
| 121 | m.def("print_myobject1_3", [](const ref<MyObject1> &obj) { py::print(obj->toString()); }); |
| 122 | m.def("print_myobject1_4", [](const ref<MyObject1> *obj) { py::print((*obj)->toString()); }); |
Wenzel Jakob | 2029171 | 2016-12-15 23:44:23 +0100 | [diff] [blame] | 123 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 124 | // Expose constructor stats for the ref type |
| 125 | m.def("cstats_ref", &ConstructorStats::get<ref_tag>); |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 126 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 127 | |
| 128 | // Object managed by a std::shared_ptr<> |
| 129 | class MyObject2 { |
| 130 | public: |
Francesco Biscani | ba33b2f | 2017-11-20 14:19:53 +0100 | [diff] [blame] | 131 | MyObject2(const MyObject2 &) = default; |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 132 | MyObject2(int value) : value(value) { print_created(this, toString()); } |
| 133 | std::string toString() const { return "MyObject2[" + std::to_string(value) + "]"; } |
| 134 | virtual ~MyObject2() { print_destroyed(this); } |
| 135 | private: |
| 136 | int value; |
| 137 | }; |
Wenzel Jakob | 6e213c9 | 2015-11-24 23:05:58 +0100 | [diff] [blame] | 138 | py::class_<MyObject2, std::shared_ptr<MyObject2>>(m, "MyObject2") |
| 139 | .def(py::init<int>()); |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 140 | m.def("make_myobject2_1", []() { return new MyObject2(6); }); |
| 141 | m.def("make_myobject2_2", []() { return std::make_shared<MyObject2>(7); }); |
| 142 | m.def("print_myobject2_1", [](const MyObject2 *obj) { py::print(obj->toString()); }); |
| 143 | m.def("print_myobject2_2", [](std::shared_ptr<MyObject2> obj) { py::print(obj->toString()); }); |
| 144 | m.def("print_myobject2_3", [](const std::shared_ptr<MyObject2> &obj) { py::print(obj->toString()); }); |
| 145 | m.def("print_myobject2_4", [](const std::shared_ptr<MyObject2> *obj) { py::print((*obj)->toString()); }); |
Wenzel Jakob | 6e213c9 | 2015-11-24 23:05:58 +0100 | [diff] [blame] | 146 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 147 | // Object managed by a std::shared_ptr<>, additionally derives from std::enable_shared_from_this<> |
| 148 | class MyObject3 : public std::enable_shared_from_this<MyObject3> { |
| 149 | public: |
Francesco Biscani | ba33b2f | 2017-11-20 14:19:53 +0100 | [diff] [blame] | 150 | MyObject3(const MyObject3 &) = default; |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 151 | MyObject3(int value) : value(value) { print_created(this, toString()); } |
| 152 | std::string toString() const { return "MyObject3[" + std::to_string(value) + "]"; } |
| 153 | virtual ~MyObject3() { print_destroyed(this); } |
| 154 | private: |
| 155 | int value; |
| 156 | }; |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 157 | py::class_<MyObject3, std::shared_ptr<MyObject3>>(m, "MyObject3") |
| 158 | .def(py::init<int>()); |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 159 | m.def("make_myobject3_1", []() { return new MyObject3(8); }); |
| 160 | m.def("make_myobject3_2", []() { return std::make_shared<MyObject3>(9); }); |
| 161 | m.def("print_myobject3_1", [](const MyObject3 *obj) { py::print(obj->toString()); }); |
| 162 | m.def("print_myobject3_2", [](std::shared_ptr<MyObject3> obj) { py::print(obj->toString()); }); |
| 163 | m.def("print_myobject3_3", [](const std::shared_ptr<MyObject3> &obj) { py::print(obj->toString()); }); |
| 164 | m.def("print_myobject3_4", [](const std::shared_ptr<MyObject3> *obj) { py::print((*obj)->toString()); }); |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 165 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 166 | // test_smart_ptr_refcounting |
| 167 | m.def("test_object1_refcounting", []() { |
| 168 | ref<MyObject1> o = new MyObject1(0); |
| 169 | bool good = o->getRefCount() == 1; |
| 170 | py::object o2 = py::cast(o, py::return_value_policy::reference); |
| 171 | // always request (partial) ownership for objects with intrusive |
| 172 | // reference counting even when using the 'reference' RVP |
| 173 | good &= o->getRefCount() == 2; |
| 174 | return good; |
| 175 | }); |
| 176 | |
| 177 | // test_unique_nodelete |
| 178 | // Object with a private destructor |
| 179 | class MyObject4 { |
| 180 | public: |
| 181 | MyObject4(int value) : value{value} { print_created(this); } |
| 182 | int value; |
| 183 | private: |
| 184 | ~MyObject4() { print_destroyed(this); } |
| 185 | }; |
Jason Rhinelander | a6495af | 2016-09-04 18:23:55 -0400 | [diff] [blame] | 186 | py::class_<MyObject4, std::unique_ptr<MyObject4, py::nodelete>>(m, "MyObject4") |
| 187 | .def(py::init<int>()) |
| 188 | .def_readwrite("value", &MyObject4::value); |
| 189 | |
Trevor Laughlin | 63c2a97 | 2018-11-11 13:36:55 -0500 | [diff] [blame] | 190 | // test_unique_deleter |
| 191 | // Object with std::unique_ptr<T, D> where D is not matching the base class |
| 192 | // Object with a protected destructor |
| 193 | class MyObject4a { |
| 194 | public: |
| 195 | MyObject4a(int i) { |
| 196 | value = i; |
| 197 | print_created(this); |
| 198 | }; |
| 199 | int value; |
| 200 | protected: |
| 201 | virtual ~MyObject4a() { print_destroyed(this); } |
| 202 | }; |
| 203 | py::class_<MyObject4a, std::unique_ptr<MyObject4a, py::nodelete>>(m, "MyObject4a") |
| 204 | .def(py::init<int>()) |
| 205 | .def_readwrite("value", &MyObject4a::value); |
| 206 | |
| 207 | // Object derived but with public destructor and no Deleter in default holder |
| 208 | class MyObject4b : public MyObject4a { |
| 209 | public: |
| 210 | MyObject4b(int i) : MyObject4a(i) { print_created(this); } |
Henry Schreiner | 5dfbe6f | 2020-09-10 22:43:53 -0400 | [diff] [blame] | 211 | ~MyObject4b() override { print_destroyed(this); } |
Trevor Laughlin | 63c2a97 | 2018-11-11 13:36:55 -0500 | [diff] [blame] | 212 | }; |
| 213 | py::class_<MyObject4b, MyObject4a>(m, "MyObject4b") |
| 214 | .def(py::init<int>()); |
| 215 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 216 | // test_large_holder |
| 217 | class MyObject5 { // managed by huge_unique_ptr |
| 218 | public: |
| 219 | MyObject5(int value) : value{value} { print_created(this); } |
| 220 | ~MyObject5() { print_destroyed(this); } |
| 221 | int value; |
| 222 | }; |
Jason Rhinelander | e45c211 | 2017-02-22 21:36:09 -0500 | [diff] [blame] | 223 | py::class_<MyObject5, huge_unique_ptr<MyObject5>>(m, "MyObject5") |
| 224 | .def(py::init<int>()) |
| 225 | .def_readwrite("value", &MyObject5::value); |
| 226 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 227 | // test_shared_ptr_and_references |
| 228 | struct SharedPtrRef { |
| 229 | struct A { |
| 230 | A() { print_created(this); } |
| 231 | A(const A &) { print_copy_created(this); } |
| 232 | A(A &&) { print_move_created(this); } |
| 233 | ~A() { print_destroyed(this); } |
| 234 | }; |
Jason Rhinelander | 3f58937 | 2016-08-07 13:05:26 -0400 | [diff] [blame] | 235 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 236 | A value = {}; |
| 237 | std::shared_ptr<A> shared = std::make_shared<A>(); |
Dean Moldovan | ab90ec6 | 2016-12-07 02:36:44 +0100 | [diff] [blame] | 238 | }; |
Dean Moldovan | ab90ec6 | 2016-12-07 02:36:44 +0100 | [diff] [blame] | 239 | using A = SharedPtrRef::A; |
| 240 | py::class_<A, std::shared_ptr<A>>(m, "A"); |
Dean Moldovan | ab90ec6 | 2016-12-07 02:36:44 +0100 | [diff] [blame] | 241 | py::class_<SharedPtrRef>(m, "SharedPtrRef") |
| 242 | .def(py::init<>()) |
| 243 | .def_readonly("ref", &SharedPtrRef::value) |
| 244 | .def_property_readonly("copy", [](const SharedPtrRef &s) { return s.value; }, |
| 245 | py::return_value_policy::copy) |
| 246 | .def_readonly("holder_ref", &SharedPtrRef::shared) |
| 247 | .def_property_readonly("holder_copy", [](const SharedPtrRef &s) { return s.shared; }, |
| 248 | py::return_value_policy::copy) |
| 249 | .def("set_ref", [](SharedPtrRef &, const A &) { return true; }) |
| 250 | .def("set_holder", [](SharedPtrRef &, std::shared_ptr<A>) { return true; }); |
| 251 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 252 | // test_shared_ptr_from_this_and_references |
| 253 | struct SharedFromThisRef { |
| 254 | struct B : std::enable_shared_from_this<B> { |
| 255 | B() { print_created(this); } |
| 256 | B(const B &) : std::enable_shared_from_this<B>() { print_copy_created(this); } |
| 257 | B(B &&) : std::enable_shared_from_this<B>() { print_move_created(this); } |
| 258 | ~B() { print_destroyed(this); } |
| 259 | }; |
| 260 | |
| 261 | B value = {}; |
| 262 | std::shared_ptr<B> shared = std::make_shared<B>(); |
| 263 | }; |
Dean Moldovan | ab90ec6 | 2016-12-07 02:36:44 +0100 | [diff] [blame] | 264 | using B = SharedFromThisRef::B; |
| 265 | py::class_<B, std::shared_ptr<B>>(m, "B"); |
Dean Moldovan | ab90ec6 | 2016-12-07 02:36:44 +0100 | [diff] [blame] | 266 | py::class_<SharedFromThisRef>(m, "SharedFromThisRef") |
| 267 | .def(py::init<>()) |
| 268 | .def_readonly("bad_wp", &SharedFromThisRef::value) |
| 269 | .def_property_readonly("ref", [](const SharedFromThisRef &s) -> const B & { return *s.shared; }) |
| 270 | .def_property_readonly("copy", [](const SharedFromThisRef &s) { return s.value; }, |
| 271 | py::return_value_policy::copy) |
| 272 | .def_readonly("holder_ref", &SharedFromThisRef::shared) |
| 273 | .def_property_readonly("holder_copy", [](const SharedFromThisRef &s) { return s.shared; }, |
| 274 | py::return_value_policy::copy) |
| 275 | .def("set_ref", [](SharedFromThisRef &, const B &) { return true; }) |
| 276 | .def("set_holder", [](SharedFromThisRef &, std::shared_ptr<B>) { return true; }); |
Dean Moldovan | ec009a7 | 2017-01-31 17:05:44 +0100 | [diff] [blame] | 277 | |
Jason Rhinelander | b8ac438 | 2017-05-19 13:34:55 -0400 | [diff] [blame] | 278 | // Issue #865: shared_from_this doesn't work with virtual inheritance |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 279 | struct SharedFromThisVBase : std::enable_shared_from_this<SharedFromThisVBase> { |
Francesco Biscani | ba33b2f | 2017-11-20 14:19:53 +0100 | [diff] [blame] | 280 | SharedFromThisVBase() = default; |
| 281 | SharedFromThisVBase(const SharedFromThisVBase &) = default; |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 282 | virtual ~SharedFromThisVBase() = default; |
| 283 | }; |
| 284 | struct SharedFromThisVirt : virtual SharedFromThisVBase {}; |
Jason Rhinelander | b8ac438 | 2017-05-19 13:34:55 -0400 | [diff] [blame] | 285 | static std::shared_ptr<SharedFromThisVirt> sft(new SharedFromThisVirt()); |
| 286 | py::class_<SharedFromThisVirt, std::shared_ptr<SharedFromThisVirt>>(m, "SharedFromThisVirt") |
| 287 | .def_static("get", []() { return sft.get(); }); |
| 288 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 289 | // test_move_only_holder |
Dean Moldovan | ec009a7 | 2017-01-31 17:05:44 +0100 | [diff] [blame] | 290 | struct C { |
| 291 | C() { print_created(this); } |
| 292 | ~C() { print_destroyed(this); } |
| 293 | }; |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 294 | py::class_<C, custom_unique_ptr<C>>(m, "TypeWithMoveOnlyHolder") |
Yannick Jadoul | d54d6d8 | 2020-07-01 01:53:09 +0200 | [diff] [blame] | 295 | .def_static("make", []() { return custom_unique_ptr<C>(new C); }) |
| 296 | .def_static("make_as_object", []() { return py::cast(custom_unique_ptr<C>(new C)); }); |
Dean Moldovan | ec009a7 | 2017-01-31 17:05:44 +0100 | [diff] [blame] | 297 | |
Khachajantc Michael | e3cb2a6 | 2018-06-20 18:33:50 +0300 | [diff] [blame] | 298 | // test_holder_with_addressof_operator |
| 299 | struct TypeForHolderWithAddressOf { |
| 300 | TypeForHolderWithAddressOf() { print_created(this); } |
| 301 | TypeForHolderWithAddressOf(const TypeForHolderWithAddressOf &) { print_copy_created(this); } |
| 302 | TypeForHolderWithAddressOf(TypeForHolderWithAddressOf &&) { print_move_created(this); } |
| 303 | ~TypeForHolderWithAddressOf() { print_destroyed(this); } |
| 304 | std::string toString() const { |
| 305 | return "TypeForHolderWithAddressOf[" + std::to_string(value) + "]"; |
| 306 | } |
| 307 | int value = 42; |
| 308 | }; |
| 309 | using HolderWithAddressOf = shared_ptr_with_addressof_operator<TypeForHolderWithAddressOf>; |
| 310 | py::class_<TypeForHolderWithAddressOf, HolderWithAddressOf>(m, "TypeForHolderWithAddressOf") |
| 311 | .def_static("make", []() { return HolderWithAddressOf(new TypeForHolderWithAddressOf); }) |
| 312 | .def("get", [](const HolderWithAddressOf &self) { return self.get(); }) |
| 313 | .def("print_object_1", [](const TypeForHolderWithAddressOf *obj) { py::print(obj->toString()); }) |
| 314 | .def("print_object_2", [](HolderWithAddressOf obj) { py::print(obj.get()->toString()); }) |
| 315 | .def("print_object_3", [](const HolderWithAddressOf &obj) { py::print(obj.get()->toString()); }) |
| 316 | .def("print_object_4", [](const HolderWithAddressOf *obj) { py::print((*obj).get()->toString()); }); |
| 317 | |
| 318 | // test_move_only_holder_with_addressof_operator |
| 319 | struct TypeForMoveOnlyHolderWithAddressOf { |
| 320 | TypeForMoveOnlyHolderWithAddressOf(int value) : value{value} { print_created(this); } |
| 321 | ~TypeForMoveOnlyHolderWithAddressOf() { print_destroyed(this); } |
| 322 | std::string toString() const { |
| 323 | return "MoveOnlyHolderWithAddressOf[" + std::to_string(value) + "]"; |
| 324 | } |
| 325 | int value; |
| 326 | }; |
| 327 | using MoveOnlyHolderWithAddressOf = unique_ptr_with_addressof_operator<TypeForMoveOnlyHolderWithAddressOf>; |
| 328 | py::class_<TypeForMoveOnlyHolderWithAddressOf, MoveOnlyHolderWithAddressOf>(m, "TypeForMoveOnlyHolderWithAddressOf") |
| 329 | .def_static("make", []() { return MoveOnlyHolderWithAddressOf(new TypeForMoveOnlyHolderWithAddressOf(0)); }) |
| 330 | .def_readwrite("value", &TypeForMoveOnlyHolderWithAddressOf::value) |
| 331 | .def("print_object", [](const TypeForMoveOnlyHolderWithAddressOf *obj) { py::print(obj->toString()); }); |
| 332 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 333 | // test_smart_ptr_from_default |
Dean Moldovan | cd3d1fc | 2017-03-20 21:53:24 +0100 | [diff] [blame] | 334 | struct HeldByDefaultHolder { }; |
Dean Moldovan | cd3d1fc | 2017-03-20 21:53:24 +0100 | [diff] [blame] | 335 | py::class_<HeldByDefaultHolder>(m, "HeldByDefaultHolder") |
| 336 | .def(py::init<>()) |
| 337 | .def_static("load_shared_ptr", [](std::shared_ptr<HeldByDefaultHolder>) {}); |
Dean Moldovan | bdfb50f | 2017-06-07 16:52:50 +0200 | [diff] [blame] | 338 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 339 | // test_shared_ptr_gc |
Dean Moldovan | bdfb50f | 2017-06-07 16:52:50 +0200 | [diff] [blame] | 340 | // #187: issue involving std::shared_ptr<> return value policy & garbage collection |
Henry Schreiner | ae951ca | 2019-04-06 19:09:39 +0200 | [diff] [blame] | 341 | struct ElementBase { |
| 342 | virtual ~ElementBase() { } /* Force creation of virtual table */ |
Henry Schreiner | e428a7f | 2020-07-23 21:16:54 -0400 | [diff] [blame] | 343 | ElementBase() = default; |
| 344 | ElementBase(const ElementBase&) = delete; |
Henry Schreiner | ae951ca | 2019-04-06 19:09:39 +0200 | [diff] [blame] | 345 | }; |
Dean Moldovan | bdfb50f | 2017-06-07 16:52:50 +0200 | [diff] [blame] | 346 | py::class_<ElementBase, std::shared_ptr<ElementBase>>(m, "ElementBase"); |
| 347 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 348 | struct ElementA : ElementBase { |
| 349 | ElementA(int v) : v(v) { } |
| 350 | int value() { return v; } |
| 351 | int v; |
| 352 | }; |
Dean Moldovan | bdfb50f | 2017-06-07 16:52:50 +0200 | [diff] [blame] | 353 | py::class_<ElementA, ElementBase, std::shared_ptr<ElementA>>(m, "ElementA") |
| 354 | .def(py::init<int>()) |
| 355 | .def("value", &ElementA::value); |
| 356 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 357 | struct ElementList { |
| 358 | void add(std::shared_ptr<ElementBase> e) { l.push_back(e); } |
| 359 | std::vector<std::shared_ptr<ElementBase>> l; |
| 360 | }; |
Dean Moldovan | bdfb50f | 2017-06-07 16:52:50 +0200 | [diff] [blame] | 361 | py::class_<ElementList, std::shared_ptr<ElementList>>(m, "ElementList") |
| 362 | .def(py::init<>()) |
| 363 | .def("add", &ElementList::add) |
| 364 | .def("get", [](ElementList &el) { |
| 365 | py::list list; |
| 366 | for (auto &e : el.l) |
| 367 | list.append(py::cast(e)); |
| 368 | return list; |
| 369 | }); |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame] | 370 | } |