-
Notifications
You must be signed in to change notification settings - Fork 722
/
Copy pathunrolled_loop.hpp
executable file
·103 lines (93 loc) · 2.85 KB
/
unrolled_loop.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
100
101
102
103
#ifndef __UNROLLEDLOOP_HPP__
#define __UNROLLEDLOOP_HPP__
#include <type_traits>
#include <utility>
#include "metaprogramming_utils.hpp"
namespace fpga_tools {
///////////////////////////////////////////////////////////////////////////////
//
// Example usage for UnrolledLoop constexpr:
//
// Base
// UnrolledLoop(std::integer_sequence<int,5,2,7,8>{},[&](auto i) {
// /* i = 5,2,7,8 */
// });
//
// Case A
// UnrolledLoop<10>([&](auto i) {
// /* i = 0,1,...,9 */
// });
//
// Case B
// UnrolledLoop<10>([&](auto i) {
// /* i = 0,1,...,9 */
// });
//
// Case C
// UnrolledLoop<char, 1, 10>([&](auto i) {
// /* i = 1,2,...,9 */
// });
// UnrolledLoop<char, 10, 1>([&](auto i) {
// /* i = 10,9,...,2 */
// });
//
// Case D
// UnrolledLoop<1, 10>([&](auto i) {
// /* i = 1,2,...,9 */
// });
// UnrolledLoop<10, 1>([&](auto i) {
// /* i = 10,9,...,2 */
// });
//
///////////////////////////////////////////////////////////////////////////////
//
// Base implementation
// Templated on:
// ItType - the type of the iterator (size_t, int, char, ...)
// ItType... - the indices to iterate on
// F - the function to run for each index (i.e. the lambda)
//
template <class ItType, ItType... inds, class F>
constexpr void UnrolledLoop(std::integer_sequence<ItType, inds...>, F&& f) {
(f(std::integral_constant<ItType, inds>{}), ...);
}
//
// Convience implementation (A)
// performs UnrolledLoop in range [0,n) with iterator of type ItType
//
template <class ItType, ItType n, class F>
constexpr void UnrolledLoop(F&& f) {
UnrolledLoop(std::make_integer_sequence<ItType, n>{}, std::forward<F>(f));
}
//
// Convenience implementation (B)
// performs UnrolledLoop in range [0,n) with an iterator of type std::size_t
//
template <std::size_t n, class F>
constexpr void UnrolledLoop(F&& f) {
UnrolledLoop(std::make_index_sequence<n>{}, std::forward<F>(f));
}
//
// Convenience implementation (C)
// performs UnrolledLoop from start...end with an iterator of type ItType
// NOTE: start is INCLUSIVE, end is EXCLUSIVE
// NOTE: if start<=end, sequence is start,start+1,...,end-1
// if end<=start, sequence is start,start-1,...,end+1
//
template <class ItType, ItType start, ItType end, class F>
constexpr void UnrolledLoop(F&& f) {
UnrolledLoop(make_integer_range<ItType, start, end>{}, std::forward<F>(f));
}
//
// Convenience implementation (D)
// performs UnrolledLoop from start...end with an iterator of type size_t
// NOTE: start is INCLUSIVE, end is EXCLUSIVE
// NOTE: if start<=end, sequence is start,start+1,...,end-1
// if end<=start, sequence is start,start-1,...,end+1
//
template <std::size_t start, std::size_t end, class F>
constexpr void UnrolledLoop(F&& f) {
UnrolledLoop(make_index_range<start, end>{}, std::forward<F>(f));
}
} // namespace fpga_tools
#endif /* __UNROLLEDLOOP_HPP__ */