blob: 2baf5a4662e38332f06ae79a02dcc09439203183 [file] [log] [blame]
/*
* Copyright (c) 2019 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 "test/network/traffic_route.h"
#include <algorithm>
#include <memory>
#include <utility>
#include "absl/types/optional.h"
#include "rtc_base/logging.h"
#include "rtc_base/numerics/safe_minmax.h"
namespace webrtc {
namespace test {
namespace {
class NullReceiver : public EmulatedNetworkReceiverInterface {
public:
void OnPacketReceived(EmulatedIpPacket packet) override {}
};
class ActionReceiver : public EmulatedNetworkReceiverInterface {
public:
ActionReceiver(std::function<void()> action, EmulatedEndpoint* endpoint)
: action_(action), endpoint_(endpoint) {}
~ActionReceiver() override = default;
void OnPacketReceived(EmulatedIpPacket packet) override {
RTC_DCHECK(port_);
action_();
endpoint_->UnbindReceiver(port_.value());
}
// We can't set port in constructor, because port will be provided by
// endpoint, when this receiver will be binded to that endpoint.
void SetPort(uint16_t port) { port_ = port; }
private:
std::function<void()> action_;
// Endpoint and port will be used to free port in the endpoint after action
// will be done.
EmulatedEndpoint* endpoint_;
absl::optional<uint16_t> port_ = absl::nullopt;
};
} // namespace
TrafficRoute::TrafficRoute(Clock* clock,
EmulatedNetworkReceiverInterface* receiver,
EmulatedEndpoint* endpoint)
: clock_(clock), receiver_(receiver), endpoint_(endpoint) {
null_receiver_ = std::make_unique<NullReceiver>();
absl::optional<uint16_t> port =
endpoint_->BindReceiver(0, null_receiver_.get());
RTC_DCHECK(port);
null_receiver_port_ = port.value();
}
TrafficRoute::~TrafficRoute() = default;
void TrafficRoute::TriggerPacketBurst(size_t num_packets, size_t packet_size) {
for (size_t i = 0; i < num_packets; ++i) {
SendPacket(packet_size);
}
}
void TrafficRoute::NetworkDelayedAction(size_t packet_size,
std::function<void()> action) {
auto action_receiver = std::make_unique<ActionReceiver>(action, endpoint_);
absl::optional<uint16_t> port =
endpoint_->BindReceiver(0, action_receiver.get());
RTC_DCHECK(port);
action_receiver->SetPort(port.value());
actions_.push_back(std::move(action_receiver));
SendPacket(packet_size, port.value());
}
void TrafficRoute::SendPacket(size_t packet_size) {
SendPacket(packet_size, null_receiver_port_);
}
void TrafficRoute::SendPacket(size_t packet_size, uint16_t dest_port) {
rtc::CopyOnWriteBuffer data(packet_size);
std::fill_n(data.data<uint8_t>(), data.size(), 0);
receiver_->OnPacketReceived(EmulatedIpPacket(
/*from=*/rtc::SocketAddress(),
rtc::SocketAddress(endpoint_->GetPeerLocalAddress(), dest_port), data,
clock_->CurrentTime()));
}
} // namespace test
} // namespace webrtc