| /* | 
 |  *  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 |