blob: 1ef199849eeb672becfe911c0bbd50d7871d65c5 [file] [log] [blame]
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +01001/*
2 * Copyright (c) 2017 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#include "modules/rtp_rtcp/source/rtcp_transceiver.h"
12
Danil Chapovalova32d7102017-12-14 17:28:27 +010013#include <memory>
14
Karl Wiberg918f50c2018-07-05 11:40:33 +020015#include "absl/memory/memory.h"
Danil Chapovalova32d7102017-12-14 17:28:27 +010016#include "modules/rtp_rtcp/source/rtcp_packet/sender_report.h"
Danil Chapovalovd5cae4d2017-12-14 11:14:35 +010017#include "modules/rtp_rtcp/source/rtcp_packet/transport_feedback.h"
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010018#include "rtc_base/event.h"
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010019#include "test/gmock.h"
20#include "test/gtest.h"
21#include "test/mock_transport.h"
22
23namespace {
24
25using ::testing::AtLeast;
Danil Chapovalovd5cae4d2017-12-14 11:14:35 +010026using ::testing::Invoke;
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010027using ::testing::InvokeWithoutArgs;
Danil Chapovalova32d7102017-12-14 17:28:27 +010028using ::testing::IsNull;
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010029using ::testing::NiceMock;
30using ::testing::_;
31using ::webrtc::MockTransport;
32using ::webrtc::RtcpTransceiver;
33using ::webrtc::RtcpTransceiverConfig;
Danil Chapovalovd5cae4d2017-12-14 11:14:35 +010034using ::webrtc::rtcp::TransportFeedback;
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010035
Danil Chapovalova32d7102017-12-14 17:28:27 +010036class MockMediaReceiverRtcpObserver : public webrtc::MediaReceiverRtcpObserver {
37 public:
38 MOCK_METHOD3(OnSenderReport, void(uint32_t, webrtc::NtpTime, uint32_t));
39};
40
41constexpr int kTimeoutMs = 1000;
42
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010043void WaitPostedTasks(rtc::TaskQueue* queue) {
44 rtc::Event done(false, false);
45 queue->PostTask([&done] { done.Set(); });
Danil Chapovalova32d7102017-12-14 17:28:27 +010046 ASSERT_TRUE(done.Wait(kTimeoutMs));
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010047}
48
49TEST(RtcpTransceiverTest, SendsRtcpOnTaskQueueWhenCreatedOffTaskQueue) {
50 rtc::TaskQueue queue("rtcp");
51 MockTransport outgoing_transport;
52 RtcpTransceiverConfig config;
53 config.outgoing_transport = &outgoing_transport;
54 config.task_queue = &queue;
55 EXPECT_CALL(outgoing_transport, SendRtcp(_, _))
56 .WillRepeatedly(InvokeWithoutArgs([&] {
57 EXPECT_TRUE(queue.IsCurrent());
58 return true;
59 }));
60
61 RtcpTransceiver rtcp_transceiver(config);
62 rtcp_transceiver.SendCompoundPacket();
63 WaitPostedTasks(&queue);
64}
65
66TEST(RtcpTransceiverTest, SendsRtcpOnTaskQueueWhenCreatedOnTaskQueue) {
67 rtc::TaskQueue queue("rtcp");
68 MockTransport outgoing_transport;
69 RtcpTransceiverConfig config;
70 config.outgoing_transport = &outgoing_transport;
71 config.task_queue = &queue;
72 EXPECT_CALL(outgoing_transport, SendRtcp(_, _))
73 .WillRepeatedly(InvokeWithoutArgs([&] {
74 EXPECT_TRUE(queue.IsCurrent());
75 return true;
76 }));
77
78 std::unique_ptr<RtcpTransceiver> rtcp_transceiver;
79 queue.PostTask([&] {
Karl Wiberg918f50c2018-07-05 11:40:33 +020080 rtcp_transceiver = absl::make_unique<RtcpTransceiver>(config);
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010081 rtcp_transceiver->SendCompoundPacket();
82 });
83 WaitPostedTasks(&queue);
84}
85
Danil Chapovalovf0076d32018-09-05 16:46:40 +020086TEST(RtcpTransceiverTest, CanBeDestroyedOnTaskQueueAfterStop) {
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010087 rtc::TaskQueue queue("rtcp");
88 NiceMock<MockTransport> outgoing_transport;
89 RtcpTransceiverConfig config;
90 config.outgoing_transport = &outgoing_transport;
91 config.task_queue = &queue;
Karl Wiberg918f50c2018-07-05 11:40:33 +020092 auto rtcp_transceiver = absl::make_unique<RtcpTransceiver>(config);
Danil Chapovalovf0076d32018-09-05 16:46:40 +020093 rtcp_transceiver->Stop(rtc::NewClosure([] {}));
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +010094
95 queue.PostTask([&] { rtcp_transceiver.reset(); });
96 WaitPostedTasks(&queue);
97}
98
Danil Chapovalovf0076d32018-09-05 16:46:40 +020099TEST(RtcpTransceiverTest, CanBeDestroyedWithoutBlockingAfterStop) {
100 rtc::TaskQueue queue("rtcp");
101 NiceMock<MockTransport> outgoing_transport;
102 RtcpTransceiverConfig config;
103 config.outgoing_transport = &outgoing_transport;
104 config.task_queue = &queue;
105 auto rtcp_transceiver = absl::make_unique<RtcpTransceiver>(config);
106 rtcp_transceiver->SendCompoundPacket();
107
108 rtc::Event heavy_task(false, false);
109 queue.PostTask(
110 rtc::NewClosure([&] { EXPECT_TRUE(heavy_task.Wait(kTimeoutMs)); }));
111 rtc::Event done(false, false);
112 rtcp_transceiver->Stop(rtc::NewClosure([&done] { done.Set(); }));
113 rtcp_transceiver = nullptr;
114
115 heavy_task.Set();
116 EXPECT_TRUE(done.Wait(kTimeoutMs));
117}
118
Danil Chapovalova32d7102017-12-14 17:28:27 +0100119// Use rtp timestamp to distinguish different incoming sender reports.
120rtc::CopyOnWriteBuffer CreateSenderReport(uint32_t ssrc, uint32_t rtp_time) {
121 webrtc::rtcp::SenderReport sr;
122 sr.SetSenderSsrc(ssrc);
123 sr.SetRtpTimestamp(rtp_time);
124 rtc::Buffer buffer = sr.Build();
125 // Switch to an efficient way creating CopyOnWriteBuffer from RtcpPacket when
126 // there is one. Until then do not worry about extra memcpy in test.
127 return rtc::CopyOnWriteBuffer(buffer.data(), buffer.size());
128}
129
130TEST(RtcpTransceiverTest, DoesntPostToRtcpObserverAfterCallToRemove) {
131 const uint32_t kRemoteSsrc = 1234;
132 MockTransport null_transport;
133 rtc::TaskQueue queue("rtcp");
134 RtcpTransceiverConfig config;
135 config.outgoing_transport = &null_transport;
136 config.task_queue = &queue;
137 RtcpTransceiver rtcp_transceiver(config);
138 rtc::Event observer_deleted(false, false);
139
Karl Wiberg918f50c2018-07-05 11:40:33 +0200140 auto observer = absl::make_unique<MockMediaReceiverRtcpObserver>();
Danil Chapovalova32d7102017-12-14 17:28:27 +0100141 EXPECT_CALL(*observer, OnSenderReport(kRemoteSsrc, _, 1));
142 EXPECT_CALL(*observer, OnSenderReport(kRemoteSsrc, _, 2)).Times(0);
143
144 rtcp_transceiver.AddMediaReceiverRtcpObserver(kRemoteSsrc, observer.get());
145 rtcp_transceiver.ReceivePacket(CreateSenderReport(kRemoteSsrc, 1));
146 rtcp_transceiver.RemoveMediaReceiverRtcpObserver(
147 kRemoteSsrc, observer.get(),
148 /*on_removed=*/rtc::NewClosure([&] {
149 observer.reset();
150 observer_deleted.Set();
151 }));
152 rtcp_transceiver.ReceivePacket(CreateSenderReport(kRemoteSsrc, 2));
153
154 EXPECT_TRUE(observer_deleted.Wait(kTimeoutMs));
155 WaitPostedTasks(&queue);
156}
157
158TEST(RtcpTransceiverTest, RemoveMediaReceiverRtcpObserverIsNonBlocking) {
159 const uint32_t kRemoteSsrc = 1234;
160 MockTransport null_transport;
161 rtc::TaskQueue queue("rtcp");
162 RtcpTransceiverConfig config;
163 config.outgoing_transport = &null_transport;
164 config.task_queue = &queue;
165 RtcpTransceiver rtcp_transceiver(config);
Karl Wiberg918f50c2018-07-05 11:40:33 +0200166 auto observer = absl::make_unique<MockMediaReceiverRtcpObserver>();
Danil Chapovalova32d7102017-12-14 17:28:27 +0100167 rtcp_transceiver.AddMediaReceiverRtcpObserver(kRemoteSsrc, observer.get());
168
169 rtc::Event queue_blocker(false, false);
170 rtc::Event observer_deleted(false, false);
171 queue.PostTask([&] { EXPECT_TRUE(queue_blocker.Wait(kTimeoutMs)); });
172 rtcp_transceiver.RemoveMediaReceiverRtcpObserver(
173 kRemoteSsrc, observer.get(),
174 /*on_removed=*/rtc::NewClosure([&] {
175 observer.reset();
176 observer_deleted.Set();
177 }));
178
179 EXPECT_THAT(observer, Not(IsNull()));
180 queue_blocker.Set();
181 EXPECT_TRUE(observer_deleted.Wait(kTimeoutMs));
182}
183
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +0100184TEST(RtcpTransceiverTest, CanCallSendCompoundPacketFromAnyThread) {
185 MockTransport outgoing_transport;
186 rtc::TaskQueue queue("rtcp");
187 RtcpTransceiverConfig config;
188 config.outgoing_transport = &outgoing_transport;
189 config.task_queue = &queue;
190
191 EXPECT_CALL(outgoing_transport, SendRtcp(_, _))
192 // If test is slow, a periodic task may send an extra packet.
193 .Times(AtLeast(3))
194 .WillRepeatedly(InvokeWithoutArgs([&] {
195 EXPECT_TRUE(queue.IsCurrent());
196 return true;
197 }));
198
199 RtcpTransceiver rtcp_transceiver(config);
200
201 // Call from the construction thread.
202 rtcp_transceiver.SendCompoundPacket();
203 // Call from the same queue transceiver use for processing.
204 queue.PostTask([&] { rtcp_transceiver.SendCompoundPacket(); });
205 // Call from unrelated task queue.
206 rtc::TaskQueue queue_send("send_packet");
207 queue_send.PostTask([&] { rtcp_transceiver.SendCompoundPacket(); });
208
209 WaitPostedTasks(&queue_send);
210 WaitPostedTasks(&queue);
211}
212
Danil Chapovalovf0076d32018-09-05 16:46:40 +0200213TEST(RtcpTransceiverTest, DoesntSendPacketsAfterStopCallback) {
214 NiceMock<MockTransport> outgoing_transport;
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +0100215 rtc::TaskQueue queue("rtcp");
216 RtcpTransceiverConfig config;
217 config.outgoing_transport = &outgoing_transport;
218 config.task_queue = &queue;
Danil Chapovalovf0076d32018-09-05 16:46:40 +0200219 config.schedule_periodic_compound_packets = true;
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +0100220
Karl Wiberg918f50c2018-07-05 11:40:33 +0200221 auto rtcp_transceiver = absl::make_unique<RtcpTransceiver>(config);
Danil Chapovalovf0076d32018-09-05 16:46:40 +0200222 rtc::Event done(false, false);
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +0100223 rtcp_transceiver->SendCompoundPacket();
Danil Chapovalovf0076d32018-09-05 16:46:40 +0200224 rtcp_transceiver->Stop(rtc::NewClosure([&] {
225 EXPECT_CALL(outgoing_transport, SendRtcp).Times(0);
226 done.Set();
227 }));
228 rtcp_transceiver = nullptr;
229 EXPECT_TRUE(done.Wait(kTimeoutMs));
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +0100230}
231
Danil Chapovalovd5cae4d2017-12-14 11:14:35 +0100232TEST(RtcpTransceiverTest, SendsTransportFeedbackOnTaskQueue) {
233 static constexpr uint32_t kSenderSsrc = 12345;
234 MockTransport outgoing_transport;
235 rtc::TaskQueue queue("rtcp");
236 RtcpTransceiverConfig config;
237 config.feedback_ssrc = kSenderSsrc;
238 config.outgoing_transport = &outgoing_transport;
239 config.task_queue = &queue;
240 config.schedule_periodic_compound_packets = false;
241 RtcpTransceiver rtcp_transceiver(config);
242
243 EXPECT_CALL(outgoing_transport, SendRtcp(_, _))
244 .WillOnce(Invoke([&](const uint8_t* buffer, size_t size) {
245 EXPECT_TRUE(queue.IsCurrent());
246
247 std::unique_ptr<TransportFeedback> transport_feedback =
248 TransportFeedback::ParseFrom(buffer, size);
249 EXPECT_TRUE(transport_feedback);
250 EXPECT_EQ(transport_feedback->sender_ssrc(), kSenderSsrc);
251 return true;
252 }));
253
254 // Create minimalistic transport feedback packet.
255 TransportFeedback transport_feedback;
256 transport_feedback.SetSenderSsrc(rtcp_transceiver.SSRC());
257 transport_feedback.AddReceivedPacket(321, 10000);
258
259 EXPECT_TRUE(rtcp_transceiver.SendFeedbackPacket(transport_feedback));
260
261 WaitPostedTasks(&queue);
262}
263
Danil Chapovalovc0fd5f92017-11-16 14:35:32 +0100264} // namespace