| /* |
| * 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/testclient.h" |
| |
| #include <string.h> |
| #include <utility> |
| |
| #include "absl/memory/memory.h" |
| #include "rtc_base/gunit.h" |
| #include "rtc_base/thread.h" |
| #include "rtc_base/timeutils.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<AsyncPacketSocket> socket) |
| : TestClient(std::move(socket), nullptr) {} |
| |
| TestClient::TestClient(std::unique_ptr<AsyncPacketSocket> socket, |
| FakeClock* fake_clock) |
| : fake_clock_(fake_clock), |
| socket_(std::move(socket)), |
| prev_packet_timestamp_(-1) { |
| socket_->SignalReadPacket.connect(this, &TestClient::OnPacket); |
| 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::Packet> 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) { |
| { |
| CritScope cs(&crit_); |
| if (packets_.size() != 0) { |
| break; |
| } |
| } |
| AdvanceTime(1); |
| } |
| |
| // Return the first packet placed in the queue. |
| std::unique_ptr<Packet> packet; |
| CritScope cs(&crit_); |
| 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> packet = NextPacket(kTimeoutMs); |
| if (packet) { |
| res = (packet->size == size && memcmp(packet->buf, buf, size) == 0 && |
| CheckTimestamp(packet->packet_time_us)); |
| if (addr) |
| *addr = packet->addr; |
| } |
| return res; |
| } |
| |
| bool TestClient::CheckTimestamp(int64_t packet_timestamp) { |
| bool res = true; |
| if (packet_timestamp == -1) { |
| res = false; |
| } |
| if (prev_packet_timestamp_ != -1) { |
| 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 char* buf, |
| size_t size, |
| const SocketAddress& remote_addr, |
| const int64_t& packet_time_us) { |
| CritScope cs(&crit_); |
| packets_.push_back( |
| absl::make_unique<Packet>(remote_addr, buf, size, packet_time_us)); |
| } |
| |
| void TestClient::OnReadyToSend(AsyncPacketSocket* socket) { |
| ++ready_to_send_count_; |
| } |
| |
| TestClient::Packet::Packet(const SocketAddress& a, |
| const char* b, |
| size_t s, |
| int64_t packet_time_us) |
| : addr(a), buf(0), size(s), packet_time_us(packet_time_us) { |
| buf = new char[size]; |
| memcpy(buf, b, size); |
| } |
| |
| TestClient::Packet::Packet(const Packet& p) |
| : addr(p.addr), buf(0), size(p.size), packet_time_us(p.packet_time_us) { |
| buf = new char[size]; |
| memcpy(buf, p.buf, size); |
| } |
| |
| TestClient::Packet::~Packet() { |
| delete[] buf; |
| } |
| |
| } // namespace rtc |