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
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
|
// Copyright 2019 The TCMalloc Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#ifndef TCMALLOC_INTERNAL_TIMESERIES_TRACKER_H_
#define TCMALLOC_INTERNAL_TIMESERIES_TRACKER_H_
#include <stddef.h>
#include <stdint.h>
#include <algorithm>
#include <atomic>
#include <limits>
#include "absl/base/internal/cycleclock.h"
#include "absl/functional/function_ref.h"
#include "absl/numeric/bits.h"
#include "absl/numeric/int128.h"
#include "absl/time/clock.h"
#include "absl/time/time.h"
#include "tcmalloc/internal/clock.h"
#include "tcmalloc/internal/logging.h"
GOOGLE_MALLOC_SECTION_BEGIN
namespace tcmalloc {
namespace tcmalloc_internal {
// Aggregates a series of reported values of type S in a set of entries of type
// T, one entry per epoch. This class factors out common functionality of
// different time series trackers. S can be any type, T needs to implement:
// Nil(), Report(S val), empty()
template <typename T, typename S, size_t kEpochs = 16>
class TimeSeriesTracker {
public:
enum SkipEntriesSetting { kSkipEmptyEntries, kDoNotSkipEmptyEntries };
explicit constexpr TimeSeriesTracker(Clock clock, absl::Duration w)
: window_(w), epoch_length_(window_ / kEpochs), clock_(clock) {
// See comment in GetCurrentEpoch().
auto d = static_cast<uint64_t>(absl::ToDoubleSeconds(epoch_length_) *
clock.freq());
div_precision_ = 63 + absl::bit_width(d);
epoch_ticks_m_ =
static_cast<uint64_t>(
(static_cast<absl::uint128>(1) << div_precision_) / d) +
1;
}
bool Report(S val);
// Iterates over the time series, starting from the oldest entry. The callback
// receives the offset of the entry, its timestamp according to the clock and
// the entry itself. Offsets are relative to the beginning of the buffer.
void Iter(absl::FunctionRef<void(size_t, int64_t, const T&)> f,
SkipEntriesSetting skip_entries) const;
// Iterates over the last num_epochs data points (if -1, iterate to the
// oldest entry). Offsets are relative to the end of the buffer.
void IterBackwards(absl::FunctionRef<void(size_t, int64_t, const T&)> f,
int64_t num_epochs = -1) const;
// This retrieves a particular data point (if offset is outside the valid
// range, the default data point will be returned).
const T GetEpochAtOffset(size_t offset);
// Updates the time base to the current time. This is useful to report the
// most recent time window rather than the last time window that had any
// reported values.
void UpdateTimeBase() { UpdateClock(); }
private:
// Returns true if the tracker moved to a different epoch.
bool UpdateClock();
// Returns the current epoch based on the clock.
int64_t GetCurrentEpoch() {
// This is equivalent to
// `clock_.now() / (absl::ToDoubleSeconds(epoch_length_) * clock_.freq())`.
// We basically follow the technique from
// https://ridiculousfish.com/blog/posts/labor-of-division-episode-i.html,
// except that we use one fewer bit of precision than necessary to always
// get the correct answer if the numerator were a 64-bit unsigned number. In
// this case, because clock_.now() returns a signed 64-bit number (i.e. max
// is <2^63), it shouldn't cause a problem. This way, we don't need to
// handle overflow so it's simpler. See also:
// https://lemire.me/blog/2019/02/20/more-fun-with-fast-remainders-when-the-divisor-is-a-constant/.
return static_cast<int64_t>(static_cast<absl::uint128>(epoch_ticks_m_) *
clock_.now() >>
div_precision_);
}
const absl::Duration window_;
const absl::Duration epoch_length_;
T entries_[kEpochs]{};
size_t last_epoch_{0};
size_t current_epoch_{0};
// This is the magic constant from
// https://ridiculousfish.com/blog/posts/labor-of-division-episode-i.html.
uint64_t epoch_ticks_m_;
uint8_t div_precision_;
Clock clock_;
};
// Erases values from the window that are out of date; sets the current epoch
// to the current location in the ringbuffer.
template <class T, class S, size_t kEpochs>
bool TimeSeriesTracker<T, S, kEpochs>::UpdateClock() {
const size_t epoch = GetCurrentEpoch();
// How many time steps did we take? (Since we only record kEpochs
// time steps, we can pretend it was at most that.)
size_t delta = epoch - last_epoch_;
delta = std::min(delta, kEpochs);
last_epoch_ = epoch;
if (delta == 0) {
return false;
}
// At each tick, we move our current location by one, to a new location
// that contains too-old data (which must be zeroed.)
for (size_t offset = 0; offset < delta; ++offset) {
current_epoch_++;
if (current_epoch_ == kEpochs) current_epoch_ = 0;
entries_[current_epoch_] = T::Nil();
}
return true;
}
template <class T, class S, size_t kEpochs>
void TimeSeriesTracker<T, S, kEpochs>::Iter(
absl::FunctionRef<void(size_t, int64_t, const T&)> f,
SkipEntriesSetting skip_entries) const {
size_t j = current_epoch_ + 1;
if (j == kEpochs) j = 0;
int64_t timestamp =
(last_epoch_ - kEpochs) * absl::ToInt64Nanoseconds(epoch_length_);
for (int offset = 0; offset < kEpochs; offset++) {
timestamp += absl::ToInt64Nanoseconds(epoch_length_);
if (skip_entries == kDoNotSkipEmptyEntries || !entries_[j].empty()) {
f(offset, timestamp, entries_[j]);
}
j++;
if (j == kEpochs) j = 0;
}
}
template <class T, class S, size_t kEpochs>
void TimeSeriesTracker<T, S, kEpochs>::IterBackwards(
absl::FunctionRef<void(size_t, int64_t, const T&)> f,
int64_t num_epochs) const {
// -1 means that we are outputting all epochs.
num_epochs = (num_epochs == -1) ? kEpochs : num_epochs;
size_t j = current_epoch_;
ASSERT(num_epochs <= kEpochs);
int64_t timestamp = last_epoch_ * absl::ToInt64Nanoseconds(epoch_length_);
for (size_t offset = 0; offset < num_epochs; ++offset) {
// This is deliberately int64_t and not a time unit, since clock_ is not
// guaranteed to be a real time base.
f(offset, timestamp, entries_[j]);
timestamp -= absl::ToInt64Nanoseconds(epoch_length_);
if (j == 0) j = kEpochs;
--j;
}
}
template <class T, class S, size_t kEpochs>
const T TimeSeriesTracker<T, S, kEpochs>::GetEpochAtOffset(size_t offset) {
return (offset >= kEpochs)
? T::Nil()
: entries_[(current_epoch_ + kEpochs - offset) % kEpochs];
}
template <class T, class S, size_t kEpochs>
bool TimeSeriesTracker<T, S, kEpochs>::Report(S val) {
bool updated_clock = UpdateClock();
entries_[current_epoch_].Report(val);
return updated_clock;
}
} // namespace tcmalloc_internal
} // namespace tcmalloc
GOOGLE_MALLOC_SECTION_END
#endif // TCMALLOC_INTERNAL_TIMESERIES_TRACKER_H_
|