summaryrefslogtreecommitdiffstats
path: root/src/boost/libs/hana/test/pair/cnstr.move.cpp
blob: cb13e724382b593b6c7a4a44fcdbf5e1f53414bc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
// Copyright Louis Dionne 2013-2017
// Distributed under the Boost Software License, Version 1.0.
// (See accompanying file LICENSE.md or copy at http://boost.org/LICENSE_1_0.txt)

#include <boost/hana/assert.hpp>
#include <boost/hana/first.hpp>
#include <boost/hana/pair.hpp>
#include <boost/hana/second.hpp>

#include <type_traits>
#include <utility>
namespace hana = boost::hana;


struct MoveOnly {
    int data_;
    MoveOnly(MoveOnly const&) = delete;
    MoveOnly& operator=(MoveOnly const&) = delete;
    MoveOnly(int data) : data_(data) { }
    MoveOnly(MoveOnly&& x) : data_(x.data_) { x.data_ = 0; }

    MoveOnly& operator=(MoveOnly&& x)
    { data_ = x.data_; x.data_ = 0; return *this; }

    bool operator==(const MoveOnly& x) const { return data_ == x.data_; }
};

struct MoveOnlyDerived : MoveOnly {
    MoveOnlyDerived(MoveOnlyDerived&&) = default;
    MoveOnlyDerived(int data = 1) : MoveOnly(data) { }
};

template <typename Target>
struct implicit_to {
    constexpr operator Target() const { return Target{}; }
};

struct NoMove {
    NoMove() = default;
    NoMove(NoMove const&) = delete;
    NoMove(NoMove&&) = delete;
};

// Note: It is also useful to check with a non-empty class, because that
//       triggers different instantiations due to EBO.
struct NoMove_nonempty {
    NoMove_nonempty() = default;
    NoMove_nonempty(NoMove_nonempty const&) = delete;
    NoMove_nonempty(NoMove_nonempty&&) = delete;
    int i;
};

int main() {
    {
        hana::pair<MoveOnly, short> p1(MoveOnly{3}, 4);
        hana::pair<MoveOnly, short> p2(std::move(p1));
        BOOST_HANA_RUNTIME_CHECK(hana::first(p2) == MoveOnly{3});
        BOOST_HANA_RUNTIME_CHECK(hana::second(p2) == 4);
    }

    // Make sure it works across pair types
    {
        hana::pair<MoveOnlyDerived, short> p1(MoveOnlyDerived{3}, 4);
        hana::pair<MoveOnly, long> p2 = std::move(p1);
        BOOST_HANA_RUNTIME_CHECK(hana::first(p2) == MoveOnly{3});
        BOOST_HANA_RUNTIME_CHECK(hana::second(p2) == 4);
    }
    {
        struct target1 {
            target1() = default;
            target1(target1 const&) = delete;
            target1(target1&&) = default;
        };

        struct target2 {
            target2() = default;
            target2(target2 const&) = delete;
            target2(target2&&) = default;
        };
        using Target = hana::pair<target1, target2>;
        Target p1(hana::make_pair(target1{}, target2{})); (void)p1;
        Target p2(hana::make_pair(implicit_to<target1>{}, target2{})); (void)p2;
        Target p3(hana::make_pair(target1{}, implicit_to<target2>{})); (void)p3;
        Target p4(hana::make_pair(implicit_to<target1>{}, implicit_to<target2>{})); (void)p4;
    }

    // Make sure we don't define the move constructor when it shouldn't be defined.
    {
        using Pair1 = hana::pair<NoMove, NoMove>;
        Pair1 pair1; (void)pair1;
        static_assert(!std::is_move_constructible<Pair1>::value, "");

        using Pair2 = hana::pair<NoMove_nonempty, NoMove_nonempty>;
        Pair2 pair2; (void)pair2;
        static_assert(!std::is_move_constructible<Pair2>::value, "");
    }
}