mp_helper.h 1.32 KB
Newer Older
traveller59's avatar
traveller59 committed
1
2
3
4
5
#ifndef MP_HELPER_H_
#define MP_HELPER_H_
#include <type_traits>
#include <utility>

6
namespace tv {
traveller59's avatar
traveller59 committed
7
8
9
10
11
12
13
template <class... T> struct mp_list {};

template <class T, T... I>
using mp_list_c = mp_list<std::integral_constant<T, I>...>;

namespace detail {

14
15
16
17
template <class... Ts, class F>
constexpr F mp_for_each_impl(mp_list<Ts...>, F &&f) {
  return (void)(std::initializer_list<int>{(f(Ts()), 0)...}),
         std::forward<F>(f);
traveller59's avatar
traveller59 committed
18
19
20
21
22
23
24
25
}

template <class F> constexpr F mp_for_each_impl(mp_list<>, F &&f) {
  return std::forward<F>(f);
}

} // namespace detail

yanyan's avatar
yanyan committed
26
27
28
template <class... T>
using mp_length = std::integral_constant<std::size_t, sizeof...(T)>;

traveller59's avatar
traveller59 committed
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
namespace detail {

template <class A, template <class...> class B> struct mp_rename_impl {
  // An error "no type named 'type'" here means that the first argument to
  // mp_rename is not a list
};

template <template <class...> class A, class... T, template <class...> class B>
struct mp_rename_impl<A<T...>, B> {
  using type = B<T...>;
};

} // namespace detail

template <class A, template <class...> class B>
using mp_rename = typename detail::mp_rename_impl<A, B>::type;

yanyan's avatar
yanyan committed
46
47
template <class L> using mp_size = mp_rename<L, mp_length>;

traveller59's avatar
traveller59 committed
48
49
50
template <class L, class F> constexpr F mp_for_each(F &&f) {
  return detail::mp_for_each_impl(mp_rename<L, mp_list>(), std::forward<F>(f));
}
51
} // namespace tv
traveller59's avatar
traveller59 committed
52
53

#endif