henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2014 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 | #include "modules/audio_coding/acm2/acm_send_test.h" |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 12 | |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 13 | #include <stdio.h> |
| 14 | #include <string.h> |
| 15 | |
Fredrik Solenberg | f693bfa | 2018-12-11 12:22:10 +0100 | [diff] [blame] | 16 | #include "absl/strings/match.h" |
Ali Tofigh | 714e3cb | 2022-07-20 12:53:07 +0200 | [diff] [blame] | 17 | #include "absl/strings/string_view.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 18 | #include "api/audio_codecs/audio_encoder.h" |
Karl Wiberg | 5817d3d | 2018-04-06 10:06:42 +0200 | [diff] [blame] | 19 | #include "api/audio_codecs/builtin_audio_decoder_factory.h" |
Karl Wiberg | 801500c | 2018-08-16 15:01:12 +0200 | [diff] [blame] | 20 | #include "api/audio_codecs/builtin_audio_encoder_factory.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 21 | #include "modules/audio_coding/include/audio_coding_module.h" |
| 22 | #include "modules/audio_coding/neteq/tools/input_audio_file.h" |
| 23 | #include "modules/audio_coding/neteq/tools/packet.h" |
| 24 | #include "rtc_base/checks.h" |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 25 | #include "rtc_base/string_encode.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 26 | #include "test/gtest.h" |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 27 | |
| 28 | namespace webrtc { |
| 29 | namespace test { |
| 30 | |
| 31 | AcmSendTestOldApi::AcmSendTestOldApi(InputAudioFile* audio_source, |
| 32 | int source_rate_hz, |
| 33 | int test_duration_ms) |
| 34 | : clock_(0), |
Henrik Lundin | 84f7569 | 2023-02-01 12:07:10 +0000 | [diff] [blame] | 35 | acm_(webrtc::AudioCodingModule::Create()), |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 36 | audio_source_(audio_source), |
| 37 | source_rate_hz_(source_rate_hz), |
Peter Kasting | dce40cf | 2015-08-24 14:52:23 -0700 | [diff] [blame] | 38 | input_block_size_samples_( |
| 39 | static_cast<size_t>(source_rate_hz_ * kBlockSizeMs / 1000)), |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 40 | codec_registered_(false), |
| 41 | test_duration_ms_(test_duration_ms), |
Niels Möller | c936cb6 | 2019-03-19 14:10:16 +0100 | [diff] [blame] | 42 | frame_type_(AudioFrameType::kAudioFrameSpeech), |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 43 | payload_type_(0), |
| 44 | timestamp_(0), |
| 45 | sequence_number_(0) { |
| 46 | input_frame_.sample_rate_hz_ = source_rate_hz_; |
| 47 | input_frame_.num_channels_ = 1; |
| 48 | input_frame_.samples_per_channel_ = input_block_size_samples_; |
Mirko Bonadei | 25ab322 | 2021-07-08 20:08:20 +0200 | [diff] [blame] | 49 | RTC_DCHECK_LE(input_block_size_samples_ * input_frame_.num_channels_, |
| 50 | AudioFrame::kMaxDataSizeSamples); |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 51 | acm_->RegisterTransportCallback(this); |
| 52 | } |
| 53 | |
kwiberg | 65fc8b9 | 2016-08-29 10:05:24 -0700 | [diff] [blame] | 54 | AcmSendTestOldApi::~AcmSendTestOldApi() = default; |
| 55 | |
Ali Tofigh | 714e3cb | 2022-07-20 12:53:07 +0200 | [diff] [blame] | 56 | bool AcmSendTestOldApi::RegisterCodec(absl::string_view payload_name, |
Fredrik Solenberg | f693bfa | 2018-12-11 12:22:10 +0100 | [diff] [blame] | 57 | int clockrate_hz, |
| 58 | int num_channels, |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 59 | int payload_type, |
| 60 | int frame_size_samples) { |
Fredrik Solenberg | f693bfa | 2018-12-11 12:22:10 +0100 | [diff] [blame] | 61 | SdpAudioFormat format(payload_name, clockrate_hz, num_channels); |
| 62 | if (absl::EqualsIgnoreCase(payload_name, "g722")) { |
| 63 | RTC_CHECK_EQ(16000, clockrate_hz); |
| 64 | format.clockrate_hz = 8000; |
| 65 | } else if (absl::EqualsIgnoreCase(payload_name, "opus")) { |
| 66 | RTC_CHECK(num_channels == 1 || num_channels == 2); |
| 67 | if (num_channels == 2) { |
| 68 | format.parameters["stereo"] = "1"; |
| 69 | } |
| 70 | format.num_channels = 2; |
| 71 | } |
Karl Wiberg | 801500c | 2018-08-16 15:01:12 +0200 | [diff] [blame] | 72 | format.parameters["ptime"] = rtc::ToString(rtc::CheckedDivExact( |
Fredrik Solenberg | f693bfa | 2018-12-11 12:22:10 +0100 | [diff] [blame] | 73 | frame_size_samples, rtc::CheckedDivExact(clockrate_hz, 1000))); |
| 74 | auto factory = CreateBuiltinAudioEncoderFactory(); |
Karl Wiberg | 801500c | 2018-08-16 15:01:12 +0200 | [diff] [blame] | 75 | acm_->SetEncoder( |
| 76 | factory->MakeAudioEncoder(payload_type, format, absl::nullopt)); |
| 77 | codec_registered_ = true; |
Fredrik Solenberg | f693bfa | 2018-12-11 12:22:10 +0100 | [diff] [blame] | 78 | input_frame_.num_channels_ = num_channels; |
Mirko Bonadei | 25ab322 | 2021-07-08 20:08:20 +0200 | [diff] [blame] | 79 | RTC_DCHECK_LE(input_block_size_samples_ * input_frame_.num_channels_, |
| 80 | AudioFrame::kMaxDataSizeSamples); |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 81 | return codec_registered_; |
| 82 | } |
| 83 | |
Karl Wiberg | 801500c | 2018-08-16 15:01:12 +0200 | [diff] [blame] | 84 | void AcmSendTestOldApi::RegisterExternalCodec( |
| 85 | std::unique_ptr<AudioEncoder> external_speech_encoder) { |
Karl Wiberg | 7e0c7d4 | 2015-05-18 14:52:29 +0200 | [diff] [blame] | 86 | input_frame_.num_channels_ = external_speech_encoder->NumChannels(); |
Karl Wiberg | 801500c | 2018-08-16 15:01:12 +0200 | [diff] [blame] | 87 | acm_->SetEncoder(std::move(external_speech_encoder)); |
Mirko Bonadei | 25ab322 | 2021-07-08 20:08:20 +0200 | [diff] [blame] | 88 | RTC_DCHECK_LE(input_block_size_samples_ * input_frame_.num_channels_, |
| 89 | AudioFrame::kMaxDataSizeSamples); |
Karl Wiberg | 801500c | 2018-08-16 15:01:12 +0200 | [diff] [blame] | 90 | codec_registered_ = true; |
Karl Wiberg | 7e0c7d4 | 2015-05-18 14:52:29 +0200 | [diff] [blame] | 91 | } |
| 92 | |
henrik.lundin | 46ba49c | 2016-05-24 22:50:47 -0700 | [diff] [blame] | 93 | std::unique_ptr<Packet> AcmSendTestOldApi::NextPacket() { |
Mirko Bonadei | 25ab322 | 2021-07-08 20:08:20 +0200 | [diff] [blame] | 94 | RTC_DCHECK(codec_registered_); |
pkasting@chromium.org | 0e81fdf | 2015-02-02 23:54:03 +0000 | [diff] [blame] | 95 | if (filter_.test(static_cast<size_t>(payload_type_))) { |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 96 | // This payload type should be filtered out. Since the payload type is the |
| 97 | // same throughout the whole test run, no packet at all will be delivered. |
| 98 | // We can just as well signal that the test is over by returning NULL. |
henrik.lundin | 46ba49c | 2016-05-24 22:50:47 -0700 | [diff] [blame] | 99 | return nullptr; |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 100 | } |
| 101 | // Insert audio and process until one packet is produced. |
| 102 | while (clock_.TimeInMilliseconds() < test_duration_ms_) { |
| 103 | clock_.AdvanceTimeMilliseconds(kBlockSizeMs); |
Alex Loiko | 6543881 | 2019-02-22 10:13:44 +0100 | [diff] [blame] | 104 | RTC_CHECK(audio_source_->Read( |
| 105 | input_block_size_samples_ * input_frame_.num_channels_, |
| 106 | input_frame_.mutable_data())); |
henrik.lundin@webrtc.org | f56c162 | 2015-03-02 12:29:30 +0000 | [diff] [blame] | 107 | data_to_send_ = false; |
henrikg | 91d6ede | 2015-09-17 00:24:34 -0700 | [diff] [blame] | 108 | RTC_CHECK_GE(acm_->Add10MsData(input_frame_), 0); |
Peter Kasting | b7e5054 | 2015-06-11 12:55:50 -0700 | [diff] [blame] | 109 | input_frame_.timestamp_ += static_cast<uint32_t>(input_block_size_samples_); |
henrik.lundin@webrtc.org | f56c162 | 2015-03-02 12:29:30 +0000 | [diff] [blame] | 110 | if (data_to_send_) { |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 111 | // Encoded packet received. |
| 112 | return CreatePacket(); |
| 113 | } |
| 114 | } |
| 115 | // Test ended. |
henrik.lundin | 46ba49c | 2016-05-24 22:50:47 -0700 | [diff] [blame] | 116 | return nullptr; |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 117 | } |
| 118 | |
| 119 | // This method receives the callback from ACM when a new packet is produced. |
Niels Möller | c35b6e6 | 2019-04-25 16:31:18 +0200 | [diff] [blame] | 120 | int32_t AcmSendTestOldApi::SendData(AudioFrameType frame_type, |
| 121 | uint8_t payload_type, |
| 122 | uint32_t timestamp, |
| 123 | const uint8_t* payload_data, |
Minyue Li | ff0e4db | 2020-01-23 13:45:50 +0100 | [diff] [blame] | 124 | size_t payload_len_bytes, |
| 125 | int64_t absolute_capture_timestamp_ms) { |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 126 | // Store the packet locally. |
| 127 | frame_type_ = frame_type; |
| 128 | payload_type_ = payload_type; |
| 129 | timestamp_ = timestamp; |
| 130 | last_payload_vec_.assign(payload_data, payload_data + payload_len_bytes); |
Mirko Bonadei | 25ab322 | 2021-07-08 20:08:20 +0200 | [diff] [blame] | 131 | RTC_DCHECK_EQ(last_payload_vec_.size(), payload_len_bytes); |
henrik.lundin@webrtc.org | f56c162 | 2015-03-02 12:29:30 +0000 | [diff] [blame] | 132 | data_to_send_ = true; |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 133 | return 0; |
| 134 | } |
| 135 | |
henrik.lundin | 46ba49c | 2016-05-24 22:50:47 -0700 | [diff] [blame] | 136 | std::unique_ptr<Packet> AcmSendTestOldApi::CreatePacket() { |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 137 | const size_t kRtpHeaderSize = 12; |
Danil Chapovalov | b4100ad | 2021-06-16 14:23:22 +0200 | [diff] [blame] | 138 | rtc::CopyOnWriteBuffer packet_buffer(last_payload_vec_.size() + |
| 139 | kRtpHeaderSize); |
| 140 | uint8_t* packet_memory = packet_buffer.MutableData(); |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 141 | // Populate the header bytes. |
| 142 | packet_memory[0] = 0x80; |
pkasting@chromium.org | 0e81fdf | 2015-02-02 23:54:03 +0000 | [diff] [blame] | 143 | packet_memory[1] = static_cast<uint8_t>(payload_type_); |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 144 | packet_memory[2] = (sequence_number_ >> 8) & 0xFF; |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 145 | packet_memory[3] = (sequence_number_)&0xFF; |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 146 | packet_memory[4] = (timestamp_ >> 24) & 0xFF; |
| 147 | packet_memory[5] = (timestamp_ >> 16) & 0xFF; |
| 148 | packet_memory[6] = (timestamp_ >> 8) & 0xFF; |
| 149 | packet_memory[7] = timestamp_ & 0xFF; |
| 150 | // Set SSRC to 0x12345678. |
| 151 | packet_memory[8] = 0x12; |
| 152 | packet_memory[9] = 0x34; |
| 153 | packet_memory[10] = 0x56; |
| 154 | packet_memory[11] = 0x78; |
| 155 | |
| 156 | ++sequence_number_; |
| 157 | |
| 158 | // Copy the payload data. |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 159 | memcpy(packet_memory + kRtpHeaderSize, &last_payload_vec_[0], |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 160 | last_payload_vec_.size()); |
Danil Chapovalov | b4100ad | 2021-06-16 14:23:22 +0200 | [diff] [blame] | 161 | auto packet = std::make_unique<Packet>(std::move(packet_buffer), |
| 162 | clock_.TimeInMilliseconds()); |
henrik.lundin | 46ba49c | 2016-05-24 22:50:47 -0700 | [diff] [blame] | 163 | RTC_DCHECK(packet); |
| 164 | RTC_DCHECK(packet->valid_header()); |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 165 | return packet; |
| 166 | } |
| 167 | |
| 168 | } // namespace test |
| 169 | } // namespace webrtc |