ZeroErr
Loading...
Searching...
No Matches
typetraits.h
Go to the documentation of this file.
1#pragma once
3
4#include <ostream>
5#include <sstream>
6#include <string>
7#include <tuple> // this should be removed
8#include <type_traits>
9#include <complex>
10#include <memory>
11
13
14
15namespace zeroerr {
16
17
31template <unsigned N>
32struct rank : rank<N - 1> {};
33template <>
34struct rank<0> {};
35
36
37namespace detail {
38
39// C++11 void_t
40template <typename... Ts>
41using void_t = void;
42
43// Type dependent "true"/"false".
44// Useful for SFINAE and static_asserts where we need a type dependent
45// expression that happens to be constant.
46template <typename T>
48 static std::false_type value;
49};
50
51template <typename T>
53 static std::true_type value;
54};
55
56
57// Generate sequence of integers from 0 to N-1
58// Usage: detail::gen_seq<N> then use <size_t... I> to match it
59template <unsigned...>
60struct seq {};
61
62template <unsigned N, unsigned... Is>
63struct gen_seq : gen_seq<N - 1, N - 1, Is...> {};
64
65template <unsigned... Is>
66struct gen_seq<0, Is...> : seq<Is...> {};
67
68
69// Some utility structs to check template specialization
70template <typename Test, template <typename...> class Ref>
71struct is_specialization : std::false_type {};
72
73template <template <typename...> class Ref, typename... Args>
74struct is_specialization<Ref<Args...>, Ref> : std::true_type {};
75
76
77// Check if a type is stream writable, i.e., std::cout << foo;
78// Usage: is_streamable<std::ostream, int>::value
79template <typename S, typename T>
81
82template <typename S, typename T, typename = void>
83struct is_streamable : std::false_type {};
84
85template <typename S, typename T>
86struct is_streamable<S, T, has_stream_operator<S, T>> : std::true_type {};
87
88
89// Check if a type is a container type
90// Usage: is_container<std::vector<int>>::value
91template <typename T>
93 void_t<decltype(std::declval<T>().begin()), decltype(std::declval<T>().end())>;
94
95template <typename T, typename = void>
96struct is_container : std::false_type {};
97
98template <typename T>
99struct is_container<T, has_begin_end<T>> : std::true_type {};
100
101
102template <typename T>
104 void_t<decltype(std::declval<T>().begin()), decltype(std::declval<T>().end()),
105 decltype(std::declval<T>().find(std::declval<typename T::key_type>())),
106 decltype(std::declval<T>().insert(std::declval<typename T::value_type>()))>;
107
108template <typename T, typename = void>
109struct is_associative_container : std::false_type {};
110
111template <typename T>
112struct is_associative_container<T, has_begin_end_find_insert<T>> : std::true_type {};
113
114
115#if ZEROERR_CXX_STANDARD >= 17
116#define ZEROERR_STRING_VIEW std::is_same<T, std::string_view>::value
117#else
118#define ZEROERR_STRING_VIEW 0
119#endif
120
121// Check if a type is a string type
122template <class T>
124 : std::integral_constant<bool, std::is_same<T, std::string>::value ||
125 std::is_same<T, const char*>::value || ZEROERR_STRING_VIEW> {
126};
127
128
129// Check if a type can use arr[0] like an array
130template <typename T, typename = void>
131struct is_array : std::false_type {};
132
133template <typename T>
134struct is_array<T, void_t<decltype(std::declval<T>()[0])>> : std::true_type {};
135
136
137template <typename T, typename = void>
138struct is_modifiable : std::false_type {};
139
140template <typename T>
141struct is_modifiable<T, void_t<decltype(
142 // Iterable
143 T().begin(), T().end(), T().size(),
144 // Values are mutable
145 // This rejects associative containers, for example
146 // *T().begin() = std::declval<value_type_t<T>>(),
147 // Can insert and erase elements
148 T().insert(T().end(), std::declval<typename T::value_type>()),
149 T().erase(T().begin()), (void)0)>> : std::true_type {};
150
151
152
153// Check if a type has the element type as std::pair
154template <typename T>
157 decltype(std::declval<typename T::value_type>().second)>;
158template <typename T, typename = void>
159struct ele_type_is_pair : std::false_type {};
160
161template <typename T>
162struct ele_type_is_pair<T, has_pair_type<T>> : std::true_type {};
163
164template <typename T, typename V = void>
166 using type = T;
167};
168
169template <>
170struct to_store_type<const char*> {
171 using type = std::string;
172};
173
174template <>
175struct to_store_type<const char (&)[]> {
176 using type = std::string;
177};
178
179template <typename T>
180using is_not_array = typename std::enable_if<!std::is_array<T>::value>::type;
181template <typename T>
183 using type = T;
184};
185
186template <typename T>
187struct to_store_type<T&&> {
188 using type = T;
189};
190
191template <typename T>
193
194
195template <size_t I>
197 template <typename T, typename F>
198 static void visit(T& tup, size_t idx, F&& fun) {
199 if (idx == I - 1)
200 fun(std::get<I - 1>(tup));
201 else
202 visit_impl<I - 1>::visit(tup, idx, std::forward<F>(fun));
203 }
204};
205
206template <>
207struct visit_impl<0> {
208 template <typename T, typename F>
209 static void visit(T&, size_t, F&&) {}
210};
211
212template <typename F, typename... Ts>
213void visit_at(const std::tuple<Ts...>& tup, size_t idx, F&& fun) {
214 visit_impl<sizeof...(Ts)>::visit(tup, idx, std::forward<F>(fun));
215}
216
217template <typename F, typename... Ts>
218void visit_at(std::tuple<Ts...>& tup, size_t idx, F&& fun) {
219 visit_impl<sizeof...(Ts)>::visit(tup, idx, std::forward<F>(fun));
220}
221
222
223template <size_t I>
225 template <typename T1, typename T2, typename F>
226 static void visit(T1 tup1, T2 tup2, size_t idx, F&& fun) {
227 if (idx == I - 1)
228 fun(std::get<I - 1>(tup1), std::get<I - 1>(tup2));
229 else
230 visit2_impl<I - 1>::visit(tup1, tup2, idx, std::forward<F>(fun));
231 }
232};
233
234template <>
235struct visit2_impl<0> {
236 template <typename T1, typename T2, typename F>
237 static void visit(T1&, T2&, size_t, F&&) {}
238};
239
240template <typename F, typename... Ts, typename... T2s>
241void visit2_at(const std::tuple<Ts...>& tup1, const std::tuple<T2s...>& tup2, size_t idx, F&& fun) {
242 visit2_impl<sizeof...(Ts)>::visit(tup1, tup2, idx, std::forward<F>(fun));
243}
244
245template <typename F, typename... Ts, typename... T2s>
246void visit2_at(std::tuple<Ts...>& tup1, std::tuple<T2s...>& tup2, size_t idx, F&& fun) {
247 visit2_impl<sizeof...(Ts)>::visit(tup1, tup2, idx, std::forward<F>(fun));
248}
249
250template <typename F, typename... Ts, typename... T2s>
251void visit2_at(const std::tuple<Ts...>& tup1, std::tuple<T2s...>& tup2, size_t idx, F&& fun) {
252 visit2_impl<sizeof...(Ts)>::visit(tup1, tup2, idx, std::forward<F>(fun));
253}
254
255template <typename F, typename... Ts, typename... T2s>
256void visit2_at(std::tuple<Ts...>& tup1, const std::tuple<T2s...>& tup2, size_t idx, F&& fun) {
257 visit2_impl<sizeof...(Ts)>::visit(tup1, tup2, idx, std::forward<F>(fun));
258}
259
260#define ZEROERR_ENABLE_IF(x) \
261 template <typename T> \
262 typename std::enable_if<x, void>::type
263#define ZEROERR_IS_INT std::is_integral<T>::value
264#define ZEROERR_IS_FLOAT std::is_floating_point<T>::value
265#define ZEROERR_IS_CONTAINER detail::is_container<T>::value
266#define ZEROERR_IS_STRING detail::is_string<T>::value
267#define ZEROERR_IS_POINTER std::is_pointer<T>::value
268#define ZEROERR_IS_CHAR std::is_same<T, char>::value
269#define ZEROERR_IS_WCHAR std::is_same<T, wchar_t>::value
270#define ZEROERR_IS_CLASS std::is_class<T>::value
271#define ZEROERR_IS_STREAMABLE detail::is_streamable<std::ostream, T>::value
272#define ZEROERR_IS_ARRAY detail::is_array<T>::value
273#define ZEROERR_IS_COMPLEX detail::is_specialization<T, std::complex>::value
274#define ZEROERR_IS_BOOL std::is_same<T, bool>::value
275#define ZEROERR_IS_AUTOPTR \
276 (detail::is_specialization<T, std::unique_ptr>::value || \
277 detail::is_specialization<T, std::shared_ptr>::value || \
278 detail::is_specialization<T, std::weak_ptr>::value)
279#define ZEROERR_IS_MAP detail::ele_type_is_pair<T>::value
280#define ZEROERR_IS_POD std::is_standard_layout<T>::value
281#define ZEROERR_IS_EXT detail::has_extension<T>::value
282
283} // namespace detail
284
285} // namespace zeroerr
286
287
#define ZEROERR_SUPPRESS_COMMON_WARNINGS_POP
Definition config.h:265
#define ZEROERR_SUPPRESS_COMMON_WARNINGS_PUSH
Definition config.h:218
void_t< typename T::value_type, decltype(std::declval< typename T::value_type >().first), decltype(std::declval< typename T::value_type >().second)> has_pair_type
Definition typetraits.h:157
typename to_store_type< T >::type to_store_type_t
Definition typetraits.h:192
void_t< decltype(std::declval< S & >()<< std::declval< T >())> has_stream_operator
Definition typetraits.h:80
void_t< decltype(std::declval< T >().begin()), decltype(std::declval< T >().end()), decltype(std::declval< T >().find(std::declval< typename T::key_type >())), decltype(std::declval< T >().insert(std::declval< typename T::value_type >()))> has_begin_end_find_insert
Definition typetraits.h:106
void_t< decltype(std::declval< T >().begin()), decltype(std::declval< T >().end())> has_begin_end
Definition typetraits.h:93
void void_t
Definition typetraits.h:41
typename std::enable_if<!std::is_array< T >::value >::type is_not_array
Definition typetraits.h:180
void visit_at(const std::tuple< Ts... > &tup, size_t idx, F &&fun)
Definition typetraits.h:213
void visit2_at(const std::tuple< Ts... > &tup1, const std::tuple< T2s... > &tup2, size_t idx, F &&fun)
Definition typetraits.h:241
Definition benchmark.cpp:17
Definition typetraits.h:47
static std::false_type value
Definition typetraits.h:48
Definition typetraits.h:52
static std::true_type value
Definition typetraits.h:53
Definition typetraits.h:159
Definition typetraits.h:63
Definition typetraits.h:131
Definition typetraits.h:96
Definition typetraits.h:138
Definition typetraits.h:71
Definition typetraits.h:83
Definition typetraits.h:125
Definition typetraits.h:60
T type
Definition typetraits.h:188
std::string type
Definition typetraits.h:171
std::string type
Definition typetraits.h:176
Definition typetraits.h:165
T type
Definition typetraits.h:166
static void visit(T1 &, T2 &, size_t, F &&)
Definition typetraits.h:237
Definition typetraits.h:224
static void visit(T1 tup1, T2 tup2, size_t idx, F &&fun)
Definition typetraits.h:226
static void visit(T &, size_t, F &&)
Definition typetraits.h:209
Definition typetraits.h:196
static void visit(T &tup, size_t idx, F &&fun)
Definition typetraits.h:198
rank is a helper class for Printer to define the priority of overloaded functions.
Definition typetraits.h:32