-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmultipleVariadicArguments_Impl.hpp
99 lines (83 loc) · 3.72 KB
/
multipleVariadicArguments_Impl.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
#ifndef MULTIPLE_VARIADIC_ARGUMENTS_IMPL_HPP
#define MULTIPLE_VARIADIC_ARGUMENTS_IMPL_HPP
#if __cplusplus >= 201103L
#define USE_CPP11
#endif
#ifdef USE_CPP11
#include <tuple>
#include <utility>
#endif
struct Empty {};
struct MultipleVariadicArgumentSeparator {} multipleVariadicArgumentSeparator;
#ifdef USE_CPP11
namespace detail {
// Implementation details, user should not call any of these
template<typename... T>
struct MultipleVariadicArgumentSeparatorHandler_Impl {
// Less specialized function: we only get here when we have an empty parameter pack T
enum { SeparatorPosition = 0 };
typedef std::tuple<Empty> TailType;
};
template<>
struct MultipleVariadicArgumentSeparatorHandler_Impl<MultipleVariadicArgumentSeparator> {
enum { SeparatorPosition = 0 };
typedef std::tuple<Empty> TailType;
};
template<typename T>
struct MultipleVariadicArgumentSeparatorHandler_Impl<T> {
enum { SeparatorPosition = 1 + MultipleVariadicArgumentSeparatorHandler_Impl<MultipleVariadicArgumentSeparator>::SeparatorPosition };
typedef std::tuple<Empty> TailType;
};
template<typename... Tail>
struct MultipleVariadicArgumentSeparatorHandler_Impl<MultipleVariadicArgumentSeparator, Tail...> {
enum { SeparatorPosition = MultipleVariadicArgumentSeparatorHandler_Impl<MultipleVariadicArgumentSeparator>::SeparatorPosition };
typedef std::tuple<Tail...> TailType;
};
template<typename T, typename... Tail>
struct MultipleVariadicArgumentSeparatorHandler_Impl<T, Tail...> {
enum { SeparatorPosition = MultipleVariadicArgumentSeparatorHandler_Impl<Tail...>::SeparatorPosition + 1 };
typedef typename MultipleVariadicArgumentSeparatorHandler_Impl<Tail...>::TailType TailType;
};
// Unpack tuple implementation
// Heavily based on: http://stackoverflow.com/questions/10766112/c11-i-can-go-from-multiple-args-to-tuple-but-can-i-go-from-tuple-to-multiple
template <bool done, typename F, typename Tuple, int nbOfArgumentsToTake, int offset, int... N>
struct MultipleVariadicArgumentCall_Impl {
static void call (F& f, Tuple&& t) {
MultipleVariadicArgumentCall_Impl<nbOfArgumentsToTake == 1+sizeof...(N), F, Tuple, nbOfArgumentsToTake, offset, N..., sizeof...(N) + offset>::call(f,t);
}
};
template <typename F, typename Tuple, int nbOfArgumentsToTake, int offset, int... N>
struct MultipleVariadicArgumentCall_Impl<true, F, Tuple, nbOfArgumentsToTake, offset, N...> {
static void call(F& f, Tuple&& t) {
f(std::get<N>(std::forward<Tuple>(t))...);
}
};
template <bool offsetStarted, typename F, typename Tuple, int nbOfArgumentsToTake, int offset, int... N>
struct MultipleVariadicArgumentCallWithOffset {
static void call(F& f, Tuple&& t) {
MultipleVariadicArgumentCallWithOffset<offsetStarted == sizeof...(N), F, Tuple, nbOfArgumentsToTake, offset, N..., sizeof...(N)>::call(f,t);
}
};
template <typename F, typename Tuple, int nbOfArgumentsToTake, int offset, int... N>
struct MultipleVariadicArgumentCallWithOffset<true, F, Tuple, nbOfArgumentsToTake, offset, N...> {
static void call(F& f, Tuple&& t) {
MultipleVariadicArgumentCall_Impl<0 == nbOfArgumentsToTake, F, Tuple, nbOfArgumentsToTake, offset>::call(f,t);
}
};
template <int nbOfArgumentsToTake, int offset, bool execute, typename F, typename Tuple>
struct callMultipleArgumentFunctionTuple {
static void call(F& f, Tuple&& t) {
MultipleVariadicArgumentCallWithOffset<0 == offset, F, Tuple, nbOfArgumentsToTake, offset>::call(f, std::forward<Tuple>(t));
}
};
template <int nbOfArgumentsToTake, int offset, typename F, typename Tuple>
struct callMultipleArgumentFunctionTuple<nbOfArgumentsToTake, offset, false, F, Tuple> {
static void call(F& f, Tuple&& t) {
// prevent unused errors
(void) f;
(void) t;
}
};
}
#endif
#endif