lin
functors.hpp
Go to the documentation of this file.
1 // vim: set tabstop=2:softtabstop=2:shiftwidth=2:expandtab
2 
7 #ifndef LIN_CORE_OPERATIONS_FUNCTORS_HPP_
8 #define LIN_CORE_OPERATIONS_FUNCTORS_HPP_
9 
10 #include <type_traits>
11 #include <utility>
12 
13 namespace lin {
14 namespace internal {
15 
16 struct add {
17  template <typename T, typename U>
18  using expression = decltype(std::declval<T &>() + std::declval<U &>());
19 
20  constexpr add() = default;
21  template <typename T, typename U>
22  inline constexpr auto operator()(T const &t, U const &u) const { return t + u; }
23 };
24 
25 template <typename T>
26 struct add_st {
27  template <typename U>
28  using expression = typename add::template expression<T, U>;
29 
30  T const t;
31  constexpr add_st() = default;
32  constexpr add_st(T const &t) : t(t) { }
33  template <typename U>
34  inline constexpr auto operator()(U const &u) const { return add()(t, u); }
35 };
36 
37 template <typename T>
38 using add_ts = add_st<T>;
39 
40 template <typename T>
41 struct cast {
42  template <typename U>
43  using expression = decltype(static_cast<T>(std::declval<U &>()));
44 
45  constexpr cast() = default;
46  template <typename U>
47  inline constexpr auto operator()(U const &u) const { return static_cast<T>(u); }
48 };
49 
50 struct divide {
51  template <typename T, typename U>
52  using expression = decltype(std::declval<T &>() / std::declval<U &>());
53 
54  constexpr divide() = default;
55  template <typename T, typename U>
56  inline constexpr auto operator()(T const &t, U const &u) const { return t / u; }
57 };
58 
59 template <typename T>
60 struct divide_st {
61  template <typename U>
62  using expression = typename divide::template expression<T, U>;
63 
64  T const t;
65  constexpr divide_st() = default;
66  constexpr divide_st(T const &t) : t(t) { }
67  template <typename U>
68  inline constexpr auto operator()(U const u) const { return divide()(t, u); }
69 };
70 
71 template <typename T>
72 struct divide_ts {
73  template <typename U>
74  using expression = typename divide::template expression<U, T>;
75 
76  T const t;
77  constexpr divide_ts() = default;
78  constexpr divide_ts(T const &t) : t(t) { }
79  template <typename U>
80  inline constexpr auto operator()(U const &u) const { return divide()(u, t); }
81 };
82 
83 struct multiply {
84  template <typename T, typename U>
85  using expression = decltype(std::declval<T &>() * std::declval<U &>());
86 
87  constexpr multiply() = default;
88  template <typename T, typename U>
89  inline constexpr auto operator()(T const &t, U const &u) const { return t * u; }
90 };
91 
92 template <typename T>
93 struct multiply_st {
94  template <typename U>
95  using expression = typename multiply::template expression<T, U>;
96 
97  T const t;
98  constexpr multiply_st() = default;
99  constexpr multiply_st(T const &t) : t(t) { }
100  template <typename U>
101  inline constexpr auto operator()(U const &u) const { return multiply()(t, u); }
102 };
103 
104 template <typename T>
106 
107 struct negate {
108  template <typename T>
109  using expression = decltype(-std::declval<T &>());
110 
111  constexpr negate() = default;
112  template <typename T>
113  inline constexpr auto operator()(T const &t) const { return -t; }
114 };
115 
116 struct sign {
117  template <typename T>
118  inline constexpr static T _sign(T const &t) {
119  if (std::is_signed<T>::value) return T((T(0) < t) - (t < T(0)));
120  else return T(T(0) < t);
121  }
122 
123  template <typename T>
124  using expression = decltype(_sign(std::declval<T &>()));
125 
126  constexpr sign() = default;
127  template <typename T>
128  inline constexpr auto operator()(T const &t) const { return _sign(t); }
129 };
130 
131 struct square {
132  template <typename T>
133  using expression = decltype(std::declval<T &>() * std::declval<T &>());
134 
135  constexpr square() = default;
136  template <typename T>
137  inline constexpr auto operator()(T const &t) const { return t * t; }
138 };
139 
140 struct subtract {
141  template <typename T, typename U>
142  using expression = decltype(std::declval<T &>() - std::declval<U &>());
143 
144  constexpr subtract() = default;
145  template <typename T, typename U>
146  inline constexpr auto operator()(T const &t, U const &u) const { return t - u; }
147 };
148 
149 template <typename T>
150 struct subtract_st {
151  template <typename U>
152  using expression = typename subtract::template expression<T, U>;
153 
154  T const t;
155  constexpr subtract_st() = default;
156  constexpr subtract_st(T const &t) : t(t) { }
157  template <typename U>
158  inline constexpr auto operator()(U const &u) const { return subtract()(t, u); }
159 };
160 
161 template <typename T>
162 struct subtract_ts {
163  template <typename U>
164  using expression = typename subtract::template expression<U, T>;
165 
166  T const t;
167  constexpr subtract_ts() = default;
168  constexpr subtract_ts(T const &t) : t(t) { }
169  template <typename U>
170  inline constexpr auto operator()(U const &u) const { return subtract()(u, t); }
171 };
172 } // namespace internal
173 } // namespace lin
174 
175 #endif
Definition: functors.hpp:16
Definition: functors.hpp:140
Definition: functors.hpp:41
Definition: functors.hpp:83
Definition: functors.hpp:107
Definition: functors.hpp:60
Definition: functors.hpp:131
Definition: functors.hpp:162
Definition: functors.hpp:93
Definition: functors.hpp:72
Definition: config.hpp:27
Definition: functors.hpp:150
Definition: functors.hpp:116
Definition: functors.hpp:50
Definition: functors.hpp:26