mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-16 15:20:42 +01:00

Bug: webrtc:9719 Change-Id: I568da8720377342cf44ee8caa316e14b4cd8beba Reviewed-on: https://webrtc-review.googlesource.com/c/111960 Commit-Queue: Niels Moller <nisse@webrtc.org> Reviewed-by: Karl Wiberg <kwiberg@webrtc.org> Cr-Commit-Position: refs/heads/master@{#25826}
218 lines
7.6 KiB
C++
218 lines
7.6 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.
|
|
*/
|
|
|
|
#include <algorithm>
|
|
#include <memory>
|
|
#include <vector>
|
|
|
|
#include "api/test/loopback_media_transport.h"
|
|
#include "test/gmock.h"
|
|
|
|
namespace webrtc {
|
|
|
|
namespace {
|
|
|
|
class MockMediaTransportAudioSinkInterface
|
|
: public MediaTransportAudioSinkInterface {
|
|
public:
|
|
MOCK_METHOD2(OnData, void(uint64_t, MediaTransportEncodedAudioFrame));
|
|
};
|
|
|
|
class MockMediaTransportVideoSinkInterface
|
|
: public MediaTransportVideoSinkInterface {
|
|
public:
|
|
MOCK_METHOD2(OnData, void(uint64_t, MediaTransportEncodedVideoFrame));
|
|
MOCK_METHOD1(OnKeyFrameRequested, void(uint64_t));
|
|
};
|
|
|
|
class MockDataChannelSink : public DataChannelSink {
|
|
public:
|
|
MOCK_METHOD3(OnDataReceived,
|
|
void(int, DataMessageType, const rtc::CopyOnWriteBuffer&));
|
|
MOCK_METHOD1(OnChannelClosing, void(int));
|
|
MOCK_METHOD1(OnChannelClosed, void(int));
|
|
};
|
|
|
|
class MockStateCallback : public MediaTransportStateCallback {
|
|
public:
|
|
MOCK_METHOD1(OnStateChanged, void(MediaTransportState));
|
|
};
|
|
|
|
// Test only uses the sequence number.
|
|
MediaTransportEncodedAudioFrame CreateAudioFrame(int sequence_number) {
|
|
static constexpr int kSamplingRateHz = 48000;
|
|
static constexpr int kStartingSampleIndex = 0;
|
|
static constexpr int kSamplesPerChannel = 480;
|
|
static constexpr uint8_t kPayloadType = 17;
|
|
|
|
return MediaTransportEncodedAudioFrame(
|
|
kSamplingRateHz, kStartingSampleIndex, kSamplesPerChannel,
|
|
sequence_number, MediaTransportEncodedAudioFrame::FrameType::kSpeech,
|
|
kPayloadType, std::vector<uint8_t>(kSamplesPerChannel));
|
|
}
|
|
|
|
MediaTransportEncodedVideoFrame CreateVideoFrame(
|
|
int frame_id,
|
|
const webrtc::EncodedImage& encoded_image) {
|
|
return MediaTransportEncodedVideoFrame(frame_id, /*referenced_frame_ids=*/{},
|
|
kVideoCodecVP8, encoded_image);
|
|
}
|
|
|
|
} // namespace
|
|
|
|
TEST(LoopbackMediaTransport, AudioWithNoSinkSilentlyIgnored) {
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
thread->Start();
|
|
MediaTransportPair transport_pair(thread.get());
|
|
transport_pair.first()->SendAudioFrame(1, CreateAudioFrame(0));
|
|
transport_pair.second()->SendAudioFrame(2, CreateAudioFrame(0));
|
|
transport_pair.FlushAsyncInvokes();
|
|
}
|
|
|
|
TEST(LoopbackMediaTransport, AudioDeliveredToSink) {
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
thread->Start();
|
|
MediaTransportPair transport_pair(thread.get());
|
|
testing::StrictMock<MockMediaTransportAudioSinkInterface> sink;
|
|
EXPECT_CALL(sink,
|
|
OnData(1, testing::Property(
|
|
&MediaTransportEncodedAudioFrame::sequence_number,
|
|
testing::Eq(10))));
|
|
transport_pair.second()->SetReceiveAudioSink(&sink);
|
|
transport_pair.first()->SendAudioFrame(1, CreateAudioFrame(10));
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
transport_pair.second()->SetReceiveAudioSink(nullptr);
|
|
}
|
|
|
|
TEST(LoopbackMediaTransport, VideoDeliveredToSink) {
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
thread->Start();
|
|
MediaTransportPair transport_pair(thread.get());
|
|
testing::StrictMock<MockMediaTransportVideoSinkInterface> sink;
|
|
uint8_t encoded_data[] = {1, 2, 3};
|
|
EncodedImage encoded_image;
|
|
encoded_image._buffer = encoded_data;
|
|
encoded_image._length = sizeof(encoded_data);
|
|
|
|
EXPECT_CALL(sink, OnData(1, testing::Property(
|
|
&MediaTransportEncodedVideoFrame::frame_id,
|
|
testing::Eq(10))))
|
|
.WillOnce(testing::Invoke(
|
|
[&encoded_image](int frame_id,
|
|
const MediaTransportEncodedVideoFrame& frame) {
|
|
EXPECT_NE(frame.encoded_image()._buffer, encoded_image._buffer);
|
|
EXPECT_EQ(frame.encoded_image()._length, encoded_image._length);
|
|
EXPECT_EQ(
|
|
0, memcmp(frame.encoded_image()._buffer, encoded_image._buffer,
|
|
std::min(frame.encoded_image()._length,
|
|
encoded_image._length)));
|
|
}));
|
|
|
|
transport_pair.second()->SetReceiveVideoSink(&sink);
|
|
transport_pair.first()->SendVideoFrame(1,
|
|
CreateVideoFrame(10, encoded_image));
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
transport_pair.second()->SetReceiveVideoSink(nullptr);
|
|
}
|
|
|
|
TEST(LoopbackMediaTransport, DataDeliveredToSink) {
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
thread->Start();
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
MockDataChannelSink sink;
|
|
transport_pair.first()->SetDataSink(&sink);
|
|
|
|
const int channel_id = 1;
|
|
EXPECT_CALL(sink,
|
|
OnDataReceived(
|
|
channel_id, DataMessageType::kText,
|
|
testing::Property<rtc::CopyOnWriteBuffer, const char*>(
|
|
&rtc::CopyOnWriteBuffer::cdata, testing::StrEq("foo"))));
|
|
|
|
SendDataParams params;
|
|
params.type = DataMessageType::kText;
|
|
rtc::CopyOnWriteBuffer buffer("foo");
|
|
transport_pair.second()->SendData(channel_id, params, buffer);
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
transport_pair.first()->SetDataSink(nullptr);
|
|
}
|
|
|
|
TEST(LoopbackMediaTransport, CloseDeliveredToSink) {
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
thread->Start();
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
MockDataChannelSink first_sink;
|
|
transport_pair.first()->SetDataSink(&first_sink);
|
|
|
|
MockDataChannelSink second_sink;
|
|
transport_pair.second()->SetDataSink(&second_sink);
|
|
|
|
const int channel_id = 1;
|
|
{
|
|
testing::InSequence s;
|
|
EXPECT_CALL(second_sink, OnChannelClosing(channel_id));
|
|
EXPECT_CALL(second_sink, OnChannelClosed(channel_id));
|
|
EXPECT_CALL(first_sink, OnChannelClosed(channel_id));
|
|
}
|
|
|
|
transport_pair.first()->CloseChannel(channel_id);
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
transport_pair.first()->SetDataSink(nullptr);
|
|
transport_pair.second()->SetDataSink(nullptr);
|
|
}
|
|
|
|
TEST(LoopbackMediaTransport, InitialStateDeliveredWhenCallbackSet) {
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
thread->Start();
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
MockStateCallback state_callback;
|
|
|
|
EXPECT_CALL(state_callback, OnStateChanged(MediaTransportState::kPending));
|
|
transport_pair.first()->SetMediaTransportStateCallback(&state_callback);
|
|
transport_pair.FlushAsyncInvokes();
|
|
}
|
|
|
|
TEST(LoopbackMediaTransport, ChangedStateDeliveredWhenCallbackSet) {
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
thread->Start();
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
transport_pair.SetState(MediaTransportState::kWritable);
|
|
transport_pair.FlushAsyncInvokes();
|
|
|
|
MockStateCallback state_callback;
|
|
|
|
EXPECT_CALL(state_callback, OnStateChanged(MediaTransportState::kWritable));
|
|
transport_pair.first()->SetMediaTransportStateCallback(&state_callback);
|
|
transport_pair.FlushAsyncInvokes();
|
|
}
|
|
|
|
TEST(LoopbackMediaTransport, StateChangeDeliveredToCallback) {
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
thread->Start();
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
MockStateCallback state_callback;
|
|
|
|
EXPECT_CALL(state_callback, OnStateChanged(MediaTransportState::kPending));
|
|
EXPECT_CALL(state_callback, OnStateChanged(MediaTransportState::kWritable));
|
|
transport_pair.first()->SetMediaTransportStateCallback(&state_callback);
|
|
transport_pair.SetState(MediaTransportState::kWritable);
|
|
transport_pair.FlushAsyncInvokes();
|
|
}
|
|
|
|
} // namespace webrtc
|