/* * Copyright (c) 2021 The WebRTC project authors. All Rights Reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #include "modules/remote_bitrate_estimator/packet_arrival_map.h" #include #include #include "api/units/timestamp.h" #include "rtc_base/checks.h" namespace webrtc { void PacketArrivalTimeMap::AddPacket(int64_t sequence_number, Timestamp arrival_time) { RTC_DCHECK_GE(arrival_time, Timestamp::Zero()); if (!has_seen_packet()) { // First packet. Reallocate(kMinCapacity); begin_sequence_number_ = sequence_number; end_sequence_number_ = sequence_number + 1; arrival_times_[Index(sequence_number)] = arrival_time; return; } if (sequence_number >= begin_sequence_number() && sequence_number < end_sequence_number()) { // The packet is within the buffer - no need to expand it. arrival_times_[Index(sequence_number)] = arrival_time; return; } if (sequence_number < begin_sequence_number()) { // The packet goes before the current buffer. Expand to add packet, but only // if it fits within kMaxNumberOfPackets. int64_t new_size = end_sequence_number() - sequence_number; if (new_size > kMaxNumberOfPackets) { // Don't expand the buffer further, as that would remove newly received // packets. return; } AdjustToSize(new_size); arrival_times_[Index(sequence_number)] = arrival_time; SetNotReceived(sequence_number + 1, begin_sequence_number_); begin_sequence_number_ = sequence_number; return; } // The packet goes after the buffer. RTC_DCHECK_GE(sequence_number, end_sequence_number_); int64_t new_end_sequence_number = sequence_number + 1; if (new_end_sequence_number >= end_sequence_number_ + kMaxNumberOfPackets) { // All old packets have to be removed. begin_sequence_number_ = sequence_number; end_sequence_number_ = new_end_sequence_number; arrival_times_[Index(sequence_number)] = arrival_time; return; } if (begin_sequence_number_ < new_end_sequence_number - kMaxNumberOfPackets) { // Remove oldest entries begin_sequence_number_ = new_end_sequence_number - kMaxNumberOfPackets; RTC_DCHECK_GT(end_sequence_number_, begin_sequence_number_); } AdjustToSize(new_end_sequence_number - begin_sequence_number_); // Packets can be received out-of-order. If this isn't the next expected // packet, add enough placeholders to fill the gap. SetNotReceived(end_sequence_number_, sequence_number); end_sequence_number_ = new_end_sequence_number; arrival_times_[Index(sequence_number)] = arrival_time; } void PacketArrivalTimeMap::SetNotReceived( int64_t begin_sequence_number_inclusive, int64_t end_sequence_number_exclusive) { static constexpr Timestamp value = Timestamp::MinusInfinity(); int begin_index = Index(begin_sequence_number_inclusive); int end_index = Index(end_sequence_number_exclusive); if (begin_index <= end_index) { // Entries to clear are in single block: // [......{-----}....] std::fill(arrival_times_.get() + begin_index, arrival_times_.get() + end_index, value); } else { // Entries to clear span across arrival_times_ border: // [--}..........{---] std::fill(arrival_times_.get() + begin_index, arrival_times_.get() + capacity(), value); std::fill(arrival_times_.get(), arrival_times_.get() + end_index, value); } } void PacketArrivalTimeMap::RemoveOldPackets(int64_t sequence_number, Timestamp arrival_time_limit) { int64_t check_to = std::min(sequence_number, end_sequence_number_); while (begin_sequence_number_ < check_to && arrival_times_[Index(begin_sequence_number_)] <= arrival_time_limit) { ++begin_sequence_number_; } AdjustToSize(end_sequence_number_ - begin_sequence_number_); } void PacketArrivalTimeMap::EraseTo(int64_t sequence_number) { if (sequence_number < begin_sequence_number_) { return; } if (sequence_number >= end_sequence_number_) { // Erase all. begin_sequence_number_ = end_sequence_number_; return; } // Remove some. begin_sequence_number_ = sequence_number; AdjustToSize(end_sequence_number_ - begin_sequence_number_); } void PacketArrivalTimeMap::AdjustToSize(int new_size) { if (new_size > capacity()) { int new_capacity = capacity(); while (new_capacity < new_size) new_capacity *= 2; Reallocate(new_capacity); } if (capacity() > std::max(kMinCapacity, 4 * new_size)) { int new_capacity = capacity(); while (new_capacity > 2 * std::max(new_size, kMinCapacity)) { new_capacity /= 2; } Reallocate(new_capacity); } RTC_DCHECK_LE(new_size, capacity()); } void PacketArrivalTimeMap::Reallocate(int new_capacity) { int new_capacity_minus_1 = new_capacity - 1; // Check capacity is a power of 2. RTC_DCHECK_EQ(new_capacity & new_capacity_minus_1, 0); // Create uninitialized memory. // All valid entries should be set by `AddPacket` before use. void* raw = operator new[](new_capacity * sizeof(Timestamp)); Timestamp* new_buffer = static_cast(raw); for (int64_t sequence_number = begin_sequence_number_; sequence_number < end_sequence_number_; ++sequence_number) { new_buffer[sequence_number & new_capacity_minus_1] = arrival_times_[sequence_number & capacity_minus_1_]; } arrival_times_.reset(new_buffer); capacity_minus_1_ = new_capacity_minus_1; } } // namespace webrtc