| /* |
| * Copyright (c) 2017 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/audio_coding/neteq/statistics_calculator.h" |
| |
| #include "test/gtest.h" |
| |
| namespace webrtc { |
| |
| TEST(LifetimeStatistics, TotalSamplesReceived) { |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| stats.DecodedOutputPlayed(); |
| for (int i = 0; i < 10; ++i) { |
| stats.IncreaseCounter(480, 48000); // 10 ms at 48 kHz. |
| } |
| EXPECT_EQ(10 * 480u, stats.GetLifetimeStatistics().total_samples_received); |
| } |
| |
| TEST(LifetimeStatistics, SamplesConcealed) { |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| stats.DecodedOutputPlayed(); |
| stats.ExpandedVoiceSamples(100, false); |
| stats.ExpandedNoiseSamples(17, false); |
| EXPECT_EQ(100u + 17u, stats.GetLifetimeStatistics().concealed_samples); |
| } |
| |
| // This test verifies that a negative correction of concealed_samples does not |
| // result in a decrease in the stats value (because stats-consuming applications |
| // would not expect the value to decrease). Instead, the correction should be |
| // made to future increments to the stat. |
| TEST(LifetimeStatistics, SamplesConcealedCorrection) { |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| stats.DecodedOutputPlayed(); |
| stats.ExpandedVoiceSamples(100, false); |
| EXPECT_EQ(100u, stats.GetLifetimeStatistics().concealed_samples); |
| stats.ExpandedVoiceSamplesCorrection(-10); |
| // Do not subtract directly, but keep the correction for later. |
| EXPECT_EQ(100u, stats.GetLifetimeStatistics().concealed_samples); |
| stats.ExpandedVoiceSamplesCorrection(20); |
| // The total correction is 20 - 10. |
| EXPECT_EQ(110u, stats.GetLifetimeStatistics().concealed_samples); |
| |
| // Also test correction done to the next ExpandedVoiceSamples call. |
| stats.ExpandedVoiceSamplesCorrection(-17); |
| EXPECT_EQ(110u, stats.GetLifetimeStatistics().concealed_samples); |
| stats.ExpandedVoiceSamples(100, false); |
| EXPECT_EQ(110u + 100u - 17u, stats.GetLifetimeStatistics().concealed_samples); |
| } |
| |
| // This test verifies that neither "accelerate" nor "pre-emptive expand" reults |
| // in a modification to concealed_samples stats. Only PLC operations (i.e., |
| // "expand" and "merge") should affect the stat. |
| TEST(LifetimeStatistics, NoUpdateOnTimeStretch) { |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| stats.DecodedOutputPlayed(); |
| stats.ExpandedVoiceSamples(100, false); |
| stats.AcceleratedSamples(4711); |
| stats.PreemptiveExpandedSamples(17); |
| stats.ExpandedVoiceSamples(100, false); |
| EXPECT_EQ(200u, stats.GetLifetimeStatistics().concealed_samples); |
| } |
| |
| TEST(StatisticsCalculator, ExpandedSamplesCorrection) { |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| stats.DecodedOutputPlayed(); |
| NetEqNetworkStatistics stats_output; |
| constexpr int kSampleRateHz = 48000; |
| constexpr int k10MsSamples = kSampleRateHz / 100; |
| constexpr int kPacketSizeMs = 20; |
| constexpr size_t kSamplesPerPacket = kPacketSizeMs * kSampleRateHz / 1000; |
| |
| // Advance time by 10 ms. |
| stats.IncreaseCounter(k10MsSamples, kSampleRateHz); |
| |
| stats.GetNetworkStatistics(kSamplesPerPacket, &stats_output); |
| |
| EXPECT_EQ(0u, stats_output.expand_rate); |
| EXPECT_EQ(0u, stats_output.speech_expand_rate); |
| |
| // Correct with a negative value. |
| stats.ExpandedVoiceSamplesCorrection(-100); |
| stats.ExpandedNoiseSamplesCorrection(-100); |
| stats.IncreaseCounter(k10MsSamples, kSampleRateHz); |
| stats.GetNetworkStatistics(kSamplesPerPacket, &stats_output); |
| // Expect no change, since negative values are disallowed. |
| EXPECT_EQ(0u, stats_output.expand_rate); |
| EXPECT_EQ(0u, stats_output.speech_expand_rate); |
| |
| // Correct with a positive value. |
| stats.ExpandedVoiceSamplesCorrection(50); |
| stats.ExpandedNoiseSamplesCorrection(200); |
| stats.IncreaseCounter(k10MsSamples, kSampleRateHz); |
| stats.GetNetworkStatistics(kSamplesPerPacket, &stats_output); |
| // Calculate expected rates in Q14. Expand rate is noise + voice, while |
| // speech expand rate is only voice. |
| EXPECT_EQ(((50u + 200u) << 14) / k10MsSamples, stats_output.expand_rate); |
| EXPECT_EQ((50u << 14) / k10MsSamples, stats_output.speech_expand_rate); |
| } |
| |
| TEST(StatisticsCalculator, RelativePacketArrivalDelay) { |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| |
| stats.RelativePacketArrivalDelay(50); |
| NetEqLifetimeStatistics stats_output = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(50u, stats_output.relative_packet_arrival_delay_ms); |
| |
| stats.RelativePacketArrivalDelay(20); |
| stats_output = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(70u, stats_output.relative_packet_arrival_delay_ms); |
| } |
| |
| TEST(StatisticsCalculator, ReceivedPacket) { |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| |
| stats.ReceivedPacket(); |
| NetEqLifetimeStatistics stats_output = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(1u, stats_output.jitter_buffer_packets_received); |
| |
| stats.ReceivedPacket(); |
| stats_output = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(2u, stats_output.jitter_buffer_packets_received); |
| } |
| |
| TEST(StatisticsCalculator, InterruptionCounter) { |
| constexpr int fs_khz = 48; |
| constexpr int fs_hz = fs_khz * 1000; |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| stats.DecodedOutputPlayed(); |
| stats.EndExpandEvent(fs_hz); |
| auto lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(0, lts.interruption_count); |
| EXPECT_EQ(0, lts.total_interruption_duration_ms); |
| |
| // Add an event that is shorter than 150 ms. Should not be logged. |
| stats.ExpandedVoiceSamples(10 * fs_khz, false); // 10 ms. |
| stats.ExpandedNoiseSamples(139 * fs_khz, false); // 139 ms. |
| stats.EndExpandEvent(fs_hz); |
| lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(0, lts.interruption_count); |
| |
| // Add an event that is longer than 150 ms. Should be logged. |
| stats.ExpandedVoiceSamples(140 * fs_khz, false); // 140 ms. |
| stats.ExpandedNoiseSamples(11 * fs_khz, false); // 11 ms. |
| stats.EndExpandEvent(fs_hz); |
| lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(1, lts.interruption_count); |
| EXPECT_EQ(151, lts.total_interruption_duration_ms); |
| |
| // Add one more long event. |
| stats.ExpandedVoiceSamples(100 * fs_khz, false); // 100 ms. |
| stats.ExpandedNoiseSamples(5000 * fs_khz, false); // 5000 ms. |
| stats.EndExpandEvent(fs_hz); |
| lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(2, lts.interruption_count); |
| EXPECT_EQ(5100 + 151, lts.total_interruption_duration_ms); |
| } |
| |
| TEST(StatisticsCalculator, InterruptionCounterDoNotLogBeforeDecoding) { |
| constexpr int fs_khz = 48; |
| constexpr int fs_hz = fs_khz * 1000; |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| |
| // Add an event that is longer than 150 ms. Should normally be logged, but we |
| // have not called DecodedOutputPlayed() yet, so it shouldn't this time. |
| stats.ExpandedVoiceSamples(151 * fs_khz, false); // 151 ms. |
| stats.EndExpandEvent(fs_hz); |
| auto lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(0, lts.interruption_count); |
| |
| // Call DecodedOutputPlayed(). Logging should happen after this. |
| stats.DecodedOutputPlayed(); |
| |
| // Add one more long event. |
| stats.ExpandedVoiceSamples(151 * fs_khz, false); // 151 ms. |
| stats.EndExpandEvent(fs_hz); |
| lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(1, lts.interruption_count); |
| } |
| |
| TEST(StatisticsCalculator, DiscardedPackets) { |
| TickTimer timer; |
| StatisticsCalculator statistics_calculator(&timer); |
| EXPECT_EQ(0u, |
| statistics_calculator.GetLifetimeStatistics().packets_discarded); |
| |
| statistics_calculator.PacketsDiscarded(1); |
| EXPECT_EQ(1u, |
| statistics_calculator.GetLifetimeStatistics().packets_discarded); |
| |
| statistics_calculator.PacketsDiscarded(10); |
| EXPECT_EQ(11u, |
| statistics_calculator.GetLifetimeStatistics().packets_discarded); |
| |
| // Calling `SecondaryPacketsDiscarded` does not modify `packets_discarded`. |
| statistics_calculator.SecondaryPacketsDiscarded(1); |
| EXPECT_EQ(11u, |
| statistics_calculator.GetLifetimeStatistics().packets_discarded); |
| |
| // Calling `FlushedPacketBuffer` does not modify `packets_discarded`. |
| statistics_calculator.FlushedPacketBuffer(); |
| EXPECT_EQ(11u, |
| statistics_calculator.GetLifetimeStatistics().packets_discarded); |
| } |
| |
| TEST(StatisticsCalculator, JitterBufferDelay) { |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| NetEqLifetimeStatistics lts; |
| lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(lts.total_processing_delay_us, 0ul); |
| stats.JitterBufferDelay(/*num_samples=*/480, |
| /*waiting_time_ms=*/90ul, |
| /*target_delay_ms=*/80ul, |
| /*unlimited_target_delay_ms=*/70, |
| /*processing_delay_us=*/100 * 1000ul); |
| lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(lts.jitter_buffer_delay_ms / 480, 90ul); |
| EXPECT_EQ(lts.jitter_buffer_target_delay_ms / 480, 80ul); |
| EXPECT_EQ(lts.jitter_buffer_minimum_delay_ms / 480, 70ul); |
| EXPECT_EQ(lts.total_processing_delay_us / 480, 100 * 1000ul); |
| EXPECT_EQ(lts.jitter_buffer_emitted_count, 480ul); |
| stats.JitterBufferDelay(/*num_samples=*/480, |
| /*waiting_time_ms=*/90ul, |
| /*target_delay_ms=*/80ul, |
| /*unlimited_target_delay_ms=*/70, |
| /*processing_delay_us=*/100 * 1000ul); |
| lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(lts.jitter_buffer_delay_ms / 960, 90ul); |
| EXPECT_EQ(lts.jitter_buffer_target_delay_ms / 960, 80ul); |
| EXPECT_EQ(lts.jitter_buffer_minimum_delay_ms / 960, 70ul); |
| EXPECT_EQ(lts.total_processing_delay_us / 960, 100 * 1000ul); |
| EXPECT_EQ(lts.jitter_buffer_emitted_count, 960ul); |
| } |
| |
| TEST(StatisticsCalculator, CountStatsAfterFirstDecodedPacket) { |
| TickTimer timer; |
| StatisticsCalculator stats(&timer); |
| stats.IncreaseCounter(/*num_samples=*/480, /*fs_hz=*/48000); |
| stats.ExpandedVoiceSamples(/*num_samples=*/480, |
| /*is_new_concealment_event=*/true); |
| NetEqLifetimeStatistics lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(lts.total_samples_received, 0u); |
| EXPECT_EQ(lts.concealed_samples, 0u); |
| stats.DecodedOutputPlayed(); |
| stats.IncreaseCounter(/*num_samples=*/480, /*fs_hz=*/48000); |
| lts = stats.GetLifetimeStatistics(); |
| EXPECT_EQ(lts.total_samples_received, 480u); |
| } |
| |
| } // namespace webrtc |