blob: e8f325b5a254472399ccfcb91078ff84907064a3 [file] [log] [blame]
Sebastian Jansson98b07e92018-09-27 13:47:01 +02001/*
2 * Copyright 2018 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#include "test/scenario/network_node.h"
11
12#include <algorithm>
13#include <vector>
14
Steve Anton40d55332019-01-07 10:21:47 -080015#include "absl/memory/memory.h"
Yves Gerey2e00abc2018-10-05 15:39:24 +020016#include "rtc_base/numerics/safe_minmax.h"
17
Sebastian Jansson98b07e92018-09-27 13:47:01 +020018namespace webrtc {
19namespace test {
20namespace {
Sebastian Jansson4124dab2019-04-01 14:33:53 +020021constexpr char kDummyTransportName[] = "dummy";
Sebastian Janssonef86d142019-04-15 14:42:42 +020022SimulatedNetwork::Config CreateSimulationConfig(
23 NetworkSimulationConfig config) {
Sebastian Jansson98b07e92018-09-27 13:47:01 +020024 SimulatedNetwork::Config sim_config;
Sebastian Janssonef86d142019-04-15 14:42:42 +020025 sim_config.link_capacity_kbps = config.bandwidth.kbps_or(0);
26 sim_config.loss_percent = config.loss_rate * 100;
27 sim_config.queue_delay_ms = config.delay.ms();
28 sim_config.delay_standard_deviation_ms = config.delay_std_dev.ms();
Sebastian Jansson8c8feb92019-01-29 15:59:17 +010029 sim_config.packet_overhead = config.packet_overhead.bytes<int>();
Sebastian Jansson2b08e312019-02-25 10:24:46 +010030 sim_config.codel_active_queue_management =
Sebastian Janssonef86d142019-04-15 14:42:42 +020031 config.codel_active_queue_management;
Sebastian Jansson98b07e92018-09-27 13:47:01 +020032 return sim_config;
33}
34} // namespace
35
Artem Titov40f51152019-01-04 15:45:01 +010036void NullReceiver::OnPacketReceived(EmulatedIpPacket packet) {}
Sebastian Jansson98b07e92018-09-27 13:47:01 +020037
38ActionReceiver::ActionReceiver(std::function<void()> action)
39 : action_(action) {}
40
Artem Titov40f51152019-01-04 15:45:01 +010041void ActionReceiver::OnPacketReceived(EmulatedIpPacket packet) {
Sebastian Jansson98b07e92018-09-27 13:47:01 +020042 action_();
Sebastian Jansson98b07e92018-09-27 13:47:01 +020043}
44
Sebastian Jansson98b07e92018-09-27 13:47:01 +020045std::unique_ptr<SimulationNode> SimulationNode::Create(
Sebastian Jansson4124dab2019-04-01 14:33:53 +020046 Clock* clock,
47 rtc::TaskQueue* task_queue,
Sebastian Janssonef86d142019-04-15 14:42:42 +020048 NetworkSimulationConfig config) {
Sebastian Jansson98b07e92018-09-27 13:47:01 +020049 SimulatedNetwork::Config sim_config = CreateSimulationConfig(config);
50 auto network = absl::make_unique<SimulatedNetwork>(sim_config);
51 SimulatedNetwork* simulation_ptr = network.get();
Sebastian Jansson4124dab2019-04-01 14:33:53 +020052 return std::unique_ptr<SimulationNode>(new SimulationNode(
53 clock, task_queue, config, std::move(network), simulation_ptr));
Sebastian Jansson98b07e92018-09-27 13:47:01 +020054}
55
56void SimulationNode::UpdateConfig(
Sebastian Janssonef86d142019-04-15 14:42:42 +020057 std::function<void(NetworkSimulationConfig*)> modifier) {
Sebastian Jansson98b07e92018-09-27 13:47:01 +020058 modifier(&config_);
59 SimulatedNetwork::Config sim_config = CreateSimulationConfig(config_);
60 simulated_network_->SetConfig(sim_config);
61}
62
63void SimulationNode::PauseTransmissionUntil(Timestamp until) {
64 simulated_network_->PauseTransmissionUntil(until.us());
65}
66
67ColumnPrinter SimulationNode::ConfigPrinter() const {
Sebastian Janssonef86d142019-04-15 14:42:42 +020068 return ColumnPrinter::Lambda(
69 "propagation_delay capacity loss_rate",
70 [this](rtc::SimpleStringBuilder& sb) {
71 sb.AppendFormat("%.3lf %.0lf %.2lf", config_.delay.seconds<double>(),
72 config_.bandwidth.bps() / 8.0, config_.loss_rate);
73 });
Sebastian Jansson98b07e92018-09-27 13:47:01 +020074}
75
76SimulationNode::SimulationNode(
Sebastian Jansson4124dab2019-04-01 14:33:53 +020077 Clock* clock,
78 rtc::TaskQueue* task_queue,
Sebastian Janssonef86d142019-04-15 14:42:42 +020079 NetworkSimulationConfig config,
Artem Titov8ea1e9d2018-10-04 14:46:31 +020080 std::unique_ptr<NetworkBehaviorInterface> behavior,
Sebastian Jansson98b07e92018-09-27 13:47:01 +020081 SimulatedNetwork* simulation)
Sebastian Jansson4124dab2019-04-01 14:33:53 +020082 : EmulatedNetworkNode(clock, task_queue, std::move(behavior)),
Sebastian Jansson98b07e92018-09-27 13:47:01 +020083 simulated_network_(simulation),
84 config_(config) {}
85
Sebastian Janssonaa01f272019-01-30 11:28:59 +010086NetworkNodeTransport::NetworkNodeTransport(Clock* sender_clock,
Sebastian Jansson800e1212018-10-22 11:49:03 +020087 Call* sender_call)
88 : sender_clock_(sender_clock), sender_call_(sender_call) {}
Sebastian Jansson98b07e92018-09-27 13:47:01 +020089
90NetworkNodeTransport::~NetworkNodeTransport() = default;
91
92bool NetworkNodeTransport::SendRtp(const uint8_t* packet,
93 size_t length,
94 const PacketOptions& options) {
Sebastian Jansson800e1212018-10-22 11:49:03 +020095 int64_t send_time_ms = sender_clock_->TimeInMilliseconds();
Sebastian Jansson156d11d2018-09-28 17:21:34 +020096 rtc::SentPacket sent_packet;
97 sent_packet.packet_id = options.packet_id;
Sebastian Jansson03789972018-10-09 18:27:57 +020098 sent_packet.info.included_in_feedback = options.included_in_feedback;
99 sent_packet.info.included_in_allocation = options.included_in_allocation;
Sebastian Jansson156d11d2018-09-28 17:21:34 +0200100 sent_packet.send_time_ms = send_time_ms;
101 sent_packet.info.packet_size_bytes = length;
102 sent_packet.info.packet_type = rtc::PacketType::kData;
Sebastian Jansson800e1212018-10-22 11:49:03 +0200103 sender_call_->OnSentPacket(sent_packet);
Sebastian Jansson156d11d2018-09-28 17:21:34 +0200104
105 Timestamp send_time = Timestamp::ms(send_time_ms);
Sebastian Jansson800e1212018-10-22 11:49:03 +0200106 rtc::CritScope crit(&crit_sect_);
107 if (!send_net_)
108 return false;
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200109 rtc::CopyOnWriteBuffer buffer(packet, length,
110 length + packet_overhead_.bytes());
111 buffer.SetSize(length + packet_overhead_.bytes());
Artem Titov4cd433e2019-04-01 11:01:16 +0200112 send_net_->OnPacketReceived(
113 EmulatedIpPacket(local_address_, receiver_address_, buffer, send_time));
Sebastian Janssonf65309c2018-12-20 10:26:00 +0100114 return true;
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200115}
116
117bool NetworkNodeTransport::SendRtcp(const uint8_t* packet, size_t length) {
118 rtc::CopyOnWriteBuffer buffer(packet, length);
Sebastian Jansson800e1212018-10-22 11:49:03 +0200119 Timestamp send_time = Timestamp::ms(sender_clock_->TimeInMilliseconds());
120 rtc::CritScope crit(&crit_sect_);
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200121 buffer.SetSize(length + packet_overhead_.bytes());
Sebastian Jansson800e1212018-10-22 11:49:03 +0200122 if (!send_net_)
123 return false;
Artem Titov4cd433e2019-04-01 11:01:16 +0200124 send_net_->OnPacketReceived(
125 EmulatedIpPacket(local_address_, receiver_address_, buffer, send_time));
Sebastian Janssonf65309c2018-12-20 10:26:00 +0100126 return true;
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200127}
128
Artem Titov37d18482019-01-08 15:41:45 +0100129void NetworkNodeTransport::Connect(EmulatedNetworkNode* send_node,
Artem Titov4cd433e2019-04-01 11:01:16 +0200130 rtc::IPAddress receiver_ip,
Sebastian Jansson800e1212018-10-22 11:49:03 +0200131 DataSize packet_overhead) {
Sebastian Jansson800e1212018-10-22 11:49:03 +0200132 rtc::NetworkRoute route;
133 route.connected = true;
Artem Titov4cd433e2019-04-01 11:01:16 +0200134 route.local_network_id =
135 static_cast<uint16_t>(receiver_ip.v4AddressAsHostOrderInteger());
136 route.remote_network_id =
137 static_cast<uint16_t>(receiver_ip.v4AddressAsHostOrderInteger());
Sebastian Jansson4124dab2019-04-01 14:33:53 +0200138 {
139 // Only IPv4 address is supported. We don't use full range of IPs in
140 // scenario framework and also we need a simple way to convert IP into
141 // network_id to signal network route.
142 RTC_CHECK_EQ(receiver_ip.family(), AF_INET);
143 RTC_CHECK_LE(receiver_ip.v4AddressAsHostOrderInteger(),
144 std::numeric_limits<uint16_t>::max());
145 rtc::CritScope crit(&crit_sect_);
146 send_net_ = send_node;
147 receiver_address_ = rtc::SocketAddress(receiver_ip, 0);
148 packet_overhead_ = packet_overhead;
149 current_network_route_ = route;
150 }
151
Sebastian Jansson800e1212018-10-22 11:49:03 +0200152 sender_call_->GetTransportControllerSend()->OnNetworkRouteChanged(
Sebastian Jansson4124dab2019-04-01 14:33:53 +0200153 kDummyTransportName, route);
154}
155
156void NetworkNodeTransport::Disconnect() {
157 rtc::CritScope crit(&crit_sect_);
158 current_network_route_.connected = false;
159 sender_call_->GetTransportControllerSend()->OnNetworkRouteChanged(
160 kDummyTransportName, current_network_route_);
161 current_network_route_ = {};
162 send_net_ = nullptr;
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200163}
164
Artem Titov40f51152019-01-04 15:45:01 +0100165CrossTrafficSource::CrossTrafficSource(EmulatedNetworkReceiverInterface* target,
Artem Titov4cd433e2019-04-01 11:01:16 +0200166 rtc::IPAddress receiver_ip,
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200167 CrossTrafficConfig config)
168 : target_(target),
Artem Titov4cd433e2019-04-01 11:01:16 +0200169 receiver_address_(receiver_ip, 0),
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200170 config_(config),
171 random_(config.random_seed) {}
172
173CrossTrafficSource::~CrossTrafficSource() = default;
174
175DataRate CrossTrafficSource::TrafficRate() const {
176 return config_.peak_rate * intensity_;
177}
178
179void CrossTrafficSource::Process(Timestamp at_time, TimeDelta delta) {
180 time_since_update_ += delta;
181 if (config_.mode == CrossTrafficConfig::Mode::kRandomWalk) {
182 if (time_since_update_ >= config_.random_walk.update_interval) {
183 intensity_ += random_.Gaussian(config_.random_walk.bias,
184 config_.random_walk.variance) *
185 time_since_update_.seconds<double>();
186 intensity_ = rtc::SafeClamp(intensity_, 0.0, 1.0);
187 time_since_update_ = TimeDelta::Zero();
188 }
189 } else if (config_.mode == CrossTrafficConfig::Mode::kPulsedPeaks) {
190 if (intensity_ == 0 && time_since_update_ >= config_.pulsed.hold_duration) {
191 intensity_ = 1;
192 time_since_update_ = TimeDelta::Zero();
193 } else if (intensity_ == 1 &&
194 time_since_update_ >= config_.pulsed.send_duration) {
195 intensity_ = 0;
196 time_since_update_ = TimeDelta::Zero();
197 }
198 }
199 pending_size_ += TrafficRate() * delta;
200 if (pending_size_ > config_.min_packet_size) {
Artem Titov40f51152019-01-04 15:45:01 +0100201 target_->OnPacketReceived(EmulatedIpPacket(
Artem Titovde83d0c2019-04-01 15:03:46 +0200202 /*from=*/rtc::SocketAddress(), receiver_address_,
Artem Titov4cd433e2019-04-01 11:01:16 +0200203 rtc::CopyOnWriteBuffer(pending_size_.bytes()), at_time));
Sebastian Jansson98b07e92018-09-27 13:47:01 +0200204 pending_size_ = DataSize::Zero();
205 }
206}
207
208ColumnPrinter CrossTrafficSource::StatsPrinter() {
209 return ColumnPrinter::Lambda("cross_traffic_rate",
210 [this](rtc::SimpleStringBuilder& sb) {
211 sb.AppendFormat("%.0lf",
212 TrafficRate().bps() / 8.0);
213 },
214 32);
215}
216
217} // namespace test
218} // namespace webrtc