henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2012 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 Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 11 | #ifndef MODULES_AUDIO_CODING_NETEQ_DELAY_MANAGER_H_ |
| 12 | #define MODULES_AUDIO_CODING_NETEQ_DELAY_MANAGER_H_ |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 13 | |
pbos@webrtc.org | 12dc1a3 | 2013-08-05 16:22:53 +0000 | [diff] [blame] | 14 | #include <string.h> // Provide access to size_t. |
| 15 | |
Jakob Ivarsson | db42ed2 | 2019-02-27 10:08:09 +0100 | [diff] [blame] | 16 | #include <deque> |
henrik.lundin | 8f8c96d | 2016-04-28 23:19:20 -0700 | [diff] [blame] | 17 | #include <memory> |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 18 | |
Minyue Li | 002fbb8 | 2018-10-04 11:31:03 +0200 | [diff] [blame] | 19 | #include "absl/types/optional.h" |
Ivo Creusen | 3ce44a3 | 2019-10-31 14:38:11 +0100 | [diff] [blame] | 20 | #include "api/neteq/tick_timer.h" |
Jakob Ivarsson | 1eb3d7e | 2019-02-21 15:42:31 +0100 | [diff] [blame] | 21 | #include "modules/audio_coding/neteq/histogram.h" |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 22 | #include "rtc_base/constructor_magic.h" |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 23 | |
| 24 | namespace webrtc { |
| 25 | |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 26 | class DelayManager { |
| 27 | public: |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 28 | DelayManager(int max_packets_in_buffer, |
Jakob Ivarsson | 1eb3d7e | 2019-02-21 15:42:31 +0100 | [diff] [blame] | 29 | int base_minimum_delay_ms, |
Jakob Ivarsson | db42ed2 | 2019-02-27 10:08:09 +0100 | [diff] [blame] | 30 | int histogram_quantile, |
Jakob Ivarsson | 7dff9f3 | 2020-11-11 15:26:10 +0100 | [diff] [blame] | 31 | absl::optional<int> resample_interval_ms, |
| 32 | int max_history_ms, |
Jakob Ivarsson | 1eb3d7e | 2019-02-21 15:42:31 +0100 | [diff] [blame] | 33 | const TickTimer* tick_timer, |
Jakob Ivarsson | db42ed2 | 2019-02-27 10:08:09 +0100 | [diff] [blame] | 34 | std::unique_ptr<Histogram> histogram); |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 35 | |
| 36 | // Create a DelayManager object. Notify the delay manager that the packet |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 37 | // buffer can hold no more than `max_packets_in_buffer` packets (i.e., this |
Jakob Ivarsson | 10403ae | 2018-11-27 15:45:20 +0100 | [diff] [blame] | 38 | // is the number of packet slots in the buffer) and that the target delay |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 39 | // should be greater than or equal to `base_minimum_delay_ms`. Supply a |
Jakob Ivarsson | 10403ae | 2018-11-27 15:45:20 +0100 | [diff] [blame] | 40 | // PeakDetector object to the DelayManager. |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 41 | static std::unique_ptr<DelayManager> Create(int max_packets_in_buffer, |
Jakob Ivarsson | 1eb3d7e | 2019-02-21 15:42:31 +0100 | [diff] [blame] | 42 | int base_minimum_delay_ms, |
Ivo Creusen | 53a31f7 | 2019-10-24 15:20:39 +0200 | [diff] [blame] | 43 | const TickTimer* tick_timer); |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 44 | |
pbos@webrtc.org | 2d1a55c | 2013-07-31 15:54:00 +0000 | [diff] [blame] | 45 | virtual ~DelayManager(); |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 46 | |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 47 | // Updates the delay manager with a new incoming packet, with `timestamp` from |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 48 | // the RTP header. This updates the statistics and a new target buffer level |
| 49 | // is calculated. Returns the relative delay if it can be calculated. If |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 50 | // `reset` is true, restarts the relative arrival delay calculation from this |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 51 | // packet. |
| 52 | virtual absl::optional<int> Update(uint32_t timestamp, |
| 53 | int sample_rate_hz, |
| 54 | bool reset = false); |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 55 | |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 56 | // Resets all state. |
Jakob Ivarsson | ff9f646 | 2020-10-07 12:46:42 +0000 | [diff] [blame] | 57 | virtual void Reset(); |
| 58 | |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 59 | // Gets the target buffer level in milliseconds. |
| 60 | virtual int TargetDelayMs() const; |
Jakob Ivarsson | ff9f646 | 2020-10-07 12:46:42 +0000 | [diff] [blame] | 61 | |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 62 | // Notifies the DelayManager of how much audio data is carried in each packet. |
| 63 | virtual int SetPacketAudioLength(int length_ms); |
Jakob Ivarsson | ff9f646 | 2020-10-07 12:46:42 +0000 | [diff] [blame] | 64 | |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 65 | // Accessors and mutators. |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 66 | // Assuming `delay` is in valid range. |
turaj@webrtc.org | f1efc57 | 2013-08-16 23:44:24 +0000 | [diff] [blame] | 67 | virtual bool SetMinimumDelay(int delay_ms); |
| 68 | virtual bool SetMaximumDelay(int delay_ms); |
Ruslan Burakov | edbea46 | 2019-02-04 16:17:31 +0100 | [diff] [blame] | 69 | virtual bool SetBaseMinimumDelay(int delay_ms); |
| 70 | virtual int GetBaseMinimumDelay() const; |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 71 | |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 72 | // These accessors are only intended for testing purposes. |
Ruslan Burakov | 4a68fb9 | 2019-02-13 14:25:39 +0100 | [diff] [blame] | 73 | int effective_minimum_delay_ms_for_test() const { |
| 74 | return effective_minimum_delay_ms_; |
| 75 | } |
Jakob Ivarsson | db42ed2 | 2019-02-27 10:08:09 +0100 | [diff] [blame] | 76 | int histogram_quantile() const { return histogram_quantile_; } |
Jakob Ivarsson | e9a2ee2 | 2019-05-22 16:54:09 +0200 | [diff] [blame] | 77 | Histogram* histogram() const { return histogram_.get(); } |
Jakob Ivarsson | db42ed2 | 2019-02-27 10:08:09 +0100 | [diff] [blame] | 78 | |
Minyue Li | 002fbb8 | 2018-10-04 11:31:03 +0200 | [diff] [blame] | 79 | private: |
Ruslan Burakov | 4a68fb9 | 2019-02-13 14:25:39 +0100 | [diff] [blame] | 80 | // Provides value which minimum delay can't exceed based on current buffer |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 81 | // size and given `maximum_delay_ms_`. Lower bound is a constant 0. |
Ruslan Burakov | 4a68fb9 | 2019-02-13 14:25:39 +0100 | [diff] [blame] | 82 | int MinimumDelayUpperBound() const; |
| 83 | |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 84 | // Updates `delay_history_`. |
Jakob Ivarsson | 74154e6 | 2019-08-22 15:00:16 +0200 | [diff] [blame] | 85 | void UpdateDelayHistory(int iat_delay_ms, |
| 86 | uint32_t timestamp, |
| 87 | int sample_rate_hz); |
Jakob Ivarsson | db42ed2 | 2019-02-27 10:08:09 +0100 | [diff] [blame] | 88 | |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 89 | // Calculate relative packet arrival delay from `delay_history_`. |
Jakob Ivarsson | db42ed2 | 2019-02-27 10:08:09 +0100 | [diff] [blame] | 90 | int CalculateRelativePacketArrivalDelay() const; |
| 91 | |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 92 | // Updates `effective_minimum_delay_ms_` delay based on current |
| 93 | // `minimum_delay_ms_`, `base_minimum_delay_ms_` and `maximum_delay_ms_` |
Ruslan Burakov | 4a68fb9 | 2019-02-13 14:25:39 +0100 | [diff] [blame] | 94 | // and buffer size. |
| 95 | void UpdateEffectiveMinimumDelay(); |
| 96 | |
Artem Titov | d00ce74 | 2021-07-28 20:00:17 +0200 | [diff] [blame^] | 97 | // Makes sure that `delay_ms` is less than maximum delay, if any maximum |
| 98 | // is set. Also, if possible check `delay_ms` to be less than 75% of |
| 99 | // `max_packets_in_buffer_`. |
Ruslan Burakov | 4a68fb9 | 2019-02-13 14:25:39 +0100 | [diff] [blame] | 100 | bool IsValidMinimumDelay(int delay_ms) const; |
| 101 | |
| 102 | bool IsValidBaseMinimumDelay(int delay_ms) const; |
Ruslan Burakov | edbea46 | 2019-02-04 16:17:31 +0100 | [diff] [blame] | 103 | |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 104 | bool first_packet_received_; |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 105 | // TODO(jakobi): set maximum buffer delay instead of number of packets. |
| 106 | const int max_packets_in_buffer_; |
Jakob Ivarsson | db42ed2 | 2019-02-27 10:08:09 +0100 | [diff] [blame] | 107 | std::unique_ptr<Histogram> histogram_; |
| 108 | const int histogram_quantile_; |
henrik.lundin | 8f8c96d | 2016-04-28 23:19:20 -0700 | [diff] [blame] | 109 | const TickTimer* tick_timer_; |
Jakob Ivarsson | 7dff9f3 | 2020-11-11 15:26:10 +0100 | [diff] [blame] | 110 | const absl::optional<int> resample_interval_ms_; |
| 111 | const int max_history_ms_; |
| 112 | |
Ruslan Burakov | 4a68fb9 | 2019-02-13 14:25:39 +0100 | [diff] [blame] | 113 | int base_minimum_delay_ms_; |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 114 | int effective_minimum_delay_ms_; // Used as lower bound for target delay. |
| 115 | int minimum_delay_ms_; // Externally set minimum delay. |
| 116 | int maximum_delay_ms_; // Externally set maximum allowed delay. |
Ruslan Burakov | 4a68fb9 | 2019-02-13 14:25:39 +0100 | [diff] [blame] | 117 | |
Jakob Ivarsson | 80fb978 | 2020-10-09 13:41:06 +0200 | [diff] [blame] | 118 | int packet_len_ms_ = 0; |
| 119 | std::unique_ptr<TickTimer::Stopwatch> |
| 120 | packet_iat_stopwatch_; // Time elapsed since last packet. |
| 121 | int target_level_ms_; // Currently preferred buffer level. |
| 122 | uint32_t last_timestamp_; // Timestamp for the last received packet. |
| 123 | int num_reordered_packets_ = 0; |
Jakob Ivarsson | 7dff9f3 | 2020-11-11 15:26:10 +0100 | [diff] [blame] | 124 | int max_delay_in_interval_ms_ = 0; |
| 125 | std::unique_ptr<TickTimer::Stopwatch> resample_stopwatch_; |
Jakob Ivarsson | 74154e6 | 2019-08-22 15:00:16 +0200 | [diff] [blame] | 126 | |
| 127 | struct PacketDelay { |
| 128 | int iat_delay_ms; |
| 129 | uint32_t timestamp; |
| 130 | }; |
| 131 | std::deque<PacketDelay> delay_history_; |
| 132 | |
henrikg | 3c089d7 | 2015-09-16 05:37:44 -0700 | [diff] [blame] | 133 | RTC_DISALLOW_COPY_AND_ASSIGN(DelayManager); |
henrik.lundin@webrtc.org | d94659d | 2013-01-29 12:09:21 +0000 | [diff] [blame] | 134 | }; |
| 135 | |
| 136 | } // namespace webrtc |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 137 | #endif // MODULES_AUDIO_CODING_NETEQ_DELAY_MANAGER_H_ |