-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsmf_control.hpp
173 lines (162 loc) · 6.07 KB
/
smf_control.hpp
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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
#pragma once
/*
synopsis
struct user_provided_t {
explicit user_provided_t() = default;
};
inline constexpr user_provided_t user_provided{};
template<bool cond, class T> struct default_copy_ctor_if;
template<bool cond, class T> struct default_move_ctor_if;
template<bool cond, class T> struct default_copy_assign_if;
template<bool cond, class T> struct default_move_assign_if;
template<bool cond, class T> struct delete_copy_ctor_if;
template<bool cond, class T> struct delete_move_ctor_if;
template<bool cond, class T> struct delete_copy_assign_if;
template<bool cond, class T> struct delete_move_assign_if;
*/
struct user_provided_t {
explicit user_provided_t() = default;
};
inline constexpr user_provided_t user_provided{};
// All the following class templates inherit from the template parameter T and
// introduce T's constructors and assignment operators into their own scope
// via using declaration.
// All their special member functions are defaulted except for the one
// corresponding to the template name.
// For each of default_*_if, the corresponding special member function is
// defaulted if the first template argument is not false, otherwise it is
// user-provided and calls T's constructor (if the special member function is
// a constructor) or T's member function named assign (if the special member
// function is operator=), with user_provided being the first function
// argument and the argument to the special member function being the second.
// For each of delete_*_if, the corresponding special member function is
// deleted if the first template argument is true, otherwise it is defaulted.
// Note that a defaulted function might still be implicitly deleted.
template<bool cond, class T>
struct default_copy_ctor_if : T {
using T::T;
using T::operator=;
};
template<class T>
struct default_copy_ctor_if<false, T> : T {
using T::T;
using T::operator=;
default_copy_ctor_if() = default;
constexpr default_copy_ctor_if(const default_copy_ctor_if& that)
: T(user_provided, that) {}
default_copy_ctor_if(default_copy_ctor_if&&) = default;
default_copy_ctor_if& operator=(const default_copy_ctor_if&) = default;
default_copy_ctor_if& operator=(default_copy_ctor_if&&) = default;
};
template<bool cond, class T>
struct default_move_ctor_if : T {
using T::T;
using T::operator=;
};
template<class T>
struct default_move_ctor_if<false, T> : T {
using T::T;
using T::operator=;
default_move_ctor_if() = default;
default_move_ctor_if(const default_move_ctor_if&) = default;
constexpr default_move_ctor_if(default_move_ctor_if&& that)
: T(user_provided, static_cast<decltype(that)>(that)) {}
default_move_ctor_if& operator=(const default_move_ctor_if&) = default;
default_move_ctor_if& operator=(default_move_ctor_if&&) = default;
};
template<bool cond, class T>
struct default_copy_assign_if : T {
using T::T;
using T::operator=;
};
template<class T>
struct default_copy_assign_if<false, T> : T {
using T::T;
using T::operator=;
default_copy_assign_if() = default;
default_copy_assign_if(const default_copy_assign_if&) = default;
default_copy_assign_if(default_copy_assign_if&&) = default;
constexpr auto& operator=(const default_copy_assign_if& that) {
T::assign(user_provided, that);
return *this;
}
default_copy_assign_if& operator=(default_copy_assign_if&&) = default;
};
template<bool cond, class T>
struct default_move_assign_if : T {
using T::T;
using T::operator=;
};
template<class T>
struct default_move_assign_if<false, T> : T {
using T::T;
using T::operator=;
default_move_assign_if() = default;
default_move_assign_if(const default_move_assign_if&) = default;
default_move_assign_if(default_move_assign_if&&) = default;
default_move_assign_if& operator=(const default_move_assign_if&) = default;
constexpr auto& operator=(default_move_assign_if&& that) {
T::assign(user_provided, static_cast<decltype(that)>(that));
return *this;
}
};
template<bool cond, class T>
struct delete_copy_ctor_if : T {
using T::T;
using T::operator=;
};
template<class T>
struct delete_copy_ctor_if<true, T> : T {
using T::T;
using T::operator=;
delete_copy_ctor_if() = default;
delete_copy_ctor_if(const delete_copy_ctor_if&) = delete;
delete_copy_ctor_if(delete_copy_ctor_if&&) = default;
delete_copy_ctor_if& operator=(const delete_copy_ctor_if&) = default;
delete_copy_ctor_if& operator=(delete_copy_ctor_if&&) = default;
};
template<bool cond, class T>
struct delete_move_ctor_if : T {
using T::T;
using T::operator=;
};
template<class T>
struct delete_move_ctor_if<true, T> : T {
using T::T;
using T::operator=;
delete_move_ctor_if() = default;
delete_move_ctor_if(const delete_move_ctor_if&) = default;
delete_move_ctor_if(delete_move_ctor_if&&) = delete;
delete_move_ctor_if& operator=(const delete_move_ctor_if&) = default;
delete_move_ctor_if& operator=(delete_move_ctor_if&&) = default;
};
template<bool cond, class T>
struct delete_copy_assign_if : T {
using T::T;
using T::operator=;
};
template<class T>
struct delete_copy_assign_if<true, T> : T {
using T::T;
using T::operator=;
delete_copy_assign_if() = default;
delete_copy_assign_if(const delete_copy_assign_if&) = default;
delete_copy_assign_if(delete_copy_assign_if&&) = default;
delete_copy_assign_if& operator=(const delete_copy_assign_if&) = delete;
delete_copy_assign_if& operator=(delete_copy_assign_if&&) = default;
};
template<bool cond, class T>
struct delete_move_assign_if : T {
using T::T;
using T::operator=;
};
template<class T>
struct delete_move_assign_if<true, T> : T {
using T::T;
using T::operator=;
delete_move_assign_if() = default;
delete_move_assign_if(const delete_move_assign_if&) = default;
delete_move_assign_if(delete_move_assign_if&&) = default;
delete_move_assign_if& operator=(const delete_move_assign_if&) = default;
delete_move_assign_if& operator=(delete_move_assign_if&&) = delete;
};