qxLib
All Classes Files Functions Variables Typedefs Enumerations Enumerator Macros
constexpr_sequence.h
Go to the documentation of this file.
1 /**
2 
3  @file constexpr_sequence.h
4  @author Khrapov
5  @date 25.08.2020
6  @copyright � Nick Khrapov, 2021. All right reserved.
7 
8 **/
9 #pragma once
10 
11 #include <qx/macros/config.h>
12 
13 // https://stackoverflow.com/questions/65566935/
14 // https://developercommunity.visualstudio.com/content/problem/1300886/possible-error-in-compiler-with-cpp20.html
15 #define QX_CONSTEXPR_SEQUENCE_SUPPORTED !QX_MSVC
16 
17 #if QX_CONSTEXPR_SEQUENCE_SUPPORTED
18 
19  #include <qx/meta/constexpr_flag.h>
20 
21 namespace qx
22 {
23 
24 /**
25 
26  @class constexpr_sequence
27  @brief Constexpr sequence. use unique tag for different instances
28  @details Modified version of Anthony Williams constexpr counter
29  https://stackoverflow.com/a/58200261/8021662
30  \see examples in tests\test_constexpr_sequence.cpp
31  \note that you can't vary other template parameters with same Tag
32  prefer using "using my_sequence = qx::constexpr_sequence<...>"
33  @tparam tag_t - tag for unique instances
34  @tparam T - value type
35  @tparam Start - start value
36  @tparam Func - function that modifies value
37  @author Khrapov
38  @date 25.08.2020
39 
40 **/
41 template<class tag_t, class T, T Start, T Func(T)>
43 {
44 private:
45  template<size_t nIndex>
46  struct element
47  {
48  static constexpr T value() noexcept
49  {
50  T _value = Start;
51 
52  for (size_t i = 0; i < nIndex; ++i)
53  _value = Func(_value);
54 
55  return _value;
56  }
57  };
58 
59  template<size_t nIndex>
61 
62  template<size_t nCurrent, bool bWasSet /* = false */>
63  struct checker_setter
64  {
65  static constexpr size_t index() noexcept
66  {
67  return nCurrent;
68  }
69  };
70 
71  template<size_t nCurrent>
72  struct checker_wrapper
73  {
74  template<
75  bool bWasSet = element_flag<nCurrent> {}.test(),
76  size_t nNext = checker_setter<nCurrent, bWasSet> {}.index()>
77  static constexpr size_t index() noexcept
78  {
79  return nNext;
80  }
81  };
82 
83  template<size_t nCurrent>
84  struct checker_setter<nCurrent, /* bool bWasSet = */ true>
85  {
86  template<size_t nNext = checker_wrapper<nCurrent + 1> {}.index()>
87  static constexpr size_t index() noexcept
88  {
89  return nNext;
90  }
91  };
92 
93 public:
94  /**
95  @brief Get current sequence value
96  @details As function works through generating new template instance each time,
97  return value will be the same for each code line. That means in loops
98  you will get same value each time. Use macros like BOOST_PP_REPEAT
99  to simplify the code
100  @tparam nIndex - current sequence index
101  @tparam _value - current sequence value
102  @retval - current sequence value
103  **/
104  template<size_t nIndex = checker_wrapper<0> {}.index(), T _value = element<nIndex> {}.value()>
105  static constexpr T value() noexcept
106  {
107  return _value;
108  }
109 
110  /**
111  @brief Change value by Func and return new value
112  �details As function works through generating new template instance each time,
113  return value will be the same for one code line. That means in loops
114  you will get same value each time. Use macros like BOOST_PP_REPEAT
115  to simplify the code
116  @tparam nIndex - current sequence index
117  @tparam _value - current sequence value
118  @tparam bStub - stub parameter
119  @retval - next sequence value
120  **/
121  template<
122  size_t nIndex = checker_wrapper<0> {}.index(),
123  T _value = element<nIndex> {}.value(),
124  bool bStub = element_flag<nIndex> {}.test_and_set()>
125  static constexpr T next() noexcept
126  {
127  return Func(_value);
128  }
129 };
130 
131 namespace details
132 {
133 
134 template<class T, T Term>
135 constexpr T increase(T val)
136 {
137  return val + Term;
138 }
139 
140 template<class T, T Multiplier>
141 constexpr T multiply(T val)
142 {
143  return val * Multiplier;
144 }
145 
146 } // namespace details
147 
148 template<class tag_t = struct counter_tag, class T = int, T Start = 0, T Term = 1>
149 using constexpr_counter = constexpr_sequence<tag_t, T, Start, details::increase<T, Term>>;
150 
151 template<class tag_t = struct multiplier_tag, class T = int, T Start = 1, T Multiplier = 2>
152 using constexpr_multiplier = constexpr_sequence<tag_t, T, Start, details::multiply<T, Multiplier>>;
153 
154 } // namespace qx
155 
156 #endif
Constexpr flag class.
Constexpr sequence. use unique tag for different instances.
static constexpr T value() noexcept
Get current sequence value.
static constexpr T next() noexcept
Change value by Func and return new value �details As function works through generating new template ...