-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrecursive_variant.hpp
310 lines (232 loc) · 5.85 KB
/
recursive_variant.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
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
#pragma once
#include <concepts>
#include <memory>
#include <type_traits>
#include <utility>
#include <variant>
namespace detail
{
template<class T>
class box
{
public:
box(const T& value)
: value_(std::make_unique<T>(value))
{}
box(T&& value)
: value_(std::make_unique<T>(std::move(value)))
{}
box(box&&) = default;
box(const box& other)
: box(other.value())
{}
box& operator=(box&&) = default;
box& operator=(const box& other)
{
value() = other.value();
return *this;
}
T& value() &
{
return *value_;
}
const T& value() const &
{
return *value_;
}
T&& value() &&
{
return std::move(*value_);
}
private:
std::unique_ptr<T> value_;
};
template<class T>
struct is_box : std::false_type {};
template<class T>
struct is_box<box<T>> : std::true_type {};
// see https://stackoverflow.com/a/1956217/722294
namespace
{
template<class T, int discriminator>
struct is_complete {
template<class U, int = sizeof(U)>
static std::true_type test(int);
template<class>
static std::false_type test(...);
static const bool value = decltype(test<T>(0))::value;
};
} // end anonymous namespace
#define IS_COMPLETE(X) detail::is_complete<X,__COUNTER__>::value
template<class T>
using box_if_incomplete_t = std::conditional_t<IS_COMPLETE(T), T, box<T>>;
template<class Function>
struct unbox_and_call
{
Function f_;
template<class Arg>
static decltype(auto) unbox(Arg&& arg)
{
using T = std::remove_cvref_t<Arg>;
if constexpr (is_box<T>::value)
{
return std::forward<Arg>(arg).value();
}
else
{
// arg isn't a box, just forward it
return std::forward<Arg>(arg);
}
}
// forward unboxxed arguments to f
template<class... Args>
decltype(auto) operator()(Args&&... args) const
{
return std::forward<Function>(f_)(unbox(std::forward<Args>(args))...);
}
};
template<class T>
constexpr bool is_one_of_impl()
{
return false;
}
template<class T, class Type1, class... Types>
constexpr bool is_one_of_impl()
{
if constexpr (std::same_as<T,Type1>)
{
return true;
}
else
{
return is_one_of_impl<T,Types...>();
}
}
template<class T, class... Types>
concept is_one_of = is_one_of_impl<T,Types...>();
} // end detail
template<class... Types>
class recursive_variant : public std::variant<detail::box_if_incomplete_t<Types>...>
{
public:
using super_t = std::variant<detail::box_if_incomplete_t<Types>...>;
using super_t::super_t;
// add a boxing constructor for types that need to be boxed
template<class U>
requires (detail::is_one_of<detail::box<std::remove_cvref_t<U>>,Types...> and
std::constructible_from<U,U&&>)
recursive_variant(U&& value)
: super_t(detail::box<std::remove_cvref_t<U>>(std::forward<U>(value)))
{}
};
namespace detail
{
template<class Variant>
constexpr Variant&& super(Variant&& var)
{
return std::forward<Variant>(var);
}
template<class... Types>
constexpr typename recursive_variant<Types...>::super_t& super(recursive_variant<Types...>& var)
{
return var;
}
template<class... Types>
constexpr const typename recursive_variant<Types...>::super_t& super(const recursive_variant<Types...>& var)
{
return var;
}
template<class... Types>
constexpr typename recursive_variant<Types...>::super_t&& super(recursive_variant<Types...>&& var)
{
return std::move(var);
}
template<class T>
constexpr std::size_t index_of()
{
return -1;
}
template<class T, class Head, class... Tail>
constexpr std::size_t index_of()
{
if constexpr (std::same_as<T,Head>)
{
return 0;
}
else
{
return 1 + index_of<T,Tail...>();
}
}
template<class T>
constexpr std::size_t count_occurrences()
{
return 0;
}
template<class T, class Head, class... Tail>
constexpr std::size_t count_occurrences()
{
if constexpr (std::same_as<T,Head>)
{
return 1 + count_occurrences<T,Tail...>();
}
return count_occurrences<T,Tail...>();
}
template<class T, class... Types>
constexpr bool occurs_exactly_once()
{
return 1 == count_occurrences<T,Types...>();
}
template<class T>
struct get_visitor
{
T& operator()(T& arg) const
{
return arg;
}
const T& operator()(const T& arg) const
{
return arg;
}
template<class U>
const T& operator()(const U& arg) const
{
throw std::bad_variant_access();
return *reinterpret_cast<const T*>(&arg);
}
template<class U>
T& operator()(U& arg) const
{
throw std::bad_variant_access();
return *reinterpret_cast<T*>(&arg);
}
};
} // end detail
template<class Visitor, class Variant, class... Variants>
constexpr decltype(auto) visit(Visitor&& visitor, Variant&& var, Variants&&... vars)
{
// unbox boxxed types before the visitor sees them
detail::unbox_and_call<Visitor&&> unboxing_visitor{std::forward<Visitor>(visitor)};
// unbox recursive_variants into std::variant before passing to std::visit
return std::visit(unboxing_visitor, detail::super(std::forward<Variant>(var)), detail::super(std::forward<Variants>(vars))...);
}
template<class T, class... Types>
constexpr bool holds_alternative(const recursive_variant<Types...>& var)
{
static_assert(detail::occurs_exactly_once<T,Types...>(), "T must occur exactly once in alternatives");
return detail::index_of<T,Types...>() == var.index();
}
template<class T, class... Types>
constexpr T& get(recursive_variant<Types...>& var)
{
static_assert(detail::occurs_exactly_once<T,Types...>(), "T must occur exactly once in alternatives");
return visit(detail::get_visitor<T>{}, var);
}
template<class T, class... Types>
constexpr const T& get(const recursive_variant<Types...>& var)
{
static_assert(detail::occurs_exactly_once<T,Types...>(), "T must occur exactly once in alternatives");
return visit(detail::get_visitor<T>{}, var);
}
// clean up after ourself
#undef IS_COMPLETE