OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
ossia/editor/scenario/time_value.hpp
Go to the documentation of this file.
1#pragma once
2#include <ossia/detail/config.hpp>
3
4#include <ossia/detail/flicks.hpp>
5#if defined(__APPLE__)
6#include <mach/time_value.h>
7#endif
8
9#include <ossia/math/safe_math.hpp>
10
11#include <cmath>
12
13#include <cassert>
14#include <cinttypes>
15#include <limits>
16
20namespace ossia
21{
22using physical_time = int64_t;
29struct OSSIA_EXPORT time_value
30{
31 // infinity is everything beyond 2^60 as this is already a gigantic quantity
32 // (~50 years in flicks) we set ~2^62 as the default "infinity" value to
33 // allow for some leeway and make sure we won't hit integer overflow in any
34 // reasonable cases
35 static const constexpr int64_t infinite_min = std::numeric_limits<int64_t>::max() / 8;
36 static const constexpr int64_t infinity = std::numeric_limits<int64_t>::max() / 2;
37
38 constexpr time_value& operator=(bool d) noexcept = delete;
39 constexpr time_value& operator=(double d) noexcept = delete;
40 constexpr time_value& operator=(float d) noexcept = delete;
41 constexpr time_value& operator=(uint64_t d) noexcept = delete;
42
43 constexpr time_value& operator=(int64_t d) noexcept
44 {
45 impl = d;
46 return *this;
47 }
48
50 constexpr time_value& operator+=(double d) noexcept = delete;
51 constexpr time_value& operator+=(float d) noexcept = delete;
52
53 constexpr time_value& operator+=(int64_t d) noexcept
54 {
55 if(infinite())
56 impl = 0;
57 else
58 impl += d;
59
60 return *this;
61 }
62
63 constexpr time_value& operator+=(ossia::time_value t) noexcept
64 {
65 if(infinite() || t.infinite())
66 impl = infinity;
67 else
68 impl += t.impl;
69
70 return *this;
71 }
72
74 constexpr time_value& operator-=(double d) noexcept = delete;
75 constexpr time_value& operator-=(int64_t d) noexcept
76 {
77 if(infinite())
78 impl = infinity;
79 else
80 impl -= d;
81
82 return *this;
83 }
84
85 constexpr time_value& operator-=(ossia::time_value t) noexcept
86 {
87 if(infinite() || t.infinite())
88 impl = infinity;
89 else
90 impl -= t.impl;
91
92 return *this;
93 }
94
95 constexpr time_value& operator-() noexcept
96 {
97 if(!infinite())
98 impl = -impl;
99
100 return *this;
101 }
102
104 // not constexpr because of isnan
105 /* constexpr */ time_value operator+(double d) const noexcept
106 {
107 assert(!ossia::safe_isnan(d));
108 assert(d < static_cast<double>(infinite_min));
109 return *this + time_value{int64_t(d)};
110 }
111 constexpr time_value operator+(int64_t d) const noexcept
112 {
113 return *this + time_value{d};
114 }
115 constexpr time_value operator+(uint64_t d) const noexcept
116 {
117 assert(d < infinite_min);
118 return *this + time_value{int64_t(d)};
119 }
120 constexpr time_value operator-(int64_t d) const noexcept
121 {
122 return *this + time_value{-d};
123 }
124 constexpr time_value operator-(uint64_t d) const noexcept
125 {
126 assert(d < infinite_min);
127 return *this + time_value{-int64_t(d)};
128 }
129
130 static constexpr bool
131 add_is_infinite(const ossia::time_value& lhs, const ossia::time_value& rhs) noexcept
132 {
133 if(lhs.infinite() || rhs.infinite())
134 {
135 return true;
136 }
137 else if(lhs.impl >= 0 && rhs.impl >= 0)
138 {
139 uint64_t l = lhs.impl;
140 uint64_t r = rhs.impl;
141 return l + r >= infinite_min;
142 }
143 else if(lhs.impl >= 0 && rhs.impl < 0)
144 {
145 uint64_t l = lhs.impl;
146 return l + rhs.impl >= infinite_min;
147 }
148 else if(lhs.impl < 0 && rhs.impl >= 0)
149 {
150 uint64_t r = rhs.impl;
151 return lhs.impl + r >= infinite_min;
152 }
153 else if(lhs.impl < 0 && rhs.impl < 0)
154 {
155 // TODO have a better underflow check
156 uint64_t l = -lhs.impl;
157 uint64_t r = -rhs.impl;
158 return l + r >= infinite_min;
159 }
160
161 return false;
162 }
163
164 static constexpr bool
165 sub_is_infinite(const ossia::time_value& lhs, const ossia::time_value& rhs) noexcept
166 {
167 if(lhs.infinite() || rhs.infinite())
168 {
169 return true;
170 }
171 else if(lhs.impl >= 0 && rhs.impl >= 0)
172 {
173 return false;
174 }
175 else if(lhs.impl >= 0 && rhs.impl < 0)
176 {
177 uint64_t l = lhs.impl;
178 uint64_t r = -rhs.impl;
179 return l + r >= infinite_min;
180 }
181 else if(lhs.impl < 0 && rhs.impl >= 0)
182 {
183 uint64_t l = -lhs.impl;
184 uint64_t r = rhs.impl;
185 return l + r >= infinite_min;
186 }
187 else if(lhs.impl < 0 && rhs.impl < 0)
188 {
189 return false;
190 }
191
192 return false;
193 }
194
195 constexpr time_value operator+(ossia::time_value t) const noexcept
196 {
197 if(add_is_infinite(*this, t))
198 return time_value{infinity};
199
200 return time_value{impl + t.impl};
201 }
202
204 // not constexpr because of isnan
205 /* constexpr */ time_value operator-(double d) const noexcept
206 {
207 assert(!ossia::safe_isnan(d));
208 assert(d < static_cast<double>(infinite_min));
209 return *this - time_value{int64_t(d)};
210 }
211
212 constexpr time_value operator-(ossia::time_value t) const noexcept
213 {
214 if(sub_is_infinite(*this, t))
215 return time_value{infinity};
216
217 return time_value{impl - t.impl};
218 }
219
221 constexpr time_value operator*(float d) const noexcept
222 {
223 return time_value{int64_t(impl * d)};
224 }
225
226 constexpr time_value operator*(double d) const noexcept
227 {
228 return time_value{int64_t(impl * d)};
229 }
230
231 constexpr time_value operator*(int32_t d) const noexcept
232 {
233 return time_value{impl * d};
234 }
235
236 constexpr time_value operator*(int64_t d) const noexcept
237 {
238 return time_value{impl * d};
239 }
240
241 constexpr time_value operator*(uint32_t d) const noexcept
242 {
243 return time_value{impl * d};
244 }
245
246 constexpr time_value operator*(uint64_t d) const noexcept
247 {
248 return time_value{int64_t(impl * d)};
249 }
250
251 friend constexpr double operator/(time_value lhs, time_value rhs) noexcept
252 {
253 return double(lhs.impl) / double(rhs.impl);
254 }
255
258 [[nodiscard]] constexpr bool infinite() const noexcept { return impl >= infinite_min; }
259 constexpr time_value operator%(time_value d) const noexcept
260 {
261 return time_value{impl % d.impl};
262 }
263 constexpr bool operator==(ossia::time_value rhs) const noexcept
264 {
265 return (infinite() && rhs.infinite()) || (impl == rhs.impl);
266 }
267 constexpr bool operator!=(ossia::time_value rhs) const noexcept
268 {
269 return (infinite() != rhs.infinite()) || (impl != rhs.impl);
270 }
271 constexpr bool operator<(ossia::time_value rhs) const noexcept
272 {
273 return !(infinite() && rhs.infinite()) && (impl < rhs.impl);
274 }
275 constexpr bool operator>(ossia::time_value rhs) const noexcept
276 {
277 return !(infinite() && rhs.infinite()) && (impl > rhs.impl);
278 }
279 constexpr bool operator<=(ossia::time_value rhs) const noexcept
280 {
281 return !(infinite() && rhs.infinite()) && (impl <= rhs.impl);
282 }
283 constexpr bool operator>=(ossia::time_value rhs) const noexcept
284 {
285 return !(infinite() && rhs.infinite()) && (impl >= rhs.impl);
286 }
287
288 int64_t impl;
289};
290
291constexpr inline time_value operator""_tv(long double v) noexcept
292{
293 return time_value{int64_t(v)};
294}
295
296constexpr inline time_value operator""_tv(unsigned long long v) noexcept
297{
298 return time_value{(int64_t)v};
299}
300
301const constexpr time_value Infinite{time_value::infinity};
302const constexpr time_value Zero{0};
303const constexpr time_value One{1};
304
305constexpr inline time_value abs(time_value t) noexcept
306{
307 return time_value{t.impl >= 0 ? t.impl : -t.impl};
308}
309
310constexpr inline time_value norm(time_value t1, time_value t2) noexcept
311{
312 if(t1.infinite() || t2.infinite())
313 return Infinite;
314 return time_value{t1.impl > t2.impl ? t1.impl - t2.impl : t2.impl - t1.impl};
315}
316
317inline constexpr int64_t to_sample(ossia::time_value t, double rate) noexcept
318{
319 const double samples_per_flicks = rate / ossia::flicks_per_second<double>;
320 return (rate > 0 && !t.infinite()) ? std::round(t.impl * samples_per_flicks) : 0;
321}
322
323}
324
325// static_assert(std::is_pod<ossia::time_value>::value, "bug introduced
326// somewhere");
Definition git_info.h:7
The time_value class.
Definition ossia/editor/scenario/time_value.hpp:30
time_value operator-(double d) const noexcept
substraction operator
Definition ossia/editor/scenario/time_value.hpp:205
constexpr time_value operator*(float d) const noexcept
multiplication operator
Definition ossia/editor/scenario/time_value.hpp:221
constexpr time_value & operator-=(double d) noexcept=delete
self substraction operator
time_value operator+(double d) const noexcept
addition operator
Definition ossia/editor/scenario/time_value.hpp:105
constexpr bool infinite() const noexcept
is the time value infinite ?
Definition ossia/editor/scenario/time_value.hpp:258
constexpr time_value & operator+=(double d) noexcept=delete
self addition operator