2018-10-16 10:41:42 +02:00
|
|
|
/*
|
|
|
|
|
* 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_
|
|
|
|
|
|
2018-11-08 11:23:22 -08:00
|
|
|
#include <memory>
|
2018-10-16 10:41:42 +02:00
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
|
|
#include "api/media_transport_interface.h"
|
2018-11-01 16:42:44 -07:00
|
|
|
#include "rtc_base/asyncinvoker.h"
|
|
|
|
|
#include "rtc_base/criticalsection.h"
|
|
|
|
|
#include "rtc_base/thread.h"
|
2018-11-08 11:23:22 -08:00
|
|
|
#include "rtc_base/thread_checker.h"
|
2018-10-16 10:41:42 +02:00
|
|
|
|
|
|
|
|
namespace webrtc {
|
|
|
|
|
|
2018-11-08 11:23:22 -08:00
|
|
|
// Wrapper used to hand out unique_ptrs to loopback media transports without
|
|
|
|
|
// ownership changes.
|
|
|
|
|
class WrapperMediaTransport : public MediaTransportInterface {
|
|
|
|
|
public:
|
|
|
|
|
explicit WrapperMediaTransport(MediaTransportInterface* wrapped)
|
|
|
|
|
: wrapped_(wrapped) {}
|
|
|
|
|
|
|
|
|
|
RTCError SendAudioFrame(uint64_t channel_id,
|
|
|
|
|
MediaTransportEncodedAudioFrame frame) override {
|
|
|
|
|
return wrapped_->SendAudioFrame(channel_id, std::move(frame));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RTCError SendVideoFrame(
|
|
|
|
|
uint64_t channel_id,
|
|
|
|
|
const MediaTransportEncodedVideoFrame& frame) override {
|
|
|
|
|
return wrapped_->SendVideoFrame(channel_id, frame);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RTCError RequestKeyFrame(uint64_t channel_id) override {
|
|
|
|
|
return wrapped_->RequestKeyFrame(channel_id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SetReceiveAudioSink(MediaTransportAudioSinkInterface* sink) override {
|
|
|
|
|
wrapped_->SetReceiveAudioSink(sink);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SetReceiveVideoSink(MediaTransportVideoSinkInterface* sink) override {
|
|
|
|
|
wrapped_->SetReceiveVideoSink(sink);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SetMediaTransportStateCallback(
|
|
|
|
|
MediaTransportStateCallback* callback) override {
|
|
|
|
|
wrapped_->SetMediaTransportStateCallback(callback);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RTCError SendData(int channel_id,
|
|
|
|
|
const SendDataParams& params,
|
|
|
|
|
const rtc::CopyOnWriteBuffer& buffer) override {
|
|
|
|
|
return wrapped_->SendData(channel_id, params, buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RTCError CloseChannel(int channel_id) override {
|
|
|
|
|
return wrapped_->CloseChannel(channel_id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SetDataSink(DataChannelSink* sink) override {
|
|
|
|
|
wrapped_->SetDataSink(sink);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
MediaTransportInterface* wrapped_;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class WrapperMediaTransportFactory : public MediaTransportFactory {
|
|
|
|
|
public:
|
|
|
|
|
explicit WrapperMediaTransportFactory(MediaTransportInterface* wrapped)
|
|
|
|
|
: wrapped_(wrapped) {}
|
|
|
|
|
|
|
|
|
|
RTCErrorOr<std::unique_ptr<MediaTransportInterface>> CreateMediaTransport(
|
|
|
|
|
rtc::PacketTransportInternal* packet_transport,
|
|
|
|
|
rtc::Thread* network_thread,
|
|
|
|
|
const MediaTransportSettings& settings) override {
|
|
|
|
|
return {absl::make_unique<WrapperMediaTransport>(wrapped_)};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
MediaTransportInterface* wrapped_;
|
|
|
|
|
};
|
|
|
|
|
|
2018-10-16 10:41:42 +02:00
|
|
|
// Contains two MediaTransportsInterfaces that are connected to each other.
|
|
|
|
|
// Currently supports audio only.
|
|
|
|
|
class MediaTransportPair {
|
|
|
|
|
public:
|
2018-11-20 14:52:05 +01:00
|
|
|
struct Stats {
|
|
|
|
|
int sent_audio_frames = 0;
|
|
|
|
|
int received_audio_frames = 0;
|
2018-11-28 15:34:37 +01:00
|
|
|
int sent_video_frames = 0;
|
|
|
|
|
int received_video_frames = 0;
|
2018-11-20 14:52:05 +01:00
|
|
|
};
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
explicit MediaTransportPair(rtc::Thread* thread)
|
|
|
|
|
: first_(thread, &second_), second_(thread, &first_) {}
|
2018-10-16 10:41:42 +02:00
|
|
|
|
|
|
|
|
// Ownership stays with MediaTransportPair
|
2018-11-01 16:42:44 -07:00
|
|
|
MediaTransportInterface* first() { return &first_; }
|
|
|
|
|
MediaTransportInterface* second() { return &second_; }
|
|
|
|
|
|
2018-11-08 11:23:22 -08:00
|
|
|
std::unique_ptr<MediaTransportFactory> first_factory() {
|
|
|
|
|
return absl::make_unique<WrapperMediaTransportFactory>(&first_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::unique_ptr<MediaTransportFactory> second_factory() {
|
|
|
|
|
return absl::make_unique<WrapperMediaTransportFactory>(&second_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SetState(MediaTransportState state) {
|
|
|
|
|
first_.SetState(state);
|
|
|
|
|
second_.SetState(state);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
void FlushAsyncInvokes() {
|
|
|
|
|
first_.FlushAsyncInvokes();
|
|
|
|
|
second_.FlushAsyncInvokes();
|
|
|
|
|
}
|
2018-10-16 10:41:42 +02:00
|
|
|
|
2018-11-20 14:52:05 +01:00
|
|
|
Stats FirstStats() { return first_.GetStats(); }
|
|
|
|
|
Stats SecondStats() { return second_.GetStats(); }
|
|
|
|
|
|
2018-10-16 10:41:42 +02:00
|
|
|
private:
|
|
|
|
|
class LoopbackMediaTransport : public MediaTransportInterface {
|
|
|
|
|
public:
|
2018-11-01 16:42:44 -07:00
|
|
|
LoopbackMediaTransport(rtc::Thread* thread, LoopbackMediaTransport* other)
|
|
|
|
|
: thread_(thread), other_(other) {}
|
|
|
|
|
|
|
|
|
|
~LoopbackMediaTransport() {
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
2018-11-28 15:34:37 +01:00
|
|
|
RTC_CHECK(audio_sink_ == nullptr);
|
|
|
|
|
RTC_CHECK(video_sink_ == nullptr);
|
2018-11-01 16:42:44 -07:00
|
|
|
RTC_CHECK(data_sink_ == nullptr);
|
|
|
|
|
}
|
2018-10-16 10:41:42 +02:00
|
|
|
|
|
|
|
|
RTCError SendAudioFrame(uint64_t channel_id,
|
|
|
|
|
MediaTransportEncodedAudioFrame frame) override {
|
2018-11-20 14:52:05 +01:00
|
|
|
{
|
|
|
|
|
rtc::CritScope lock(&stats_lock_);
|
|
|
|
|
++stats_.sent_audio_frames;
|
|
|
|
|
}
|
2018-11-01 16:42:44 -07:00
|
|
|
invoker_.AsyncInvoke<void>(RTC_FROM_HERE, thread_,
|
|
|
|
|
[this, channel_id, frame] {
|
|
|
|
|
other_->OnData(channel_id, std::move(frame));
|
|
|
|
|
});
|
2018-10-16 10:41:42 +02:00
|
|
|
return RTCError::OK();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
RTCError SendVideoFrame(
|
|
|
|
|
uint64_t channel_id,
|
|
|
|
|
const MediaTransportEncodedVideoFrame& frame) override {
|
2018-11-28 15:34:37 +01:00
|
|
|
{
|
|
|
|
|
rtc::CritScope lock(&stats_lock_);
|
|
|
|
|
++stats_.sent_video_frames;
|
|
|
|
|
}
|
|
|
|
|
// Ensure that we own the referenced data.
|
|
|
|
|
MediaTransportEncodedVideoFrame frame_copy = frame;
|
|
|
|
|
frame_copy.Retain();
|
|
|
|
|
invoker_.AsyncInvoke<void>(
|
|
|
|
|
RTC_FROM_HERE, thread_, [this, channel_id, frame_copy] {
|
|
|
|
|
other_->OnData(channel_id, std::move(frame_copy));
|
|
|
|
|
});
|
2018-10-16 10:41:42 +02:00
|
|
|
return RTCError::OK();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RTCError RequestKeyFrame(uint64_t channel_id) override {
|
|
|
|
|
return RTCError::OK();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SetReceiveAudioSink(MediaTransportAudioSinkInterface* sink) override {
|
2018-11-01 16:42:44 -07:00
|
|
|
rtc::CritScope lock(&sink_lock_);
|
2018-10-16 10:41:42 +02:00
|
|
|
if (sink) {
|
2018-11-28 15:34:37 +01:00
|
|
|
RTC_CHECK(audio_sink_ == nullptr);
|
2018-10-16 10:41:42 +02:00
|
|
|
}
|
2018-11-28 15:34:37 +01:00
|
|
|
audio_sink_ = sink;
|
2018-10-16 10:41:42 +02:00
|
|
|
}
|
|
|
|
|
|
2018-11-28 15:34:37 +01:00
|
|
|
void SetReceiveVideoSink(MediaTransportVideoSinkInterface* sink) override {
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
|
|
|
|
if (sink) {
|
|
|
|
|
RTC_CHECK(video_sink_ == nullptr);
|
|
|
|
|
}
|
|
|
|
|
video_sink_ = sink;
|
|
|
|
|
}
|
2018-10-16 10:41:42 +02:00
|
|
|
|
2018-10-31 15:25:32 -07:00
|
|
|
void SetMediaTransportStateCallback(
|
2018-11-08 11:23:22 -08:00
|
|
|
MediaTransportStateCallback* callback) override {
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
|
|
|
|
state_callback_ = callback;
|
|
|
|
|
invoker_.AsyncInvoke<void>(RTC_FROM_HERE, thread_, [this] {
|
|
|
|
|
RTC_DCHECK_RUN_ON(thread_);
|
|
|
|
|
OnStateChanged();
|
|
|
|
|
});
|
|
|
|
|
}
|
2018-10-31 15:25:32 -07:00
|
|
|
|
|
|
|
|
RTCError SendData(int channel_id,
|
|
|
|
|
const SendDataParams& params,
|
|
|
|
|
const rtc::CopyOnWriteBuffer& buffer) override {
|
2018-11-01 16:42:44 -07:00
|
|
|
invoker_.AsyncInvoke<void>(
|
|
|
|
|
RTC_FROM_HERE, thread_, [this, channel_id, params, buffer] {
|
|
|
|
|
other_->OnData(channel_id, params.type, buffer);
|
|
|
|
|
});
|
|
|
|
|
return RTCError::OK();
|
2018-10-31 15:25:32 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
RTCError CloseChannel(int channel_id) override {
|
2018-11-01 16:42:44 -07:00
|
|
|
invoker_.AsyncInvoke<void>(RTC_FROM_HERE, thread_, [this, channel_id] {
|
|
|
|
|
other_->OnRemoteCloseChannel(channel_id);
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
|
|
|
|
if (data_sink_) {
|
|
|
|
|
data_sink_->OnChannelClosed(channel_id);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return RTCError::OK();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SetDataSink(DataChannelSink* sink) override {
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
|
|
|
|
data_sink_ = sink;
|
2018-10-31 15:25:32 -07:00
|
|
|
}
|
|
|
|
|
|
2018-11-08 11:23:22 -08:00
|
|
|
void SetState(MediaTransportState state) {
|
|
|
|
|
invoker_.AsyncInvoke<void>(RTC_FROM_HERE, thread_, [this, state] {
|
|
|
|
|
RTC_DCHECK_RUN_ON(thread_);
|
|
|
|
|
state_ = state;
|
|
|
|
|
OnStateChanged();
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
void FlushAsyncInvokes() { invoker_.Flush(thread_); }
|
2018-10-31 15:25:32 -07:00
|
|
|
|
2018-11-20 14:52:05 +01:00
|
|
|
Stats GetStats() {
|
|
|
|
|
rtc::CritScope lock(&stats_lock_);
|
|
|
|
|
return stats_;
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-16 10:41:42 +02:00
|
|
|
private:
|
|
|
|
|
void OnData(uint64_t channel_id, MediaTransportEncodedAudioFrame frame) {
|
2018-11-20 14:52:05 +01:00
|
|
|
{
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
2018-11-28 15:34:37 +01:00
|
|
|
if (audio_sink_) {
|
|
|
|
|
audio_sink_->OnData(channel_id, frame);
|
2018-11-20 14:52:05 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
rtc::CritScope lock(&stats_lock_);
|
|
|
|
|
++stats_.received_audio_frames;
|
2018-10-16 10:41:42 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-28 15:34:37 +01:00
|
|
|
void OnData(uint64_t channel_id, MediaTransportEncodedVideoFrame frame) {
|
|
|
|
|
{
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
|
|
|
|
if (video_sink_) {
|
|
|
|
|
video_sink_->OnData(channel_id, frame);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
rtc::CritScope lock(&stats_lock_);
|
|
|
|
|
++stats_.received_video_frames;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
void OnData(int channel_id,
|
|
|
|
|
DataMessageType type,
|
|
|
|
|
const rtc::CopyOnWriteBuffer& buffer) {
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
|
|
|
|
if (data_sink_) {
|
|
|
|
|
data_sink_->OnDataReceived(channel_id, type, buffer);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void OnRemoteCloseChannel(int channel_id) {
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
|
|
|
|
if (data_sink_) {
|
|
|
|
|
data_sink_->OnChannelClosing(channel_id);
|
|
|
|
|
data_sink_->OnChannelClosed(channel_id);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-08 11:23:22 -08:00
|
|
|
void OnStateChanged() RTC_RUN_ON(thread_) {
|
|
|
|
|
rtc::CritScope lock(&sink_lock_);
|
|
|
|
|
if (state_callback_) {
|
|
|
|
|
state_callback_->OnStateChanged(state_);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
rtc::Thread* const thread_;
|
|
|
|
|
rtc::CriticalSection sink_lock_;
|
2018-11-20 14:52:05 +01:00
|
|
|
rtc::CriticalSection stats_lock_;
|
2018-11-01 16:42:44 -07:00
|
|
|
|
2018-11-28 15:34:37 +01:00
|
|
|
MediaTransportAudioSinkInterface* audio_sink_ RTC_GUARDED_BY(sink_lock_) =
|
|
|
|
|
nullptr;
|
|
|
|
|
MediaTransportVideoSinkInterface* video_sink_ RTC_GUARDED_BY(sink_lock_) =
|
2018-11-01 16:42:44 -07:00
|
|
|
nullptr;
|
|
|
|
|
DataChannelSink* data_sink_ RTC_GUARDED_BY(sink_lock_) = nullptr;
|
2018-11-08 11:23:22 -08:00
|
|
|
MediaTransportStateCallback* state_callback_ RTC_GUARDED_BY(sink_lock_) =
|
|
|
|
|
nullptr;
|
|
|
|
|
|
|
|
|
|
MediaTransportState state_ RTC_GUARDED_BY(thread_) =
|
|
|
|
|
MediaTransportState::kPending;
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
LoopbackMediaTransport* const other_;
|
|
|
|
|
|
2018-11-20 14:52:05 +01:00
|
|
|
Stats stats_ RTC_GUARDED_BY(stats_lock_);
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
rtc::AsyncInvoker invoker_;
|
2018-10-16 10:41:42 +02:00
|
|
|
};
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
LoopbackMediaTransport first_;
|
|
|
|
|
LoopbackMediaTransport second_;
|
2018-10-16 10:41:42 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace webrtc
|
|
|
|
|
|
|
|
|
|
#endif // API_TEST_LOOPBACK_MEDIA_TRANSPORT_H_
|