/* * Copyright (c) 2014 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/acm2/acm_send_test.h" #include #include #include #include #include #include "absl/strings/match.h" #include "absl/strings/str_cat.h" #include "absl/strings/string_view.h" #include "api/audio_codecs/audio_encoder.h" #include "api/audio_codecs/audio_format.h" #include "api/audio_codecs/builtin_audio_encoder_factory.h" #include "api/environment/environment_factory.h" #include "modules/audio_coding/include/audio_coding_module.h" #include "modules/audio_coding/include/audio_coding_module_typedefs.h" #include "modules/audio_coding/neteq/tools/input_audio_file.h" #include "modules/rtp_rtcp/source/rtp_packet_received.h" #include "rtc_base/checks.h" namespace webrtc { namespace test { AcmSendTestOldApi::AcmSendTestOldApi(InputAudioFile* audio_source, int source_rate_hz, int test_duration_ms) : clock_(0), env_(CreateEnvironment(&clock_)), acm_(AudioCodingModule::Create()), audio_source_(audio_source), source_rate_hz_(source_rate_hz), input_block_size_samples_( static_cast(source_rate_hz_ * kBlockSizeMs / 1000)), codec_registered_(false), test_duration_ms_(test_duration_ms), frame_type_(AudioFrameType::kAudioFrameSpeech), payload_type_(0), timestamp_(0), sequence_number_(0) { input_frame_.sample_rate_hz_ = source_rate_hz_; input_frame_.num_channels_ = 1; input_frame_.samples_per_channel_ = input_block_size_samples_; RTC_DCHECK_LE(input_block_size_samples_ * input_frame_.num_channels_, AudioFrame::kMaxDataSizeSamples); acm_->RegisterTransportCallback(this); } AcmSendTestOldApi::~AcmSendTestOldApi() = default; bool AcmSendTestOldApi::RegisterCodec(absl::string_view payload_name, int clockrate_hz, int num_channels, int payload_type, int frame_size_samples) { SdpAudioFormat format(payload_name, clockrate_hz, num_channels); if (absl::EqualsIgnoreCase(payload_name, "g722")) { RTC_CHECK_EQ(16000, clockrate_hz); format.clockrate_hz = 8000; } else if (absl::EqualsIgnoreCase(payload_name, "opus")) { RTC_CHECK(num_channels == 1 || num_channels == 2); if (num_channels == 2) { format.parameters["stereo"] = "1"; } format.num_channels = 2; } format.parameters["ptime"] = absl::StrCat( CheckedDivExact(frame_size_samples, CheckedDivExact(clockrate_hz, 1000))); auto factory = CreateBuiltinAudioEncoderFactory(); acm_->SetEncoder( factory->Create(env_, format, {.payload_type = payload_type})); codec_registered_ = true; input_frame_.num_channels_ = num_channels; RTC_DCHECK_LE(input_block_size_samples_ * input_frame_.num_channels_, AudioFrame::kMaxDataSizeSamples); return codec_registered_; } void AcmSendTestOldApi::RegisterExternalCodec( std::unique_ptr external_speech_encoder) { input_frame_.num_channels_ = external_speech_encoder->NumChannels(); acm_->SetEncoder(std::move(external_speech_encoder)); RTC_DCHECK_LE(input_block_size_samples_ * input_frame_.num_channels_, AudioFrame::kMaxDataSizeSamples); codec_registered_ = true; } std::unique_ptr AcmSendTestOldApi::NextPacket() { RTC_DCHECK(codec_registered_); if (filter_.test(static_cast(payload_type_))) { // This payload type should be filtered out. Since the payload type is the // same throughout the whole test run, no packet at all will be delivered. // We can just as well signal that the test is over by returning NULL. return nullptr; } // Insert audio and process until one packet is produced. while (clock_.TimeInMilliseconds() < test_duration_ms_) { clock_.AdvanceTimeMilliseconds(kBlockSizeMs); RTC_CHECK(audio_source_->Read( input_block_size_samples_ * input_frame_.num_channels_, input_frame_.mutable_data())); data_to_send_ = false; RTC_CHECK_GE(acm_->Add10MsData(input_frame_), 0); input_frame_.timestamp_ += static_cast(input_block_size_samples_); if (data_to_send_) { // Encoded packet received. return CreatePacket(); } } // Test ended. return nullptr; } // This method receives the callback from ACM when a new packet is produced. int32_t AcmSendTestOldApi::SendData( AudioFrameType frame_type, uint8_t payload_type, uint32_t timestamp, const uint8_t* payload_data, size_t payload_len_bytes, int64_t /* absolute_capture_timestamp_ms */) { // Store the packet locally. frame_type_ = frame_type; payload_type_ = payload_type; timestamp_ = timestamp; last_payload_vec_.assign(payload_data, payload_data + payload_len_bytes); RTC_DCHECK_EQ(last_payload_vec_.size(), payload_len_bytes); data_to_send_ = true; return 0; } std::unique_ptr AcmSendTestOldApi::CreatePacket() { auto rtp_packet = std::make_unique(); // Populate the header. rtp_packet->SetPayloadType(payload_type_); rtp_packet->SetSequenceNumber(sequence_number_); rtp_packet->SetTimestamp(timestamp_); rtp_packet->SetSsrc(0x12345678); ++sequence_number_; rtp_packet->SetPayload(last_payload_vec_); rtp_packet->set_arrival_time(clock_.CurrentTime()); return rtp_packet; } } // namespace test } // namespace webrtc