/* * Copyright 2004 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 "rtc_base/test_client.h" #include #include #include #include "api/units/timestamp.h" #include "rtc_base/gunit.h" #include "rtc_base/network/received_packet.h" #include "rtc_base/thread.h" #include "rtc_base/time_utils.h" namespace rtc { // DESIGN: Each packet received is put it into a list of packets. // Callers can retrieve received packets from any thread by calling // NextPacket. TestClient::TestClient(std::unique_ptr socket) : TestClient(std::move(socket), nullptr) {} TestClient::TestClient(std::unique_ptr socket, ThreadProcessingFakeClock* fake_clock) : fake_clock_(fake_clock), socket_(std::move(socket)) { socket_->RegisterReceivedPacketCallback( [&](rtc::AsyncPacketSocket* socket, const rtc::ReceivedPacket& packet) { OnPacket(socket, packet); }); socket_->SignalReadyToSend.connect(this, &TestClient::OnReadyToSend); } TestClient::~TestClient() {} bool TestClient::CheckConnState(AsyncPacketSocket::State state) { // Wait for our timeout value until the socket reaches the desired state. int64_t end = TimeAfter(kTimeoutMs); while (socket_->GetState() != state && TimeUntil(end) > 0) { AdvanceTime(1); } return (socket_->GetState() == state); } int TestClient::Send(const char* buf, size_t size) { rtc::PacketOptions options; return socket_->Send(buf, size, options); } int TestClient::SendTo(const char* buf, size_t size, const SocketAddress& dest) { rtc::PacketOptions options; return socket_->SendTo(buf, size, dest, options); } std::unique_ptr TestClient::NextPacket(int timeout_ms) { // If no packets are currently available, we go into a get/dispatch loop for // at most timeout_ms. If, during the loop, a packet arrives, then we can // stop early and return it. // Note that the case where no packet arrives is important. We often want to // test that a packet does not arrive. // Note also that we only try to pump our current thread's message queue. // Pumping another thread's queue could lead to messages being dispatched from // the wrong thread to non-thread-safe objects. int64_t end = TimeAfter(timeout_ms); while (TimeUntil(end) > 0) { { webrtc::MutexLock lock(&mutex_); if (packets_.size() != 0) { break; } } AdvanceTime(1); } // Return the first packet placed in the queue. std::unique_ptr packet; webrtc::MutexLock lock(&mutex_); if (packets_.size() > 0) { packet = std::move(packets_.front()); packets_.erase(packets_.begin()); } return packet; } bool TestClient::CheckNextPacket(const char* buf, size_t size, SocketAddress* addr) { bool res = false; std::unique_ptr packet = NextPacket(kTimeoutMs); if (packet) { res = (packet->buf.size() == size && memcmp(packet->buf.data(), buf, size) == 0 && CheckTimestamp(packet->packet_time)); if (addr) *addr = packet->addr; } return res; } bool TestClient::CheckTimestamp( absl::optional packet_timestamp) { bool res = true; if (!packet_timestamp) { res = false; } if (prev_packet_timestamp_) { if (packet_timestamp < prev_packet_timestamp_) { res = false; } } prev_packet_timestamp_ = packet_timestamp; return res; } void TestClient::AdvanceTime(int ms) { // If the test is using a fake clock, we must advance the fake clock to // advance time. Otherwise, ProcessMessages will work. if (fake_clock_) { SIMULATED_WAIT(false, ms, *fake_clock_); } else { Thread::Current()->ProcessMessages(1); } } bool TestClient::CheckNoPacket() { return NextPacket(kNoPacketTimeoutMs) == nullptr; } int TestClient::GetError() { return socket_->GetError(); } int TestClient::SetOption(Socket::Option opt, int value) { return socket_->SetOption(opt, value); } void TestClient::OnPacket(AsyncPacketSocket* socket, const rtc::ReceivedPacket& received_packet) { webrtc::MutexLock lock(&mutex_); packets_.push_back(std::make_unique(received_packet)); } void TestClient::OnReadyToSend(AsyncPacketSocket* socket) { ++ready_to_send_count_; } TestClient::Packet::Packet(const rtc::ReceivedPacket& received_packet) : addr(received_packet.source_address()), // Copy received_packet payload to a buffer owned by Packet. buf(received_packet.payload().data(), received_packet.payload().size()), packet_time(received_packet.arrival_time()) {} TestClient::Packet::Packet(const Packet& p) : addr(p.addr), buf(p.buf.data(), p.buf.size()), packet_time(p.packet_time) {} } // namespace rtc