|  | /* | 
|  | *  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 <algorithm> | 
|  | #include <cstdint> | 
|  |  | 
|  | #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<Timestamp*>(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 |