1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker * Copyright 2018 The WebRTC project authors. All Rights Reserved.
3*d9f75844SAndroid Build Coastguard Worker *
4*d9f75844SAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker * that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker * tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker * in the file PATENTS. All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker * be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker */
10*d9f75844SAndroid Build Coastguard Worker #include "test/scenario/network_node.h"
11*d9f75844SAndroid Build Coastguard Worker
12*d9f75844SAndroid Build Coastguard Worker #include <algorithm>
13*d9f75844SAndroid Build Coastguard Worker #include <vector>
14*d9f75844SAndroid Build Coastguard Worker
15*d9f75844SAndroid Build Coastguard Worker #include <memory>
16*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/net_helper.h"
17*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/numerics/safe_minmax.h"
18*d9f75844SAndroid Build Coastguard Worker
19*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
20*d9f75844SAndroid Build Coastguard Worker namespace test {
21*d9f75844SAndroid Build Coastguard Worker namespace {
22*d9f75844SAndroid Build Coastguard Worker constexpr char kDummyTransportName[] = "dummy";
CreateSimulationConfig(NetworkSimulationConfig config)23*d9f75844SAndroid Build Coastguard Worker SimulatedNetwork::Config CreateSimulationConfig(
24*d9f75844SAndroid Build Coastguard Worker NetworkSimulationConfig config) {
25*d9f75844SAndroid Build Coastguard Worker SimulatedNetwork::Config sim_config;
26*d9f75844SAndroid Build Coastguard Worker sim_config.link_capacity_kbps = config.bandwidth.kbps_or(0);
27*d9f75844SAndroid Build Coastguard Worker sim_config.loss_percent = config.loss_rate * 100;
28*d9f75844SAndroid Build Coastguard Worker sim_config.queue_delay_ms = config.delay.ms();
29*d9f75844SAndroid Build Coastguard Worker sim_config.delay_standard_deviation_ms = config.delay_std_dev.ms();
30*d9f75844SAndroid Build Coastguard Worker sim_config.packet_overhead = config.packet_overhead.bytes<int>();
31*d9f75844SAndroid Build Coastguard Worker sim_config.queue_length_packets =
32*d9f75844SAndroid Build Coastguard Worker config.packet_queue_length_limit.value_or(0);
33*d9f75844SAndroid Build Coastguard Worker return sim_config;
34*d9f75844SAndroid Build Coastguard Worker }
35*d9f75844SAndroid Build Coastguard Worker } // namespace
36*d9f75844SAndroid Build Coastguard Worker
SimulationNode(NetworkSimulationConfig config,SimulatedNetwork * behavior,EmulatedNetworkNode * network_node)37*d9f75844SAndroid Build Coastguard Worker SimulationNode::SimulationNode(NetworkSimulationConfig config,
38*d9f75844SAndroid Build Coastguard Worker SimulatedNetwork* behavior,
39*d9f75844SAndroid Build Coastguard Worker EmulatedNetworkNode* network_node)
40*d9f75844SAndroid Build Coastguard Worker : config_(config), simulation_(behavior), network_node_(network_node) {}
41*d9f75844SAndroid Build Coastguard Worker
CreateBehavior(NetworkSimulationConfig config)42*d9f75844SAndroid Build Coastguard Worker std::unique_ptr<SimulatedNetwork> SimulationNode::CreateBehavior(
43*d9f75844SAndroid Build Coastguard Worker NetworkSimulationConfig config) {
44*d9f75844SAndroid Build Coastguard Worker SimulatedNetwork::Config sim_config = CreateSimulationConfig(config);
45*d9f75844SAndroid Build Coastguard Worker return std::make_unique<SimulatedNetwork>(sim_config);
46*d9f75844SAndroid Build Coastguard Worker }
47*d9f75844SAndroid Build Coastguard Worker
UpdateConfig(std::function<void (NetworkSimulationConfig *)> modifier)48*d9f75844SAndroid Build Coastguard Worker void SimulationNode::UpdateConfig(
49*d9f75844SAndroid Build Coastguard Worker std::function<void(NetworkSimulationConfig*)> modifier) {
50*d9f75844SAndroid Build Coastguard Worker modifier(&config_);
51*d9f75844SAndroid Build Coastguard Worker SimulatedNetwork::Config sim_config = CreateSimulationConfig(config_);
52*d9f75844SAndroid Build Coastguard Worker simulation_->SetConfig(sim_config);
53*d9f75844SAndroid Build Coastguard Worker }
54*d9f75844SAndroid Build Coastguard Worker
PauseTransmissionUntil(Timestamp until)55*d9f75844SAndroid Build Coastguard Worker void SimulationNode::PauseTransmissionUntil(Timestamp until) {
56*d9f75844SAndroid Build Coastguard Worker simulation_->PauseTransmissionUntil(until.us());
57*d9f75844SAndroid Build Coastguard Worker }
58*d9f75844SAndroid Build Coastguard Worker
ConfigPrinter() const59*d9f75844SAndroid Build Coastguard Worker ColumnPrinter SimulationNode::ConfigPrinter() const {
60*d9f75844SAndroid Build Coastguard Worker return ColumnPrinter::Lambda(
61*d9f75844SAndroid Build Coastguard Worker "propagation_delay capacity loss_rate",
62*d9f75844SAndroid Build Coastguard Worker [this](rtc::SimpleStringBuilder& sb) {
63*d9f75844SAndroid Build Coastguard Worker sb.AppendFormat("%.3lf %.0lf %.2lf", config_.delay.seconds<double>(),
64*d9f75844SAndroid Build Coastguard Worker config_.bandwidth.bps() / 8.0, config_.loss_rate);
65*d9f75844SAndroid Build Coastguard Worker });
66*d9f75844SAndroid Build Coastguard Worker }
67*d9f75844SAndroid Build Coastguard Worker
NetworkNodeTransport(Clock * sender_clock,Call * sender_call)68*d9f75844SAndroid Build Coastguard Worker NetworkNodeTransport::NetworkNodeTransport(Clock* sender_clock,
69*d9f75844SAndroid Build Coastguard Worker Call* sender_call)
70*d9f75844SAndroid Build Coastguard Worker : sender_clock_(sender_clock), sender_call_(sender_call) {}
71*d9f75844SAndroid Build Coastguard Worker
72*d9f75844SAndroid Build Coastguard Worker NetworkNodeTransport::~NetworkNodeTransport() = default;
73*d9f75844SAndroid Build Coastguard Worker
SendRtp(const uint8_t * packet,size_t length,const PacketOptions & options)74*d9f75844SAndroid Build Coastguard Worker bool NetworkNodeTransport::SendRtp(const uint8_t* packet,
75*d9f75844SAndroid Build Coastguard Worker size_t length,
76*d9f75844SAndroid Build Coastguard Worker const PacketOptions& options) {
77*d9f75844SAndroid Build Coastguard Worker int64_t send_time_ms = sender_clock_->TimeInMilliseconds();
78*d9f75844SAndroid Build Coastguard Worker rtc::SentPacket sent_packet;
79*d9f75844SAndroid Build Coastguard Worker sent_packet.packet_id = options.packet_id;
80*d9f75844SAndroid Build Coastguard Worker sent_packet.info.included_in_feedback = options.included_in_feedback;
81*d9f75844SAndroid Build Coastguard Worker sent_packet.info.included_in_allocation = options.included_in_allocation;
82*d9f75844SAndroid Build Coastguard Worker sent_packet.send_time_ms = send_time_ms;
83*d9f75844SAndroid Build Coastguard Worker sent_packet.info.packet_size_bytes = length;
84*d9f75844SAndroid Build Coastguard Worker sent_packet.info.packet_type = rtc::PacketType::kData;
85*d9f75844SAndroid Build Coastguard Worker sender_call_->OnSentPacket(sent_packet);
86*d9f75844SAndroid Build Coastguard Worker
87*d9f75844SAndroid Build Coastguard Worker MutexLock lock(&mutex_);
88*d9f75844SAndroid Build Coastguard Worker if (!endpoint_)
89*d9f75844SAndroid Build Coastguard Worker return false;
90*d9f75844SAndroid Build Coastguard Worker rtc::CopyOnWriteBuffer buffer(packet, length);
91*d9f75844SAndroid Build Coastguard Worker endpoint_->SendPacket(local_address_, remote_address_, buffer,
92*d9f75844SAndroid Build Coastguard Worker packet_overhead_.bytes());
93*d9f75844SAndroid Build Coastguard Worker return true;
94*d9f75844SAndroid Build Coastguard Worker }
95*d9f75844SAndroid Build Coastguard Worker
SendRtcp(const uint8_t * packet,size_t length)96*d9f75844SAndroid Build Coastguard Worker bool NetworkNodeTransport::SendRtcp(const uint8_t* packet, size_t length) {
97*d9f75844SAndroid Build Coastguard Worker rtc::CopyOnWriteBuffer buffer(packet, length);
98*d9f75844SAndroid Build Coastguard Worker MutexLock lock(&mutex_);
99*d9f75844SAndroid Build Coastguard Worker if (!endpoint_)
100*d9f75844SAndroid Build Coastguard Worker return false;
101*d9f75844SAndroid Build Coastguard Worker endpoint_->SendPacket(local_address_, remote_address_, buffer,
102*d9f75844SAndroid Build Coastguard Worker packet_overhead_.bytes());
103*d9f75844SAndroid Build Coastguard Worker return true;
104*d9f75844SAndroid Build Coastguard Worker }
105*d9f75844SAndroid Build Coastguard Worker
Connect(EmulatedEndpoint * endpoint,const rtc::SocketAddress & receiver_address,DataSize packet_overhead)106*d9f75844SAndroid Build Coastguard Worker void NetworkNodeTransport::Connect(EmulatedEndpoint* endpoint,
107*d9f75844SAndroid Build Coastguard Worker const rtc::SocketAddress& receiver_address,
108*d9f75844SAndroid Build Coastguard Worker DataSize packet_overhead) {
109*d9f75844SAndroid Build Coastguard Worker rtc::NetworkRoute route;
110*d9f75844SAndroid Build Coastguard Worker route.connected = true;
111*d9f75844SAndroid Build Coastguard Worker // We assume that the address will be unique in the lower bytes.
112*d9f75844SAndroid Build Coastguard Worker route.local = rtc::RouteEndpoint::CreateWithNetworkId(static_cast<uint16_t>(
113*d9f75844SAndroid Build Coastguard Worker receiver_address.ipaddr().v4AddressAsHostOrderInteger()));
114*d9f75844SAndroid Build Coastguard Worker route.remote = rtc::RouteEndpoint::CreateWithNetworkId(static_cast<uint16_t>(
115*d9f75844SAndroid Build Coastguard Worker receiver_address.ipaddr().v4AddressAsHostOrderInteger()));
116*d9f75844SAndroid Build Coastguard Worker route.packet_overhead = packet_overhead.bytes() +
117*d9f75844SAndroid Build Coastguard Worker receiver_address.ipaddr().overhead() +
118*d9f75844SAndroid Build Coastguard Worker cricket::kUdpHeaderSize;
119*d9f75844SAndroid Build Coastguard Worker {
120*d9f75844SAndroid Build Coastguard Worker // Only IPv4 address is supported.
121*d9f75844SAndroid Build Coastguard Worker RTC_CHECK_EQ(receiver_address.family(), AF_INET);
122*d9f75844SAndroid Build Coastguard Worker MutexLock lock(&mutex_);
123*d9f75844SAndroid Build Coastguard Worker endpoint_ = endpoint;
124*d9f75844SAndroid Build Coastguard Worker local_address_ = rtc::SocketAddress(endpoint_->GetPeerLocalAddress(), 0);
125*d9f75844SAndroid Build Coastguard Worker remote_address_ = receiver_address;
126*d9f75844SAndroid Build Coastguard Worker packet_overhead_ = packet_overhead;
127*d9f75844SAndroid Build Coastguard Worker current_network_route_ = route;
128*d9f75844SAndroid Build Coastguard Worker }
129*d9f75844SAndroid Build Coastguard Worker
130*d9f75844SAndroid Build Coastguard Worker sender_call_->GetTransportControllerSend()->OnNetworkRouteChanged(
131*d9f75844SAndroid Build Coastguard Worker kDummyTransportName, route);
132*d9f75844SAndroid Build Coastguard Worker }
133*d9f75844SAndroid Build Coastguard Worker
Disconnect()134*d9f75844SAndroid Build Coastguard Worker void NetworkNodeTransport::Disconnect() {
135*d9f75844SAndroid Build Coastguard Worker MutexLock lock(&mutex_);
136*d9f75844SAndroid Build Coastguard Worker current_network_route_.connected = false;
137*d9f75844SAndroid Build Coastguard Worker sender_call_->GetTransportControllerSend()->OnNetworkRouteChanged(
138*d9f75844SAndroid Build Coastguard Worker kDummyTransportName, current_network_route_);
139*d9f75844SAndroid Build Coastguard Worker current_network_route_ = {};
140*d9f75844SAndroid Build Coastguard Worker endpoint_ = nullptr;
141*d9f75844SAndroid Build Coastguard Worker }
142*d9f75844SAndroid Build Coastguard Worker
143*d9f75844SAndroid Build Coastguard Worker } // namespace test
144*d9f75844SAndroid Build Coastguard Worker } // namespace webrtc
145