2013-07-10 00:45:36 +00:00
|
|
|
/*
|
2016-02-10 07:54:43 -08:00
|
|
|
* Copyright 2012 The WebRTC project authors. All Rights Reserved.
|
2013-07-10 00:45:36 +00:00
|
|
|
*
|
2016-02-10 07:54:43 -08:00
|
|
|
* 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.
|
2013-07-10 00:45:36 +00:00
|
|
|
*/
|
|
|
|
|
|
2018-11-28 16:47:49 +01:00
|
|
|
#include <stddef.h>
|
2019-07-05 19:08:33 +02:00
|
|
|
|
2018-11-28 16:47:49 +01:00
|
|
|
#include <cstdint>
|
2016-04-27 06:47:29 -07:00
|
|
|
#include <memory>
|
2013-07-10 00:45:36 +00:00
|
|
|
#include <string>
|
2015-12-12 01:37:01 +01:00
|
|
|
#include <utility>
|
2018-11-28 16:47:49 +01:00
|
|
|
#include <vector>
|
2013-07-10 00:45:36 +00:00
|
|
|
|
2019-02-19 12:49:22 -08:00
|
|
|
#include "absl/algorithm/container.h"
|
2018-11-28 16:47:49 +01:00
|
|
|
#include "absl/memory/memory.h"
|
|
|
|
|
#include "absl/types/optional.h"
|
|
|
|
|
#include "api/audio_options.h"
|
2019-01-11 09:11:00 -08:00
|
|
|
#include "api/crypto/crypto_options.h"
|
|
|
|
|
#include "api/crypto/frame_decryptor_interface.h"
|
|
|
|
|
#include "api/crypto/frame_encryptor_interface.h"
|
|
|
|
|
#include "api/dtmf_sender_interface.h"
|
|
|
|
|
#include "api/media_stream_interface.h"
|
|
|
|
|
#include "api/rtc_error.h"
|
2019-08-07 12:24:53 +02:00
|
|
|
#include "api/rtc_event_log/rtc_event_log.h"
|
2019-01-11 09:11:00 -08:00
|
|
|
#include "api/rtp_parameters.h"
|
2019-01-25 20:26:48 +01:00
|
|
|
#include "api/scoped_refptr.h"
|
2018-10-04 14:22:34 -07:00
|
|
|
#include "api/test/fake_frame_decryptor.h"
|
|
|
|
|
#include "api/test/fake_frame_encryptor.h"
|
2019-04-17 07:38:40 +02:00
|
|
|
#include "api/video/builtin_video_bitrate_allocator_factory.h"
|
2018-11-28 16:47:49 +01:00
|
|
|
#include "media/base/codec.h"
|
2019-01-11 09:11:00 -08:00
|
|
|
#include "media/base/fake_media_engine.h"
|
|
|
|
|
#include "media/base/media_channel.h"
|
|
|
|
|
#include "media/base/media_config.h"
|
|
|
|
|
#include "media/base/media_engine.h"
|
|
|
|
|
#include "media/base/rtp_data_engine.h"
|
|
|
|
|
#include "media/base/stream_params.h"
|
|
|
|
|
#include "media/base/test_utils.h"
|
|
|
|
|
#include "media/engine/fake_webrtc_call.h"
|
|
|
|
|
#include "p2p/base/dtls_transport_internal.h"
|
|
|
|
|
#include "p2p/base/fake_dtls_transport.h"
|
|
|
|
|
#include "p2p/base/p2p_constants.h"
|
2019-02-11 10:29:19 +01:00
|
|
|
#include "pc/audio_rtp_receiver.h"
|
2019-01-11 09:11:00 -08:00
|
|
|
#include "pc/audio_track.h"
|
2018-11-28 16:47:49 +01:00
|
|
|
#include "pc/channel.h"
|
2019-01-11 09:11:00 -08:00
|
|
|
#include "pc/channel_manager.h"
|
|
|
|
|
#include "pc/dtls_srtp_transport.h"
|
|
|
|
|
#include "pc/local_audio_source.h"
|
|
|
|
|
#include "pc/media_stream.h"
|
2019-02-16 02:07:05 +01:00
|
|
|
#include "pc/remote_audio_source.h"
|
2019-01-11 09:11:00 -08:00
|
|
|
#include "pc/rtp_receiver.h"
|
|
|
|
|
#include "pc/rtp_sender.h"
|
|
|
|
|
#include "pc/rtp_transport_internal.h"
|
|
|
|
|
#include "pc/test/fake_video_track_source.h"
|
2019-02-11 10:29:19 +01:00
|
|
|
#include "pc/video_rtp_receiver.h"
|
2019-01-11 09:11:00 -08:00
|
|
|
#include "pc/video_track.h"
|
2018-11-28 16:47:49 +01:00
|
|
|
#include "rtc_base/checks.h"
|
2017-09-15 06:47:31 +02:00
|
|
|
#include "rtc_base/gunit.h"
|
2018-11-28 16:47:49 +01:00
|
|
|
#include "rtc_base/third_party/sigslot/sigslot.h"
|
|
|
|
|
#include "rtc_base/thread.h"
|
2017-09-15 06:47:31 +02:00
|
|
|
#include "test/gmock.h"
|
|
|
|
|
#include "test/gtest.h"
|
2013-07-10 00:45:36 +00:00
|
|
|
|
|
|
|
|
using ::testing::_;
|
2019-02-19 12:49:22 -08:00
|
|
|
using ::testing::ContainerEq;
|
2013-07-10 00:45:36 +00:00
|
|
|
using ::testing::Exactly;
|
2016-05-02 16:20:01 -07:00
|
|
|
using ::testing::InvokeWithoutArgs;
|
2016-03-16 19:07:43 -07:00
|
|
|
using ::testing::Return;
|
2019-02-19 12:49:22 -08:00
|
|
|
using RidList = std::vector<std::string>;
|
2013-07-10 00:45:36 +00:00
|
|
|
|
2017-02-01 20:27:00 -08:00
|
|
|
namespace {
|
|
|
|
|
|
2018-03-02 11:34:10 -08:00
|
|
|
static const char kStreamId1[] = "local_stream_1";
|
2013-07-10 00:45:36 +00:00
|
|
|
static const char kVideoTrackId[] = "video_1";
|
|
|
|
|
static const char kAudioTrackId[] = "audio_1";
|
Use suffixed {uint,int}{8,16,32,64}_t types.
Removes the use of uint8, etc. in favor of uint8_t.
BUG=webrtc:5024
R=henrik.lundin@webrtc.org, henrikg@webrtc.org, perkj@webrtc.org, solenberg@webrtc.org, stefan@webrtc.org, tina.legrand@webrtc.org
Review URL: https://codereview.webrtc.org/1362503003 .
Cr-Commit-Position: refs/heads/master@{#10196}
2015-10-07 12:23:21 +02:00
|
|
|
static const uint32_t kVideoSsrc = 98;
|
2015-11-25 11:26:01 -08:00
|
|
|
static const uint32_t kVideoSsrc2 = 100;
|
Use suffixed {uint,int}{8,16,32,64}_t types.
Removes the use of uint8, etc. in favor of uint8_t.
BUG=webrtc:5024
R=henrik.lundin@webrtc.org, henrikg@webrtc.org, perkj@webrtc.org, solenberg@webrtc.org, stefan@webrtc.org, tina.legrand@webrtc.org
Review URL: https://codereview.webrtc.org/1362503003 .
Cr-Commit-Position: refs/heads/master@{#10196}
2015-10-07 12:23:21 +02:00
|
|
|
static const uint32_t kAudioSsrc = 99;
|
2015-11-25 11:26:01 -08:00
|
|
|
static const uint32_t kAudioSsrc2 = 101;
|
2018-10-01 22:47:20 +02:00
|
|
|
static const uint32_t kVideoSsrcSimulcast = 102;
|
|
|
|
|
static const uint32_t kVideoSimulcastLayerCount = 2;
|
2017-02-01 20:27:00 -08:00
|
|
|
static const int kDefaultTimeout = 10000; // 10 seconds.
|
2019-05-20 19:31:53 +02:00
|
|
|
|
|
|
|
|
class MockSetStreamsObserver
|
|
|
|
|
: public webrtc::RtpSenderBase::SetStreamsObserver {
|
|
|
|
|
public:
|
2020-05-15 11:16:53 +02:00
|
|
|
MOCK_METHOD(void, OnSetStreams, (), (override));
|
2019-05-20 19:31:53 +02:00
|
|
|
};
|
|
|
|
|
|
2017-02-01 20:27:00 -08:00
|
|
|
} // namespace
|
2013-07-10 00:45:36 +00:00
|
|
|
|
|
|
|
|
namespace webrtc {
|
|
|
|
|
|
2019-02-19 12:49:22 -08:00
|
|
|
class RtpSenderReceiverTest
|
2019-04-09 15:11:12 +02:00
|
|
|
: public ::testing::Test,
|
|
|
|
|
public ::testing::WithParamInterface<std::pair<RidList, RidList>>,
|
2019-02-19 12:49:22 -08:00
|
|
|
public sigslot::has_slots<> {
|
2016-06-24 19:31:47 -07:00
|
|
|
public:
|
2016-06-27 16:30:35 -07:00
|
|
|
RtpSenderReceiverTest()
|
2018-01-12 10:49:35 -08:00
|
|
|
: network_thread_(rtc::Thread::Current()),
|
|
|
|
|
worker_thread_(rtc::Thread::Current()),
|
2019-04-17 07:38:40 +02:00
|
|
|
video_bitrate_allocator_factory_(
|
|
|
|
|
webrtc::CreateBuiltinVideoBitrateAllocatorFactory()),
|
2018-01-12 10:49:35 -08:00
|
|
|
// Create fake media engine/etc. so we can create channels to use to
|
|
|
|
|
// test RtpSenders/RtpReceivers.
|
2016-06-27 16:30:35 -07:00
|
|
|
media_engine_(new cricket::FakeMediaEngine()),
|
2018-02-21 13:07:13 +01:00
|
|
|
fake_call_(),
|
2018-03-02 11:34:10 -08:00
|
|
|
local_stream_(MediaStream::Create(kStreamId1)) {
|
2021-04-01 16:49:42 +02:00
|
|
|
worker_thread_->Invoke<void>(RTC_FROM_HERE, [&]() {
|
|
|
|
|
channel_manager_ = cricket::ChannelManager::Create(
|
|
|
|
|
absl::WrapUnique(media_engine_),
|
|
|
|
|
std::make_unique<cricket::RtpDataEngine>(), false, worker_thread_,
|
|
|
|
|
network_thread_);
|
|
|
|
|
});
|
|
|
|
|
|
2016-12-13 11:29:11 -08:00
|
|
|
bool srtp_required = true;
|
2019-09-17 17:06:18 +02:00
|
|
|
rtp_dtls_transport_ = std::make_unique<cricket::FakeDtlsTransport>(
|
2018-03-30 10:48:35 -07:00
|
|
|
"fake_dtls_transport", cricket::ICE_CANDIDATE_COMPONENT_RTP);
|
|
|
|
|
rtp_transport_ = CreateDtlsSrtpTransport();
|
|
|
|
|
|
2021-04-01 16:49:42 +02:00
|
|
|
voice_channel_ = channel_manager_->CreateVoiceChannel(
|
2018-03-30 10:48:35 -07:00
|
|
|
&fake_call_, cricket::MediaConfig(), rtp_transport_.get(),
|
2020-06-16 16:39:13 +02:00
|
|
|
rtc::Thread::Current(), cricket::CN_AUDIO, srtp_required,
|
|
|
|
|
webrtc::CryptoOptions(), &ssrc_generator_, cricket::AudioOptions());
|
2021-04-01 16:49:42 +02:00
|
|
|
video_channel_ = channel_manager_->CreateVideoChannel(
|
2018-03-30 10:48:35 -07:00
|
|
|
&fake_call_, cricket::MediaConfig(), rtp_transport_.get(),
|
2020-06-16 16:39:13 +02:00
|
|
|
rtc::Thread::Current(), cricket::CN_VIDEO, srtp_required,
|
|
|
|
|
webrtc::CryptoOptions(), &ssrc_generator_, cricket::VideoOptions(),
|
|
|
|
|
video_bitrate_allocator_factory_.get());
|
2017-02-01 20:27:00 -08:00
|
|
|
voice_channel_->Enable(true);
|
|
|
|
|
video_channel_->Enable(true);
|
2016-06-27 16:30:35 -07:00
|
|
|
voice_media_channel_ = media_engine_->GetVoiceChannel(0);
|
|
|
|
|
video_media_channel_ = media_engine_->GetVideoChannel(0);
|
|
|
|
|
RTC_CHECK(voice_channel_);
|
|
|
|
|
RTC_CHECK(video_channel_);
|
|
|
|
|
RTC_CHECK(voice_media_channel_);
|
|
|
|
|
RTC_CHECK(video_media_channel_);
|
|
|
|
|
|
|
|
|
|
// Create streams for predefined SSRCs. Streams need to exist in order
|
|
|
|
|
// for the senders and receievers to apply parameters to them.
|
|
|
|
|
// Normally these would be created by SetLocalDescription and
|
|
|
|
|
// SetRemoteDescription.
|
|
|
|
|
voice_media_channel_->AddSendStream(
|
|
|
|
|
cricket::StreamParams::CreateLegacy(kAudioSsrc));
|
|
|
|
|
voice_media_channel_->AddRecvStream(
|
|
|
|
|
cricket::StreamParams::CreateLegacy(kAudioSsrc));
|
|
|
|
|
voice_media_channel_->AddSendStream(
|
|
|
|
|
cricket::StreamParams::CreateLegacy(kAudioSsrc2));
|
|
|
|
|
voice_media_channel_->AddRecvStream(
|
|
|
|
|
cricket::StreamParams::CreateLegacy(kAudioSsrc2));
|
|
|
|
|
video_media_channel_->AddSendStream(
|
|
|
|
|
cricket::StreamParams::CreateLegacy(kVideoSsrc));
|
|
|
|
|
video_media_channel_->AddRecvStream(
|
|
|
|
|
cricket::StreamParams::CreateLegacy(kVideoSsrc));
|
|
|
|
|
video_media_channel_->AddSendStream(
|
|
|
|
|
cricket::StreamParams::CreateLegacy(kVideoSsrc2));
|
|
|
|
|
video_media_channel_->AddRecvStream(
|
|
|
|
|
cricket::StreamParams::CreateLegacy(kVideoSsrc2));
|
2016-06-24 19:31:47 -07:00
|
|
|
}
|
2016-06-24 14:18:22 -07:00
|
|
|
|
2021-04-01 16:49:42 +02:00
|
|
|
~RtpSenderReceiverTest() {
|
|
|
|
|
audio_rtp_sender_ = nullptr;
|
|
|
|
|
video_rtp_sender_ = nullptr;
|
|
|
|
|
audio_rtp_receiver_ = nullptr;
|
|
|
|
|
video_rtp_receiver_ = nullptr;
|
|
|
|
|
local_stream_ = nullptr;
|
|
|
|
|
video_track_ = nullptr;
|
|
|
|
|
audio_track_ = nullptr;
|
|
|
|
|
worker_thread_->Invoke<void>(RTC_FROM_HERE,
|
|
|
|
|
[&]() { channel_manager_.reset(); });
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-30 10:48:35 -07:00
|
|
|
std::unique_ptr<webrtc::RtpTransportInternal> CreateDtlsSrtpTransport() {
|
2019-09-17 17:06:18 +02:00
|
|
|
auto dtls_srtp_transport = std::make_unique<webrtc::DtlsSrtpTransport>(
|
Use absl::make_unique and absl::WrapUnique directly
Instead of going through our wrappers in ptr_util.h.
This CL was generated by the following script:
git grep -l ptr_util | xargs perl -pi -e 's,#include "rtc_base/ptr_util.h",#include "absl/memory/memory.h",'
git grep -l MakeUnique | xargs perl -pi -e 's,\b(rtc::)?MakeUnique\b,absl::make_unique,g'
git grep -l WrapUnique | xargs perl -pi -e 's,\b(rtc::)?WrapUnique\b,absl::WrapUnique,g'
git checkout -- rtc_base/ptr_util{.h,_unittest.cc}
git cl format
Followed by manually adding dependencies on
//third_party/abseil-cpp/absl/memory until `gn check` stopped
complaining.
Bug: webrtc:9473
Change-Id: I89ccd363f070479b8c431eb2c3d404a46eaacc1c
Reviewed-on: https://webrtc-review.googlesource.com/86600
Commit-Queue: Karl Wiberg <kwiberg@webrtc.org>
Reviewed-by: Danil Chapovalov <danilchap@webrtc.org>
Cr-Commit-Position: refs/heads/master@{#23850}
2018-07-05 11:40:33 +02:00
|
|
|
/*rtcp_mux_required=*/true);
|
2018-03-30 10:48:35 -07:00
|
|
|
dtls_srtp_transport->SetDtlsTransports(rtp_dtls_transport_.get(),
|
|
|
|
|
/*rtcp_dtls_transport=*/nullptr);
|
|
|
|
|
return dtls_srtp_transport;
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-01 20:27:00 -08:00
|
|
|
// Needed to use DTMF sender.
|
|
|
|
|
void AddDtmfCodec() {
|
|
|
|
|
cricket::AudioSendParameters params;
|
|
|
|
|
const cricket::AudioCodec kTelephoneEventCodec(106, "telephone-event", 8000,
|
|
|
|
|
0, 1);
|
|
|
|
|
params.codecs.push_back(kTelephoneEventCodec);
|
|
|
|
|
voice_media_channel_->SetSendParameters(params);
|
|
|
|
|
}
|
2016-06-27 16:30:35 -07:00
|
|
|
|
2016-12-16 15:39:11 -08:00
|
|
|
void AddVideoTrack() { AddVideoTrack(false); }
|
|
|
|
|
|
|
|
|
|
void AddVideoTrack(bool is_screencast) {
|
2016-03-08 01:27:48 +01:00
|
|
|
rtc::scoped_refptr<VideoTrackSourceInterface> source(
|
2016-12-16 15:39:11 -08:00
|
|
|
FakeVideoTrackSource::Create(is_screencast));
|
2017-07-31 23:22:01 -07:00
|
|
|
video_track_ =
|
|
|
|
|
VideoTrack::Create(kVideoTrackId, source, rtc::Thread::Current());
|
2017-02-25 18:15:09 -08:00
|
|
|
EXPECT_TRUE(local_stream_->AddTrack(video_track_));
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
void CreateAudioRtpSender() { CreateAudioRtpSender(nullptr); }
|
|
|
|
|
|
2017-11-21 17:04:20 +01:00
|
|
|
void CreateAudioRtpSender(
|
|
|
|
|
const rtc::scoped_refptr<LocalAudioSource>& source) {
|
2016-06-27 16:30:35 -07:00
|
|
|
audio_track_ = AudioTrack::Create(kAudioTrackId, source);
|
2017-02-25 18:15:09 -08:00
|
|
|
EXPECT_TRUE(local_stream_->AddTrack(audio_track_));
|
2019-05-20 19:31:53 +02:00
|
|
|
std::unique_ptr<MockSetStreamsObserver> set_streams_observer =
|
2019-09-17 17:06:18 +02:00
|
|
|
std::make_unique<MockSetStreamsObserver>();
|
2018-01-12 10:49:35 -08:00
|
|
|
audio_rtp_sender_ =
|
2019-05-20 19:31:53 +02:00
|
|
|
AudioRtpSender::Create(worker_thread_, audio_track_->id(), nullptr,
|
|
|
|
|
set_streams_observer.get());
|
2018-06-25 13:03:36 -07:00
|
|
|
ASSERT_TRUE(audio_rtp_sender_->SetTrack(audio_track_));
|
2019-05-20 19:31:53 +02:00
|
|
|
EXPECT_CALL(*set_streams_observer, OnSetStreams());
|
|
|
|
|
audio_rtp_sender_->SetStreams({local_stream_->id()});
|
2018-11-13 16:26:05 -08:00
|
|
|
audio_rtp_sender_->SetMediaChannel(voice_media_channel_);
|
2015-11-25 11:26:01 -08:00
|
|
|
audio_rtp_sender_->SetSsrc(kAudioSsrc);
|
2017-02-01 20:27:00 -08:00
|
|
|
audio_rtp_sender_->GetOnDestroyedSignal()->connect(
|
|
|
|
|
this, &RtpSenderReceiverTest::OnAudioSenderDestroyed);
|
2016-06-27 16:30:35 -07:00
|
|
|
VerifyVoiceChannelInput();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2018-01-10 16:26:06 -08:00
|
|
|
void CreateAudioRtpSenderWithNoTrack() {
|
2019-02-19 15:20:21 -08:00
|
|
|
audio_rtp_sender_ =
|
2019-05-20 19:31:53 +02:00
|
|
|
AudioRtpSender::Create(worker_thread_, /*id=*/"", nullptr, nullptr);
|
2018-11-13 16:26:05 -08:00
|
|
|
audio_rtp_sender_->SetMediaChannel(voice_media_channel_);
|
2018-01-10 16:26:06 -08:00
|
|
|
}
|
|
|
|
|
|
2017-02-01 20:27:00 -08:00
|
|
|
void OnAudioSenderDestroyed() { audio_sender_destroyed_signal_fired_ = true; }
|
|
|
|
|
|
2018-05-16 16:02:32 -07:00
|
|
|
void CreateVideoRtpSender(uint32_t ssrc) {
|
|
|
|
|
CreateVideoRtpSender(false, ssrc);
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-16 15:39:11 -08:00
|
|
|
void CreateVideoRtpSender() { CreateVideoRtpSender(false); }
|
|
|
|
|
|
2019-02-19 12:49:22 -08:00
|
|
|
cricket::StreamParams CreateSimulcastStreamParams(int num_layers) {
|
2018-10-01 22:47:20 +02:00
|
|
|
std::vector<uint32_t> ssrcs;
|
2019-01-29 10:11:53 +01:00
|
|
|
ssrcs.reserve(num_layers);
|
2019-02-19 12:49:22 -08:00
|
|
|
for (int i = 0; i < num_layers; ++i) {
|
2018-10-01 22:47:20 +02:00
|
|
|
ssrcs.push_back(kVideoSsrcSimulcast + i);
|
2019-02-19 12:49:22 -08:00
|
|
|
}
|
|
|
|
|
return cricket::CreateSimStreamParams("cname", ssrcs);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint32_t CreateVideoRtpSender(const cricket::StreamParams& stream_params) {
|
2018-10-01 22:47:20 +02:00
|
|
|
video_media_channel_->AddSendStream(stream_params);
|
|
|
|
|
uint32_t primary_ssrc = stream_params.first_ssrc();
|
|
|
|
|
CreateVideoRtpSender(primary_ssrc);
|
2019-02-19 12:49:22 -08:00
|
|
|
return primary_ssrc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint32_t CreateVideoRtpSenderWithSimulcast(
|
|
|
|
|
int num_layers = kVideoSimulcastLayerCount) {
|
|
|
|
|
return CreateVideoRtpSender(CreateSimulcastStreamParams(num_layers));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint32_t CreateVideoRtpSenderWithSimulcast(
|
|
|
|
|
const std::vector<std::string>& rids) {
|
|
|
|
|
cricket::StreamParams stream_params =
|
|
|
|
|
CreateSimulcastStreamParams(rids.size());
|
|
|
|
|
std::vector<cricket::RidDescription> rid_descriptions;
|
|
|
|
|
absl::c_transform(
|
|
|
|
|
rids, std::back_inserter(rid_descriptions), [](const std::string& rid) {
|
|
|
|
|
return cricket::RidDescription(rid, cricket::RidDirection::kSend);
|
|
|
|
|
});
|
|
|
|
|
stream_params.set_rids(rid_descriptions);
|
|
|
|
|
return CreateVideoRtpSender(stream_params);
|
2018-10-01 22:47:20 +02:00
|
|
|
}
|
|
|
|
|
|
2018-05-16 16:02:32 -07:00
|
|
|
void CreateVideoRtpSender(bool is_screencast, uint32_t ssrc = kVideoSsrc) {
|
2016-12-16 15:39:11 -08:00
|
|
|
AddVideoTrack(is_screencast);
|
2019-05-20 19:31:53 +02:00
|
|
|
std::unique_ptr<MockSetStreamsObserver> set_streams_observer =
|
2019-09-17 17:06:18 +02:00
|
|
|
std::make_unique<MockSetStreamsObserver>();
|
2019-05-20 19:31:53 +02:00
|
|
|
video_rtp_sender_ = VideoRtpSender::Create(
|
|
|
|
|
worker_thread_, video_track_->id(), set_streams_observer.get());
|
2018-06-25 13:03:36 -07:00
|
|
|
ASSERT_TRUE(video_rtp_sender_->SetTrack(video_track_));
|
2019-05-20 19:31:53 +02:00
|
|
|
EXPECT_CALL(*set_streams_observer, OnSetStreams());
|
|
|
|
|
video_rtp_sender_->SetStreams({local_stream_->id()});
|
2018-11-13 16:26:05 -08:00
|
|
|
video_rtp_sender_->SetMediaChannel(video_media_channel_);
|
2018-05-16 16:02:32 -07:00
|
|
|
video_rtp_sender_->SetSsrc(ssrc);
|
|
|
|
|
VerifyVideoChannelInput(ssrc);
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
2018-01-10 16:26:06 -08:00
|
|
|
void CreateVideoRtpSenderWithNoTrack() {
|
2019-05-20 19:31:53 +02:00
|
|
|
video_rtp_sender_ =
|
|
|
|
|
VideoRtpSender::Create(worker_thread_, /*id=*/"", nullptr);
|
2018-11-13 16:26:05 -08:00
|
|
|
video_rtp_sender_->SetMediaChannel(video_media_channel_);
|
2018-01-10 16:26:06 -08:00
|
|
|
}
|
|
|
|
|
|
2015-09-28 16:53:55 -07:00
|
|
|
void DestroyAudioRtpSender() {
|
|
|
|
|
audio_rtp_sender_ = nullptr;
|
2016-06-27 16:30:35 -07:00
|
|
|
VerifyVoiceChannelNoInput();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2015-09-28 16:53:55 -07:00
|
|
|
void DestroyVideoRtpSender() {
|
|
|
|
|
video_rtp_sender_ = nullptr;
|
2016-06-27 16:30:35 -07:00
|
|
|
VerifyVideoChannelNoInput();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2017-11-21 13:41:51 +01:00
|
|
|
void CreateAudioRtpReceiver(
|
|
|
|
|
std::vector<rtc::scoped_refptr<MediaStreamInterface>> streams = {}) {
|
2019-01-31 21:38:12 +01:00
|
|
|
audio_rtp_receiver_ =
|
[Unified Plan] Don't end audio tracks when SSRC changes.
The RemoteAudioSource has an AudioDataProxy that acts as a sink, passing
along data from AudioRecvStreams to the RemoteAudioSource. If an SSRC is
changed (or other reconfiguration happens) with SDP, the recv stream and
proxy get recreated.
In Plan B, because remote tracks maps 1:1 with SSRCs, it made sense to
end remote track/audio source in response to this. In Plan B, a new
receiver, with a new track and a new proxy would be created for the new
SSRC.
In Unified Plan however, remote tracks correspond to m= sections. The
remote track should only end on port:0 (or RTCP BYE or timeout, etc),
not because the recv stream of an m= section is recreated. The code
already supports changing SSRC and this is working correctly, but
because ~AudioDataProxy() would end the source this would cause the
MediaStreamTrack of the receiver to end (even though the media engine
is still processing the remote audio stream correctly under the hood).
This issue only happened on audio tracks, and because of timing of
PostTasks the track would kEnd in Chromium *after* promise.then().
This CL fixes that issue by not ending the source when the proxy is
destroyed. Destroying a recv stream is a temporary action in Unified
Plan, unless stopped. Tests are added ensuring tracks are kLive.
I have manually verified that this CL fixes the issue and that both
audio and video is flowing through the entire pipeline:
https://jsfiddle.net/henbos/h21xec97/122/
Bug: chromium:1121454
Change-Id: Ic21ac8ea263ccf021b96a14d3e4e3b24eb756c86
Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/214136
Commit-Queue: Henrik Boström <hbos@webrtc.org>
Reviewed-by: Harald Alvestrand <hta@webrtc.org>
Cr-Commit-Position: refs/heads/master@{#33645}
2021-04-08 07:25:38 +02:00
|
|
|
new AudioRtpReceiver(rtc::Thread::Current(), kAudioTrackId, streams,
|
|
|
|
|
/*is_unified_plan=*/true);
|
2018-11-13 16:26:05 -08:00
|
|
|
audio_rtp_receiver_->SetMediaChannel(voice_media_channel_);
|
2018-01-17 17:41:02 -08:00
|
|
|
audio_rtp_receiver_->SetupMediaChannel(kAudioSsrc);
|
2016-03-24 03:16:19 -07:00
|
|
|
audio_track_ = audio_rtp_receiver_->audio_track();
|
2016-06-27 16:30:35 -07:00
|
|
|
VerifyVoiceChannelOutput();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2017-11-21 13:41:51 +01:00
|
|
|
void CreateVideoRtpReceiver(
|
|
|
|
|
std::vector<rtc::scoped_refptr<MediaStreamInterface>> streams = {}) {
|
2019-01-31 21:38:12 +01:00
|
|
|
video_rtp_receiver_ =
|
|
|
|
|
new VideoRtpReceiver(rtc::Thread::Current(), kVideoTrackId, streams);
|
2018-11-13 16:26:05 -08:00
|
|
|
video_rtp_receiver_->SetMediaChannel(video_media_channel_);
|
2018-01-17 17:41:02 -08:00
|
|
|
video_rtp_receiver_->SetupMediaChannel(kVideoSsrc);
|
2016-03-10 18:32:00 +01:00
|
|
|
video_track_ = video_rtp_receiver_->video_track();
|
2016-06-27 16:30:35 -07:00
|
|
|
VerifyVideoChannelOutput();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2018-11-20 14:08:06 +01:00
|
|
|
void CreateVideoRtpReceiverWithSimulcast(
|
|
|
|
|
std::vector<rtc::scoped_refptr<MediaStreamInterface>> streams = {},
|
|
|
|
|
int num_layers = kVideoSimulcastLayerCount) {
|
|
|
|
|
std::vector<uint32_t> ssrcs;
|
2019-01-29 10:11:53 +01:00
|
|
|
ssrcs.reserve(num_layers);
|
2018-11-20 14:08:06 +01:00
|
|
|
for (int i = 0; i < num_layers; ++i)
|
|
|
|
|
ssrcs.push_back(kVideoSsrcSimulcast + i);
|
|
|
|
|
cricket::StreamParams stream_params =
|
|
|
|
|
cricket::CreateSimStreamParams("cname", ssrcs);
|
|
|
|
|
video_media_channel_->AddRecvStream(stream_params);
|
|
|
|
|
uint32_t primary_ssrc = stream_params.first_ssrc();
|
|
|
|
|
|
2019-01-31 21:38:12 +01:00
|
|
|
video_rtp_receiver_ =
|
|
|
|
|
new VideoRtpReceiver(rtc::Thread::Current(), kVideoTrackId, streams);
|
2018-11-20 14:08:06 +01:00
|
|
|
video_rtp_receiver_->SetMediaChannel(video_media_channel_);
|
|
|
|
|
video_rtp_receiver_->SetupMediaChannel(primary_ssrc);
|
|
|
|
|
video_track_ = video_rtp_receiver_->video_track();
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-28 16:53:55 -07:00
|
|
|
void DestroyAudioRtpReceiver() {
|
|
|
|
|
audio_rtp_receiver_ = nullptr;
|
2016-06-27 16:30:35 -07:00
|
|
|
VerifyVoiceChannelNoOutput();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2015-09-28 16:53:55 -07:00
|
|
|
void DestroyVideoRtpReceiver() {
|
|
|
|
|
video_rtp_receiver_ = nullptr;
|
2016-06-27 16:30:35 -07:00
|
|
|
VerifyVideoChannelNoOutput();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void VerifyVoiceChannelInput() { VerifyVoiceChannelInput(kAudioSsrc); }
|
|
|
|
|
|
|
|
|
|
void VerifyVoiceChannelInput(uint32_t ssrc) {
|
|
|
|
|
// Verify that the media channel has an audio source, and the stream isn't
|
|
|
|
|
// muted.
|
|
|
|
|
EXPECT_TRUE(voice_media_channel_->HasSource(ssrc));
|
|
|
|
|
EXPECT_FALSE(voice_media_channel_->IsStreamMuted(ssrc));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void VerifyVideoChannelInput() { VerifyVideoChannelInput(kVideoSsrc); }
|
|
|
|
|
|
|
|
|
|
void VerifyVideoChannelInput(uint32_t ssrc) {
|
|
|
|
|
// Verify that the media channel has a video source,
|
|
|
|
|
EXPECT_TRUE(video_media_channel_->HasSource(ssrc));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void VerifyVoiceChannelNoInput() { VerifyVoiceChannelNoInput(kAudioSsrc); }
|
|
|
|
|
|
|
|
|
|
void VerifyVoiceChannelNoInput(uint32_t ssrc) {
|
|
|
|
|
// Verify that the media channel's source is reset.
|
|
|
|
|
EXPECT_FALSE(voice_media_channel_->HasSource(ssrc));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void VerifyVideoChannelNoInput() { VerifyVideoChannelNoInput(kVideoSsrc); }
|
|
|
|
|
|
|
|
|
|
void VerifyVideoChannelNoInput(uint32_t ssrc) {
|
|
|
|
|
// Verify that the media channel's source is reset.
|
|
|
|
|
EXPECT_FALSE(video_media_channel_->HasSource(ssrc));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void VerifyVoiceChannelOutput() {
|
|
|
|
|
// Verify that the volume is initialized to 1.
|
|
|
|
|
double volume;
|
|
|
|
|
EXPECT_TRUE(voice_media_channel_->GetOutputVolume(kAudioSsrc, &volume));
|
|
|
|
|
EXPECT_EQ(1, volume);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void VerifyVideoChannelOutput() {
|
|
|
|
|
// Verify that the media channel has a sink.
|
|
|
|
|
EXPECT_TRUE(video_media_channel_->HasSink(kVideoSsrc));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void VerifyVoiceChannelNoOutput() {
|
|
|
|
|
// Verify that the volume is reset to 0.
|
|
|
|
|
double volume;
|
|
|
|
|
EXPECT_TRUE(voice_media_channel_->GetOutputVolume(kAudioSsrc, &volume));
|
|
|
|
|
EXPECT_EQ(0, volume);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void VerifyVideoChannelNoOutput() {
|
|
|
|
|
// Verify that the media channel's sink is reset.
|
|
|
|
|
EXPECT_FALSE(video_media_channel_->HasSink(kVideoSsrc));
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2019-02-19 12:49:22 -08:00
|
|
|
// Verifies that the encoding layers contain the specified RIDs.
|
|
|
|
|
bool VerifyEncodingLayers(const VideoRtpSender& sender,
|
|
|
|
|
const std::vector<std::string>& rids) {
|
|
|
|
|
bool has_failure = HasFailure();
|
|
|
|
|
RtpParameters parameters = sender.GetParameters();
|
|
|
|
|
std::vector<std::string> encoding_rids;
|
|
|
|
|
absl::c_transform(
|
|
|
|
|
parameters.encodings, std::back_inserter(encoding_rids),
|
|
|
|
|
[](const RtpEncodingParameters& encoding) { return encoding.rid; });
|
|
|
|
|
EXPECT_THAT(rids, ContainerEq(encoding_rids));
|
|
|
|
|
return has_failure || !HasFailure();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Runs a test for disabling the encoding layers on the specified sender.
|
|
|
|
|
void RunDisableEncodingLayersTest(
|
|
|
|
|
const std::vector<std::string>& all_layers,
|
|
|
|
|
const std::vector<std::string>& disabled_layers,
|
|
|
|
|
VideoRtpSender* sender) {
|
|
|
|
|
std::vector<std::string> expected;
|
|
|
|
|
absl::c_copy_if(all_layers, std::back_inserter(expected),
|
|
|
|
|
[&disabled_layers](const std::string& rid) {
|
|
|
|
|
return !absl::c_linear_search(disabled_layers, rid);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
EXPECT_TRUE(VerifyEncodingLayers(*sender, all_layers));
|
|
|
|
|
sender->DisableEncodingLayers(disabled_layers);
|
|
|
|
|
EXPECT_TRUE(VerifyEncodingLayers(*sender, expected));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Runs a test for setting an encoding layer as inactive.
|
|
|
|
|
// This test assumes that some layers have already been disabled.
|
|
|
|
|
void RunSetLastLayerAsInactiveTest(VideoRtpSender* sender) {
|
|
|
|
|
auto parameters = sender->GetParameters();
|
|
|
|
|
if (parameters.encodings.size() == 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RtpEncodingParameters& encoding = parameters.encodings.back();
|
|
|
|
|
auto rid = encoding.rid;
|
|
|
|
|
EXPECT_TRUE(encoding.active);
|
|
|
|
|
encoding.active = false;
|
|
|
|
|
auto error = sender->SetParameters(parameters);
|
|
|
|
|
ASSERT_TRUE(error.ok());
|
|
|
|
|
parameters = sender->GetParameters();
|
|
|
|
|
RtpEncodingParameters& result_encoding = parameters.encodings.back();
|
|
|
|
|
EXPECT_EQ(rid, result_encoding.rid);
|
|
|
|
|
EXPECT_FALSE(result_encoding.active);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Runs a test for disabling the encoding layers on a sender without a media
|
|
|
|
|
// channel.
|
|
|
|
|
void RunDisableSimulcastLayersWithoutMediaEngineTest(
|
|
|
|
|
const std::vector<std::string>& all_layers,
|
|
|
|
|
const std::vector<std::string>& disabled_layers) {
|
2019-05-20 19:31:53 +02:00
|
|
|
auto sender = VideoRtpSender::Create(rtc::Thread::Current(), "1", nullptr);
|
2019-02-19 12:49:22 -08:00
|
|
|
RtpParameters parameters;
|
|
|
|
|
parameters.encodings.resize(all_layers.size());
|
|
|
|
|
for (size_t i = 0; i < all_layers.size(); ++i) {
|
|
|
|
|
parameters.encodings[i].rid = all_layers[i];
|
|
|
|
|
}
|
2019-02-19 15:20:21 -08:00
|
|
|
sender->set_init_send_encodings(parameters.encodings);
|
|
|
|
|
RunDisableEncodingLayersTest(all_layers, disabled_layers, sender.get());
|
|
|
|
|
RunSetLastLayerAsInactiveTest(sender.get());
|
2019-02-19 12:49:22 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Runs a test for disabling the encoding layers on a sender with a media
|
|
|
|
|
// channel.
|
|
|
|
|
void RunDisableSimulcastLayersWithMediaEngineTest(
|
|
|
|
|
const std::vector<std::string>& all_layers,
|
|
|
|
|
const std::vector<std::string>& disabled_layers) {
|
|
|
|
|
uint32_t ssrc = CreateVideoRtpSenderWithSimulcast(all_layers);
|
|
|
|
|
RunDisableEncodingLayersTest(all_layers, disabled_layers,
|
|
|
|
|
video_rtp_sender_.get());
|
|
|
|
|
|
|
|
|
|
auto channel_parameters = video_media_channel_->GetRtpSendParameters(ssrc);
|
|
|
|
|
ASSERT_EQ(channel_parameters.encodings.size(), all_layers.size());
|
|
|
|
|
for (size_t i = 0; i < all_layers.size(); ++i) {
|
|
|
|
|
EXPECT_EQ(all_layers[i], channel_parameters.encodings[i].rid);
|
|
|
|
|
bool is_active = !absl::c_linear_search(disabled_layers, all_layers[i]);
|
|
|
|
|
EXPECT_EQ(is_active, channel_parameters.encodings[i].active);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RunSetLastLayerAsInactiveTest(video_rtp_sender_.get());
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-03 19:55:33 +02:00
|
|
|
// Check that minimum Jitter Buffer delay is propagated to the underlying
|
|
|
|
|
// |media_channel|.
|
|
|
|
|
void VerifyRtpReceiverDelayBehaviour(cricket::Delayable* media_channel,
|
|
|
|
|
RtpReceiverInterface* receiver,
|
|
|
|
|
uint32_t ssrc) {
|
|
|
|
|
receiver->SetJitterBufferMinimumDelay(/*delay_seconds=*/0.5);
|
|
|
|
|
absl::optional<int> delay_ms =
|
|
|
|
|
media_channel->GetBaseMinimumPlayoutDelayMs(ssrc); // In milliseconds.
|
|
|
|
|
EXPECT_DOUBLE_EQ(0.5, delay_ms.value_or(0) / 1000.0);
|
|
|
|
|
}
|
|
|
|
|
|
2013-07-10 00:45:36 +00:00
|
|
|
protected:
|
2018-01-12 10:49:35 -08:00
|
|
|
rtc::Thread* const network_thread_;
|
|
|
|
|
rtc::Thread* const worker_thread_;
|
2019-08-07 12:24:53 +02:00
|
|
|
webrtc::RtcEventLogNull event_log_;
|
2018-03-30 10:48:35 -07:00
|
|
|
// The |rtp_dtls_transport_| and |rtp_transport_| should be destroyed after
|
|
|
|
|
// the |channel_manager|.
|
|
|
|
|
std::unique_ptr<cricket::DtlsTransportInternal> rtp_dtls_transport_;
|
|
|
|
|
std::unique_ptr<webrtc::RtpTransportInternal> rtp_transport_;
|
2019-04-17 07:38:40 +02:00
|
|
|
std::unique_ptr<webrtc::VideoBitrateAllocatorFactory>
|
|
|
|
|
video_bitrate_allocator_factory_;
|
2017-02-10 20:13:37 -08:00
|
|
|
// |media_engine_| is actually owned by |channel_manager_|.
|
2016-06-27 16:30:35 -07:00
|
|
|
cricket::FakeMediaEngine* media_engine_;
|
2021-04-01 16:49:42 +02:00
|
|
|
std::unique_ptr<cricket::ChannelManager> channel_manager_;
|
2016-06-27 16:30:35 -07:00
|
|
|
cricket::FakeCall fake_call_;
|
|
|
|
|
cricket::VoiceChannel* voice_channel_;
|
|
|
|
|
cricket::VideoChannel* video_channel_;
|
|
|
|
|
cricket::FakeVoiceMediaChannel* voice_media_channel_;
|
|
|
|
|
cricket::FakeVideoMediaChannel* video_media_channel_;
|
2015-09-28 16:53:55 -07:00
|
|
|
rtc::scoped_refptr<AudioRtpSender> audio_rtp_sender_;
|
|
|
|
|
rtc::scoped_refptr<VideoRtpSender> video_rtp_sender_;
|
|
|
|
|
rtc::scoped_refptr<AudioRtpReceiver> audio_rtp_receiver_;
|
|
|
|
|
rtc::scoped_refptr<VideoRtpReceiver> video_rtp_receiver_;
|
2017-02-25 18:15:09 -08:00
|
|
|
rtc::scoped_refptr<MediaStreamInterface> local_stream_;
|
2014-07-29 17:36:52 +00:00
|
|
|
rtc::scoped_refptr<VideoTrackInterface> video_track_;
|
|
|
|
|
rtc::scoped_refptr<AudioTrackInterface> audio_track_;
|
2017-02-01 20:27:00 -08:00
|
|
|
bool audio_sender_destroyed_signal_fired_ = false;
|
2019-01-25 17:13:56 -08:00
|
|
|
rtc::UniqueRandomIdGenerator ssrc_generator_;
|
2013-07-10 00:45:36 +00:00
|
|
|
};
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that |voice_channel_| is updated when an audio track is associated
|
2015-09-28 16:53:55 -07:00
|
|
|
// and disassociated with an AudioRtpSender.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AddAndDestroyAudioRtpSender) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
DestroyAudioRtpSender();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that |video_channel_| is updated when a video track is associated and
|
2015-09-28 16:53:55 -07:00
|
|
|
// disassociated with a VideoRtpSender.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AddAndDestroyVideoRtpSender) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
DestroyVideoRtpSender();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that |voice_channel_| is updated when a remote audio track is
|
2015-09-28 16:53:55 -07:00
|
|
|
// associated and disassociated with an AudioRtpReceiver.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AddAndDestroyAudioRtpReceiver) {
|
|
|
|
|
CreateAudioRtpReceiver();
|
|
|
|
|
DestroyAudioRtpReceiver();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that |video_channel_| is updated when a remote video track is
|
|
|
|
|
// associated and disassociated with a VideoRtpReceiver.
|
2015-09-28 16:53:55 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, AddAndDestroyVideoRtpReceiver) {
|
|
|
|
|
CreateVideoRtpReceiver();
|
|
|
|
|
DestroyVideoRtpReceiver();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2017-11-21 13:41:51 +01:00
|
|
|
TEST_F(RtpSenderReceiverTest, AddAndDestroyAudioRtpReceiverWithStreams) {
|
|
|
|
|
CreateAudioRtpReceiver({local_stream_});
|
|
|
|
|
DestroyAudioRtpReceiver();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AddAndDestroyVideoRtpReceiverWithStreams) {
|
|
|
|
|
CreateVideoRtpReceiver({local_stream_});
|
|
|
|
|
DestroyVideoRtpReceiver();
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the AudioRtpSender applies options from the local audio source.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, LocalAudioSourceOptionsApplied) {
|
|
|
|
|
cricket::AudioOptions options;
|
2017-11-16 10:54:27 +01:00
|
|
|
options.echo_cancellation = true;
|
2017-02-10 21:26:48 -08:00
|
|
|
auto source = LocalAudioSource::Create(&options);
|
2016-06-27 16:30:35 -07:00
|
|
|
CreateAudioRtpSender(source.get());
|
|
|
|
|
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(true, voice_media_channel_->options().echo_cancellation);
|
2016-06-27 16:30:35 -07:00
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Test that the stream is muted when the track is disabled, and unmuted when
|
|
|
|
|
// the track is enabled.
|
2015-09-28 16:53:55 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, LocalAudioTrackDisable) {
|
|
|
|
|
CreateAudioRtpSender();
|
2013-07-10 00:45:36 +00:00
|
|
|
|
|
|
|
|
audio_track_->set_enabled(false);
|
2016-06-27 16:30:35 -07:00
|
|
|
EXPECT_TRUE(voice_media_channel_->IsStreamMuted(kAudioSsrc));
|
2013-07-10 00:45:36 +00:00
|
|
|
|
|
|
|
|
audio_track_->set_enabled(true);
|
2016-06-27 16:30:35 -07:00
|
|
|
EXPECT_FALSE(voice_media_channel_->IsStreamMuted(kAudioSsrc));
|
2013-07-10 00:45:36 +00:00
|
|
|
|
2015-09-28 16:53:55 -07:00
|
|
|
DestroyAudioRtpSender();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the volume is set to 0 when the track is disabled, and back to
|
|
|
|
|
// 1 when the track is enabled.
|
2015-09-28 16:53:55 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, RemoteAudioTrackDisable) {
|
|
|
|
|
CreateAudioRtpReceiver();
|
2013-07-10 00:45:36 +00:00
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
double volume;
|
|
|
|
|
EXPECT_TRUE(voice_media_channel_->GetOutputVolume(kAudioSsrc, &volume));
|
|
|
|
|
EXPECT_EQ(1, volume);
|
|
|
|
|
|
2013-07-10 00:45:36 +00:00
|
|
|
audio_track_->set_enabled(false);
|
2016-06-27 16:30:35 -07:00
|
|
|
EXPECT_TRUE(voice_media_channel_->GetOutputVolume(kAudioSsrc, &volume));
|
|
|
|
|
EXPECT_EQ(0, volume);
|
2013-07-10 00:45:36 +00:00
|
|
|
|
|
|
|
|
audio_track_->set_enabled(true);
|
2016-06-27 16:30:35 -07:00
|
|
|
EXPECT_TRUE(voice_media_channel_->GetOutputVolume(kAudioSsrc, &volume));
|
|
|
|
|
EXPECT_EQ(1, volume);
|
2013-07-10 00:45:36 +00:00
|
|
|
|
2015-09-28 16:53:55 -07:00
|
|
|
DestroyAudioRtpReceiver();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Currently no action is taken when a remote video track is disabled or
|
|
|
|
|
// enabled, so there's nothing to test here, other than what is normally
|
|
|
|
|
// verified in DestroyVideoRtpSender.
|
2015-09-28 16:53:55 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, LocalVideoTrackDisable) {
|
|
|
|
|
CreateVideoRtpSender();
|
2013-07-10 00:45:36 +00:00
|
|
|
|
|
|
|
|
video_track_->set_enabled(false);
|
|
|
|
|
video_track_->set_enabled(true);
|
|
|
|
|
|
2015-09-28 16:53:55 -07:00
|
|
|
DestroyVideoRtpSender();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the state of the video track created by the VideoRtpReceiver is
|
|
|
|
|
// updated when the receiver is destroyed.
|
2016-03-10 18:32:00 +01:00
|
|
|
TEST_F(RtpSenderReceiverTest, RemoteVideoTrackState) {
|
|
|
|
|
CreateVideoRtpReceiver();
|
|
|
|
|
|
|
|
|
|
EXPECT_EQ(webrtc::MediaStreamTrackInterface::kLive, video_track_->state());
|
|
|
|
|
EXPECT_EQ(webrtc::MediaSourceInterface::kLive,
|
|
|
|
|
video_track_->GetSource()->state());
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpReceiver();
|
|
|
|
|
|
|
|
|
|
EXPECT_EQ(webrtc::MediaStreamTrackInterface::kEnded, video_track_->state());
|
|
|
|
|
EXPECT_EQ(webrtc::MediaSourceInterface::kEnded,
|
|
|
|
|
video_track_->GetSource()->state());
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Currently no action is taken when a remote video track is disabled or
|
|
|
|
|
// enabled, so there's nothing to test here, other than what is normally
|
|
|
|
|
// verified in DestroyVideoRtpReceiver.
|
2015-09-28 16:53:55 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, RemoteVideoTrackDisable) {
|
|
|
|
|
CreateVideoRtpReceiver();
|
2013-07-10 00:45:36 +00:00
|
|
|
|
|
|
|
|
video_track_->set_enabled(false);
|
|
|
|
|
video_track_->set_enabled(true);
|
|
|
|
|
|
2015-09-28 16:53:55 -07:00
|
|
|
DestroyVideoRtpReceiver();
|
2013-07-10 00:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the AudioRtpReceiver applies volume changes from the track source
|
|
|
|
|
// to the media channel.
|
2015-09-28 16:53:55 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, RemoteAudioTrackSetVolume) {
|
|
|
|
|
CreateAudioRtpReceiver();
|
2014-02-13 23:18:49 +00:00
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
double volume;
|
|
|
|
|
audio_track_->GetSource()->SetVolume(0.5);
|
|
|
|
|
EXPECT_TRUE(voice_media_channel_->GetOutputVolume(kAudioSsrc, &volume));
|
|
|
|
|
EXPECT_EQ(0.5, volume);
|
2014-02-13 23:18:49 +00:00
|
|
|
|
|
|
|
|
// Disable the audio track, this should prevent setting the volume.
|
|
|
|
|
audio_track_->set_enabled(false);
|
2016-06-27 16:30:35 -07:00
|
|
|
audio_track_->GetSource()->SetVolume(0.8);
|
|
|
|
|
EXPECT_TRUE(voice_media_channel_->GetOutputVolume(kAudioSsrc, &volume));
|
|
|
|
|
EXPECT_EQ(0, volume);
|
2014-02-13 23:18:49 +00:00
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// When the track is enabled, the previously set volume should take effect.
|
2014-02-13 23:18:49 +00:00
|
|
|
audio_track_->set_enabled(true);
|
2016-06-27 16:30:35 -07:00
|
|
|
EXPECT_TRUE(voice_media_channel_->GetOutputVolume(kAudioSsrc, &volume));
|
|
|
|
|
EXPECT_EQ(0.8, volume);
|
2014-02-13 23:18:49 +00:00
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Try changing volume one more time.
|
|
|
|
|
audio_track_->GetSource()->SetVolume(0.9);
|
|
|
|
|
EXPECT_TRUE(voice_media_channel_->GetOutputVolume(kAudioSsrc, &volume));
|
|
|
|
|
EXPECT_EQ(0.9, volume);
|
2014-02-13 23:18:49 +00:00
|
|
|
|
2015-09-28 16:53:55 -07:00
|
|
|
DestroyAudioRtpReceiver();
|
2014-02-13 23:18:49 +00:00
|
|
|
}
|
|
|
|
|
|
2019-04-03 19:55:33 +02:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioRtpReceiverDelay) {
|
|
|
|
|
CreateAudioRtpReceiver();
|
|
|
|
|
VerifyRtpReceiverDelayBehaviour(voice_media_channel_,
|
|
|
|
|
audio_rtp_receiver_.get(), kAudioSsrc);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoRtpReceiverDelay) {
|
|
|
|
|
CreateVideoRtpReceiver();
|
|
|
|
|
VerifyRtpReceiverDelayBehaviour(video_media_channel_,
|
|
|
|
|
video_rtp_receiver_.get(), kVideoSsrc);
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel isn't enabled for sending if the audio sender
|
|
|
|
|
// doesn't have both a track and SSRC.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderWithoutTrackAndSsrc) {
|
2018-01-10 16:26:06 -08:00
|
|
|
CreateAudioRtpSenderWithNoTrack();
|
2015-11-25 11:26:01 -08:00
|
|
|
rtc::scoped_refptr<AudioTrackInterface> track =
|
|
|
|
|
AudioTrack::Create(kAudioTrackId, nullptr);
|
2016-06-27 16:30:35 -07:00
|
|
|
|
|
|
|
|
// Track but no SSRC.
|
|
|
|
|
EXPECT_TRUE(audio_rtp_sender_->SetTrack(track));
|
|
|
|
|
VerifyVoiceChannelNoInput();
|
|
|
|
|
|
|
|
|
|
// SSRC but no track.
|
|
|
|
|
EXPECT_TRUE(audio_rtp_sender_->SetTrack(nullptr));
|
|
|
|
|
audio_rtp_sender_->SetSsrc(kAudioSsrc);
|
|
|
|
|
VerifyVoiceChannelNoInput();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel isn't enabled for sending if the video sender
|
|
|
|
|
// doesn't have both a track and SSRC.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderWithoutTrackAndSsrc) {
|
2018-01-10 16:26:06 -08:00
|
|
|
CreateVideoRtpSenderWithNoTrack();
|
2016-06-27 16:30:35 -07:00
|
|
|
|
|
|
|
|
// Track but no SSRC.
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetTrack(video_track_));
|
|
|
|
|
VerifyVideoChannelNoInput();
|
|
|
|
|
|
|
|
|
|
// SSRC but no track.
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetTrack(nullptr));
|
|
|
|
|
video_rtp_sender_->SetSsrc(kVideoSsrc);
|
|
|
|
|
VerifyVideoChannelNoInput();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel is enabled for sending when the audio sender
|
|
|
|
|
// has a track and SSRC, when the SSRC is set first.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderEarlyWarmupSsrcThenTrack) {
|
2018-01-10 16:26:06 -08:00
|
|
|
CreateAudioRtpSenderWithNoTrack();
|
2015-11-25 11:26:01 -08:00
|
|
|
rtc::scoped_refptr<AudioTrackInterface> track =
|
|
|
|
|
AudioTrack::Create(kAudioTrackId, nullptr);
|
2016-06-27 16:30:35 -07:00
|
|
|
audio_rtp_sender_->SetSsrc(kAudioSsrc);
|
|
|
|
|
audio_rtp_sender_->SetTrack(track);
|
|
|
|
|
VerifyVoiceChannelInput();
|
2015-11-25 11:26:01 -08:00
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
DestroyAudioRtpSender();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel is enabled for sending when the audio sender
|
|
|
|
|
// has a track and SSRC, when the SSRC is set last.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderEarlyWarmupTrackThenSsrc) {
|
2018-01-10 16:26:06 -08:00
|
|
|
CreateAudioRtpSenderWithNoTrack();
|
2015-11-25 11:26:01 -08:00
|
|
|
rtc::scoped_refptr<AudioTrackInterface> track =
|
|
|
|
|
AudioTrack::Create(kAudioTrackId, nullptr);
|
2016-06-27 16:30:35 -07:00
|
|
|
audio_rtp_sender_->SetTrack(track);
|
|
|
|
|
audio_rtp_sender_->SetSsrc(kAudioSsrc);
|
|
|
|
|
VerifyVoiceChannelInput();
|
2015-11-25 11:26:01 -08:00
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
DestroyAudioRtpSender();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel is enabled for sending when the video sender
|
|
|
|
|
// has a track and SSRC, when the SSRC is set first.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderEarlyWarmupSsrcThenTrack) {
|
2016-03-21 08:20:42 -07:00
|
|
|
AddVideoTrack();
|
2018-01-10 16:26:06 -08:00
|
|
|
CreateVideoRtpSenderWithNoTrack();
|
2016-06-27 16:30:35 -07:00
|
|
|
video_rtp_sender_->SetSsrc(kVideoSsrc);
|
|
|
|
|
video_rtp_sender_->SetTrack(video_track_);
|
|
|
|
|
VerifyVideoChannelInput();
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel is enabled for sending when the video sender
|
|
|
|
|
// has a track and SSRC, when the SSRC is set last.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderEarlyWarmupTrackThenSsrc) {
|
2016-03-21 08:20:42 -07:00
|
|
|
AddVideoTrack();
|
2018-01-10 16:26:06 -08:00
|
|
|
CreateVideoRtpSenderWithNoTrack();
|
2016-06-27 16:30:35 -07:00
|
|
|
video_rtp_sender_->SetTrack(video_track_);
|
|
|
|
|
video_rtp_sender_->SetSsrc(kVideoSsrc);
|
|
|
|
|
VerifyVideoChannelInput();
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel stops sending when the audio sender's SSRC is set
|
|
|
|
|
// to 0.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderSsrcSetToZero) {
|
2016-06-27 16:30:35 -07:00
|
|
|
CreateAudioRtpSender();
|
2016-06-24 19:31:47 -07:00
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
audio_rtp_sender_->SetSsrc(0);
|
|
|
|
|
VerifyVoiceChannelNoInput();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel stops sending when the video sender's SSRC is set
|
|
|
|
|
// to 0.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderSsrcSetToZero) {
|
2016-06-27 16:30:35 -07:00
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
audio_rtp_sender_->SetSsrc(0);
|
|
|
|
|
VerifyVideoChannelNoInput();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel stops sending when the audio sender's track is
|
|
|
|
|
// set to null.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderTrackSetToNull) {
|
2016-06-27 16:30:35 -07:00
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
EXPECT_TRUE(audio_rtp_sender_->SetTrack(nullptr));
|
|
|
|
|
VerifyVoiceChannelNoInput();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that the media channel stops sending when the video sender's track is
|
|
|
|
|
// set to null.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderTrackSetToNull) {
|
2016-06-27 16:30:35 -07:00
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
video_rtp_sender_->SetSsrc(0);
|
|
|
|
|
VerifyVideoChannelNoInput();
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that when the audio sender's SSRC is changed, the media channel stops
|
|
|
|
|
// sending with the old SSRC and starts sending with the new one.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderSsrcChanged) {
|
2016-06-27 16:30:35 -07:00
|
|
|
CreateAudioRtpSender();
|
2015-11-25 11:26:01 -08:00
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
audio_rtp_sender_->SetSsrc(kAudioSsrc2);
|
|
|
|
|
VerifyVoiceChannelNoInput(kAudioSsrc);
|
|
|
|
|
VerifyVoiceChannelInput(kAudioSsrc2);
|
2015-11-25 11:26:01 -08:00
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
audio_rtp_sender_ = nullptr;
|
|
|
|
|
VerifyVoiceChannelNoInput(kAudioSsrc2);
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
// Test that when the audio sender's SSRC is changed, the media channel stops
|
|
|
|
|
// sending with the old SSRC and starts sending with the new one.
|
2015-11-25 11:26:01 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderSsrcChanged) {
|
2016-06-27 16:30:35 -07:00
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
video_rtp_sender_->SetSsrc(kVideoSsrc2);
|
|
|
|
|
VerifyVideoChannelNoInput(kVideoSsrc);
|
|
|
|
|
VerifyVideoChannelInput(kVideoSsrc2);
|
|
|
|
|
|
|
|
|
|
video_rtp_sender_ = nullptr;
|
|
|
|
|
VerifyVideoChannelNoInput(kVideoSsrc2);
|
2015-11-25 11:26:01 -08:00
|
|
|
}
|
|
|
|
|
|
2016-03-16 19:07:43 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderCanSetParameters) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = audio_rtp_sender_->GetParameters();
|
2016-06-27 16:30:35 -07:00
|
|
|
EXPECT_EQ(1u, params.encodings.size());
|
2018-01-23 15:02:36 -08:00
|
|
|
EXPECT_TRUE(audio_rtp_sender_->SetParameters(params).ok());
|
2016-03-16 19:07:43 -07:00
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-01 22:47:20 +02:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderCanSetParametersBeforeNegotiation) {
|
2019-02-19 15:20:21 -08:00
|
|
|
audio_rtp_sender_ =
|
2019-05-20 19:31:53 +02:00
|
|
|
AudioRtpSender::Create(worker_thread_, /*id=*/"", nullptr, nullptr);
|
2018-10-01 22:47:20 +02:00
|
|
|
|
|
|
|
|
RtpParameters params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
ASSERT_EQ(1u, params.encodings.size());
|
|
|
|
|
params.encodings[0].max_bitrate_bps = 90000;
|
|
|
|
|
EXPECT_TRUE(audio_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
|
|
|
|
|
params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_TRUE(audio_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
EXPECT_EQ(params.encodings[0].max_bitrate_bps, 90000);
|
|
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderInitParametersMovedAfterNegotiation) {
|
|
|
|
|
audio_track_ = AudioTrack::Create(kAudioTrackId, nullptr);
|
|
|
|
|
EXPECT_TRUE(local_stream_->AddTrack(audio_track_));
|
|
|
|
|
|
2019-05-20 19:31:53 +02:00
|
|
|
std::unique_ptr<MockSetStreamsObserver> set_streams_observer =
|
2019-09-17 17:06:18 +02:00
|
|
|
std::make_unique<MockSetStreamsObserver>();
|
2019-05-20 19:31:53 +02:00
|
|
|
audio_rtp_sender_ = AudioRtpSender::Create(
|
|
|
|
|
worker_thread_, audio_track_->id(), nullptr, set_streams_observer.get());
|
2018-10-01 22:47:20 +02:00
|
|
|
ASSERT_TRUE(audio_rtp_sender_->SetTrack(audio_track_));
|
2019-05-20 19:31:53 +02:00
|
|
|
EXPECT_CALL(*set_streams_observer, OnSetStreams());
|
|
|
|
|
audio_rtp_sender_->SetStreams({local_stream_->id()});
|
2018-10-01 22:47:20 +02:00
|
|
|
|
|
|
|
|
std::vector<RtpEncodingParameters> init_encodings(1);
|
|
|
|
|
init_encodings[0].max_bitrate_bps = 60000;
|
|
|
|
|
audio_rtp_sender_->set_init_send_encodings(init_encodings);
|
|
|
|
|
|
|
|
|
|
RtpParameters params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
ASSERT_EQ(1u, params.encodings.size());
|
|
|
|
|
EXPECT_EQ(params.encodings[0].max_bitrate_bps, 60000);
|
|
|
|
|
|
|
|
|
|
// Simulate the setLocalDescription call
|
|
|
|
|
std::vector<uint32_t> ssrcs(1, 1);
|
|
|
|
|
cricket::StreamParams stream_params =
|
|
|
|
|
cricket::CreateSimStreamParams("cname", ssrcs);
|
|
|
|
|
voice_media_channel_->AddSendStream(stream_params);
|
2018-11-13 16:26:05 -08:00
|
|
|
audio_rtp_sender_->SetMediaChannel(voice_media_channel_);
|
2018-10-01 22:47:20 +02:00
|
|
|
audio_rtp_sender_->SetSsrc(1);
|
|
|
|
|
|
|
|
|
|
params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
ASSERT_EQ(1u, params.encodings.size());
|
|
|
|
|
EXPECT_EQ(params.encodings[0].max_bitrate_bps, 60000);
|
|
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
AudioSenderMustCallGetParametersBeforeSetParametersBeforeNegotiation) {
|
2019-02-19 15:20:21 -08:00
|
|
|
audio_rtp_sender_ =
|
2019-05-20 19:31:53 +02:00
|
|
|
AudioRtpSender::Create(worker_thread_, /*id=*/"", nullptr, nullptr);
|
2018-10-01 22:47:20 +02:00
|
|
|
|
|
|
|
|
RtpParameters params;
|
|
|
|
|
RTCError result = audio_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_STATE, result.type());
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-03 15:31:53 +02:00
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
AudioSenderMustCallGetParametersBeforeSetParameters) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params;
|
|
|
|
|
RTCError result = audio_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_STATE, result.type());
|
|
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
AudioSenderSetParametersInvalidatesTransactionId) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_EQ(1u, params.encodings.size());
|
|
|
|
|
EXPECT_TRUE(audio_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
RTCError result = audio_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_STATE, result.type());
|
|
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderDetectTransactionIdModification) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
params.transaction_id = "";
|
|
|
|
|
RTCError result = audio_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_MODIFICATION, result.type());
|
|
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderCheckTransactionIdRefresh) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = audio_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_NE(params.transaction_id.size(), 0U);
|
2018-05-03 15:31:53 +02:00
|
|
|
auto saved_transaction_id = params.transaction_id;
|
|
|
|
|
params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_NE(saved_transaction_id, params.transaction_id);
|
|
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderSetParametersOldValueFail) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
RtpParameters second_params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
|
|
|
|
|
RTCError result = audio_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_MODIFICATION, result.type());
|
2018-05-16 16:02:32 -07:00
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderCantSetUnimplementedRtpParameters) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
RtpParameters params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_EQ(1u, params.encodings.size());
|
|
|
|
|
|
2018-07-18 16:00:28 +02:00
|
|
|
// Unimplemented RtpParameters: mid
|
2018-05-16 16:02:32 -07:00
|
|
|
params.mid = "dummy_mid";
|
|
|
|
|
EXPECT_EQ(RTCErrorType::UNSUPPORTED_PARAMETER,
|
|
|
|
|
audio_rtp_sender_->SetParameters(params).type());
|
|
|
|
|
params = audio_rtp_sender_->GetParameters();
|
|
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-27 16:30:35 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, SetAudioMaxSendBitrate) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
EXPECT_EQ(-1, voice_media_channel_->max_bps());
|
|
|
|
|
webrtc::RtpParameters params = audio_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2017-02-04 12:09:01 -08:00
|
|
|
EXPECT_FALSE(params.encodings[0].max_bitrate_bps);
|
2017-11-16 10:54:27 +01:00
|
|
|
params.encodings[0].max_bitrate_bps = 1000;
|
2018-01-23 15:02:36 -08:00
|
|
|
EXPECT_TRUE(audio_rtp_sender_->SetParameters(params).ok());
|
2016-06-27 16:30:35 -07:00
|
|
|
|
|
|
|
|
// Read back the parameters and verify they have been changed.
|
|
|
|
|
params = audio_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(1000, params.encodings[0].max_bitrate_bps);
|
2016-06-27 16:30:35 -07:00
|
|
|
|
|
|
|
|
// Verify that the audio channel received the new parameters.
|
|
|
|
|
params = voice_media_channel_->GetRtpSendParameters(kAudioSsrc);
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(1000, params.encodings[0].max_bitrate_bps);
|
2016-06-27 16:30:35 -07:00
|
|
|
|
|
|
|
|
// Verify that the global bitrate limit has not been changed.
|
|
|
|
|
EXPECT_EQ(-1, voice_media_channel_->max_bps());
|
|
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2017-12-22 09:36:42 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, SetAudioBitratePriority) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
|
|
|
|
|
webrtc::RtpParameters params = audio_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2017-12-22 09:36:42 -08:00
|
|
|
EXPECT_EQ(webrtc::kDefaultBitratePriority,
|
|
|
|
|
params.encodings[0].bitrate_priority);
|
|
|
|
|
double new_bitrate_priority = 2.0;
|
|
|
|
|
params.encodings[0].bitrate_priority = new_bitrate_priority;
|
2018-01-23 15:02:36 -08:00
|
|
|
EXPECT_TRUE(audio_rtp_sender_->SetParameters(params).ok());
|
2017-12-22 09:36:42 -08:00
|
|
|
|
|
|
|
|
params = audio_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2017-12-22 09:36:42 -08:00
|
|
|
EXPECT_EQ(new_bitrate_priority, params.encodings[0].bitrate_priority);
|
|
|
|
|
|
|
|
|
|
params = voice_media_channel_->GetRtpSendParameters(kAudioSsrc);
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2017-12-22 09:36:42 -08:00
|
|
|
EXPECT_EQ(new_bitrate_priority, params.encodings[0].bitrate_priority);
|
|
|
|
|
|
|
|
|
|
DestroyAudioRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-16 19:07:43 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCanSetParameters) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
2016-06-27 16:30:35 -07:00
|
|
|
EXPECT_EQ(1u, params.encodings.size());
|
2018-01-23 15:02:36 -08:00
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
2016-03-16 19:07:43 -07:00
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-01 22:47:20 +02:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCanSetParametersBeforeNegotiation) {
|
2019-05-20 19:31:53 +02:00
|
|
|
video_rtp_sender_ =
|
|
|
|
|
VideoRtpSender::Create(worker_thread_, /*id=*/"", nullptr);
|
2018-10-01 22:47:20 +02:00
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
ASSERT_EQ(1u, params.encodings.size());
|
|
|
|
|
params.encodings[0].max_bitrate_bps = 90000;
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
EXPECT_EQ(params.encodings[0].max_bitrate_bps, 90000);
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderInitParametersMovedAfterNegotiation) {
|
|
|
|
|
AddVideoTrack(false);
|
|
|
|
|
|
2019-05-20 19:31:53 +02:00
|
|
|
std::unique_ptr<MockSetStreamsObserver> set_streams_observer =
|
2019-09-17 17:06:18 +02:00
|
|
|
std::make_unique<MockSetStreamsObserver>();
|
2019-05-20 19:31:53 +02:00
|
|
|
video_rtp_sender_ = VideoRtpSender::Create(worker_thread_, video_track_->id(),
|
|
|
|
|
set_streams_observer.get());
|
2018-10-01 22:47:20 +02:00
|
|
|
ASSERT_TRUE(video_rtp_sender_->SetTrack(video_track_));
|
2019-05-20 19:31:53 +02:00
|
|
|
EXPECT_CALL(*set_streams_observer, OnSetStreams());
|
|
|
|
|
video_rtp_sender_->SetStreams({local_stream_->id()});
|
2018-10-01 22:47:20 +02:00
|
|
|
|
|
|
|
|
std::vector<RtpEncodingParameters> init_encodings(2);
|
|
|
|
|
init_encodings[0].max_bitrate_bps = 60000;
|
|
|
|
|
init_encodings[1].max_bitrate_bps = 900000;
|
|
|
|
|
video_rtp_sender_->set_init_send_encodings(init_encodings);
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
ASSERT_EQ(2u, params.encodings.size());
|
|
|
|
|
EXPECT_EQ(params.encodings[0].max_bitrate_bps, 60000);
|
|
|
|
|
EXPECT_EQ(params.encodings[1].max_bitrate_bps, 900000);
|
|
|
|
|
|
|
|
|
|
// Simulate the setLocalDescription call
|
|
|
|
|
std::vector<uint32_t> ssrcs;
|
2019-01-29 10:11:53 +01:00
|
|
|
ssrcs.reserve(2);
|
2018-10-01 22:47:20 +02:00
|
|
|
for (int i = 0; i < 2; ++i)
|
|
|
|
|
ssrcs.push_back(kVideoSsrcSimulcast + i);
|
|
|
|
|
cricket::StreamParams stream_params =
|
|
|
|
|
cricket::CreateSimStreamParams("cname", ssrcs);
|
|
|
|
|
video_media_channel_->AddSendStream(stream_params);
|
2018-11-13 16:26:05 -08:00
|
|
|
video_rtp_sender_->SetMediaChannel(video_media_channel_);
|
2018-10-01 22:47:20 +02:00
|
|
|
video_rtp_sender_->SetSsrc(kVideoSsrcSimulcast);
|
|
|
|
|
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
ASSERT_EQ(2u, params.encodings.size());
|
|
|
|
|
EXPECT_EQ(params.encodings[0].max_bitrate_bps, 60000);
|
|
|
|
|
EXPECT_EQ(params.encodings[1].max_bitrate_bps, 900000);
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
VideoSenderInitParametersMovedAfterManualSimulcastAndNegotiation) {
|
|
|
|
|
AddVideoTrack(false);
|
|
|
|
|
|
2019-05-20 19:31:53 +02:00
|
|
|
std::unique_ptr<MockSetStreamsObserver> set_streams_observer =
|
2019-09-17 17:06:18 +02:00
|
|
|
std::make_unique<MockSetStreamsObserver>();
|
2019-05-20 19:31:53 +02:00
|
|
|
video_rtp_sender_ = VideoRtpSender::Create(worker_thread_, video_track_->id(),
|
|
|
|
|
set_streams_observer.get());
|
2018-10-01 22:47:20 +02:00
|
|
|
ASSERT_TRUE(video_rtp_sender_->SetTrack(video_track_));
|
2019-05-20 19:31:53 +02:00
|
|
|
EXPECT_CALL(*set_streams_observer, OnSetStreams());
|
|
|
|
|
video_rtp_sender_->SetStreams({local_stream_->id()});
|
2018-10-01 22:47:20 +02:00
|
|
|
|
|
|
|
|
std::vector<RtpEncodingParameters> init_encodings(1);
|
|
|
|
|
init_encodings[0].max_bitrate_bps = 60000;
|
|
|
|
|
video_rtp_sender_->set_init_send_encodings(init_encodings);
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
ASSERT_EQ(1u, params.encodings.size());
|
|
|
|
|
EXPECT_EQ(params.encodings[0].max_bitrate_bps, 60000);
|
|
|
|
|
|
|
|
|
|
// Simulate the setLocalDescription call as if the user used SDP munging
|
|
|
|
|
// to enable simulcast
|
|
|
|
|
std::vector<uint32_t> ssrcs;
|
2019-01-29 10:11:53 +01:00
|
|
|
ssrcs.reserve(2);
|
2018-10-01 22:47:20 +02:00
|
|
|
for (int i = 0; i < 2; ++i)
|
|
|
|
|
ssrcs.push_back(kVideoSsrcSimulcast + i);
|
|
|
|
|
cricket::StreamParams stream_params =
|
|
|
|
|
cricket::CreateSimStreamParams("cname", ssrcs);
|
|
|
|
|
video_media_channel_->AddSendStream(stream_params);
|
2018-11-13 16:26:05 -08:00
|
|
|
video_rtp_sender_->SetMediaChannel(video_media_channel_);
|
2018-10-01 22:47:20 +02:00
|
|
|
video_rtp_sender_->SetSsrc(kVideoSsrcSimulcast);
|
|
|
|
|
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
ASSERT_EQ(2u, params.encodings.size());
|
|
|
|
|
EXPECT_EQ(params.encodings[0].max_bitrate_bps, 60000);
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
VideoSenderMustCallGetParametersBeforeSetParametersBeforeNegotiation) {
|
2019-05-20 19:31:53 +02:00
|
|
|
video_rtp_sender_ =
|
|
|
|
|
VideoRtpSender::Create(worker_thread_, /*id=*/"", nullptr);
|
2018-10-01 22:47:20 +02:00
|
|
|
|
|
|
|
|
RtpParameters params;
|
|
|
|
|
RTCError result = video_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_STATE, result.type());
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-03 15:31:53 +02:00
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
VideoSenderMustCallGetParametersBeforeSetParameters) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params;
|
|
|
|
|
RTCError result = video_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_STATE, result.type());
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
VideoSenderSetParametersInvalidatesTransactionId) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_EQ(1u, params.encodings.size());
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
RTCError result = video_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_STATE, result.type());
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderDetectTransactionIdModification) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
params.transaction_id = "";
|
|
|
|
|
RTCError result = video_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_MODIFICATION, result.type());
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCheckTransactionIdRefresh) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_NE(params.transaction_id.size(), 0U);
|
2018-05-03 15:31:53 +02:00
|
|
|
auto saved_transaction_id = params.transaction_id;
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_NE(saved_transaction_id, params.transaction_id);
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderSetParametersOldValueFail) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
RtpParameters second_params = video_rtp_sender_->GetParameters();
|
|
|
|
|
|
|
|
|
|
RTCError result = video_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_MODIFICATION, result.type());
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-16 16:02:32 -07:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCantSetUnimplementedRtpParameters) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_EQ(1u, params.encodings.size());
|
|
|
|
|
|
2018-07-18 16:00:28 +02:00
|
|
|
// Unimplemented RtpParameters: mid
|
2018-05-16 16:02:32 -07:00
|
|
|
params.mid = "dummy_mid";
|
|
|
|
|
EXPECT_EQ(RTCErrorType::UNSUPPORTED_PARAMETER,
|
|
|
|
|
video_rtp_sender_->SetParameters(params).type());
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-29 14:26:48 +01:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCanSetScaleResolutionDownBy) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
params.encodings[0].scale_resolution_down_by = 2;
|
|
|
|
|
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_EQ(2, params.encodings[0].scale_resolution_down_by);
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderDetectInvalidScaleResolutionDownBy) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
params.encodings[0].scale_resolution_down_by = 0.5;
|
|
|
|
|
RTCError result = video_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_RANGE, result.type());
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-06 15:03:19 +01:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCanSetMaxFramerate) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
params.encodings[0].max_framerate = 20;
|
|
|
|
|
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_EQ(20., params.encodings[0].max_framerate);
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCanSetMaxFramerateZero) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
params.encodings[0].max_framerate = 0.;
|
|
|
|
|
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_EQ(0., params.encodings[0].max_framerate);
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderDetectInvalidMaxFramerate) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
params.encodings[0].max_framerate = -5.;
|
|
|
|
|
RTCError result = video_rtp_sender_->SetParameters(params);
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_RANGE, result.type());
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-16 16:02:32 -07:00
|
|
|
// A video sender can have multiple simulcast layers, in which case it will
|
|
|
|
|
// contain multiple RtpEncodingParameters. This tests that if this is the case
|
|
|
|
|
// (simulcast), then we can't set the bitrate_priority, or max_bitrate_bps
|
|
|
|
|
// for any encodings besides at index 0, because these are both implemented
|
|
|
|
|
// "per-sender."
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCantSetPerSenderEncodingParameters) {
|
|
|
|
|
// Add a simulcast specific send stream that contains 2 encoding parameters.
|
2018-10-01 22:47:20 +02:00
|
|
|
CreateVideoRtpSenderWithSimulcast();
|
2018-05-16 16:02:32 -07:00
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
2018-10-01 22:47:20 +02:00
|
|
|
EXPECT_EQ(kVideoSimulcastLayerCount, params.encodings.size());
|
2018-05-16 16:02:32 -07:00
|
|
|
|
|
|
|
|
params.encodings[1].bitrate_priority = 2.0;
|
|
|
|
|
EXPECT_EQ(RTCErrorType::UNSUPPORTED_PARAMETER,
|
|
|
|
|
video_rtp_sender_->SetParameters(params).type());
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-01 22:47:20 +02:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCantSetReadOnlyEncodingParameters) {
|
|
|
|
|
// Add a simulcast specific send stream that contains 2 encoding parameters.
|
|
|
|
|
CreateVideoRtpSenderWithSimulcast();
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
|
|
|
|
EXPECT_EQ(kVideoSimulcastLayerCount, params.encodings.size());
|
|
|
|
|
|
|
|
|
|
for (size_t i = 0; i < params.encodings.size(); i++) {
|
|
|
|
|
params.encodings[i].ssrc = 1337;
|
|
|
|
|
EXPECT_EQ(RTCErrorType::INVALID_MODIFICATION,
|
|
|
|
|
video_rtp_sender_->SetParameters(params).type());
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-18 17:51:32 +02:00
|
|
|
TEST_F(RtpSenderReceiverTest, SetVideoMinMaxSendBitrate) {
|
2016-06-27 16:30:35 -07:00
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
EXPECT_EQ(-1, video_media_channel_->max_bps());
|
|
|
|
|
webrtc::RtpParameters params = video_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2018-06-18 17:51:32 +02:00
|
|
|
EXPECT_FALSE(params.encodings[0].min_bitrate_bps);
|
2017-02-04 12:09:01 -08:00
|
|
|
EXPECT_FALSE(params.encodings[0].max_bitrate_bps);
|
2018-06-18 17:51:32 +02:00
|
|
|
params.encodings[0].min_bitrate_bps = 100;
|
2017-11-16 10:54:27 +01:00
|
|
|
params.encodings[0].max_bitrate_bps = 1000;
|
2018-01-23 15:02:36 -08:00
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
2016-06-27 16:30:35 -07:00
|
|
|
|
|
|
|
|
// Read back the parameters and verify they have been changed.
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2018-06-18 17:51:32 +02:00
|
|
|
EXPECT_EQ(100, params.encodings[0].min_bitrate_bps);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(1000, params.encodings[0].max_bitrate_bps);
|
2016-06-27 16:30:35 -07:00
|
|
|
|
|
|
|
|
// Verify that the video channel received the new parameters.
|
|
|
|
|
params = video_media_channel_->GetRtpSendParameters(kVideoSsrc);
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2018-06-18 17:51:32 +02:00
|
|
|
EXPECT_EQ(100, params.encodings[0].min_bitrate_bps);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(1000, params.encodings[0].max_bitrate_bps);
|
2016-06-27 16:30:35 -07:00
|
|
|
|
|
|
|
|
// Verify that the global bitrate limit has not been changed.
|
|
|
|
|
EXPECT_EQ(-1, video_media_channel_->max_bps());
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-18 17:51:32 +02:00
|
|
|
TEST_F(RtpSenderReceiverTest, SetVideoMinMaxSendBitrateSimulcast) {
|
|
|
|
|
// Add a simulcast specific send stream that contains 2 encoding parameters.
|
2018-10-01 22:47:20 +02:00
|
|
|
CreateVideoRtpSenderWithSimulcast();
|
2018-06-18 17:51:32 +02:00
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_sender_->GetParameters();
|
2018-10-01 22:47:20 +02:00
|
|
|
EXPECT_EQ(kVideoSimulcastLayerCount, params.encodings.size());
|
2018-06-18 17:51:32 +02:00
|
|
|
params.encodings[0].min_bitrate_bps = 100;
|
|
|
|
|
params.encodings[0].max_bitrate_bps = 1000;
|
|
|
|
|
params.encodings[1].min_bitrate_bps = 200;
|
|
|
|
|
params.encodings[1].max_bitrate_bps = 2000;
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
|
|
|
|
|
|
|
|
|
// Verify that the video channel received the new parameters.
|
2018-10-01 22:47:20 +02:00
|
|
|
params = video_media_channel_->GetRtpSendParameters(kVideoSsrcSimulcast);
|
|
|
|
|
EXPECT_EQ(kVideoSimulcastLayerCount, params.encodings.size());
|
2018-06-18 17:51:32 +02:00
|
|
|
EXPECT_EQ(100, params.encodings[0].min_bitrate_bps);
|
|
|
|
|
EXPECT_EQ(1000, params.encodings[0].max_bitrate_bps);
|
|
|
|
|
EXPECT_EQ(200, params.encodings[1].min_bitrate_bps);
|
|
|
|
|
EXPECT_EQ(2000, params.encodings[1].max_bitrate_bps);
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2017-12-22 09:36:42 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, SetVideoBitratePriority) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
webrtc::RtpParameters params = video_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2017-12-22 09:36:42 -08:00
|
|
|
EXPECT_EQ(webrtc::kDefaultBitratePriority,
|
|
|
|
|
params.encodings[0].bitrate_priority);
|
|
|
|
|
double new_bitrate_priority = 2.0;
|
|
|
|
|
params.encodings[0].bitrate_priority = new_bitrate_priority;
|
2018-01-23 15:02:36 -08:00
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(params).ok());
|
2017-12-22 09:36:42 -08:00
|
|
|
|
|
|
|
|
params = video_rtp_sender_->GetParameters();
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2017-12-22 09:36:42 -08:00
|
|
|
EXPECT_EQ(new_bitrate_priority, params.encodings[0].bitrate_priority);
|
|
|
|
|
|
|
|
|
|
params = video_media_channel_->GetRtpSendParameters(kVideoSsrc);
|
2018-07-03 12:53:23 +02:00
|
|
|
EXPECT_EQ(1U, params.encodings.size());
|
2017-12-22 09:36:42 -08:00
|
|
|
EXPECT_EQ(new_bitrate_priority, params.encodings[0].bitrate_priority);
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-20 14:08:06 +01:00
|
|
|
TEST_F(RtpSenderReceiverTest, VideoReceiverCanGetParametersWithSimulcast) {
|
|
|
|
|
CreateVideoRtpReceiverWithSimulcast({}, 2);
|
|
|
|
|
|
|
|
|
|
RtpParameters params = video_rtp_receiver_->GetParameters();
|
|
|
|
|
EXPECT_EQ(2u, params.encodings.size());
|
|
|
|
|
|
|
|
|
|
DestroyVideoRtpReceiver();
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-16 15:39:11 -08:00
|
|
|
// Test that makes sure that a video track content hint translates to the proper
|
|
|
|
|
// value for sources that are not screencast.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, PropagatesVideoTrackContentHint) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
|
|
|
|
|
video_track_->set_enabled(true);
|
|
|
|
|
|
|
|
|
|
// |video_track_| is not screencast by default.
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(false, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
// No content hint should be set by default.
|
|
|
|
|
EXPECT_EQ(VideoTrackInterface::ContentHint::kNone,
|
|
|
|
|
video_track_->content_hint());
|
|
|
|
|
// Setting detailed should turn a non-screencast source into screencast mode.
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kDetailed);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(true, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
// Removing the content hint should turn the track back into non-screencast
|
|
|
|
|
// mode.
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kNone);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(false, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
// Setting fluid should remain in non-screencast mode (its default).
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kFluid);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(false, video_media_channel_->options().is_screencast);
|
2018-06-18 08:53:10 +02:00
|
|
|
// Setting text should have the same effect as Detailed
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kText);
|
|
|
|
|
EXPECT_EQ(true, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Test that makes sure that a video track content hint translates to the proper
|
|
|
|
|
// value for screencast sources.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
PropagatesVideoTrackContentHintForScreencastSource) {
|
|
|
|
|
CreateVideoRtpSender(true);
|
|
|
|
|
|
|
|
|
|
video_track_->set_enabled(true);
|
|
|
|
|
|
|
|
|
|
// |video_track_| with a screencast source should be screencast by default.
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(true, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
// No content hint should be set by default.
|
|
|
|
|
EXPECT_EQ(VideoTrackInterface::ContentHint::kNone,
|
|
|
|
|
video_track_->content_hint());
|
|
|
|
|
// Setting fluid should turn a screencast source into non-screencast mode.
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kFluid);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(false, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
// Removing the content hint should turn the track back into screencast mode.
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kNone);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(true, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
// Setting detailed should still remain in screencast mode (its default).
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kDetailed);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(true, video_media_channel_->options().is_screencast);
|
2018-06-18 08:53:10 +02:00
|
|
|
// Setting text should have the same effect as Detailed
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kText);
|
|
|
|
|
EXPECT_EQ(true, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Test that makes sure any content hints that are set on a track before
|
|
|
|
|
// VideoRtpSender is ready to send are still applied when it gets ready to send.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
PropagatesVideoTrackContentHintSetBeforeEnabling) {
|
|
|
|
|
AddVideoTrack();
|
2019-05-20 19:31:53 +02:00
|
|
|
std::unique_ptr<MockSetStreamsObserver> set_streams_observer =
|
2019-09-17 17:06:18 +02:00
|
|
|
std::make_unique<MockSetStreamsObserver>();
|
2016-12-16 15:39:11 -08:00
|
|
|
// Setting detailed overrides the default non-screencast mode. This should be
|
|
|
|
|
// applied even if the track is set on construction.
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kDetailed);
|
2019-05-20 19:31:53 +02:00
|
|
|
video_rtp_sender_ = VideoRtpSender::Create(worker_thread_, video_track_->id(),
|
|
|
|
|
set_streams_observer.get());
|
2018-06-25 13:03:36 -07:00
|
|
|
ASSERT_TRUE(video_rtp_sender_->SetTrack(video_track_));
|
2019-05-20 19:31:53 +02:00
|
|
|
EXPECT_CALL(*set_streams_observer, OnSetStreams());
|
|
|
|
|
video_rtp_sender_->SetStreams({local_stream_->id()});
|
2018-11-13 16:26:05 -08:00
|
|
|
video_rtp_sender_->SetMediaChannel(video_media_channel_);
|
2016-12-16 15:39:11 -08:00
|
|
|
video_track_->set_enabled(true);
|
|
|
|
|
|
|
|
|
|
// Sender is not ready to send (no SSRC) so no option should have been set.
|
2018-06-19 16:47:43 +02:00
|
|
|
EXPECT_EQ(absl::nullopt, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
|
|
|
|
|
// Verify that the content hint is accounted for when video_rtp_sender_ does
|
|
|
|
|
// get enabled.
|
|
|
|
|
video_rtp_sender_->SetSsrc(kVideoSsrc);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(true, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
|
|
|
|
|
// And removing the hint should go back to false (to verify that false was
|
|
|
|
|
// default correctly).
|
|
|
|
|
video_track_->set_content_hint(VideoTrackInterface::ContentHint::kNone);
|
2017-11-16 10:54:27 +01:00
|
|
|
EXPECT_EQ(false, video_media_channel_->options().is_screencast);
|
2016-12-16 15:39:11 -08:00
|
|
|
|
|
|
|
|
DestroyVideoRtpSender();
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-01 20:27:00 -08:00
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderHasDtmfSender) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
EXPECT_NE(nullptr, audio_rtp_sender_->GetDtmfSender());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderDoesNotHaveDtmfSender) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
EXPECT_EQ(nullptr, video_rtp_sender_->GetDtmfSender());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Test that the DTMF sender is really using |voice_channel_|, and thus returns
|
|
|
|
|
// true/false from CanSendDtmf based on what |voice_channel_| returns.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, CanInsertDtmf) {
|
|
|
|
|
AddDtmfCodec();
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
auto dtmf_sender = audio_rtp_sender_->GetDtmfSender();
|
|
|
|
|
ASSERT_NE(nullptr, dtmf_sender);
|
|
|
|
|
EXPECT_TRUE(dtmf_sender->CanInsertDtmf());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, CanNotInsertDtmf) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
auto dtmf_sender = audio_rtp_sender_->GetDtmfSender();
|
|
|
|
|
ASSERT_NE(nullptr, dtmf_sender);
|
|
|
|
|
// DTMF codec has not been added, as it was in the above test.
|
|
|
|
|
EXPECT_FALSE(dtmf_sender->CanInsertDtmf());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, InsertDtmf) {
|
|
|
|
|
AddDtmfCodec();
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
auto dtmf_sender = audio_rtp_sender_->GetDtmfSender();
|
|
|
|
|
ASSERT_NE(nullptr, dtmf_sender);
|
|
|
|
|
|
|
|
|
|
EXPECT_EQ(0U, voice_media_channel_->dtmf_info_queue().size());
|
|
|
|
|
|
|
|
|
|
// Insert DTMF
|
|
|
|
|
const int expected_duration = 90;
|
|
|
|
|
dtmf_sender->InsertDtmf("012", expected_duration, 100);
|
|
|
|
|
|
|
|
|
|
// Verify
|
|
|
|
|
ASSERT_EQ_WAIT(3U, voice_media_channel_->dtmf_info_queue().size(),
|
|
|
|
|
kDefaultTimeout);
|
|
|
|
|
const uint32_t send_ssrc =
|
|
|
|
|
voice_media_channel_->send_streams()[0].first_ssrc();
|
|
|
|
|
EXPECT_TRUE(CompareDtmfInfo(voice_media_channel_->dtmf_info_queue()[0],
|
|
|
|
|
send_ssrc, 0, expected_duration));
|
|
|
|
|
EXPECT_TRUE(CompareDtmfInfo(voice_media_channel_->dtmf_info_queue()[1],
|
|
|
|
|
send_ssrc, 1, expected_duration));
|
|
|
|
|
EXPECT_TRUE(CompareDtmfInfo(voice_media_channel_->dtmf_info_queue()[2],
|
|
|
|
|
send_ssrc, 2, expected_duration));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure the signal from "GetOnDestroyedSignal()" fires when the sender is
|
|
|
|
|
// destroyed, which is needed for the DTMF sender.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, TestOnDestroyedSignal) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
EXPECT_FALSE(audio_sender_destroyed_signal_fired_);
|
|
|
|
|
audio_rtp_sender_ = nullptr;
|
|
|
|
|
EXPECT_TRUE(audio_sender_destroyed_signal_fired_);
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-04 14:22:34 -07:00
|
|
|
// Validate that the default FrameEncryptor setting is nullptr.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderCanSetFrameEncryptor) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
rtc::scoped_refptr<FrameEncryptorInterface> fake_frame_encryptor(
|
|
|
|
|
new FakeFrameEncryptor());
|
|
|
|
|
EXPECT_EQ(nullptr, audio_rtp_sender_->GetFrameEncryptor());
|
|
|
|
|
audio_rtp_sender_->SetFrameEncryptor(fake_frame_encryptor);
|
|
|
|
|
EXPECT_EQ(fake_frame_encryptor.get(),
|
|
|
|
|
audio_rtp_sender_->GetFrameEncryptor().get());
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-26 13:16:16 -07:00
|
|
|
// Validate that setting a FrameEncryptor after the send stream is stopped does
|
|
|
|
|
// nothing.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AudioSenderCannotSetFrameEncryptorAfterStop) {
|
|
|
|
|
CreateAudioRtpSender();
|
|
|
|
|
rtc::scoped_refptr<FrameEncryptorInterface> fake_frame_encryptor(
|
|
|
|
|
new FakeFrameEncryptor());
|
|
|
|
|
EXPECT_EQ(nullptr, audio_rtp_sender_->GetFrameEncryptor());
|
|
|
|
|
audio_rtp_sender_->Stop();
|
|
|
|
|
audio_rtp_sender_->SetFrameEncryptor(fake_frame_encryptor);
|
|
|
|
|
// TODO(webrtc:9926) - Validate media channel not set once fakes updated.
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-04 14:22:34 -07:00
|
|
|
// Validate that the default FrameEncryptor setting is nullptr.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AudioReceiverCanSetFrameDecryptor) {
|
|
|
|
|
CreateAudioRtpReceiver();
|
|
|
|
|
rtc::scoped_refptr<FrameDecryptorInterface> fake_frame_decryptor(
|
|
|
|
|
new FakeFrameDecryptor());
|
|
|
|
|
EXPECT_EQ(nullptr, audio_rtp_receiver_->GetFrameDecryptor());
|
|
|
|
|
audio_rtp_receiver_->SetFrameDecryptor(fake_frame_decryptor);
|
|
|
|
|
EXPECT_EQ(fake_frame_decryptor.get(),
|
|
|
|
|
audio_rtp_receiver_->GetFrameDecryptor().get());
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-26 13:16:16 -07:00
|
|
|
// Validate that the default FrameEncryptor setting is nullptr.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, AudioReceiverCannotSetFrameDecryptorAfterStop) {
|
|
|
|
|
CreateAudioRtpReceiver();
|
|
|
|
|
rtc::scoped_refptr<FrameDecryptorInterface> fake_frame_decryptor(
|
|
|
|
|
new FakeFrameDecryptor());
|
|
|
|
|
EXPECT_EQ(nullptr, audio_rtp_receiver_->GetFrameDecryptor());
|
|
|
|
|
audio_rtp_receiver_->Stop();
|
|
|
|
|
audio_rtp_receiver_->SetFrameDecryptor(fake_frame_decryptor);
|
|
|
|
|
// TODO(webrtc:9926) - Validate media channel not set once fakes updated.
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Validate that the default FrameEncryptor setting is nullptr.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCanSetFrameEncryptor) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
rtc::scoped_refptr<FrameEncryptorInterface> fake_frame_encryptor(
|
|
|
|
|
new FakeFrameEncryptor());
|
|
|
|
|
EXPECT_EQ(nullptr, video_rtp_sender_->GetFrameEncryptor());
|
|
|
|
|
video_rtp_sender_->SetFrameEncryptor(fake_frame_encryptor);
|
|
|
|
|
EXPECT_EQ(fake_frame_encryptor.get(),
|
|
|
|
|
video_rtp_sender_->GetFrameEncryptor().get());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Validate that setting a FrameEncryptor after the send stream is stopped does
|
|
|
|
|
// nothing.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoSenderCannotSetFrameEncryptorAfterStop) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
rtc::scoped_refptr<FrameEncryptorInterface> fake_frame_encryptor(
|
|
|
|
|
new FakeFrameEncryptor());
|
|
|
|
|
EXPECT_EQ(nullptr, video_rtp_sender_->GetFrameEncryptor());
|
|
|
|
|
video_rtp_sender_->Stop();
|
|
|
|
|
video_rtp_sender_->SetFrameEncryptor(fake_frame_encryptor);
|
|
|
|
|
// TODO(webrtc:9926) - Validate media channel not set once fakes updated.
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Validate that the default FrameEncryptor setting is nullptr.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoReceiverCanSetFrameDecryptor) {
|
|
|
|
|
CreateVideoRtpReceiver();
|
|
|
|
|
rtc::scoped_refptr<FrameDecryptorInterface> fake_frame_decryptor(
|
|
|
|
|
new FakeFrameDecryptor());
|
|
|
|
|
EXPECT_EQ(nullptr, video_rtp_receiver_->GetFrameDecryptor());
|
|
|
|
|
video_rtp_receiver_->SetFrameDecryptor(fake_frame_decryptor);
|
|
|
|
|
EXPECT_EQ(fake_frame_decryptor.get(),
|
|
|
|
|
video_rtp_receiver_->GetFrameDecryptor().get());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Validate that the default FrameEncryptor setting is nullptr.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest, VideoReceiverCannotSetFrameDecryptorAfterStop) {
|
|
|
|
|
CreateVideoRtpReceiver();
|
|
|
|
|
rtc::scoped_refptr<FrameDecryptorInterface> fake_frame_decryptor(
|
|
|
|
|
new FakeFrameDecryptor());
|
|
|
|
|
EXPECT_EQ(nullptr, video_rtp_receiver_->GetFrameDecryptor());
|
|
|
|
|
video_rtp_receiver_->Stop();
|
|
|
|
|
video_rtp_receiver_->SetFrameDecryptor(fake_frame_decryptor);
|
|
|
|
|
// TODO(webrtc:9926) - Validate media channel not set once fakes updated.
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-26 15:55:19 -08:00
|
|
|
// Checks that calling the internal methods for get/set parameters do not
|
|
|
|
|
// invalidate any parameters retreived by clients.
|
|
|
|
|
TEST_F(RtpSenderReceiverTest,
|
|
|
|
|
InternalParameterMethodsDoNotInvalidateTransaction) {
|
|
|
|
|
CreateVideoRtpSender();
|
|
|
|
|
RtpParameters parameters = video_rtp_sender_->GetParameters();
|
|
|
|
|
RtpParameters new_parameters = video_rtp_sender_->GetParametersInternal();
|
|
|
|
|
new_parameters.encodings[0].active = false;
|
|
|
|
|
video_rtp_sender_->SetParametersInternal(new_parameters);
|
|
|
|
|
new_parameters.encodings[0].active = true;
|
|
|
|
|
video_rtp_sender_->SetParametersInternal(new_parameters);
|
|
|
|
|
parameters.encodings[0].active = false;
|
|
|
|
|
EXPECT_TRUE(video_rtp_sender_->SetParameters(parameters).ok());
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-19 12:49:22 -08:00
|
|
|
// Helper method for syntactic sugar for accepting a vector with '{}' notation.
|
|
|
|
|
std::pair<RidList, RidList> CreatePairOfRidVectors(
|
|
|
|
|
const std::vector<std::string>& first,
|
|
|
|
|
const std::vector<std::string>& second) {
|
|
|
|
|
return std::make_pair(first, second);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// These parameters are used to test disabling simulcast layers.
|
|
|
|
|
const std::pair<RidList, RidList> kDisableSimulcastLayersParameters[] = {
|
|
|
|
|
// Tests removing the first layer. This is a special case because
|
|
|
|
|
// the first layer's SSRC is also the 'primary' SSRC used to associate the
|
|
|
|
|
// parameters to the media channel.
|
|
|
|
|
CreatePairOfRidVectors({"1", "2", "3", "4"}, {"1"}),
|
|
|
|
|
// Tests removing some layers.
|
|
|
|
|
CreatePairOfRidVectors({"1", "2", "3", "4"}, {"2", "4"}),
|
|
|
|
|
// Tests simulcast rejected scenario all layers except first are rejected.
|
|
|
|
|
CreatePairOfRidVectors({"1", "2", "3", "4"}, {"2", "3", "4"}),
|
|
|
|
|
// Tests removing all layers.
|
|
|
|
|
CreatePairOfRidVectors({"1", "2", "3", "4"}, {"1", "2", "3", "4"}),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Runs test for disabling layers on a sender without a media engine set.
|
|
|
|
|
TEST_P(RtpSenderReceiverTest, DisableSimulcastLayersWithoutMediaEngine) {
|
|
|
|
|
auto parameter = GetParam();
|
|
|
|
|
RunDisableSimulcastLayersWithoutMediaEngineTest(parameter.first,
|
|
|
|
|
parameter.second);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Runs test for disabling layers on a sender with a media engine set.
|
|
|
|
|
TEST_P(RtpSenderReceiverTest, DisableSimulcastLayersWithMediaEngine) {
|
|
|
|
|
auto parameter = GetParam();
|
|
|
|
|
RunDisableSimulcastLayersWithMediaEngineTest(parameter.first,
|
|
|
|
|
parameter.second);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(
|
|
|
|
|
DisableSimulcastLayersInSender,
|
|
|
|
|
RtpSenderReceiverTest,
|
|
|
|
|
::testing::ValuesIn(kDisableSimulcastLayersParameters));
|
|
|
|
|
|
2013-07-10 00:45:36 +00:00
|
|
|
} // namespace webrtc
|