test_unique_ptr_member.cpp 2.46 KB
Newer Older
1
2
#include "pybind11_tests.h"

3
#include <iostream>
4
5
6
7
8
#include <memory>

namespace pybind11_tests {
namespace unique_ptr_member {

9
10
inline void to_cout(std::string text) { std::cout << text << std::endl; }

11
12
13
14
class pointee { // NOT copyable.
  public:
    pointee() = default;

15
16
17
18
19
20
    int get_int() const {
        to_cout("pointee::get_int()");
        return 213;
    }

    ~pointee() { to_cout("~pointee()"); }
21
22
23
24
25
26
27
28

  private:
    pointee(const pointee &) = delete;
    pointee(pointee &&) = delete;
    pointee &operator=(const pointee &) = delete;
    pointee &operator=(pointee &&) = delete;
};

29
30
31
32
inline std::unique_ptr<pointee> make_unique_pointee() {
    return std::unique_ptr<pointee>(new pointee);
}

33
34
35
36
class ptr_owner {
  public:
    explicit ptr_owner(std::unique_ptr<pointee> ptr) : ptr_(std::move(ptr)) {}

37
38
39
40
41
42
43
44
45
    bool is_owner() const { return bool(ptr_); }

    std::unique_ptr<pointee> give_up_ownership_via_unique_ptr() {
        return std::move(ptr_);
    }
    std::shared_ptr<pointee> give_up_ownership_via_shared_ptr() {
        return std::move(ptr_);
    }

46
47
48
49
50
51
  private:
    std::unique_ptr<pointee> ptr_;
};

// Just to have a minimal example of a typical C++ pattern.
inline int cpp_pattern() {
52
53
54
    auto obj = make_unique_pointee();
    int result = (obj ? 1 : 8);
    obj->get_int();
55
    ptr_owner owner(std::move(obj));
56
57
58
59
60
61
62
63
64
65
    result = result * 10 + (obj ? 8 : 1);
    result = result * 10 + (owner.is_owner() ? 1 : 8);
    to_cout("before give up");
    auto reclaimed = owner.give_up_ownership_via_shared_ptr();
    to_cout("after give up");
    result = result * 10 + (owner.is_owner() ? 8 : 1);
    result = result * 10 + (reclaimed ? 1 : 8);
    reclaimed.reset();
    to_cout("after del");
    result = result * 10 + (reclaimed ? 8 : 1);
66
67
68
69
    return result;
}

TEST_SUBMODULE(unique_ptr_member, m) {
70
    m.def("to_cout", to_cout);
71

72
    py::class_<pointee, std::shared_ptr<pointee>>(m, "pointee")
73
74
75
        .def(py::init<>())
        .def("get_int", &pointee::get_int);

76
77
    m.def("make_unique_pointee", make_unique_pointee);

78
    py::class_<ptr_owner>(m, "ptr_owner")
79
80
81
82
83
84
85
86
        //.def(py::init<std::unique_ptr<pointee>>(), py::arg("ptr"))
        .def("is_owner", &ptr_owner::is_owner)
        .def("give_up_ownership_via_unique_ptr",
             &ptr_owner::give_up_ownership_via_unique_ptr)
        .def("give_up_ownership_via_shared_ptr",
             &ptr_owner::give_up_ownership_via_shared_ptr);

    m.def("cpp_pattern", cpp_pattern);
87
88
89
90
}

} // namespace unique_ptr_member
} // namespace pybind11_tests