Currently, datagram transports must report identical transport parameters in order to negotiate use of the datagram transport. This is not strictly necessary, they just need parameters that fit some notion of "compatability" (eg. both ends share some mutually-supported version of the datagram protocol). This change allows datagram transports to implement their own notion of compatible transport parameters, by adding a SetRemoteTransportParameters method to DatagramTransportInterface which checks if the remote parameters are compatible with the local endpoint and returns an error if they are not. Bug: webrtc:9719 Change-Id: I166c787b468b89d9082d7e3c9995a6ed50a1650a Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/167741 Commit-Queue: Bjorn Mellem <mellem@webrtc.org> Reviewed-by: Steve Anton <steveanton@webrtc.org> Cr-Commit-Position: refs/heads/master@{#30412}
270 lines
9.1 KiB
C++
270 lines
9.1 KiB
C++
/*
|
|
* Copyright 2018 The WebRTC Project Authors. All rights reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#ifndef API_TEST_LOOPBACK_MEDIA_TRANSPORT_H_
|
|
#define API_TEST_LOOPBACK_MEDIA_TRANSPORT_H_
|
|
|
|
#include <memory>
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "api/transport/datagram_transport_interface.h"
|
|
#include "api/transport/media/media_transport_interface.h"
|
|
#include "rtc_base/async_invoker.h"
|
|
#include "rtc_base/critical_section.h"
|
|
#include "rtc_base/thread.h"
|
|
#include "rtc_base/thread_checker.h"
|
|
|
|
namespace webrtc {
|
|
|
|
// Wrapper used to hand out unique_ptrs to loopback media
|
|
// transport without ownership changes to the underlying
|
|
// transport.
|
|
// It works in two modes:
|
|
// It can either wrap a factory, or it can wrap an existing interface.
|
|
// In the former mode, it delegates the work to the wrapped factory.
|
|
// In the latter mode, it always returns static instance of the transport
|
|
// interface.
|
|
//
|
|
// Example use:
|
|
// Factory wrap_static_interface = Wrapper(media_transport_interface);
|
|
// Factory wrap_factory = Wrapper(wrap_static_interface);
|
|
// The second factory may be created multiple times, and ownership may be passed
|
|
// to the client. The first factory counts the number of invocations of
|
|
// CreateMediaTransport();
|
|
class WrapperMediaTransportFactory : public MediaTransportFactory {
|
|
public:
|
|
explicit WrapperMediaTransportFactory(
|
|
DatagramTransportInterface* wrapped_datagram_transport);
|
|
explicit WrapperMediaTransportFactory(MediaTransportFactory* wrapped);
|
|
|
|
RTCErrorOr<std::unique_ptr<MediaTransportInterface>> CreateMediaTransport(
|
|
rtc::PacketTransportInternal* packet_transport,
|
|
rtc::Thread* network_thread,
|
|
const MediaTransportSettings& settings) override;
|
|
|
|
RTCErrorOr<std::unique_ptr<MediaTransportInterface>> CreateMediaTransport(
|
|
rtc::Thread* network_thread,
|
|
const MediaTransportSettings& settings) override;
|
|
|
|
RTCErrorOr<std::unique_ptr<DatagramTransportInterface>>
|
|
CreateDatagramTransport(rtc::Thread* network_thread,
|
|
const MediaTransportSettings& settings) override;
|
|
|
|
std::string GetTransportName() const override;
|
|
|
|
int created_transport_count() const;
|
|
|
|
private:
|
|
DatagramTransportInterface* wrapped_datagram_transport_ = nullptr;
|
|
MediaTransportFactory* wrapped_factory_ = nullptr;
|
|
int created_transport_count_ = 0;
|
|
};
|
|
|
|
// Contains two MediaTransportsInterfaces that are connected to each other.
|
|
// Currently supports audio only.
|
|
class MediaTransportPair {
|
|
public:
|
|
struct Stats {
|
|
int sent_audio_frames = 0;
|
|
int received_audio_frames = 0;
|
|
int sent_video_frames = 0;
|
|
int received_video_frames = 0;
|
|
};
|
|
|
|
explicit MediaTransportPair(rtc::Thread* thread);
|
|
~MediaTransportPair();
|
|
|
|
DatagramTransportInterface* first_datagram_transport() {
|
|
return &first_datagram_transport_;
|
|
}
|
|
DatagramTransportInterface* second_datagram_transport() {
|
|
return &second_datagram_transport_;
|
|
}
|
|
|
|
std::unique_ptr<MediaTransportFactory> first_factory() {
|
|
return std::make_unique<WrapperMediaTransportFactory>(&first_factory_);
|
|
}
|
|
|
|
std::unique_ptr<MediaTransportFactory> second_factory() {
|
|
return std::make_unique<WrapperMediaTransportFactory>(&second_factory_);
|
|
}
|
|
|
|
void SetState(MediaTransportState state) {
|
|
first_datagram_transport_.SetState(state);
|
|
second_datagram_transport_.SetState(state);
|
|
}
|
|
|
|
void SetFirstState(MediaTransportState state) {
|
|
first_datagram_transport_.SetState(state);
|
|
}
|
|
|
|
void SetSecondStateAfterConnect(MediaTransportState state) {
|
|
second_datagram_transport_.SetState(state);
|
|
}
|
|
|
|
void SetFirstDatagramTransportParameters(const std::string& params) {
|
|
first_datagram_transport_.set_transport_parameters(params);
|
|
}
|
|
|
|
void SetSecondDatagramTransportParameters(const std::string& params) {
|
|
second_datagram_transport_.set_transport_parameters(params);
|
|
}
|
|
|
|
void SetFirstDatagramTransportParametersComparison(
|
|
std::function<bool(absl::string_view, absl::string_view)> comparison) {
|
|
first_datagram_transport_.set_transport_parameters_comparison(
|
|
std::move(comparison));
|
|
}
|
|
|
|
void SetSecondDatagramTransportParametersComparison(
|
|
std::function<bool(absl::string_view, absl::string_view)> comparison) {
|
|
second_datagram_transport_.set_transport_parameters_comparison(
|
|
std::move(comparison));
|
|
}
|
|
|
|
void FlushAsyncInvokes() {
|
|
first_datagram_transport_.FlushAsyncInvokes();
|
|
second_datagram_transport_.FlushAsyncInvokes();
|
|
}
|
|
|
|
int first_factory_transport_count() const {
|
|
return first_factory_.created_transport_count();
|
|
}
|
|
|
|
int second_factory_transport_count() const {
|
|
return second_factory_.created_transport_count();
|
|
}
|
|
|
|
private:
|
|
class LoopbackDataChannelTransport : public DataChannelTransportInterface {
|
|
public:
|
|
explicit LoopbackDataChannelTransport(rtc::Thread* thread);
|
|
~LoopbackDataChannelTransport() override;
|
|
|
|
void Connect(LoopbackDataChannelTransport* other);
|
|
|
|
RTCError OpenChannel(int channel_id) override;
|
|
|
|
RTCError SendData(int channel_id,
|
|
const SendDataParams& params,
|
|
const rtc::CopyOnWriteBuffer& buffer) override;
|
|
|
|
RTCError CloseChannel(int channel_id) override;
|
|
|
|
bool IsReadyToSend() const override;
|
|
|
|
void SetDataSink(DataChannelSink* sink) override;
|
|
|
|
void OnReadyToSend(bool ready_to_send);
|
|
|
|
void FlushAsyncInvokes();
|
|
|
|
private:
|
|
void OnData(int channel_id,
|
|
DataMessageType type,
|
|
const rtc::CopyOnWriteBuffer& buffer);
|
|
|
|
void OnRemoteCloseChannel(int channel_id);
|
|
|
|
rtc::Thread* const thread_;
|
|
rtc::CriticalSection sink_lock_;
|
|
DataChannelSink* data_sink_ RTC_GUARDED_BY(sink_lock_) = nullptr;
|
|
|
|
bool ready_to_send_ RTC_GUARDED_BY(sink_lock_) = false;
|
|
|
|
LoopbackDataChannelTransport* other_;
|
|
|
|
rtc::AsyncInvoker invoker_;
|
|
};
|
|
|
|
class LoopbackDatagramTransport : public DatagramTransportInterface {
|
|
public:
|
|
explicit LoopbackDatagramTransport(rtc::Thread* thread);
|
|
|
|
void Connect(LoopbackDatagramTransport* other);
|
|
|
|
// Datagram transport overrides.
|
|
void Connect(rtc::PacketTransportInternal* packet_transport) override;
|
|
CongestionControlInterface* congestion_control() override;
|
|
void SetTransportStateCallback(
|
|
MediaTransportStateCallback* callback) override;
|
|
RTCError SendDatagram(rtc::ArrayView<const uint8_t> data,
|
|
DatagramId datagram_id) override;
|
|
size_t GetLargestDatagramSize() const override;
|
|
void SetDatagramSink(DatagramSinkInterface* sink) override;
|
|
std::string GetTransportParameters() const override;
|
|
RTCError SetRemoteTransportParameters(
|
|
absl::string_view remote_parameters) override;
|
|
|
|
// Data channel overrides.
|
|
RTCError OpenChannel(int channel_id) override;
|
|
RTCError SendData(int channel_id,
|
|
const SendDataParams& params,
|
|
const rtc::CopyOnWriteBuffer& buffer) override;
|
|
RTCError CloseChannel(int channel_id) override;
|
|
void SetDataSink(DataChannelSink* sink) override;
|
|
bool IsReadyToSend() const override;
|
|
|
|
// Loopback-specific functionality.
|
|
void SetState(MediaTransportState state);
|
|
|
|
// When Connect() is called, the datagram transport will enter this state.
|
|
// This is useful for mimicking zero-RTT connectivity, for example.
|
|
void SetStateAfterConnect(MediaTransportState state);
|
|
void FlushAsyncInvokes();
|
|
|
|
void set_transport_parameters(const std::string& value) {
|
|
transport_parameters_ = value;
|
|
}
|
|
|
|
void set_transport_parameters_comparison(
|
|
std::function<bool(absl::string_view, absl::string_view)> comparison) {
|
|
thread_->Invoke<void>(
|
|
RTC_FROM_HERE, [this, comparison = std::move(comparison)] {
|
|
RTC_DCHECK_RUN_ON(thread_);
|
|
transport_parameters_comparison_ = std::move(comparison);
|
|
});
|
|
}
|
|
|
|
private:
|
|
void DeliverDatagram(rtc::CopyOnWriteBuffer buffer);
|
|
|
|
rtc::Thread* thread_;
|
|
LoopbackDataChannelTransport dc_transport_;
|
|
|
|
MediaTransportState state_ RTC_GUARDED_BY(thread_) =
|
|
MediaTransportState::kPending;
|
|
DatagramSinkInterface* sink_ RTC_GUARDED_BY(thread_) = nullptr;
|
|
MediaTransportStateCallback* state_callback_ RTC_GUARDED_BY(thread_) =
|
|
nullptr;
|
|
LoopbackDatagramTransport* other_;
|
|
|
|
std::string transport_parameters_;
|
|
std::function<bool(absl::string_view, absl::string_view)>
|
|
transport_parameters_comparison_ RTC_GUARDED_BY(thread_) =
|
|
[](absl::string_view a, absl::string_view b) { return a == b; };
|
|
|
|
absl::optional<MediaTransportState> state_after_connect_;
|
|
|
|
rtc::AsyncInvoker invoker_;
|
|
};
|
|
|
|
LoopbackDatagramTransport first_datagram_transport_;
|
|
LoopbackDatagramTransport second_datagram_transport_;
|
|
WrapperMediaTransportFactory first_factory_;
|
|
WrapperMediaTransportFactory second_factory_;
|
|
};
|
|
|
|
} // namespace webrtc
|
|
|
|
#endif // API_TEST_LOOPBACK_MEDIA_TRANSPORT_H_
|