forked from h4tr3d/avcpp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtimestamp.cpp
145 lines (118 loc) · 3.32 KB
/
timestamp.cpp
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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
#include <algorithm>
#include <limits>
#include "timestamp.h"
#include "avutils.h"
namespace av {
Timestamp::Timestamp() noexcept
{
}
Timestamp::Timestamp(int64_t timestamp, const Rational &timebase) noexcept
: m_timestamp(timestamp),
m_timebase(timebase)
{
}
int64_t Timestamp::timestamp() const noexcept
{
return m_timestamp;
}
int64_t Timestamp::timestamp(const Rational &timebase) const noexcept
{
return m_timebase.rescale(m_timestamp, timebase);
}
const Rational &Timestamp::timebase() const noexcept
{
return m_timebase;
}
bool Timestamp::isValid() const noexcept
{
return m_timestamp != av::NoPts;
}
bool Timestamp::isNoPts() const noexcept
{
return m_timestamp == av::NoPts;
}
Timestamp::operator double() const noexcept
{
return seconds();
}
double Timestamp::seconds() const noexcept
{
if (isNoPts()) {
return std::numeric_limits<double>::quiet_NaN();
}
return m_timebase.getDouble() * m_timestamp;
}
Timestamp &Timestamp::operator+=(const Timestamp &other)
{
m_timestamp = av_add_stable(m_timebase, m_timestamp,
other.timebase(), other.timestamp());
return *this;
}
Timestamp &Timestamp::operator-=(const Timestamp &other)
{
auto tmp = *this - other;
m_timestamp = tmp.timestamp(m_timebase);
return *this;
}
Timestamp &Timestamp::operator*=(const Timestamp &other)
{
auto tmp = *this * other;
m_timestamp = tmp.timestamp(m_timebase);
return *this;
}
Timestamp &Timestamp::operator/=(const Timestamp &other)
{
auto tmp = *this / other;
m_timestamp = tmp.timestamp(m_timebase);
return *this;
}
Timestamp::operator bool() const noexcept
{
return m_timestamp != av::NoPts;
}
Timestamp operator+(const Timestamp &left, const Timestamp &right) noexcept
{
// av_add_stable() required that rhs value less than lhs
if (left >= right) {
auto const ts = av_add_stable(left.timebase(), left.timestamp(), right.timebase(), right.timestamp());
return {ts, left.timebase()};
} else {
auto ts = av_add_stable(right.timebase(), right.timestamp(), left.timebase(), left.timestamp());
return {ts, right.timebase()};
}
}
Timestamp operator-(const Timestamp &left, const Timestamp &right) noexcept
{
// Use more good precision
auto tb = std::min(left.timebase(), right.timebase());
auto tsleft = left.timebase().rescale(left.timestamp(), tb);
auto tsright = right.timebase().rescale(right.timestamp(), tb);
auto ts = tsleft - tsright;
return {ts, tb};
}
Timestamp operator*(const Timestamp &left, const Timestamp &right) noexcept
{
auto ts = left.timestamp() * right.timestamp();
auto tb = left.timebase() * right.timebase();
return {ts, tb};
}
Timestamp operator/(const Timestamp &left, const Timestamp &right) noexcept
{
int num, den;
// Use more good precision
auto tb = std::min(left.timebase(), right.timebase());
auto tsleft = left.timebase().rescale(left.timestamp(), tb);
auto tsright = right.timebase().rescale(right.timestamp(), tb);
int64_t ts = 1;
if (tsleft > tsright) {
ts = tsleft / tsright;
tsleft %= tsright;
if (tsleft == 0) {
tsleft = 1;
tsright = 1;
}
}
av_reduce(&num, &den, tsleft, tsright, INT64_MAX);
return {ts, {num, den}};
}
} // ::av