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 | |
| 13 | #include <assert.h> |
| 14 | #include <stdio.h> |
| 15 | #include <string.h> |
| 16 | |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 17 | #include "api/audio_codecs/audio_encoder.h" |
Karl Wiberg | 5817d3d | 2018-04-06 10:06:42 +0200 | [diff] [blame] | 18 | #include "api/audio_codecs/builtin_audio_decoder_factory.h" |
Karl Wiberg | 801500c | 2018-08-16 15:01:12 +0200 | [diff] [blame] | 19 | #include "api/audio_codecs/builtin_audio_encoder_factory.h" |
| 20 | #include "modules/audio_coding/codecs/audio_format_conversion.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" |
Karl Wiberg | 801500c | 2018-08-16 15:01:12 +0200 | [diff] [blame] | 25 | #include "rtc_base/stringencode.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), |
Karl Wiberg | 5817d3d | 2018-04-06 10:06:42 +0200 | [diff] [blame] | 35 | acm_(webrtc::AudioCodingModule::Create([this] { |
| 36 | AudioCodingModule::Config config; |
| 37 | config.clock = &clock_; |
| 38 | config.decoder_factory = CreateBuiltinAudioDecoderFactory(); |
| 39 | return config; |
| 40 | }())), |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 41 | audio_source_(audio_source), |
| 42 | source_rate_hz_(source_rate_hz), |
Peter Kasting | dce40cf | 2015-08-24 14:52:23 -0700 | [diff] [blame] | 43 | input_block_size_samples_( |
| 44 | static_cast<size_t>(source_rate_hz_ * kBlockSizeMs / 1000)), |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 45 | codec_registered_(false), |
| 46 | test_duration_ms_(test_duration_ms), |
| 47 | frame_type_(kAudioFrameSpeech), |
| 48 | payload_type_(0), |
| 49 | timestamp_(0), |
| 50 | sequence_number_(0) { |
| 51 | input_frame_.sample_rate_hz_ = source_rate_hz_; |
| 52 | input_frame_.num_channels_ = 1; |
| 53 | input_frame_.samples_per_channel_ = input_block_size_samples_; |
| 54 | assert(input_block_size_samples_ * input_frame_.num_channels_ <= |
| 55 | AudioFrame::kMaxDataSizeSamples); |
| 56 | acm_->RegisterTransportCallback(this); |
| 57 | } |
| 58 | |
kwiberg | 65fc8b9 | 2016-08-29 10:05:24 -0700 | [diff] [blame] | 59 | AcmSendTestOldApi::~AcmSendTestOldApi() = default; |
| 60 | |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 61 | bool AcmSendTestOldApi::RegisterCodec(const char* payload_name, |
| 62 | int sampling_freq_hz, |
| 63 | int channels, |
| 64 | int payload_type, |
| 65 | int frame_size_samples) { |
Karl Wiberg | 7e0c7d4 | 2015-05-18 14:52:29 +0200 | [diff] [blame] | 66 | CodecInst codec; |
henrikg | 91d6ede | 2015-09-17 00:24:34 -0700 | [diff] [blame] | 67 | RTC_CHECK_EQ(0, AudioCodingModule::Codec(payload_name, &codec, |
| 68 | sampling_freq_hz, channels)); |
Karl Wiberg | 7e0c7d4 | 2015-05-18 14:52:29 +0200 | [diff] [blame] | 69 | codec.pltype = payload_type; |
| 70 | codec.pacsize = frame_size_samples; |
Karl Wiberg | 801500c | 2018-08-16 15:01:12 +0200 | [diff] [blame] | 71 | auto factory = CreateBuiltinAudioEncoderFactory(); |
| 72 | SdpAudioFormat format = CodecInstToSdp(codec); |
| 73 | format.parameters["ptime"] = rtc::ToString(rtc::CheckedDivExact( |
| 74 | frame_size_samples, rtc::CheckedDivExact(sampling_freq_hz, 1000))); |
| 75 | acm_->SetEncoder( |
| 76 | factory->MakeAudioEncoder(payload_type, format, absl::nullopt)); |
| 77 | codec_registered_ = true; |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 78 | input_frame_.num_channels_ = channels; |
| 79 | assert(input_block_size_samples_ * input_frame_.num_channels_ <= |
| 80 | AudioFrame::kMaxDataSizeSamples); |
| 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)); |
Karl Wiberg | 7e0c7d4 | 2015-05-18 14:52:29 +0200 | [diff] [blame] | 88 | assert(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() { |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 94 | assert(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); |
yujo | 36b1a5f | 2017-06-12 12:45:32 -0700 | [diff] [blame] | 104 | RTC_CHECK(audio_source_->Read(input_block_size_samples_, |
| 105 | input_frame_.mutable_data())); |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 106 | if (input_frame_.num_channels_ > 1) { |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 107 | InputAudioFile::DuplicateInterleaved( |
| 108 | input_frame_.data(), input_block_size_samples_, |
| 109 | input_frame_.num_channels_, input_frame_.mutable_data()); |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 110 | } |
henrik.lundin@webrtc.org | f56c162 | 2015-03-02 12:29:30 +0000 | [diff] [blame] | 111 | data_to_send_ = false; |
henrikg | 91d6ede | 2015-09-17 00:24:34 -0700 | [diff] [blame] | 112 | RTC_CHECK_GE(acm_->Add10MsData(input_frame_), 0); |
Peter Kasting | b7e5054 | 2015-06-11 12:55:50 -0700 | [diff] [blame] | 113 | 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] | 114 | if (data_to_send_) { |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 115 | // Encoded packet received. |
| 116 | return CreatePacket(); |
| 117 | } |
| 118 | } |
| 119 | // Test ended. |
henrik.lundin | 46ba49c | 2016-05-24 22:50:47 -0700 | [diff] [blame] | 120 | return nullptr; |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 121 | } |
| 122 | |
| 123 | // This method receives the callback from ACM when a new packet is produced. |
| 124 | int32_t AcmSendTestOldApi::SendData( |
| 125 | FrameType frame_type, |
| 126 | uint8_t payload_type, |
| 127 | uint32_t timestamp, |
| 128 | const uint8_t* payload_data, |
pkasting@chromium.org | 4591fbd | 2014-11-20 22:28:14 +0000 | [diff] [blame] | 129 | size_t payload_len_bytes, |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 130 | const RTPFragmentationHeader* fragmentation) { |
| 131 | // Store the packet locally. |
| 132 | frame_type_ = frame_type; |
| 133 | payload_type_ = payload_type; |
| 134 | timestamp_ = timestamp; |
| 135 | last_payload_vec_.assign(payload_data, payload_data + payload_len_bytes); |
| 136 | assert(last_payload_vec_.size() == payload_len_bytes); |
henrik.lundin@webrtc.org | f56c162 | 2015-03-02 12:29:30 +0000 | [diff] [blame] | 137 | data_to_send_ = true; |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 138 | return 0; |
| 139 | } |
| 140 | |
henrik.lundin | 46ba49c | 2016-05-24 22:50:47 -0700 | [diff] [blame] | 141 | std::unique_ptr<Packet> AcmSendTestOldApi::CreatePacket() { |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 142 | const size_t kRtpHeaderSize = 12; |
| 143 | size_t allocated_bytes = last_payload_vec_.size() + kRtpHeaderSize; |
| 144 | uint8_t* packet_memory = new uint8_t[allocated_bytes]; |
| 145 | // Populate the header bytes. |
| 146 | packet_memory[0] = 0x80; |
pkasting@chromium.org | 0e81fdf | 2015-02-02 23:54:03 +0000 | [diff] [blame] | 147 | packet_memory[1] = static_cast<uint8_t>(payload_type_); |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 148 | packet_memory[2] = (sequence_number_ >> 8) & 0xFF; |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 149 | packet_memory[3] = (sequence_number_)&0xFF; |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 150 | packet_memory[4] = (timestamp_ >> 24) & 0xFF; |
| 151 | packet_memory[5] = (timestamp_ >> 16) & 0xFF; |
| 152 | packet_memory[6] = (timestamp_ >> 8) & 0xFF; |
| 153 | packet_memory[7] = timestamp_ & 0xFF; |
| 154 | // Set SSRC to 0x12345678. |
| 155 | packet_memory[8] = 0x12; |
| 156 | packet_memory[9] = 0x34; |
| 157 | packet_memory[10] = 0x56; |
| 158 | packet_memory[11] = 0x78; |
| 159 | |
| 160 | ++sequence_number_; |
| 161 | |
| 162 | // Copy the payload data. |
Yves Gerey | 665174f | 2018-06-19 15:03:05 +0200 | [diff] [blame] | 163 | memcpy(packet_memory + kRtpHeaderSize, &last_payload_vec_[0], |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 164 | last_payload_vec_.size()); |
henrik.lundin | 46ba49c | 2016-05-24 22:50:47 -0700 | [diff] [blame] | 165 | std::unique_ptr<Packet> packet( |
| 166 | new Packet(packet_memory, allocated_bytes, clock_.TimeInMilliseconds())); |
| 167 | RTC_DCHECK(packet); |
| 168 | RTC_DCHECK(packet->valid_header()); |
henrik.lundin@webrtc.org | 0e6e4d2 | 2014-09-23 12:05:34 +0000 | [diff] [blame] | 169 | return packet; |
| 170 | } |
| 171 | |
| 172 | } // namespace test |
| 173 | } // namespace webrtc |