TeDDy 4.1.0
Decision diagram library.
Loading...
Searching...
No Matches
tools.hpp
1#ifndef LIBTEDDY_DETAILS_UTILS_HPP
2#define LIBTEDDY_DETAILS_UTILS_HPP
3
4#include <libteddy/details/types.hpp>
5
6#include <charconv>
7#include <concepts>
8#include <cstddef>
9#include <optional>
10#include <string_view>
11#include <type_traits>
12#include <vector>
13
14namespace teddy::utils
15{
19template<class Base>
20auto constexpr int_pow(Base base, int32 exponent) -> Base
21{
22 Base result = 1;
23
24 for (;;)
25 {
26 if (exponent & 1)
27 {
28 result *= base;
29 }
30
31 exponent >>= 1;
32
33 if (0 == exponent)
34 {
35 break;
36 }
37
38 base *= base;
39 }
40
41 return result;
42}
43
49template<class Num>
50auto parse (std::string_view const input) -> std::optional<Num>
51{
52 auto ret = Num {};
53 auto result
54 = std::from_chars(input.data(), input.data() + input.size(), ret);
55 return std::errc {} == result.ec
56 && result.ptr == input.data() + input.size()
57 ? std::optional<Num>(ret)
58 : std::nullopt;
59}
60
64inline auto do_hash (void* const p) -> std::size_t
65{
66 return reinterpret_cast<std::size_t>(p) >> 4;
67}
68
72inline auto do_hash (int32 const x) -> std::size_t
73{
74 return static_cast<std::size_t>(x);
75}
76
80template<class T>
81auto add_hash (std::size_t& hash, T const& elem) -> void
82{
83 // see boost::hash_combine
84 hash ^= do_hash(elem) + 0x9e3779b9 + (hash << 6) + (hash >> 2);
85}
86
90template<class... Ts>
91auto pack_hash (Ts const&... args) -> std::size_t
92{
93 std::size_t result = 0;
94 (add_hash(result, args), ...);
95 return result;
96}
97
103template<class... Args>
104auto any (Args... args)
105{
106 return (args || ...);
107}
108
112template<class T>
113constexpr auto min (T lhs, T rhs) -> T
114{
115 return lhs < rhs ? lhs : rhs;
116}
117
121template<class T>
122constexpr auto max (T lhs, T rhs) -> T
123{
124 return lhs > rhs ? lhs : rhs;
125}
126
130template<class X>
131constexpr auto pack_min (X x) -> X
132{
133 return x;
134}
135
139template<class X, class... Xs>
140constexpr auto pack_min (X x, Xs... xs) -> X
141{
142 return min(x, pack_min(xs...));
143}
144
149template<class It>
150auto max_elem (It first, It const last) -> It
151{
152 It maxIt = first;
153 while (first != last)
154 {
155 if (*first > *maxIt)
156 {
157 maxIt = first;
158 }
159 ++first;
160 }
161 return maxIt;
162}
163
168template<class It, class Predicate>
169auto find_if (It first, It const last, Predicate test) -> It
170{
171 while (first != last)
172 {
173 if (test(*first))
174 {
175 return first;
176 }
177 ++first;
178 }
179 return last;
180}
181
186template<class It, class Predicate>
187auto find_if_not (It first, It const last, Predicate test) -> It
188{
189 while (first != last)
190 {
191 if (not test(*first))
192 {
193 return first;
194 }
195 ++first;
196 }
197 return last;
198}
199
204template<class T, class U = T>
205auto exchange (T& var, U newVal) noexcept -> T
206{
207 auto oldVal = var;
208 var = newVal;
209 return oldVal;
210}
211
216template<typename T>
217constexpr auto swap (T& first, T& second) noexcept -> void
218{
219 auto tmp = first;
220 first = second;
221 second = tmp;
222}
223
227template<class T, class Compare>
228auto sort (std::vector<T>& xs, Compare cmp) -> void
229{
230 if (xs.empty())
231 {
232 return;
233 }
234
235 auto const sift_down = [&xs, cmp] (uint32 parent, uint32 const size)
236 {
237 uint32 left = 2 * parent + 1;
238 uint32 right = left + 1;
239 while (left < size)
240 {
241 uint32 swap = parent;
242 if (cmp(xs[swap], xs[left]))
243 {
244 swap = left;
245 }
246
247 if (right < size && cmp(xs[swap], xs[right]))
248 {
249 swap = right;
250 }
251
252 if (swap == parent)
253 {
254 break;
255 }
256
257 utils::swap(xs[parent], xs[swap]);
258 parent = swap;
259 left = 2 * parent + 1;
260 right = left + 1;
261 }
262 };
263
264 uint32 const size = static_cast<uint32>(xs.size());
265
266 // make-heap
267 for (uint32 i = size / 2 + 1; i > 0;)
268 {
269 --i;
270 sift_down(i, size);
271 }
272
273 // pop-heap
274 for (uint32 last = size - 1; last > 0; --last)
275 {
276 utils::swap(xs[last], xs[0]);
277 sift_down(0, last);
278 }
279}
280
281template<class T>
283{
284 static constexpr bool value = false;
285};
286
287template<>
288struct is_void<void>
289{
290 static constexpr bool value = true;
291};
292
293template<class T, class U>
295{
296 static constexpr bool value = false;
297};
298
299template<class T>
300struct is_same<T, T>
301{
302 static constexpr bool value = true;
303};
304
305template<class T, class U>
306concept same_as = std::is_same<T, U>::value;
307
308template<class T>
310 T,
311 std::vector<typename T::value_type, typename T::allocator_type>
312>;
313
318template<bool B, class T, class F>
319struct type_if;
320
324template<class T, class F>
325struct type_if<true, T, F>
326{
327 using type = T;
328};
329
333template<class T, class F>
334struct type_if<false, T, F>
335{
336 using type = F;
337};
338
342template<class X, class T>
343using second_t = type_if<false, X, T>::type;
344
345// TODO asi nebude treba
346template<class T>
348{
349 T member_;
350};
351
352template<>
353struct optional_member<void>
354{
355};
356} // namespace teddy::utils
357
358#endif
Definition tools.hpp:309
Definition tools.hpp:306
Definition tools.hpp:295
Definition tools.hpp:283
Definition tools.hpp:348
Provides member typedef based on the value of B Implementation of std::conditional.
Definition tools.hpp:319