|  | /* | 
|  | *  Copyright (c) 2018 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 "absl/memory/memory.h" | 
|  | #include "api/audio_codecs/audio_decoder_factory_template.h" | 
|  | #include "api/audio_codecs/audio_encoder_factory_template.h" | 
|  | #include "api/audio_codecs/opus/audio_decoder_opus.h" | 
|  | #include "api/audio_codecs/opus/audio_encoder_opus.h" | 
|  | #include "api/test/loopback_media_transport.h" | 
|  | #include "api/test/mock_audio_mixer.h" | 
|  | #include "audio/audio_receive_stream.h" | 
|  | #include "audio/audio_send_stream.h" | 
|  | #include "call/test/mock_bitrate_allocator.h" | 
|  | #include "logging/rtc_event_log/rtc_event_log.h" | 
|  | #include "modules/audio_device/include/test_audio_device.h" | 
|  | #include "modules/audio_mixer/audio_mixer_impl.h" | 
|  | #include "modules/audio_processing/include/mock_audio_processing.h" | 
|  | #include "modules/utility/include/process_thread.h" | 
|  | #include "rtc_base/task_queue.h" | 
|  | #include "rtc_base/time_utils.h" | 
|  | #include "test/gtest.h" | 
|  | #include "test/mock_transport.h" | 
|  |  | 
|  | namespace webrtc { | 
|  | namespace test { | 
|  |  | 
|  | namespace { | 
|  | constexpr int kPayloadTypeOpus = 17; | 
|  | constexpr int kSamplingFrequency = 48000; | 
|  | constexpr int kNumChannels = 2; | 
|  | constexpr int kWantedSamples = 3000; | 
|  | constexpr int kTestTimeoutMs = 2 * rtc::kNumMillisecsPerSec; | 
|  |  | 
|  | class TestRenderer : public TestAudioDeviceModule::Renderer { | 
|  | public: | 
|  | TestRenderer(int sampling_frequency, int num_channels, size_t wanted_samples) | 
|  | : sampling_frequency_(sampling_frequency), | 
|  | num_channels_(num_channels), | 
|  | wanted_samples_(wanted_samples) {} | 
|  | ~TestRenderer() override = default; | 
|  |  | 
|  | int SamplingFrequency() const override { return sampling_frequency_; } | 
|  | int NumChannels() const override { return num_channels_; } | 
|  |  | 
|  | bool Render(rtc::ArrayView<const int16_t> data) override { | 
|  | if (data.size() >= wanted_samples_) { | 
|  | return false; | 
|  | } | 
|  | wanted_samples_ -= data.size(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | private: | 
|  | const int sampling_frequency_; | 
|  | const int num_channels_; | 
|  | size_t wanted_samples_; | 
|  | }; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | TEST(AudioWithMediaTransport, DeliversAudio) { | 
|  | std::unique_ptr<rtc::Thread> transport_thread = rtc::Thread::Create(); | 
|  | transport_thread->Start(); | 
|  | MediaTransportPair transport_pair(transport_thread.get()); | 
|  | MockTransport rtcp_send_transport; | 
|  | MockTransport send_transport; | 
|  | std::unique_ptr<RtcEventLog> null_event_log = RtcEventLog::CreateNull(); | 
|  | MockBitrateAllocator bitrate_allocator; | 
|  |  | 
|  | rtc::scoped_refptr<TestAudioDeviceModule> audio_device = | 
|  | TestAudioDeviceModule::CreateTestAudioDeviceModule( | 
|  | TestAudioDeviceModule::CreatePulsedNoiseCapturer( | 
|  | /* max_amplitude= */ 10000, kSamplingFrequency, kNumChannels), | 
|  | absl::make_unique<TestRenderer>(kSamplingFrequency, kNumChannels, | 
|  | kWantedSamples)); | 
|  |  | 
|  | AudioState::Config audio_config; | 
|  | audio_config.audio_mixer = AudioMixerImpl::Create(); | 
|  | // TODO(nisse): Is a mock AudioProcessing enough? | 
|  | audio_config.audio_processing = | 
|  | new rtc::RefCountedObject<MockAudioProcessing>(); | 
|  | audio_config.audio_device_module = audio_device; | 
|  | rtc::scoped_refptr<AudioState> audio_state = AudioState::Create(audio_config); | 
|  |  | 
|  | // TODO(nisse): Use some lossless codec? | 
|  | const SdpAudioFormat audio_format("opus", kSamplingFrequency, kNumChannels); | 
|  |  | 
|  | // Setup receive stream; | 
|  | webrtc::AudioReceiveStream::Config receive_config; | 
|  | // TODO(nisse): Update AudioReceiveStream to not require rtcp_send_transport | 
|  | // when a MediaTransport is provided. | 
|  | receive_config.rtcp_send_transport = &rtcp_send_transport; | 
|  | receive_config.media_transport = transport_pair.first(); | 
|  | receive_config.decoder_map.emplace(kPayloadTypeOpus, audio_format); | 
|  | receive_config.decoder_factory = | 
|  | CreateAudioDecoderFactory<AudioDecoderOpus>(); | 
|  |  | 
|  | std::unique_ptr<ProcessThread> receive_process_thread = | 
|  | ProcessThread::Create("audio recv thread"); | 
|  |  | 
|  | webrtc::internal::AudioReceiveStream receive_stream( | 
|  | /*rtp_stream_receiver_controller=*/nullptr, | 
|  | /*packet_router=*/nullptr, receive_process_thread.get(), receive_config, | 
|  | audio_state, null_event_log.get()); | 
|  |  | 
|  | // TODO(nisse): Update AudioSendStream to not require send_transport when a | 
|  | // MediaTransport is provided. | 
|  | AudioSendStream::Config send_config(&send_transport, transport_pair.second()); | 
|  | send_config.send_codec_spec = | 
|  | AudioSendStream::Config::SendCodecSpec(kPayloadTypeOpus, audio_format); | 
|  | send_config.encoder_factory = CreateAudioEncoderFactory<AudioEncoderOpus>(); | 
|  | rtc::TaskQueue send_tq("audio send queue"); | 
|  | std::unique_ptr<ProcessThread> send_process_thread = | 
|  | ProcessThread::Create("audio send thread"); | 
|  | webrtc::internal::AudioSendStream send_stream( | 
|  | send_config, audio_state, &send_tq, send_process_thread.get(), | 
|  | /*transport=*/nullptr, &bitrate_allocator, null_event_log.get(), | 
|  | /*rtcp_rtt_stats=*/nullptr, absl::optional<RtpState>()); | 
|  |  | 
|  | audio_device->Init();  // Starts thread. | 
|  | audio_device->RegisterAudioCallback(audio_state->audio_transport()); | 
|  |  | 
|  | receive_stream.Start(); | 
|  | send_stream.Start(); | 
|  | audio_device->StartPlayout(); | 
|  | audio_device->StartRecording(); | 
|  |  | 
|  | EXPECT_TRUE(audio_device->WaitForPlayoutEnd(kTestTimeoutMs)); | 
|  |  | 
|  | audio_device->StopRecording(); | 
|  | audio_device->StopPlayout(); | 
|  | receive_stream.Stop(); | 
|  | send_stream.Stop(); | 
|  | } | 
|  |  | 
|  | }  // namespace test | 
|  | }  // namespace webrtc |