This cl/ adds the feature actually injecting a FieldTrialsView into PeerConnectionFactory, or into a PeerConnection or both. The field trials used for a PeerConnection is those specified in PeerConnectionDependencies. Otherwise will those from PeerConnectionFactoryDependencies be used (and until we're finished with this conversion, the global string fallback is used as last resort). Note that it is currently not possible to create 2 FieldTrials objects concurrently...due to global string, so this cl/ is mostly (but entirely) for show, i.e one _can_ realistically inject them into a PeerConnectionFactory. Bug: webrtc:10335 Change-Id: Id2e60525f48a1f8293c1dd0be771e3ed03790963 Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/258134 Reviewed-by: Mirko Bonadei <mbonadei@webrtc.org> Reviewed-by: Harald Alvestrand <hta@webrtc.org> Commit-Queue: Jonas Oreland <jonaso@webrtc.org> Cr-Commit-Position: refs/heads/main@{#36578}
185 lines
6.6 KiB
C++
185 lines
6.6 KiB
C++
/*
|
|
* Copyright 2020 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/connection_context.h"
|
|
|
|
#include <type_traits>
|
|
#include <utility>
|
|
|
|
#include "api/transport/field_trial_based_config.h"
|
|
#include "media/base/media_engine.h"
|
|
#include "media/sctp/sctp_transport_factory.h"
|
|
#include "rtc_base/helpers.h"
|
|
#include "rtc_base/internal/default_socket_server.h"
|
|
#include "rtc_base/socket_server.h"
|
|
#include "rtc_base/task_utils/to_queued_task.h"
|
|
#include "rtc_base/time_utils.h"
|
|
|
|
namespace webrtc {
|
|
|
|
namespace {
|
|
|
|
rtc::Thread* MaybeStartNetworkThread(
|
|
rtc::Thread* old_thread,
|
|
std::unique_ptr<rtc::SocketFactory>& socket_factory_holder,
|
|
std::unique_ptr<rtc::Thread>& thread_holder) {
|
|
if (old_thread) {
|
|
return old_thread;
|
|
}
|
|
std::unique_ptr<rtc::SocketServer> socket_server =
|
|
rtc::CreateDefaultSocketServer();
|
|
thread_holder = std::make_unique<rtc::Thread>(socket_server.get());
|
|
socket_factory_holder = std::move(socket_server);
|
|
|
|
thread_holder->SetName("pc_network_thread", nullptr);
|
|
thread_holder->Start();
|
|
return thread_holder.get();
|
|
}
|
|
|
|
rtc::Thread* MaybeStartWorkerThread(
|
|
rtc::Thread* old_thread,
|
|
std::unique_ptr<rtc::Thread>& thread_holder) {
|
|
if (old_thread) {
|
|
return old_thread;
|
|
}
|
|
thread_holder = rtc::Thread::Create();
|
|
thread_holder->SetName("pc_worker_thread", nullptr);
|
|
thread_holder->Start();
|
|
return thread_holder.get();
|
|
}
|
|
|
|
rtc::Thread* MaybeWrapThread(rtc::Thread* signaling_thread,
|
|
bool& wraps_current_thread) {
|
|
wraps_current_thread = false;
|
|
if (signaling_thread) {
|
|
return signaling_thread;
|
|
}
|
|
auto this_thread = rtc::Thread::Current();
|
|
if (!this_thread) {
|
|
// If this thread isn't already wrapped by an rtc::Thread, create a
|
|
// wrapper and own it in this class.
|
|
this_thread = rtc::ThreadManager::Instance()->WrapCurrentThread();
|
|
wraps_current_thread = true;
|
|
}
|
|
return this_thread;
|
|
}
|
|
|
|
std::unique_ptr<SctpTransportFactoryInterface> MaybeCreateSctpFactory(
|
|
std::unique_ptr<SctpTransportFactoryInterface> factory,
|
|
rtc::Thread* network_thread,
|
|
const FieldTrialsView& field_trials) {
|
|
if (factory) {
|
|
return factory;
|
|
}
|
|
#ifdef WEBRTC_HAVE_SCTP
|
|
return std::make_unique<cricket::SctpTransportFactory>(network_thread);
|
|
#else
|
|
return nullptr;
|
|
#endif
|
|
}
|
|
|
|
} // namespace
|
|
|
|
// Static
|
|
rtc::scoped_refptr<ConnectionContext> ConnectionContext::Create(
|
|
PeerConnectionFactoryDependencies* dependencies) {
|
|
return rtc::scoped_refptr<ConnectionContext>(
|
|
new ConnectionContext(dependencies));
|
|
}
|
|
|
|
ConnectionContext::ConnectionContext(
|
|
PeerConnectionFactoryDependencies* dependencies)
|
|
: network_thread_(MaybeStartNetworkThread(dependencies->network_thread,
|
|
owned_socket_factory_,
|
|
owned_network_thread_)),
|
|
worker_thread_(MaybeStartWorkerThread(dependencies->worker_thread,
|
|
owned_worker_thread_)),
|
|
signaling_thread_(MaybeWrapThread(dependencies->signaling_thread,
|
|
wraps_current_thread_)),
|
|
trials_(dependencies->trials ? std::move(dependencies->trials)
|
|
: std::make_unique<FieldTrialBasedConfig>()),
|
|
network_monitor_factory_(
|
|
std::move(dependencies->network_monitor_factory)),
|
|
call_factory_(std::move(dependencies->call_factory)),
|
|
sctp_factory_(
|
|
MaybeCreateSctpFactory(std::move(dependencies->sctp_factory),
|
|
network_thread(),
|
|
*trials_.get())) {
|
|
signaling_thread_->AllowInvokesToThread(worker_thread_);
|
|
signaling_thread_->AllowInvokesToThread(network_thread_);
|
|
worker_thread_->AllowInvokesToThread(network_thread_);
|
|
if (network_thread_->IsCurrent()) {
|
|
// TODO(https://crbug.com/webrtc/12802) switch to DisallowAllInvokes
|
|
network_thread_->AllowInvokesToThread(network_thread_);
|
|
} else {
|
|
network_thread_->PostTask(ToQueuedTask([thread = network_thread_] {
|
|
thread->DisallowBlockingCalls();
|
|
// TODO(https://crbug.com/webrtc/12802) switch to DisallowAllInvokes
|
|
thread->AllowInvokesToThread(thread);
|
|
}));
|
|
}
|
|
|
|
RTC_DCHECK_RUN_ON(signaling_thread_);
|
|
rtc::InitRandom(rtc::Time32());
|
|
|
|
rtc::SocketFactory* socket_factory = dependencies->socket_factory;
|
|
if (socket_factory == nullptr) {
|
|
if (owned_socket_factory_) {
|
|
socket_factory = owned_socket_factory_.get();
|
|
} else {
|
|
// TODO(bugs.webrtc.org/13145): This case should be deleted. Either
|
|
// require that a PacketSocketFactory and NetworkManager always are
|
|
// injected (with no need to construct these default objects), or require
|
|
// that if a network_thread is injected, an approprite rtc::SocketServer
|
|
// should be injected too.
|
|
socket_factory = network_thread()->socketserver();
|
|
}
|
|
}
|
|
// If network_monitor_factory_ is non-null, it will be used to create a
|
|
// network monitor while on the network thread.
|
|
default_network_manager_ = std::make_unique<rtc::BasicNetworkManager>(
|
|
network_monitor_factory_.get(), socket_factory, &field_trials());
|
|
|
|
default_socket_factory_ =
|
|
std::make_unique<rtc::BasicPacketSocketFactory>(socket_factory);
|
|
|
|
channel_manager_ = cricket::ChannelManager::Create(
|
|
std::move(dependencies->media_engine),
|
|
/*enable_rtx=*/true, worker_thread(), network_thread());
|
|
|
|
// Set warning levels on the threads, to give warnings when response
|
|
// may be slower than is expected of the thread.
|
|
// Since some of the threads may be the same, start with the least
|
|
// restrictive limits and end with the least permissive ones.
|
|
// This will give warnings for all cases.
|
|
signaling_thread_->SetDispatchWarningMs(100);
|
|
worker_thread_->SetDispatchWarningMs(30);
|
|
network_thread_->SetDispatchWarningMs(10);
|
|
}
|
|
|
|
ConnectionContext::~ConnectionContext() {
|
|
RTC_DCHECK_RUN_ON(signaling_thread_);
|
|
channel_manager_.reset(nullptr);
|
|
|
|
// Make sure `worker_thread()` and `signaling_thread()` outlive
|
|
// `default_socket_factory_` and `default_network_manager_`.
|
|
default_socket_factory_ = nullptr;
|
|
default_network_manager_ = nullptr;
|
|
|
|
if (wraps_current_thread_)
|
|
rtc::ThreadManager::Instance()->UnwrapCurrentThread();
|
|
}
|
|
|
|
cricket::ChannelManager* ConnectionContext::channel_manager() const {
|
|
return channel_manager_.get();
|
|
}
|
|
|
|
} // namespace webrtc
|