blob: e3ecfeaed27073c66ca2ae2aa0d2e1bd846db1e9 [file] [log] [blame]
turaj@webrtc.org7959e162013-09-12 18:30:26 +00001/*
2 * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "modules/audio_coding/neteq/nack_tracker.h"
turaj@webrtc.org7959e162013-09-12 18:30:26 +000012
Yves Gerey988cc082018-10-23 12:03:01 +020013#include <assert.h>
14#include <cstdint>
15#include <utility>
turaj@webrtc.org7959e162013-09-12 18:30:26 +000016
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020017#include "rtc_base/checks.h"
turaj@webrtc.org7959e162013-09-12 18:30:26 +000018
19namespace webrtc {
turaj@webrtc.org7959e162013-09-12 18:30:26 +000020namespace {
21
22const int kDefaultSampleRateKhz = 48;
23const int kDefaultPacketSizeMs = 20;
24
25} // namespace
26
henrik.lundin91951862016-06-08 06:43:41 -070027NackTracker::NackTracker(int nack_threshold_packets)
turaj@webrtc.org7959e162013-09-12 18:30:26 +000028 : nack_threshold_packets_(nack_threshold_packets),
29 sequence_num_last_received_rtp_(0),
30 timestamp_last_received_rtp_(0),
31 any_rtp_received_(false),
32 sequence_num_last_decoded_rtp_(0),
33 timestamp_last_decoded_rtp_(0),
34 any_rtp_decoded_(false),
35 sample_rate_khz_(kDefaultSampleRateKhz),
36 samples_per_packet_(sample_rate_khz_ * kDefaultPacketSizeMs),
37 max_nack_list_size_(kNackListSizeLimit) {}
38
henrik.lundin91951862016-06-08 06:43:41 -070039NackTracker::~NackTracker() = default;
Karl Wiberg2519c452015-04-07 16:12:57 +020040
henrik.lundin91951862016-06-08 06:43:41 -070041NackTracker* NackTracker::Create(int nack_threshold_packets) {
42 return new NackTracker(nack_threshold_packets);
turaj@webrtc.org7959e162013-09-12 18:30:26 +000043}
44
henrik.lundin91951862016-06-08 06:43:41 -070045void NackTracker::UpdateSampleRate(int sample_rate_hz) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +000046 assert(sample_rate_hz > 0);
47 sample_rate_khz_ = sample_rate_hz / 1000;
48}
49
henrik.lundin91951862016-06-08 06:43:41 -070050void NackTracker::UpdateLastReceivedPacket(uint16_t sequence_number,
51 uint32_t timestamp) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +000052 // Just record the value of sequence number and timestamp if this is the
53 // first packet.
54 if (!any_rtp_received_) {
55 sequence_num_last_received_rtp_ = sequence_number;
56 timestamp_last_received_rtp_ = timestamp;
57 any_rtp_received_ = true;
58 // If no packet is decoded, to have a reasonable estimate of time-to-play
59 // use the given values.
60 if (!any_rtp_decoded_) {
61 sequence_num_last_decoded_rtp_ = sequence_number;
62 timestamp_last_decoded_rtp_ = timestamp;
63 }
64 return;
65 }
66
67 if (sequence_number == sequence_num_last_received_rtp_)
68 return;
69
70 // Received RTP should not be in the list.
71 nack_list_.erase(sequence_number);
72
73 // If this is an old sequence number, no more action is required, return.
74 if (IsNewerSequenceNumber(sequence_num_last_received_rtp_, sequence_number))
75 return;
76
77 UpdateSamplesPerPacket(sequence_number, timestamp);
78
79 UpdateList(sequence_number);
80
81 sequence_num_last_received_rtp_ = sequence_number;
82 timestamp_last_received_rtp_ = timestamp;
83 LimitNackListSize();
84}
85
henrik.lundin91951862016-06-08 06:43:41 -070086void NackTracker::UpdateSamplesPerPacket(
87 uint16_t sequence_number_current_received_rtp,
88 uint32_t timestamp_current_received_rtp) {
henrik.lundin48ed9302015-10-29 05:36:24 -070089 uint32_t timestamp_increase =
90 timestamp_current_received_rtp - timestamp_last_received_rtp_;
91 uint16_t sequence_num_increase =
92 sequence_number_current_received_rtp - sequence_num_last_received_rtp_;
turaj@webrtc.org7959e162013-09-12 18:30:26 +000093
94 samples_per_packet_ = timestamp_increase / sequence_num_increase;
95}
96
henrik.lundin91951862016-06-08 06:43:41 -070097void NackTracker::UpdateList(uint16_t sequence_number_current_received_rtp) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +000098 // Some of the packets which were considered late, now are considered missing.
99 ChangeFromLateToMissing(sequence_number_current_received_rtp);
100
101 if (IsNewerSequenceNumber(sequence_number_current_received_rtp,
102 sequence_num_last_received_rtp_ + 1))
103 AddToList(sequence_number_current_received_rtp);
104}
105
henrik.lundin91951862016-06-08 06:43:41 -0700106void NackTracker::ChangeFromLateToMissing(
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000107 uint16_t sequence_number_current_received_rtp) {
henrik.lundin48ed9302015-10-29 05:36:24 -0700108 NackList::const_iterator lower_bound =
109 nack_list_.lower_bound(static_cast<uint16_t>(
110 sequence_number_current_received_rtp - nack_threshold_packets_));
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000111
112 for (NackList::iterator it = nack_list_.begin(); it != lower_bound; ++it)
113 it->second.is_missing = true;
114}
115
henrik.lundin91951862016-06-08 06:43:41 -0700116uint32_t NackTracker::EstimateTimestamp(uint16_t sequence_num) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000117 uint16_t sequence_num_diff = sequence_num - sequence_num_last_received_rtp_;
118 return sequence_num_diff * samples_per_packet_ + timestamp_last_received_rtp_;
119}
120
henrik.lundin91951862016-06-08 06:43:41 -0700121void NackTracker::AddToList(uint16_t sequence_number_current_received_rtp) {
henrik.lundin48ed9302015-10-29 05:36:24 -0700122 assert(!any_rtp_decoded_ ||
123 IsNewerSequenceNumber(sequence_number_current_received_rtp,
124 sequence_num_last_decoded_rtp_));
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000125
126 // Packets with sequence numbers older than |upper_bound_missing| are
127 // considered missing, and the rest are considered late.
henrik.lundin48ed9302015-10-29 05:36:24 -0700128 uint16_t upper_bound_missing =
129 sequence_number_current_received_rtp - nack_threshold_packets_;
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000130
131 for (uint16_t n = sequence_num_last_received_rtp_ + 1;
henrik.lundin48ed9302015-10-29 05:36:24 -0700132 IsNewerSequenceNumber(sequence_number_current_received_rtp, n); ++n) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000133 bool is_missing = IsNewerSequenceNumber(upper_bound_missing, n);
134 uint32_t timestamp = EstimateTimestamp(n);
135 NackElement nack_element(TimeToPlay(timestamp), timestamp, is_missing);
136 nack_list_.insert(nack_list_.end(), std::make_pair(n, nack_element));
137 }
138}
139
henrik.lundin91951862016-06-08 06:43:41 -0700140void NackTracker::UpdateEstimatedPlayoutTimeBy10ms() {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000141 while (!nack_list_.empty() &&
henrik.lundin48ed9302015-10-29 05:36:24 -0700142 nack_list_.begin()->second.time_to_play_ms <= 10)
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000143 nack_list_.erase(nack_list_.begin());
144
145 for (NackList::iterator it = nack_list_.begin(); it != nack_list_.end(); ++it)
146 it->second.time_to_play_ms -= 10;
147}
148
henrik.lundin91951862016-06-08 06:43:41 -0700149void NackTracker::UpdateLastDecodedPacket(uint16_t sequence_number,
150 uint32_t timestamp) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000151 if (IsNewerSequenceNumber(sequence_number, sequence_num_last_decoded_rtp_) ||
152 !any_rtp_decoded_) {
153 sequence_num_last_decoded_rtp_ = sequence_number;
154 timestamp_last_decoded_rtp_ = timestamp;
155 // Packets in the list with sequence numbers less than the
156 // sequence number of the decoded RTP should be removed from the lists.
157 // They will be discarded by the jitter buffer if they arrive.
henrik.lundin48ed9302015-10-29 05:36:24 -0700158 nack_list_.erase(nack_list_.begin(),
159 nack_list_.upper_bound(sequence_num_last_decoded_rtp_));
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000160
161 // Update estimated time-to-play.
162 for (NackList::iterator it = nack_list_.begin(); it != nack_list_.end();
henrik.lundin48ed9302015-10-29 05:36:24 -0700163 ++it)
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000164 it->second.time_to_play_ms = TimeToPlay(it->second.estimated_timestamp);
165 } else {
166 assert(sequence_number == sequence_num_last_decoded_rtp_);
167
168 // Same sequence number as before. 10 ms is elapsed, update estimations for
169 // time-to-play.
170 UpdateEstimatedPlayoutTimeBy10ms();
171
172 // Update timestamp for better estimate of time-to-play, for packets which
173 // are added to NACK list later on.
174 timestamp_last_decoded_rtp_ += sample_rate_khz_ * 10;
175 }
176 any_rtp_decoded_ = true;
177}
178
henrik.lundin91951862016-06-08 06:43:41 -0700179NackTracker::NackList NackTracker::GetNackList() const {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000180 return nack_list_;
181}
182
henrik.lundin91951862016-06-08 06:43:41 -0700183void NackTracker::Reset() {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000184 nack_list_.clear();
185
186 sequence_num_last_received_rtp_ = 0;
187 timestamp_last_received_rtp_ = 0;
188 any_rtp_received_ = false;
189 sequence_num_last_decoded_rtp_ = 0;
190 timestamp_last_decoded_rtp_ = 0;
191 any_rtp_decoded_ = false;
192 sample_rate_khz_ = kDefaultSampleRateKhz;
193 samples_per_packet_ = sample_rate_khz_ * kDefaultPacketSizeMs;
194}
195
henrik.lundin91951862016-06-08 06:43:41 -0700196void NackTracker::SetMaxNackListSize(size_t max_nack_list_size) {
kwibergaf476c72016-11-28 15:21:39 -0800197 RTC_CHECK_GT(max_nack_list_size, 0);
henrik.lundin48ed9302015-10-29 05:36:24 -0700198 // Ugly hack to get around the problem of passing static consts by reference.
henrik.lundin91951862016-06-08 06:43:41 -0700199 const size_t kNackListSizeLimitLocal = NackTracker::kNackListSizeLimit;
henrik.lundin48ed9302015-10-29 05:36:24 -0700200 RTC_CHECK_LE(max_nack_list_size, kNackListSizeLimitLocal);
201
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000202 max_nack_list_size_ = max_nack_list_size;
203 LimitNackListSize();
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000204}
205
henrik.lundin91951862016-06-08 06:43:41 -0700206void NackTracker::LimitNackListSize() {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000207 uint16_t limit = sequence_num_last_received_rtp_ -
henrik.lundin48ed9302015-10-29 05:36:24 -0700208 static_cast<uint16_t>(max_nack_list_size_) - 1;
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000209 nack_list_.erase(nack_list_.begin(), nack_list_.upper_bound(limit));
210}
211
henrik.lundin91951862016-06-08 06:43:41 -0700212int64_t NackTracker::TimeToPlay(uint32_t timestamp) const {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000213 uint32_t timestamp_increase = timestamp - timestamp_last_decoded_rtp_;
214 return timestamp_increase / sample_rate_khz_;
215}
216
217// We don't erase elements with time-to-play shorter than round-trip-time.
henrik.lundin91951862016-06-08 06:43:41 -0700218std::vector<uint16_t> NackTracker::GetNackList(
219 int64_t round_trip_time_ms) const {
henrik.lundin48ed9302015-10-29 05:36:24 -0700220 RTC_DCHECK_GE(round_trip_time_ms, 0);
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000221 std::vector<uint16_t> sequence_numbers;
222 for (NackList::const_iterator it = nack_list_.begin(); it != nack_list_.end();
henrik.lundin48ed9302015-10-29 05:36:24 -0700223 ++it) {
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000224 if (it->second.is_missing &&
225 it->second.time_to_play_ms > round_trip_time_ms)
226 sequence_numbers.push_back(it->first);
227 }
228 return sequence_numbers;
229}
230
turaj@webrtc.org7959e162013-09-12 18:30:26 +0000231} // namespace webrtc