summaryrefslogtreecommitdiffstats
path: root/third_party/libwebrtc/pc/peer_connection_factory.cc
diff options
context:
space:
mode:
Diffstat (limited to 'third_party/libwebrtc/pc/peer_connection_factory.cc')
-rw-r--r--third_party/libwebrtc/pc/peer_connection_factory.cc359
1 files changed, 359 insertions, 0 deletions
diff --git a/third_party/libwebrtc/pc/peer_connection_factory.cc b/third_party/libwebrtc/pc/peer_connection_factory.cc
new file mode 100644
index 0000000000..afebdd79a5
--- /dev/null
+++ b/third_party/libwebrtc/pc/peer_connection_factory.cc
@@ -0,0 +1,359 @@
+/*
+ * Copyright 2004 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#include "pc/peer_connection_factory.h"
+
+#include <type_traits>
+#include <utility>
+
+#include "absl/strings/match.h"
+#include "api/async_resolver_factory.h"
+#include "api/call/call_factory_interface.h"
+#include "api/fec_controller.h"
+#include "api/ice_transport_interface.h"
+#include "api/network_state_predictor.h"
+#include "api/packet_socket_factory.h"
+#include "api/rtc_event_log/rtc_event_log.h"
+#include "api/sequence_checker.h"
+#include "api/transport/bitrate_settings.h"
+#include "api/units/data_rate.h"
+#include "call/audio_state.h"
+#include "call/rtp_transport_controller_send_factory.h"
+#include "media/base/media_engine.h"
+#include "p2p/base/basic_async_resolver_factory.h"
+#include "p2p/base/basic_packet_socket_factory.h"
+#include "p2p/base/default_ice_transport_factory.h"
+#include "p2p/base/port_allocator.h"
+#include "p2p/client/basic_port_allocator.h"
+#include "pc/audio_track.h"
+#include "pc/local_audio_source.h"
+#include "pc/media_stream.h"
+#include "pc/media_stream_proxy.h"
+#include "pc/media_stream_track_proxy.h"
+#include "pc/peer_connection.h"
+#include "pc/peer_connection_factory_proxy.h"
+#include "pc/peer_connection_proxy.h"
+#include "pc/rtp_parameters_conversion.h"
+#include "pc/session_description.h"
+#include "pc/video_track.h"
+#include "rtc_base/checks.h"
+#include "rtc_base/experiments/field_trial_parser.h"
+#include "rtc_base/experiments/field_trial_units.h"
+#include "rtc_base/logging.h"
+#include "rtc_base/numerics/safe_conversions.h"
+#include "rtc_base/rtc_certificate_generator.h"
+#include "rtc_base/system/file_wrapper.h"
+
+namespace webrtc {
+
+rtc::scoped_refptr<PeerConnectionFactoryInterface>
+CreateModularPeerConnectionFactory(
+ PeerConnectionFactoryDependencies dependencies) {
+ // The PeerConnectionFactory must be created on the signaling thread.
+ if (dependencies.signaling_thread &&
+ !dependencies.signaling_thread->IsCurrent()) {
+ return dependencies.signaling_thread->BlockingCall([&dependencies] {
+ return CreateModularPeerConnectionFactory(std::move(dependencies));
+ });
+ }
+
+ auto pc_factory = PeerConnectionFactory::Create(std::move(dependencies));
+ if (!pc_factory) {
+ return nullptr;
+ }
+ // Verify that the invocation and the initialization ended up agreeing on the
+ // thread.
+ RTC_DCHECK_RUN_ON(pc_factory->signaling_thread());
+ return PeerConnectionFactoryProxy::Create(
+ pc_factory->signaling_thread(), pc_factory->worker_thread(), pc_factory);
+}
+
+// Static
+rtc::scoped_refptr<PeerConnectionFactory> PeerConnectionFactory::Create(
+ PeerConnectionFactoryDependencies dependencies) {
+ auto context = ConnectionContext::Create(&dependencies);
+ if (!context) {
+ return nullptr;
+ }
+ return rtc::make_ref_counted<PeerConnectionFactory>(context, &dependencies);
+}
+
+PeerConnectionFactory::PeerConnectionFactory(
+ rtc::scoped_refptr<ConnectionContext> context,
+ PeerConnectionFactoryDependencies* dependencies)
+ : context_(context),
+ task_queue_factory_(std::move(dependencies->task_queue_factory)),
+ event_log_factory_(std::move(dependencies->event_log_factory)),
+ fec_controller_factory_(std::move(dependencies->fec_controller_factory)),
+ network_state_predictor_factory_(
+ std::move(dependencies->network_state_predictor_factory)),
+ injected_network_controller_factory_(
+ std::move(dependencies->network_controller_factory)),
+ neteq_factory_(std::move(dependencies->neteq_factory)),
+ transport_controller_send_factory_(
+ (dependencies->transport_controller_send_factory)
+ ? std::move(dependencies->transport_controller_send_factory)
+ : std::make_unique<RtpTransportControllerSendFactory>()),
+ metronome_(std::move(dependencies->metronome)) {}
+
+PeerConnectionFactory::PeerConnectionFactory(
+ PeerConnectionFactoryDependencies dependencies)
+ : PeerConnectionFactory(ConnectionContext::Create(&dependencies),
+ &dependencies) {}
+
+PeerConnectionFactory::~PeerConnectionFactory() {
+ RTC_DCHECK_RUN_ON(signaling_thread());
+ worker_thread()->BlockingCall([this] {
+ RTC_DCHECK_RUN_ON(worker_thread());
+ metronome_ = nullptr;
+ });
+}
+
+void PeerConnectionFactory::SetOptions(const Options& options) {
+ RTC_DCHECK_RUN_ON(signaling_thread());
+ options_ = options;
+}
+
+RtpCapabilities PeerConnectionFactory::GetRtpSenderCapabilities(
+ cricket::MediaType kind) const {
+ RTC_DCHECK_RUN_ON(signaling_thread());
+ switch (kind) {
+ case cricket::MEDIA_TYPE_AUDIO: {
+ cricket::AudioCodecs cricket_codecs;
+ cricket_codecs = media_engine()->voice().send_codecs();
+ auto extensions =
+ GetDefaultEnabledRtpHeaderExtensions(media_engine()->voice());
+ return ToRtpCapabilities(cricket_codecs, extensions);
+ }
+ case cricket::MEDIA_TYPE_VIDEO: {
+ cricket::VideoCodecs cricket_codecs;
+ cricket_codecs = media_engine()->video().send_codecs();
+ auto extensions =
+ GetDefaultEnabledRtpHeaderExtensions(media_engine()->video());
+ return ToRtpCapabilities(cricket_codecs, extensions);
+ }
+ case cricket::MEDIA_TYPE_DATA:
+ return RtpCapabilities();
+ case cricket::MEDIA_TYPE_UNSUPPORTED:
+ return RtpCapabilities();
+ }
+ RTC_DLOG(LS_ERROR) << "Got unexpected MediaType " << kind;
+ RTC_CHECK_NOTREACHED();
+}
+
+RtpCapabilities PeerConnectionFactory::GetRtpReceiverCapabilities(
+ cricket::MediaType kind) const {
+ RTC_DCHECK_RUN_ON(signaling_thread());
+ switch (kind) {
+ case cricket::MEDIA_TYPE_AUDIO: {
+ cricket::AudioCodecs cricket_codecs;
+ cricket_codecs = media_engine()->voice().recv_codecs();
+ auto extensions =
+ GetDefaultEnabledRtpHeaderExtensions(media_engine()->voice());
+ return ToRtpCapabilities(cricket_codecs, extensions);
+ }
+ case cricket::MEDIA_TYPE_VIDEO: {
+ cricket::VideoCodecs cricket_codecs =
+ media_engine()->video().recv_codecs(context_->use_rtx());
+ auto extensions =
+ GetDefaultEnabledRtpHeaderExtensions(media_engine()->video());
+ return ToRtpCapabilities(cricket_codecs, extensions);
+ }
+ case cricket::MEDIA_TYPE_DATA:
+ return RtpCapabilities();
+ case cricket::MEDIA_TYPE_UNSUPPORTED:
+ return RtpCapabilities();
+ }
+ RTC_DLOG(LS_ERROR) << "Got unexpected MediaType " << kind;
+ RTC_CHECK_NOTREACHED();
+}
+
+rtc::scoped_refptr<AudioSourceInterface>
+PeerConnectionFactory::CreateAudioSource(const cricket::AudioOptions& options) {
+ RTC_DCHECK(signaling_thread()->IsCurrent());
+ rtc::scoped_refptr<LocalAudioSource> source(
+ LocalAudioSource::Create(&options));
+ return source;
+}
+
+bool PeerConnectionFactory::StartAecDump(FILE* file, int64_t max_size_bytes) {
+ RTC_DCHECK_RUN_ON(worker_thread());
+ return media_engine()->voice().StartAecDump(FileWrapper(file),
+ max_size_bytes);
+}
+
+void PeerConnectionFactory::StopAecDump() {
+ RTC_DCHECK_RUN_ON(worker_thread());
+ media_engine()->voice().StopAecDump();
+}
+
+cricket::MediaEngineInterface* PeerConnectionFactory::media_engine() const {
+ RTC_DCHECK(context_);
+ return context_->media_engine();
+}
+
+RTCErrorOr<rtc::scoped_refptr<PeerConnectionInterface>>
+PeerConnectionFactory::CreatePeerConnectionOrError(
+ const PeerConnectionInterface::RTCConfiguration& configuration,
+ PeerConnectionDependencies dependencies) {
+ RTC_DCHECK_RUN_ON(signaling_thread());
+
+ // Set internal defaults if optional dependencies are not set.
+ if (!dependencies.cert_generator) {
+ dependencies.cert_generator =
+ std::make_unique<rtc::RTCCertificateGenerator>(signaling_thread(),
+ network_thread());
+ }
+ if (!dependencies.allocator) {
+ const FieldTrialsView* trials =
+ dependencies.trials ? dependencies.trials.get() : &field_trials();
+ dependencies.allocator = std::make_unique<cricket::BasicPortAllocator>(
+ context_->default_network_manager(), context_->default_socket_factory(),
+ configuration.turn_customizer, /*relay_port_factory=*/nullptr, trials);
+ dependencies.allocator->SetPortRange(
+ configuration.port_allocator_config.min_port,
+ configuration.port_allocator_config.max_port);
+ dependencies.allocator->set_flags(
+ configuration.port_allocator_config.flags);
+ }
+
+ if (!dependencies.async_resolver_factory) {
+ dependencies.async_resolver_factory =
+ std::make_unique<webrtc::BasicAsyncResolverFactory>();
+ }
+
+ if (!dependencies.ice_transport_factory) {
+ dependencies.ice_transport_factory =
+ std::make_unique<DefaultIceTransportFactory>();
+ }
+
+ dependencies.allocator->SetNetworkIgnoreMask(options().network_ignore_mask);
+ dependencies.allocator->SetVpnList(configuration.vpn_list);
+
+ std::unique_ptr<RtcEventLog> event_log =
+ worker_thread()->BlockingCall([this] { return CreateRtcEventLog_w(); });
+
+ const FieldTrialsView* trials =
+ dependencies.trials ? dependencies.trials.get() : &field_trials();
+ std::unique_ptr<Call> call =
+ worker_thread()->BlockingCall([this, &event_log, trials, &configuration] {
+ return CreateCall_w(event_log.get(), *trials, configuration);
+ });
+
+ auto result = PeerConnection::Create(context_, options_, std::move(event_log),
+ std::move(call), configuration,
+ std::move(dependencies));
+ if (!result.ok()) {
+ return result.MoveError();
+ }
+ // We configure the proxy with a pointer to the network thread for methods
+ // that need to be invoked there rather than on the signaling thread.
+ // Internally, the proxy object has a member variable named `worker_thread_`
+ // which will point to the network thread (and not the factory's
+ // worker_thread()). All such methods have thread checks though, so the code
+ // should still be clear (outside of macro expansion).
+ rtc::scoped_refptr<PeerConnectionInterface> result_proxy =
+ PeerConnectionProxy::Create(signaling_thread(), network_thread(),
+ result.MoveValue());
+ return result_proxy;
+}
+
+rtc::scoped_refptr<MediaStreamInterface>
+PeerConnectionFactory::CreateLocalMediaStream(const std::string& stream_id) {
+ RTC_DCHECK(signaling_thread()->IsCurrent());
+ return MediaStreamProxy::Create(signaling_thread(),
+ MediaStream::Create(stream_id));
+}
+
+rtc::scoped_refptr<VideoTrackInterface> PeerConnectionFactory::CreateVideoTrack(
+ const std::string& id,
+ VideoTrackSourceInterface* source) {
+ RTC_DCHECK(signaling_thread()->IsCurrent());
+ rtc::scoped_refptr<VideoTrackInterface> track = VideoTrack::Create(
+ id, rtc::scoped_refptr<VideoTrackSourceInterface>(source),
+ worker_thread());
+ return VideoTrackProxy::Create(signaling_thread(), worker_thread(), track);
+}
+
+rtc::scoped_refptr<AudioTrackInterface> PeerConnectionFactory::CreateAudioTrack(
+ const std::string& id,
+ AudioSourceInterface* source) {
+ RTC_DCHECK(signaling_thread()->IsCurrent());
+ rtc::scoped_refptr<AudioTrackInterface> track =
+ AudioTrack::Create(id, rtc::scoped_refptr<AudioSourceInterface>(source));
+ return AudioTrackProxy::Create(signaling_thread(), track);
+}
+
+std::unique_ptr<RtcEventLog> PeerConnectionFactory::CreateRtcEventLog_w() {
+ RTC_DCHECK_RUN_ON(worker_thread());
+
+ auto encoding_type = RtcEventLog::EncodingType::Legacy;
+ if (IsTrialEnabled("WebRTC-RtcEventLogNewFormat"))
+ encoding_type = RtcEventLog::EncodingType::NewFormat;
+ return event_log_factory_ ? event_log_factory_->Create(encoding_type)
+ : std::make_unique<RtcEventLogNull>();
+}
+
+std::unique_ptr<Call> PeerConnectionFactory::CreateCall_w(
+ RtcEventLog* event_log,
+ const FieldTrialsView& field_trials,
+ const PeerConnectionInterface::RTCConfiguration& configuration) {
+ RTC_DCHECK_RUN_ON(worker_thread());
+
+ webrtc::Call::Config call_config(event_log, network_thread());
+ if (!media_engine() || !context_->call_factory()) {
+ return nullptr;
+ }
+ call_config.audio_state = media_engine()->voice().GetAudioState();
+
+ FieldTrialParameter<DataRate> min_bandwidth("min",
+ DataRate::KilobitsPerSec(30));
+ FieldTrialParameter<DataRate> start_bandwidth("start",
+ DataRate::KilobitsPerSec(300));
+ FieldTrialParameter<DataRate> max_bandwidth("max",
+ DataRate::KilobitsPerSec(2000));
+ ParseFieldTrial({&min_bandwidth, &start_bandwidth, &max_bandwidth},
+ field_trials.Lookup("WebRTC-PcFactoryDefaultBitrates"));
+
+ call_config.bitrate_config.min_bitrate_bps =
+ rtc::saturated_cast<int>(min_bandwidth->bps());
+ call_config.bitrate_config.start_bitrate_bps =
+ rtc::saturated_cast<int>(start_bandwidth->bps());
+ call_config.bitrate_config.max_bitrate_bps =
+ rtc::saturated_cast<int>(max_bandwidth->bps());
+
+ call_config.fec_controller_factory = fec_controller_factory_.get();
+ call_config.task_queue_factory = task_queue_factory_.get();
+ call_config.network_state_predictor_factory =
+ network_state_predictor_factory_.get();
+ call_config.neteq_factory = neteq_factory_.get();
+
+ if (IsTrialEnabled("WebRTC-Bwe-InjectedCongestionController")) {
+ RTC_LOG(LS_INFO) << "Using injected network controller factory";
+ call_config.network_controller_factory =
+ injected_network_controller_factory_.get();
+ } else {
+ RTC_LOG(LS_INFO) << "Using default network controller factory";
+ }
+
+ call_config.trials = &field_trials;
+ call_config.rtp_transport_controller_send_factory =
+ transport_controller_send_factory_.get();
+ call_config.metronome = metronome_.get();
+ call_config.pacer_burst_interval = configuration.pacer_burst_interval;
+ return std::unique_ptr<Call>(
+ context_->call_factory()->CreateCall(call_config));
+}
+
+bool PeerConnectionFactory::IsTrialEnabled(absl::string_view key) const {
+ return absl::StartsWith(field_trials().Lookup(key), "Enabled");
+}
+
+} // namespace webrtc