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 | a0c1ccf | 2016-08-12 13:50:00 +0200 | [diff] [blame] | 11 | #include "pybind11_tests.h" |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 12 | #include "object.h" |
| 13 | |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 14 | /// Custom object with builtin reference counting (see 'object.h' for the implementation) |
| 15 | class MyObject1 : public Object { |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 16 | public: |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 17 | MyObject1(int value) : value(value) { |
Jason Rhinelander | 3f58937 | 2016-08-07 13:05:26 -0400 | [diff] [blame] | 18 | print_created(this, toString()); |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 19 | } |
| 20 | |
| 21 | std::string toString() const { |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 22 | return "MyObject1[" + std::to_string(value) + "]"; |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 23 | } |
| 24 | |
| 25 | protected: |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 26 | virtual ~MyObject1() { |
Jason Rhinelander | 3f58937 | 2016-08-07 13:05:26 -0400 | [diff] [blame] | 27 | print_destroyed(this); |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 28 | } |
| 29 | |
| 30 | private: |
| 31 | int value; |
| 32 | }; |
| 33 | |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 34 | /// Object managed by a std::shared_ptr<> |
| 35 | class MyObject2 { |
Wenzel Jakob | 6e213c9 | 2015-11-24 23:05:58 +0100 | [diff] [blame] | 36 | public: |
| 37 | MyObject2(int value) : value(value) { |
Jason Rhinelander | 3f58937 | 2016-08-07 13:05:26 -0400 | [diff] [blame] | 38 | print_created(this, toString()); |
Wenzel Jakob | 6e213c9 | 2015-11-24 23:05:58 +0100 | [diff] [blame] | 39 | } |
| 40 | |
| 41 | std::string toString() const { |
| 42 | return "MyObject2[" + std::to_string(value) + "]"; |
| 43 | } |
| 44 | |
| 45 | virtual ~MyObject2() { |
Jason Rhinelander | 3f58937 | 2016-08-07 13:05:26 -0400 | [diff] [blame] | 46 | print_destroyed(this); |
Wenzel Jakob | 6e213c9 | 2015-11-24 23:05:58 +0100 | [diff] [blame] | 47 | } |
| 48 | |
| 49 | private: |
| 50 | int value; |
| 51 | }; |
| 52 | |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 53 | /// Object managed by a std::shared_ptr<>, additionally derives from std::enable_shared_from_this<> |
| 54 | class MyObject3 : public std::enable_shared_from_this<MyObject3> { |
| 55 | public: |
| 56 | MyObject3(int value) : value(value) { |
Jason Rhinelander | 3f58937 | 2016-08-07 13:05:26 -0400 | [diff] [blame] | 57 | print_created(this, toString()); |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 58 | } |
| 59 | |
| 60 | std::string toString() const { |
| 61 | return "MyObject3[" + std::to_string(value) + "]"; |
| 62 | } |
| 63 | |
| 64 | virtual ~MyObject3() { |
Jason Rhinelander | 3f58937 | 2016-08-07 13:05:26 -0400 | [diff] [blame] | 65 | print_destroyed(this); |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 66 | } |
| 67 | |
| 68 | private: |
| 69 | int value; |
| 70 | }; |
| 71 | |
Jason Rhinelander | a6495af | 2016-09-04 18:23:55 -0400 | [diff] [blame] | 72 | class MyObject4 { |
| 73 | public: |
| 74 | MyObject4(int value) : value{value} { |
| 75 | print_created(this); |
| 76 | } |
| 77 | int value; |
| 78 | private: |
| 79 | ~MyObject4() { |
| 80 | print_destroyed(this); |
| 81 | } |
| 82 | }; |
| 83 | |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 84 | /// Make pybind aware of the ref-counted wrapper type (s) |
Wenzel Jakob | b1b7140 | 2015-10-18 16:48:30 +0200 | [diff] [blame] | 85 | PYBIND11_DECLARE_HOLDER_TYPE(T, ref<T>); |
Wenzel Jakob | 6e213c9 | 2015-11-24 23:05:58 +0100 | [diff] [blame] | 86 | PYBIND11_DECLARE_HOLDER_TYPE(T, std::shared_ptr<T>); |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 87 | |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 88 | Object *make_object_1() { return new MyObject1(1); } |
| 89 | ref<Object> make_object_2() { return new MyObject1(2); } |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 90 | |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 91 | MyObject1 *make_myobject1_1() { return new MyObject1(4); } |
| 92 | ref<MyObject1> make_myobject1_2() { return new MyObject1(5); } |
| 93 | |
| 94 | MyObject2 *make_myobject2_1() { return new MyObject2(6); } |
| 95 | std::shared_ptr<MyObject2> make_myobject2_2() { return std::make_shared<MyObject2>(7); } |
| 96 | |
| 97 | MyObject3 *make_myobject3_1() { return new MyObject3(8); } |
| 98 | std::shared_ptr<MyObject3> make_myobject3_2() { return std::make_shared<MyObject3>(9); } |
Wenzel Jakob | 6e213c9 | 2015-11-24 23:05:58 +0100 | [diff] [blame] | 99 | |
Dean Moldovan | 81511be | 2016-09-07 00:50:10 +0200 | [diff] [blame] | 100 | void print_object_1(const Object *obj) { py::print(obj->toString()); } |
| 101 | void print_object_2(ref<Object> obj) { py::print(obj->toString()); } |
| 102 | void print_object_3(const ref<Object> &obj) { py::print(obj->toString()); } |
| 103 | void print_object_4(const ref<Object> *obj) { py::print((*obj)->toString()); } |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 104 | |
Dean Moldovan | 81511be | 2016-09-07 00:50:10 +0200 | [diff] [blame] | 105 | void print_myobject1_1(const MyObject1 *obj) { py::print(obj->toString()); } |
| 106 | void print_myobject1_2(ref<MyObject1> obj) { py::print(obj->toString()); } |
| 107 | void print_myobject1_3(const ref<MyObject1> &obj) { py::print(obj->toString()); } |
| 108 | void print_myobject1_4(const ref<MyObject1> *obj) { py::print((*obj)->toString()); } |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 109 | |
Dean Moldovan | 81511be | 2016-09-07 00:50:10 +0200 | [diff] [blame] | 110 | void print_myobject2_1(const MyObject2 *obj) { py::print(obj->toString()); } |
| 111 | void print_myobject2_2(std::shared_ptr<MyObject2> obj) { py::print(obj->toString()); } |
| 112 | void print_myobject2_3(const std::shared_ptr<MyObject2> &obj) { py::print(obj->toString()); } |
| 113 | void 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] | 114 | |
Dean Moldovan | 81511be | 2016-09-07 00:50:10 +0200 | [diff] [blame] | 115 | void print_myobject3_1(const MyObject3 *obj) { py::print(obj->toString()); } |
| 116 | void print_myobject3_2(std::shared_ptr<MyObject3> obj) { py::print(obj->toString()); } |
| 117 | void print_myobject3_3(const std::shared_ptr<MyObject3> &obj) { py::print(obj->toString()); } |
| 118 | void 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] | 119 | |
Jason Rhinelander | 52f4be8 | 2016-09-03 14:54:22 -0400 | [diff] [blame] | 120 | test_initializer smart_ptr([](py::module &m) { |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 121 | py::class_<Object, ref<Object>> obj(m, "Object"); |
| 122 | obj.def("getRefCount", &Object::getRefCount); |
| 123 | |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 124 | py::class_<MyObject1, ref<MyObject1>>(m, "MyObject1", obj) |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 125 | .def(py::init<int>()); |
| 126 | |
| 127 | m.def("make_object_1", &make_object_1); |
| 128 | m.def("make_object_2", &make_object_2); |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 129 | m.def("make_myobject1_1", &make_myobject1_1); |
| 130 | m.def("make_myobject1_2", &make_myobject1_2); |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 131 | m.def("print_object_1", &print_object_1); |
| 132 | m.def("print_object_2", &print_object_2); |
| 133 | m.def("print_object_3", &print_object_3); |
| 134 | m.def("print_object_4", &print_object_4); |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 135 | m.def("print_myobject1_1", &print_myobject1_1); |
| 136 | m.def("print_myobject1_2", &print_myobject1_2); |
| 137 | m.def("print_myobject1_3", &print_myobject1_3); |
| 138 | m.def("print_myobject1_4", &print_myobject1_4); |
Wenzel Jakob | 38bd711 | 2015-07-05 20:05:44 +0200 | [diff] [blame] | 139 | |
Wenzel Jakob | 6e213c9 | 2015-11-24 23:05:58 +0100 | [diff] [blame] | 140 | py::class_<MyObject2, std::shared_ptr<MyObject2>>(m, "MyObject2") |
| 141 | .def(py::init<int>()); |
| 142 | m.def("make_myobject2_1", &make_myobject2_1); |
| 143 | m.def("make_myobject2_2", &make_myobject2_2); |
| 144 | m.def("print_myobject2_1", &print_myobject2_1); |
| 145 | m.def("print_myobject2_2", &print_myobject2_2); |
| 146 | m.def("print_myobject2_3", &print_myobject2_3); |
| 147 | m.def("print_myobject2_4", &print_myobject2_4); |
| 148 | |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 149 | py::class_<MyObject3, std::shared_ptr<MyObject3>>(m, "MyObject3") |
| 150 | .def(py::init<int>()); |
| 151 | m.def("make_myobject3_1", &make_myobject3_1); |
| 152 | m.def("make_myobject3_2", &make_myobject3_2); |
| 153 | m.def("print_myobject3_1", &print_myobject3_1); |
| 154 | m.def("print_myobject3_2", &print_myobject3_2); |
| 155 | m.def("print_myobject3_3", &print_myobject3_3); |
| 156 | m.def("print_myobject3_4", &print_myobject3_4); |
| 157 | |
Jason Rhinelander | a6495af | 2016-09-04 18:23:55 -0400 | [diff] [blame] | 158 | py::class_<MyObject4, std::unique_ptr<MyObject4, py::nodelete>>(m, "MyObject4") |
| 159 | .def(py::init<int>()) |
| 160 | .def_readwrite("value", &MyObject4::value); |
| 161 | |
Wenzel Jakob | b2c2c79 | 2016-01-17 22:36:40 +0100 | [diff] [blame] | 162 | py::implicitly_convertible<py::int_, MyObject1>(); |
Jason Rhinelander | 3f58937 | 2016-08-07 13:05:26 -0400 | [diff] [blame] | 163 | |
| 164 | // Expose constructor stats for the ref type |
| 165 | m.def("cstats_ref", &ConstructorStats::get<ref_tag>); |
Jason Rhinelander | 52f4be8 | 2016-09-03 14:54:22 -0400 | [diff] [blame] | 166 | }); |