Project
Loading...
Searching...
No Matches
StructToTuple.h
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11#ifndef O2_FRAMEWORK_STRUCTTOTUPLE_H_
12#define O2_FRAMEWORK_STRUCTTOTUPLE_H_
13
14#include <Framework/Traits.h>
15#include <array>
16
17namespace
18{
19template <class T, typename... Args>
20decltype(void(T{std::declval<Args>()...}), std::true_type())
21 brace_test(int);
22
23template <class T, typename... Args>
24std::false_type
25 brace_test(...);
26} // namespace
27
28namespace o2::framework
29{
30struct any_type {
31 template <class T>
32 constexpr operator T(); // non explicit
33};
34
35template <class T, typename... Args>
36struct is_braces_constructible : decltype(brace_test<T, Args...>(0)) {
37};
38
39#define DPL_REPEAT_0(x)
40#define DPL_REPEAT_1(x) x
41#define DPL_REPEAT_2(x) x, x
42#define DPL_REPEAT_3(x) x, x, x
43#define DPL_REPEAT_4(x) x, x, x, x
44#define DPL_REPEAT_5(x) x, x, x, x, x
45#define DPL_REPEAT_6(x) x, x, x, x, x, x
46#define DPL_REPEAT_7(x) x, x, x, x, x, x, x
47#define DPL_REPEAT_8(x) x, x, x, x, x, x, x, x
48#define DPL_REPEAT_9(x) x, x, x, x, x, x, x, x, x
49#define DPL_REPEAT_10(x) x, x, x, x, x, x, x, x, x, x
50#define DPL_REPEAT(x, d, u) DPL_REPEAT_##d(DPL_REPEAT_10(x)), DPL_REPEAT_##u(x)
51
52#define DPL_ENUM_0(pre, post)
53#define DPL_ENUM_1(pre, post) pre##0##post
54#define DPL_ENUM_2(pre, post) pre##0##post, pre##1##post
55#define DPL_ENUM_3(pre, post) pre##0##post, pre##1##post, pre##2##post
56#define DPL_ENUM_4(pre, post) pre##0##post, pre##1##post, pre##2##post, pre##3##post
57#define DPL_ENUM_5(pre, post) pre##0##post, pre##1##post, pre##2##post, pre##3##post, pre##4##post
58#define DPL_ENUM_6(pre, post) pre##0##post, pre##1##post, pre##2##post, pre##3##post, pre##4##post, pre##5##post
59#define DPL_ENUM_7(pre, post) pre##0##post, pre##1##post, pre##2##post, pre##3##post, pre##4##post, pre##5##post, pre##6##post
60#define DPL_ENUM_8(pre, post) pre##0##post, pre##1##post, pre##2##post, pre##3##post, pre##4##post, pre##5##post, pre##6##post, pre##7##post
61#define DPL_ENUM_9(pre, post) pre##0##post, pre##1##post, pre##2##post, pre##3##post, pre##4##post, pre##5##post, pre##6##post, pre##7##post, pre##8##post
62#define DPL_ENUM_10(pre, post) pre##0##post, pre##1##post, pre##2##post, pre##3##post, pre##4##post, pre##5##post, pre##6##post, pre##7##post, pre##8##post, pre##9##post
63
64#define DPL_ENUM_20(pre, post) DPL_ENUM_10(pre, post), DPL_ENUM_10(pre##1, post)
65#define DPL_ENUM_30(pre, post) DPL_ENUM_20(pre, post), DPL_ENUM_10(pre##2, post)
66#define DPL_ENUM_40(pre, post) DPL_ENUM_30(pre, post), DPL_ENUM_10(pre##3, post)
67#define DPL_ENUM_50(pre, post) DPL_ENUM_40(pre, post), DPL_ENUM_10(pre##4, post)
68#define DPL_ENUM_60(pre, post) DPL_ENUM_50(pre, post), DPL_ENUM_10(pre##5, post)
69#define DPL_ENUM_70(pre, post) DPL_ENUM_60(pre, post), DPL_ENUM_10(pre##6, post)
70#define DPL_ENUM_80(pre, post) DPL_ENUM_70(pre, post), DPL_ENUM_10(pre##7, post)
71#define DPL_ENUM_90(pre, post) DPL_ENUM_80(pre, post), DPL_ENUM_10(pre##8, post)
72#define DPL_ENUM_100(pre, post) DPL_ENUM_90(pre, post), DPL_ENUM_10(pre##9, post)
73
74#define DPL_ENUM(pre, post, d, u) DPL_ENUM_##d##0(pre, post), DPL_ENUM_##u(pre##d, post)
75
76#define DPL_FENUM_0(f, pre, post)
77#define DPL_FENUM_1(f, pre, post) f(pre##0##post)
78#define DPL_FENUM_2(f, pre, post) f(pre##0##post), f(pre##1##post)
79#define DPL_FENUM_3(f, pre, post) f(pre##0##post), f(pre##1##post), f(pre##2##post)
80#define DPL_FENUM_4(f, pre, post) f(pre##0##post), f(pre##1##post), f(pre##2##post), f(pre##3##post)
81#define DPL_FENUM_5(f, pre, post) f(pre##0##post), f(pre##1##post), f(pre##2##post), f(pre##3##post), f(pre##4##post)
82#define DPL_FENUM_6(f, pre, post) f(pre##0##post), f(pre##1##post), f(pre##2##post), f(pre##3##post), f(pre##4##post), f(pre##5##post)
83#define DPL_FENUM_7(f, pre, post) f(pre##0##post), f(pre##1##post), f(pre##2##post), f(pre##3##post), f(pre##4##post), f(pre##5##post), f(pre##6##post)
84#define DPL_FENUM_8(f, pre, post) f(pre##0##post), f(pre##1##post), f(pre##2##post), f(pre##3##post), f(pre##4##post), f(pre##5##post), f(pre##6##post), f(pre##7##post)
85#define DPL_FENUM_9(f, pre, post) f(pre##0##post), f(pre##1##post), f(pre##2##post), f(pre##3##post), f(pre##4##post), f(pre##5##post), f(pre##6##post), f(pre##7##post), f(pre##8##post)
86#define DPL_FENUM_10(f, pre, post) f(pre##0##post), f(pre##1##post), f(pre##2##post), f(pre##3##post), f(pre##4##post), f(pre##5##post), f(pre##6##post), f(pre##7##post), f(pre##8##post), f(pre##9##post)
87
88#define DPL_FENUM_20(f, pre, post) DPL_FENUM_10(f, pre, post), DPL_FENUM_10(f, pre##1, post)
89#define DPL_FENUM_30(f, pre, post) DPL_FENUM_20(f, pre, post), DPL_FENUM_10(f, pre##2, post)
90#define DPL_FENUM_40(f, pre, post) DPL_FENUM_30(f, pre, post), DPL_FENUM_10(f, pre##3, post)
91#define DPL_FENUM_50(f, pre, post) DPL_FENUM_40(f, pre, post), DPL_FENUM_10(f, pre##4, post)
92#define DPL_FENUM_60(f, pre, post) DPL_FENUM_50(f, pre, post), DPL_FENUM_10(f, pre##5, post)
93#define DPL_FENUM_70(f, pre, post) DPL_FENUM_60(f, pre, post), DPL_FENUM_10(f, pre##6, post)
94#define DPL_FENUM_80(f, pre, post) DPL_FENUM_70(f, pre, post), DPL_FENUM_10(f, pre##7, post)
95#define DPL_FENUM_90(f, pre, post) DPL_FENUM_80(f, pre, post), DPL_FENUM_10(f, pre##8, post)
96#define DPL_FENUM_100(f, pre, post) DPL_FENUM_90(f, pre, post), DPL_FENUM_10(f, pre##9, post)
97
98#define DPL_FENUM(f, pre, post, d, u) DPL_FENUM_##d##0(f, pre, post), DPL_FENUM_##u(f, pre##d, post)
99
100#define DPL_10_As DPL_REPEAT_10(A)
101#define DPL_20_As DPL_10_As, DPL_10_As
102#define DPL_30_As DPL_20_As, DPL_10_As
103#define DPL_40_As DPL_30_As, DPL_10_As
104#define DPL_50_As DPL_40_As, DPL_10_As
105#define DPL_60_As DPL_50_As, DPL_10_As
106#define DPL_70_As DPL_60_As, DPL_10_As
107#define DPL_80_As DPL_70_As, DPL_10_As
108#define DPL_90_As DPL_80_As, DPL_10_As
109#define DPL_100_As DPL_90_As, DPL_10_As
110
111#define DPL_0_9(pre, po) pre##0##po, pre##1##po, pre##2##po, pre##3##po, pre##4##po, pre##5##po, pre##6##po, pre##7##po, pre##8##po, pre##9##po
112
113#define BRACE_CONSTRUCTIBLE_ENTRY_LOW(u) \
114 constexpr(is_braces_constructible<type, DPL_REPEAT_##u(A)>{}) \
115 { \
116 return u; \
117 }
118#define BRACE_CONSTRUCTIBLE_ENTRY(d, u) \
119 constexpr(is_braces_constructible<type, DPL_REPEAT(A, d, u)>{}) \
120 { \
121 return d##u; \
122 }
123
124#define BRACE_CONSTRUCTIBLE_ENTRY_TENS(d) \
125 constexpr(is_braces_constructible<type, DPL_##d##0_As>{}) \
126 { \
127 return d##0; \
128 }
129
131 template <typename T>
132 operator T()
133 {
134 }
135};
136
137template <typename T>
138consteval auto brace_constructible_size(auto... Members)
139{
140 if constexpr (requires { T{Members...}; } == false) {
141 static_assert(sizeof...(Members) != 0, "You need to make sure that you have implicit constructors or that you call the explicit constructor correctly.");
142 return sizeof...(Members) - 1;
143 } else {
144 return brace_constructible_size<T>(Members..., UniversalType{});
145 }
146}
147
148#define DPL_HOMOGENEOUS_APPLY_ENTRY_LOW(u) \
149 constexpr(numElements == u) \
150 { \
151 auto&& [DPL_ENUM_##u(p, )] = object; \
152 return std::array<decltype(l(p0)), u>{DPL_FENUM_##u(l, p, )}; \
153 }
154
155#define DPL_HOMOGENEOUS_APPLY_ENTRY(d, u) \
156 constexpr(numElements == d##u) \
157 { \
158 auto&& [DPL_ENUM(p, , d, u)] = object; \
159 return std::array<decltype(l(p0)), d##u>{DPL_FENUM(l, p, , d, u)}; \
160 }
161
162#define DPL_HOMOGENEOUS_APPLY_ENTRY_TENS(d) \
163 constexpr(numElements == d##0) \
164 { \
165 auto&& [DPL_ENUM_##d##0(p, )] = object; \
166 return std::array<decltype(l(p0)), d##0>{DPL_FENUM_##d##0(l, p, )}; \
167 }
168
169template <bool B, typename T>
171{
172 using type = std::decay_t<T>;
173 constexpr int nesting = B ? 1 : 0;
174 return brace_constructible_size<type>() - nesting;
175}
176
177template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
178 requires(D == 9)
179auto homogeneous_apply_refs(L l, T&& object)
180{
181 constexpr int numElements = nested_brace_constructible_size<B, T>();
182 // clang-format off
184 else if DPL_HOMOGENEOUS_APPLY_ENTRY (9, 8)
185 else if DPL_HOMOGENEOUS_APPLY_ENTRY (9, 7)
186 else if DPL_HOMOGENEOUS_APPLY_ENTRY (9, 6)
187 else if DPL_HOMOGENEOUS_APPLY_ENTRY (9, 5)
188 else if DPL_HOMOGENEOUS_APPLY_ENTRY (9, 4)
189 else if DPL_HOMOGENEOUS_APPLY_ENTRY (9, 3)
190 else if DPL_HOMOGENEOUS_APPLY_ENTRY (9, 2)
191 else if DPL_HOMOGENEOUS_APPLY_ENTRY (9, 1)
193 else { return std::array<bool,0>(); }
194 // clang-format on
195}
196
197template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
198 requires(D == 8)
199auto homogeneous_apply_refs(L l, T&& object)
200{
201 constexpr int numElements = nested_brace_constructible_size<B, T>();
202 // clang-format off
204 else if DPL_HOMOGENEOUS_APPLY_ENTRY (8, 8)
205 else if DPL_HOMOGENEOUS_APPLY_ENTRY (8, 7)
206 else if DPL_HOMOGENEOUS_APPLY_ENTRY (8, 6)
207 else if DPL_HOMOGENEOUS_APPLY_ENTRY (8, 5)
208 else if DPL_HOMOGENEOUS_APPLY_ENTRY (8, 4)
209 else if DPL_HOMOGENEOUS_APPLY_ENTRY (8, 3)
210 else if DPL_HOMOGENEOUS_APPLY_ENTRY (8, 2)
211 else if DPL_HOMOGENEOUS_APPLY_ENTRY (8, 1)
213 else { return std::array<bool,0>(); }
214 // clang-format on
215}
216
217template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
218 requires(D == 7)
219auto homogeneous_apply_refs(L l, T&& object)
220{
221 constexpr int numElements = nested_brace_constructible_size<B, T>();
222 // clang-format off
224 else if DPL_HOMOGENEOUS_APPLY_ENTRY (7, 8)
225 else if DPL_HOMOGENEOUS_APPLY_ENTRY (7, 7)
226 else if DPL_HOMOGENEOUS_APPLY_ENTRY (7, 6)
227 else if DPL_HOMOGENEOUS_APPLY_ENTRY (7, 5)
228 else if DPL_HOMOGENEOUS_APPLY_ENTRY (7, 4)
229 else if DPL_HOMOGENEOUS_APPLY_ENTRY (7, 3)
230 else if DPL_HOMOGENEOUS_APPLY_ENTRY (7, 2)
231 else if DPL_HOMOGENEOUS_APPLY_ENTRY (7, 1)
233 else { return std::array<bool,0>(); }
234 // clang-format on
235}
236
237template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
238 requires(D == 6)
239auto homogeneous_apply_refs(L l, T&& object)
240{
241 constexpr int numElements = nested_brace_constructible_size<B, T>();
242 // clang-format off
244 else if DPL_HOMOGENEOUS_APPLY_ENTRY (6, 8)
245 else if DPL_HOMOGENEOUS_APPLY_ENTRY (6, 7)
246 else if DPL_HOMOGENEOUS_APPLY_ENTRY (6, 6)
247 else if DPL_HOMOGENEOUS_APPLY_ENTRY (6, 5)
248 else if DPL_HOMOGENEOUS_APPLY_ENTRY (6, 4)
249 else if DPL_HOMOGENEOUS_APPLY_ENTRY (6, 3)
250 else if DPL_HOMOGENEOUS_APPLY_ENTRY (6, 2)
251 else if DPL_HOMOGENEOUS_APPLY_ENTRY (6, 1)
253 else { return std::array<bool,0>(); }
254 // clang-format on
255}
256
257template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
258 requires(D == 5)
259auto homogeneous_apply_refs(L l, T&& object)
260{
261 constexpr int numElements = nested_brace_constructible_size<B, T>();
262 // clang-format off
264 else if DPL_HOMOGENEOUS_APPLY_ENTRY (5, 8)
265 else if DPL_HOMOGENEOUS_APPLY_ENTRY (5, 7)
266 else if DPL_HOMOGENEOUS_APPLY_ENTRY (5, 6)
267 else if DPL_HOMOGENEOUS_APPLY_ENTRY (5, 5)
268 else if DPL_HOMOGENEOUS_APPLY_ENTRY (5, 4)
269 else if DPL_HOMOGENEOUS_APPLY_ENTRY (5, 3)
270 else if DPL_HOMOGENEOUS_APPLY_ENTRY (5, 2)
271 else if DPL_HOMOGENEOUS_APPLY_ENTRY (5, 1)
273 else { return std::array<bool,0>(); }
274 // clang-format on
275}
276
277template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
278 requires(D == 4)
279auto homogeneous_apply_refs(L l, T&& object)
280{
281 constexpr int numElements = nested_brace_constructible_size<B, T>();
282 // clang-format off
284 else if DPL_HOMOGENEOUS_APPLY_ENTRY (4, 8)
285 else if DPL_HOMOGENEOUS_APPLY_ENTRY (4, 7)
286 else if DPL_HOMOGENEOUS_APPLY_ENTRY (4, 6)
287 else if DPL_HOMOGENEOUS_APPLY_ENTRY (4, 5)
288 else if DPL_HOMOGENEOUS_APPLY_ENTRY (4, 4)
289 else if DPL_HOMOGENEOUS_APPLY_ENTRY (4, 3)
290 else if DPL_HOMOGENEOUS_APPLY_ENTRY (4, 2)
291 else if DPL_HOMOGENEOUS_APPLY_ENTRY (4, 1)
293 else { return std::array<bool,0>(); }
294 // clang-format on
295}
296
297template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
298 requires(D == 3)
299auto homogeneous_apply_refs(L l, T&& object)
300{
301 constexpr int numElements = nested_brace_constructible_size<B, T>();
302 // clang-format off
304 else if DPL_HOMOGENEOUS_APPLY_ENTRY (3, 8)
305 else if DPL_HOMOGENEOUS_APPLY_ENTRY (3, 7)
306 else if DPL_HOMOGENEOUS_APPLY_ENTRY (3, 6)
307 else if DPL_HOMOGENEOUS_APPLY_ENTRY (3, 5)
308 else if DPL_HOMOGENEOUS_APPLY_ENTRY (3, 4)
309 else if DPL_HOMOGENEOUS_APPLY_ENTRY (3, 3)
310 else if DPL_HOMOGENEOUS_APPLY_ENTRY (3, 2)
311 else if DPL_HOMOGENEOUS_APPLY_ENTRY (3, 1)
313 else { return std::array<bool,0>(); }
314 // clang-format on
315}
316
317template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
318 requires(D == 2)
319auto homogeneous_apply_refs(L l, T&& object)
320{
321 constexpr int numElements = nested_brace_constructible_size<B, T>();
322 // clang-format off
324 else if DPL_HOMOGENEOUS_APPLY_ENTRY (2, 8)
325 else if DPL_HOMOGENEOUS_APPLY_ENTRY (2, 7)
326 else if DPL_HOMOGENEOUS_APPLY_ENTRY (2, 6)
327 else if DPL_HOMOGENEOUS_APPLY_ENTRY (2, 5)
328 else if DPL_HOMOGENEOUS_APPLY_ENTRY (2, 4)
329 else if DPL_HOMOGENEOUS_APPLY_ENTRY (2, 3)
330 else if DPL_HOMOGENEOUS_APPLY_ENTRY (2, 2)
331 else if DPL_HOMOGENEOUS_APPLY_ENTRY (2, 1)
333 else { return std::array<bool,0>(); }
334 // clang-format on
335}
336
337template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
338 requires(D == 1)
339auto homogeneous_apply_refs(L l, T&& object)
340{
341 constexpr int numElements = nested_brace_constructible_size<B, T>();
342 // clang-format off
344 else if DPL_HOMOGENEOUS_APPLY_ENTRY (1, 8)
345 else if DPL_HOMOGENEOUS_APPLY_ENTRY (1, 7)
346 else if DPL_HOMOGENEOUS_APPLY_ENTRY (1, 6)
347 else if DPL_HOMOGENEOUS_APPLY_ENTRY (1, 5)
348 else if DPL_HOMOGENEOUS_APPLY_ENTRY (1, 4)
349 else if DPL_HOMOGENEOUS_APPLY_ENTRY (1, 3)
350 else if DPL_HOMOGENEOUS_APPLY_ENTRY (1, 2)
351 else if DPL_HOMOGENEOUS_APPLY_ENTRY (1, 1)
353 else { return std::array<bool,0>(); }
354 // clang-format on
355}
356
357template <bool B = false, typename L, class T, int D = nested_brace_constructible_size<B, T>() / 10>
358 requires(D == 0)
359auto homogeneous_apply_refs(L l, T&& object)
360{
361 constexpr int numElements = nested_brace_constructible_size<B, T>();
362 // clang-format off
372 else { return std::array<bool,0>(); }
373 // clang-format on
374}
375
376} // namespace o2::framework
377
378#endif // O2_FRAMEWORK_STRUCTTOTUPLE_H_
#define DPL_HOMOGENEOUS_APPLY_ENTRY_TENS(d)
#define DPL_HOMOGENEOUS_APPLY_ENTRY(d, u)
#define DPL_HOMOGENEOUS_APPLY_ENTRY_LOW(u)
Definition B.h:16
GLint GLint GLsizei GLint GLenum GLenum type
Definition glcorearb.h:275
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
Defining PrimaryVertex explicitly as messageable.
Definition TFIDInfo.h:20
auto homogeneous_apply_refs(L l, T &&object)
consteval int nested_brace_constructible_size()
consteval auto brace_constructible_size(auto... Members)