OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
math.hpp
Go to the documentation of this file.
1#pragma once
2#include <ossia/detail/config.hpp>
3
4#include <ossia/detail/ssize.hpp>
5
6#include <cmath>
7
8#include <algorithm>
9#include <cinttypes>
10#include <utility>
17namespace ossia
18{
19#if defined(__ANDROID_API__)
20using ::asinh;
21using ::exp2;
22using ::fma;
23using ::llround;
24using ::log1p;
25using ::remainder;
26#else
27using std::asinh;
28using std::exp2;
29using std::fma;
30using std::llround;
31using std::log1p;
32using std::remainder;
33#endif
34const constexpr auto pi = 3.141592653589793238462643383279502884;
35const constexpr auto two_pi = 6.283185307179586476925286766559005768;
36const constexpr auto half_pi = 1.570796326794896619231321691639751442;
37const constexpr auto rad_to_deg = 5.729577951308232087679815481410517033e+01;
38const constexpr auto deg_to_rad = 1.745329251994329576923690768488612713e-02;
39const constexpr auto ln_2 = 6.931471805599453094172321214581765680e-01;
40const constexpr auto sqrt_2 = 1.414213562373095048801688724209698078e+00;
41
42template <typename T>
43T ipow(T x, int N) noexcept
44{
45 T res = x;
46 for(int i = 1; i < N; i++)
47 {
48 res *= x;
49 }
50 return res;
51}
52
53template <typename T>
54auto norm(T x, T y) noexcept
55{
56 return std::sqrt(ipow(x, 2) + ipow(y, 2));
57}
58
59template <typename T>
60auto norm(T x, T y, T z) noexcept
61{
62 return std::sqrt(ipow(x, 2) + ipow(y, 2) + ipow(z, 2));
63}
64
65inline constexpr size_t constexpr_log2(size_t n) noexcept
66{
67 return ((n < 2) ? 0 : 1 + constexpr_log2(n / 2));
68}
69
70template <typename T>
71inline constexpr size_t constexpr_abs(T n) noexcept
72{
73 return n < 0 ? -n : n;
74}
75
76// https://github.com/a2flo/floor/blob/master/constexpr/const_math.hpp#L251
77template <typename T>
78inline constexpr int64_t constexpr_floor(T val) noexcept
79{
80 const auto val_int = (int64_t)val;
81 const T fval_int = (T)val_int;
82 return (val >= (T)0 ? fval_int : (val == fval_int ? val : fval_int - 1));
83}
84
85template <typename T>
86inline constexpr int64_t constexpr_ceil(T f)
87{
88 const int64_t i = f;
89 return f > i ? i + 1 : i;
90}
91
95template <class T, class U>
96OSSIA_INLINE constexpr auto max(const T a, const U b) noexcept ->
97 typename std::conditional<(sizeof(T) > sizeof(U)), T, U>::type
98{
99 static_assert(std::is_integral_v<T> && std::is_integral_v<U>);
100 static_assert(std::is_signed_v<T> == std::is_signed_v<U>);
101 return a < b ? b : a;
102}
103
104static OSSIA_INLINE constexpr float max(float a, const float b) noexcept
105{
106 return a < b ? b : a;
107}
108static OSSIA_INLINE constexpr double max(double a, const double b) noexcept
109{
110 return a < b ? b : a;
111}
112static OSSIA_INLINE constexpr double max(float a, const double b) noexcept
113{
114 return a < b ? b : a;
115}
116static OSSIA_INLINE constexpr double max(double a, const float b) noexcept
117{
118 return a < b ? b : a;
119}
120
124template <class T, class U>
125OSSIA_INLINE constexpr auto min(const T a, const U b) noexcept ->
126 typename std::conditional<(sizeof(T) > sizeof(U)), T, U>::type
127{
128 static_assert(std::is_integral_v<T> && std::is_integral_v<U>);
129 static_assert(std::is_signed_v<T> == std::is_signed_v<U>);
130 return a > b ? b : a;
131}
132
133static OSSIA_INLINE constexpr float min(float a, const float b) noexcept
134{
135 return a > b ? b : a;
136}
137static OSSIA_INLINE constexpr double min(double a, const double b) noexcept
138{
139 return a > b ? b : a;
140}
141static OSSIA_INLINE constexpr double min(float a, const double b) noexcept
142{
143 return a > b ? b : a;
144}
145static OSSIA_INLINE constexpr double min(double a, const float b) noexcept
146{
147 return a > b ? b : a;
148}
149
153template <class T>
154OSSIA_INLINE constexpr T clamp(T d, const T min, const T max) noexcept
155{
156 return ossia::max(ossia::min(d, max), min);
157}
158
159static OSSIA_INLINE constexpr int64_t
160clamp(int64_t d, const int64_t min, const int64_t max) noexcept
161{
162 return ossia::max(ossia::min(d, max), min);
163}
164static OSSIA_INLINE constexpr int64_t
165clamp(int64_t d, const int32_t min, const int64_t max) noexcept
166{
167 return ossia::max(ossia::min(d, max), min);
168}
169static OSSIA_INLINE constexpr int64_t
170clamp(int64_t d, const int32_t min, const int32_t max) noexcept
171{
172 return ossia::max(ossia::min(d, max), min);
173}
174static OSSIA_INLINE constexpr int64_t
175clamp(int64_t d, const int64_t min, const int32_t max) noexcept
176{
177 return ossia::max(ossia::min(d, max), min);
178}
179static OSSIA_INLINE constexpr int32_t
180clamp(int32_t d, const int32_t min, const int32_t max) noexcept
181{
182 return ossia::max(ossia::min(d, max), min);
183}
184static OSSIA_INLINE constexpr int64_t
185clamp(int32_t d, const int64_t min, const int64_t max) noexcept
186{
187 return ossia::max(ossia::min(d, max), min);
188}
189static OSSIA_INLINE constexpr int64_t
190clamp(int32_t d, const int32_t min, const int64_t max) noexcept
191{
192 return ossia::max(ossia::min(d, max), min);
193}
194static OSSIA_INLINE constexpr int64_t
195clamp(int32_t d, const int64_t min, const int32_t max) noexcept
196{
197 return ossia::max(ossia::min(d, max), min);
198}
199
200static OSSIA_INLINE constexpr uint64_t
201clamp(uint64_t d, const uint64_t min, const uint64_t max) noexcept
202{
203 return ossia::max(ossia::min(d, max), min);
204}
205static OSSIA_INLINE constexpr uint64_t
206clamp(uint64_t d, const uint32_t min, const uint64_t max) noexcept
207{
208 return ossia::max(ossia::min(d, max), min);
209}
210static OSSIA_INLINE constexpr uint64_t
211clamp(uint64_t d, const uint32_t min, const uint32_t max) noexcept
212{
213 return ossia::max(ossia::min(d, max), min);
214}
215static OSSIA_INLINE constexpr uint64_t
216clamp(uint64_t d, const uint64_t min, const uint32_t max) noexcept
217{
218 return ossia::max(ossia::min(d, max), min);
219}
220static OSSIA_INLINE constexpr uint32_t
221clamp(uint32_t d, const uint32_t min, const uint32_t max) noexcept
222{
223 return ossia::max(ossia::min(d, max), min);
224}
225static OSSIA_INLINE constexpr uint64_t
226clamp(uint32_t d, const uint64_t min, const uint64_t max) noexcept
227{
228 return ossia::max(ossia::min(d, max), min);
229}
230static OSSIA_INLINE constexpr uint64_t
231clamp(uint32_t d, const uint32_t min, const uint64_t max) noexcept
232{
233 return ossia::max(ossia::min(d, max), min);
234}
235static OSSIA_INLINE constexpr uint64_t
236clamp(uint32_t d, const uint64_t min, const uint32_t max) noexcept
237{
238 return ossia::max(ossia::min(d, max), min);
239}
240
241static OSSIA_INLINE constexpr double
242clamp(double d, const double min, const double max) noexcept
243{
244 return ossia::max(ossia::min(d, max), min);
245}
246static OSSIA_INLINE constexpr double
247clamp(double d, const float min, const double max) noexcept
248{
249 return ossia::max(ossia::min(d, max), min);
250}
251static OSSIA_INLINE constexpr double
252clamp(double d, const float min, const float max) noexcept
253{
254 return ossia::max(ossia::min(d, max), min);
255}
256static OSSIA_INLINE constexpr double
257clamp(double d, const double min, const float max) noexcept
258{
259 return ossia::max(ossia::min(d, max), min);
260}
261static OSSIA_INLINE constexpr float
262clamp(float d, const float min, const float max) noexcept
263{
264 return ossia::max(ossia::min(d, max), min);
265}
266static OSSIA_INLINE constexpr double
267clamp(float d, const double min, const double max) noexcept
268{
269 return ossia::max(ossia::min(d, max), min);
270}
271static OSSIA_INLINE constexpr double
272clamp(float d, const float min, const double max) noexcept
273{
274 return ossia::max(ossia::min(d, max), min);
275}
276static OSSIA_INLINE constexpr double
277clamp(float d, const double min, const float max) noexcept
278{
279 return ossia::max(ossia::min(d, max), min);
280}
281
282// Wrap & Fold code taken from Jamoma TTLimits.h.
283// Credits : Nils Peters, Nov. 2008
284template <class T>
285OSSIA_INLINE constexpr T wrap(const T val, const T low, const T high) noexcept
286{
287 if(((val >= low) && (val < high)) || low == high)
288 return val;
289 else if(val >= low)
290 return low + std::fmod(val - low, std::fabs(low - high));
291 else
292 return high - std::fmod(low - val, std::fabs(low - high));
293}
294
295template <class T>
296OSSIA_INLINE constexpr T fold(const T val, const T low, const T high) noexcept
297{
298 if(((val >= low) && (val <= high)) || low == high)
299 return val;
300 else
301 {
302 return low + std::fabs(ossia::remainder(val - low, 2. * std::fabs(low - high)));
303 }
304}
305
306template <class T>
307OSSIA_INLINE constexpr std::pair<T, T> div(T num, T denom) noexcept
308{
309 return {num / denom, num % denom};
310}
311
312// normalized means x in [min, min + range] => f(x) in [0; 1] with log increase
313OSSIA_INLINE float log_to_normalized(float min, float range, float val) noexcept
314{
315 return val <= min ? 0.f : std::log1pf(val - min) / std::log1pf(range);
316}
317OSSIA_INLINE float normalized_to_log(float min, float range, float val) noexcept
318{
319 const float res = std::expm1f(val * std::log1pf(range)) + min;
320 return std::min(res, min + range);
321}
322
323template <typename T, typename R>
324constexpr OSSIA_INLINE bool valid_index(T idx, const R& range) noexcept
325{
326 using range_size_type = typename R::size_type;
327 if constexpr(std::is_signed_v<T>)
328 {
329 if constexpr(std::is_signed_v<range_size_type>)
330 return idx >= 0 && idx < range.size();
331 else
332 return idx >= 0 && idx < std::ssize(range);
333 }
334 else
335 {
336 if constexpr(std::is_signed_v<range_size_type>)
337 return idx < static_cast<std::size_t>(range.size());
338 else
339 return idx < range.size();
340 }
341}
342
343}
Definition git_info.h:7
OSSIA_INLINE constexpr auto min(const T a, const U b) noexcept -> typename std::conditional<(sizeof(T) > sizeof(U)), T, U >::type
min function tailored for values
Definition math.hpp:125
OSSIA_INLINE constexpr T clamp(T d, const T min, const T max) noexcept
clamp Returns the value bounded by a min and a max
Definition math.hpp:154
OSSIA_INLINE constexpr auto max(const T a, const U b) noexcept -> typename std::conditional<(sizeof(T) > sizeof(U)), T, U >::type
max function tailored for values
Definition math.hpp:96