blob: d9caac0197910281dd93ba31e3fe98151dc19844 [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
kjellander1afca732016-02-07 20:46:45 -08002 * Copyright (c) 2004 The WebRTC project authors. All Rights Reserved.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00003 *
kjellander1afca732016-02-07 20:46:45 -08004 * 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.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00009 */
10
kjellandera96e2d72016-02-04 23:52:28 -080011#ifndef WEBRTC_MEDIA_BASE_FAKENETWORKINTERFACE_H_
12#define WEBRTC_MEDIA_BASE_FAKENETWORKINTERFACE_H_
henrike@webrtc.org28e20752013-07-10 00:45:36 +000013
henrike@webrtc.org28e20752013-07-10 00:45:36 +000014#include <map>
buildbot@webrtc.orga09a9992014-08-13 17:26:08 +000015#include <vector>
henrike@webrtc.org28e20752013-07-10 00:45:36 +000016
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000017#include "webrtc/base/buffer.h"
18#include "webrtc/base/byteorder.h"
19#include "webrtc/base/criticalsection.h"
20#include "webrtc/base/dscp.h"
21#include "webrtc/base/messagehandler.h"
22#include "webrtc/base/messagequeue.h"
23#include "webrtc/base/thread.h"
kjellandera96e2d72016-02-04 23:52:28 -080024#include "webrtc/media/base/mediachannel.h"
25#include "webrtc/media/base/rtputils.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000026
27namespace cricket {
28
29// Fake NetworkInterface that sends/receives RTP/RTCP packets.
30class FakeNetworkInterface : public MediaChannel::NetworkInterface,
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000031 public rtc::MessageHandler {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000032 public:
33 FakeNetworkInterface()
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000034 : thread_(rtc::Thread::Current()),
henrike@webrtc.org28e20752013-07-10 00:45:36 +000035 dest_(NULL),
36 conf_(false),
37 sendbuf_size_(-1),
wu@webrtc.orgde305012013-10-31 15:40:38 +000038 recvbuf_size_(-1),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000039 dscp_(rtc::DSCP_NO_CHANGE) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000040 }
41
42 void SetDestination(MediaChannel* dest) { dest_ = dest; }
43
44 // Conference mode is a mode where instead of simply forwarding the packets,
45 // the transport will send multiple copies of the packet with the specified
46 // SSRCs. This allows us to simulate receiving media from multiple sources.
Peter Boström0c4e06b2015-10-07 12:23:21 +020047 void SetConferenceMode(bool conf, const std::vector<uint32_t>& ssrcs) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000048 rtc::CritScope cs(&crit_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000049 conf_ = conf;
50 conf_sent_ssrcs_ = ssrcs;
51 }
52
53 int NumRtpBytes() {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000054 rtc::CritScope cs(&crit_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000055 int bytes = 0;
56 for (size_t i = 0; i < rtp_packets_.size(); ++i) {
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +000057 bytes += static_cast<int>(rtp_packets_[i].size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +000058 }
59 return bytes;
60 }
61
Peter Boström0c4e06b2015-10-07 12:23:21 +020062 int NumRtpBytes(uint32_t ssrc) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000063 rtc::CritScope cs(&crit_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000064 int bytes = 0;
65 GetNumRtpBytesAndPackets(ssrc, &bytes, NULL);
66 return bytes;
67 }
68
69 int NumRtpPackets() {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000070 rtc::CritScope cs(&crit_);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +000071 return static_cast<int>(rtp_packets_.size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +000072 }
73
Peter Boström0c4e06b2015-10-07 12:23:21 +020074 int NumRtpPackets(uint32_t ssrc) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000075 rtc::CritScope cs(&crit_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000076 int packets = 0;
77 GetNumRtpBytesAndPackets(ssrc, NULL, &packets);
78 return packets;
79 }
80
81 int NumSentSsrcs() {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000082 rtc::CritScope cs(&crit_);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +000083 return static_cast<int>(sent_ssrcs_.size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +000084 }
85
86 // Note: callers are responsible for deleting the returned buffer.
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000087 const rtc::Buffer* GetRtpPacket(int index) {
88 rtc::CritScope cs(&crit_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000089 if (index >= NumRtpPackets()) {
90 return NULL;
91 }
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000092 return new rtc::Buffer(rtp_packets_[index]);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000093 }
94
95 int NumRtcpPackets() {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000096 rtc::CritScope cs(&crit_);
henrike@webrtc.org28654cb2013-07-22 21:07:49 +000097 return static_cast<int>(rtcp_packets_.size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +000098 }
99
100 // Note: callers are responsible for deleting the returned buffer.
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000101 const rtc::Buffer* GetRtcpPacket(int index) {
102 rtc::CritScope cs(&crit_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000103 if (index >= NumRtcpPackets()) {
104 return NULL;
105 }
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000106 return new rtc::Buffer(rtcp_packets_[index]);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000107 }
108
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000109 int sendbuf_size() const { return sendbuf_size_; }
110 int recvbuf_size() const { return recvbuf_size_; }
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000111 rtc::DiffServCodePoint dscp() const { return dscp_; }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000112
113 protected:
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000114 virtual bool SendPacket(rtc::Buffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700115 const rtc::PacketOptions& options) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000116 rtc::CritScope cs(&crit_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000117
Peter Boström0c4e06b2015-10-07 12:23:21 +0200118 uint32_t cur_ssrc = 0;
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000119 if (!GetRtpSsrc(packet->data(), packet->size(), &cur_ssrc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000120 return false;
121 }
122 sent_ssrcs_[cur_ssrc]++;
123
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000124 rtp_packets_.push_back(*packet);
125 if (conf_) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000126 rtc::Buffer buffer_copy(*packet);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000127 for (size_t i = 0; i < conf_sent_ssrcs_.size(); ++i) {
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000128 if (!SetRtpSsrc(buffer_copy.data(), buffer_copy.size(),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000129 conf_sent_ssrcs_[i])) {
130 return false;
131 }
132 PostMessage(ST_RTP, buffer_copy);
133 }
134 } else {
135 PostMessage(ST_RTP, *packet);
136 }
137 return true;
138 }
139
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000140 virtual bool SendRtcp(rtc::Buffer* packet,
stefanc1aeaf02015-10-15 07:26:07 -0700141 const rtc::PacketOptions& options) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000142 rtc::CritScope cs(&crit_);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000143 rtcp_packets_.push_back(*packet);
144 if (!conf_) {
145 // don't worry about RTCP in conf mode for now
146 PostMessage(ST_RTCP, *packet);
147 }
148 return true;
149 }
150
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000151 virtual int SetOption(SocketType type, rtc::Socket::Option opt,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000152 int option) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000153 if (opt == rtc::Socket::OPT_SNDBUF) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000154 sendbuf_size_ = option;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000155 } else if (opt == rtc::Socket::OPT_RCVBUF) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000156 recvbuf_size_ = option;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000157 } else if (opt == rtc::Socket::OPT_DSCP) {
158 dscp_ = static_cast<rtc::DiffServCodePoint>(option);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000159 }
160 return 0;
161 }
162
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000163 void PostMessage(int id, const rtc::Buffer& packet) {
164 thread_->Post(this, id, rtc::WrapMessageData(packet));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000165 }
166
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000167 virtual void OnMessage(rtc::Message* msg) {
168 rtc::TypedMessageData<rtc::Buffer>* msg_data =
169 static_cast<rtc::TypedMessageData<rtc::Buffer>*>(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000170 msg->pdata);
171 if (dest_) {
172 if (msg->message_id == ST_RTP) {
wu@webrtc.orga9890802013-12-13 00:21:03 +0000173 dest_->OnPacketReceived(&msg_data->data(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000174 rtc::CreatePacketTime(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000175 } else {
wu@webrtc.orga9890802013-12-13 00:21:03 +0000176 dest_->OnRtcpReceived(&msg_data->data(),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000177 rtc::CreatePacketTime(0));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000178 }
179 }
180 delete msg_data;
181 }
182
183 private:
Peter Boström0c4e06b2015-10-07 12:23:21 +0200184 void GetNumRtpBytesAndPackets(uint32_t ssrc, int* bytes, int* packets) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000185 if (bytes) {
186 *bytes = 0;
187 }
188 if (packets) {
189 *packets = 0;
190 }
Peter Boström0c4e06b2015-10-07 12:23:21 +0200191 uint32_t cur_ssrc = 0;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000192 for (size_t i = 0; i < rtp_packets_.size(); ++i) {
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000193 if (!GetRtpSsrc(rtp_packets_[i].data(), rtp_packets_[i].size(),
194 &cur_ssrc)) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000195 return;
196 }
197 if (ssrc == cur_ssrc) {
198 if (bytes) {
kwiberg@webrtc.orgeebcab52015-03-24 09:19:06 +0000199 *bytes += static_cast<int>(rtp_packets_[i].size());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000200 }
201 if (packets) {
202 ++(*packets);
203 }
204 }
205 }
206 }
207
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000208 rtc::Thread* thread_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000209 MediaChannel* dest_;
210 bool conf_;
211 // The ssrcs used in sending out packets in conference mode.
Peter Boström0c4e06b2015-10-07 12:23:21 +0200212 std::vector<uint32_t> conf_sent_ssrcs_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000213 // Map to track counts of packets that have been sent per ssrc.
214 // This includes packets that are dropped.
Peter Boström0c4e06b2015-10-07 12:23:21 +0200215 std::map<uint32_t, uint32_t> sent_ssrcs_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000216 // Map to track packet-number that needs to be dropped per ssrc.
Peter Boström0c4e06b2015-10-07 12:23:21 +0200217 std::map<uint32_t, std::set<uint32_t> > drop_map_;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000218 rtc::CriticalSection crit_;
219 std::vector<rtc::Buffer> rtp_packets_;
220 std::vector<rtc::Buffer> rtcp_packets_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000221 int sendbuf_size_;
222 int recvbuf_size_;
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000223 rtc::DiffServCodePoint dscp_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000224};
225
226} // namespace cricket
227
kjellandera96e2d72016-02-04 23:52:28 -0800228#endif // WEBRTC_MEDIA_BASE_FAKENETWORKINTERFACE_H_