| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 1 | /* | 
 | 2 |  *  Copyright 2004 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 |  | 
| Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 11 | #include "rtc_base/testclient.h" | 
| deadbeef | 22e0814 | 2017-06-12 14:30:28 -0700 | [diff] [blame] | 12 |  | 
| Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 13 | #include "rtc_base/gunit.h" | 
 | 14 | #include "rtc_base/ptr_util.h" | 
 | 15 | #include "rtc_base/thread.h" | 
 | 16 | #include "rtc_base/timeutils.h" | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 17 |  | 
 | 18 | namespace rtc { | 
 | 19 |  | 
 | 20 | // DESIGN: Each packet received is put it into a list of packets. | 
 | 21 | //         Callers can retrieve received packets from any thread by calling | 
 | 22 | //         NextPacket. | 
 | 23 |  | 
| nisse | 32f2505 | 2017-05-08 01:57:18 -0700 | [diff] [blame] | 24 | TestClient::TestClient(std::unique_ptr<AsyncPacketSocket> socket) | 
| deadbeef | 22e0814 | 2017-06-12 14:30:28 -0700 | [diff] [blame] | 25 |     : TestClient(std::move(socket), nullptr) {} | 
 | 26 |  | 
 | 27 | TestClient::TestClient(std::unique_ptr<AsyncPacketSocket> socket, | 
 | 28 |                        FakeClock* fake_clock) | 
 | 29 |     : fake_clock_(fake_clock), | 
 | 30 |       socket_(std::move(socket)), | 
 | 31 |       prev_packet_timestamp_(-1) { | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 32 |   socket_->SignalReadPacket.connect(this, &TestClient::OnPacket); | 
 | 33 |   socket_->SignalReadyToSend.connect(this, &TestClient::OnReadyToSend); | 
 | 34 | } | 
 | 35 |  | 
| nisse | 32f2505 | 2017-05-08 01:57:18 -0700 | [diff] [blame] | 36 | TestClient::~TestClient() {} | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 37 |  | 
 | 38 | bool TestClient::CheckConnState(AsyncPacketSocket::State state) { | 
 | 39 |   // Wait for our timeout value until the socket reaches the desired state. | 
| Taylor Brandstetter | 2b3bf6b | 2016-05-19 14:57:31 -0700 | [diff] [blame] | 40 |   int64_t end = TimeAfter(kTimeoutMs); | 
 | 41 |   while (socket_->GetState() != state && TimeUntil(end) > 0) { | 
| deadbeef | 22e0814 | 2017-06-12 14:30:28 -0700 | [diff] [blame] | 42 |     AdvanceTime(1); | 
| Taylor Brandstetter | 2b3bf6b | 2016-05-19 14:57:31 -0700 | [diff] [blame] | 43 |   } | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 44 |   return (socket_->GetState() == state); | 
 | 45 | } | 
 | 46 |  | 
 | 47 | int TestClient::Send(const char* buf, size_t size) { | 
 | 48 |   rtc::PacketOptions options; | 
 | 49 |   return socket_->Send(buf, size, options); | 
 | 50 | } | 
 | 51 |  | 
 | 52 | int TestClient::SendTo(const char* buf, size_t size, | 
 | 53 |                        const SocketAddress& dest) { | 
 | 54 |   rtc::PacketOptions options; | 
 | 55 |   return socket_->SendTo(buf, size, dest, options); | 
 | 56 | } | 
 | 57 |  | 
| nisse | 32f2505 | 2017-05-08 01:57:18 -0700 | [diff] [blame] | 58 | std::unique_ptr<TestClient::Packet> TestClient::NextPacket(int timeout_ms) { | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 59 |   // If no packets are currently available, we go into a get/dispatch loop for | 
| jlmiller@webrtc.org | ec499be | 2015-02-07 22:37:59 +0000 | [diff] [blame] | 60 |   // at most timeout_ms.  If, during the loop, a packet arrives, then we can | 
 | 61 |   // stop early and return it. | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 62 |  | 
 | 63 |   // Note that the case where no packet arrives is important.  We often want to | 
 | 64 |   // test that a packet does not arrive. | 
 | 65 |  | 
 | 66 |   // Note also that we only try to pump our current thread's message queue. | 
 | 67 |   // Pumping another thread's queue could lead to messages being dispatched from | 
 | 68 |   // the wrong thread to non-thread-safe objects. | 
 | 69 |  | 
| Taylor Brandstetter | 2b3bf6b | 2016-05-19 14:57:31 -0700 | [diff] [blame] | 70 |   int64_t end = TimeAfter(timeout_ms); | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 71 |   while (TimeUntil(end) > 0) { | 
 | 72 |     { | 
 | 73 |       CritScope cs(&crit_); | 
| nisse | 32f2505 | 2017-05-08 01:57:18 -0700 | [diff] [blame] | 74 |       if (packets_.size() != 0) { | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 75 |         break; | 
 | 76 |       } | 
 | 77 |     } | 
| deadbeef | 22e0814 | 2017-06-12 14:30:28 -0700 | [diff] [blame] | 78 |     AdvanceTime(1); | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 79 |   } | 
 | 80 |  | 
 | 81 |   // Return the first packet placed in the queue. | 
| nisse | 32f2505 | 2017-05-08 01:57:18 -0700 | [diff] [blame] | 82 |   std::unique_ptr<Packet> packet; | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 83 |   CritScope cs(&crit_); | 
| nisse | 32f2505 | 2017-05-08 01:57:18 -0700 | [diff] [blame] | 84 |   if (packets_.size() > 0) { | 
 | 85 |     packet = std::move(packets_.front()); | 
 | 86 |     packets_.erase(packets_.begin()); | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 87 |   } | 
 | 88 |  | 
 | 89 |   return packet; | 
 | 90 | } | 
 | 91 |  | 
 | 92 | bool TestClient::CheckNextPacket(const char* buf, size_t size, | 
 | 93 |                                  SocketAddress* addr) { | 
 | 94 |   bool res = false; | 
| nisse | 32f2505 | 2017-05-08 01:57:18 -0700 | [diff] [blame] | 95 |   std::unique_ptr<Packet> packet = NextPacket(kTimeoutMs); | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 96 |   if (packet) { | 
| Stefan Holmer | 9131efd | 2016-05-23 18:19:26 +0200 | [diff] [blame] | 97 |     res = (packet->size == size && memcmp(packet->buf, buf, size) == 0 && | 
 | 98 |            CheckTimestamp(packet->packet_time.timestamp)); | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 99 |     if (addr) | 
 | 100 |       *addr = packet->addr; | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 101 |   } | 
 | 102 |   return res; | 
 | 103 | } | 
 | 104 |  | 
| Stefan Holmer | 9131efd | 2016-05-23 18:19:26 +0200 | [diff] [blame] | 105 | bool TestClient::CheckTimestamp(int64_t packet_timestamp) { | 
 | 106 |   bool res = true; | 
 | 107 |   if (packet_timestamp == -1) { | 
 | 108 |     res = false; | 
 | 109 |   } | 
| Stefan Holmer | 9131efd | 2016-05-23 18:19:26 +0200 | [diff] [blame] | 110 |   if (prev_packet_timestamp_ != -1) { | 
 | 111 |     if (packet_timestamp < prev_packet_timestamp_) { | 
 | 112 |       res = false; | 
 | 113 |     } | 
| Stefan Holmer | 9131efd | 2016-05-23 18:19:26 +0200 | [diff] [blame] | 114 |   } | 
 | 115 |   prev_packet_timestamp_ = packet_timestamp; | 
| Stefan Holmer | 9131efd | 2016-05-23 18:19:26 +0200 | [diff] [blame] | 116 |   return res; | 
 | 117 | } | 
 | 118 |  | 
| deadbeef | 22e0814 | 2017-06-12 14:30:28 -0700 | [diff] [blame] | 119 | void TestClient::AdvanceTime(int ms) { | 
 | 120 |   // If the test is using a fake clock, we must advance the fake clock to | 
 | 121 |   // advance time. Otherwise, ProcessMessages will work. | 
 | 122 |   if (fake_clock_) { | 
 | 123 |     SIMULATED_WAIT(false, ms, *fake_clock_); | 
 | 124 |   } else { | 
 | 125 |     Thread::Current()->ProcessMessages(1); | 
 | 126 |   } | 
 | 127 | } | 
 | 128 |  | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 129 | bool TestClient::CheckNoPacket() { | 
| nisse | 32f2505 | 2017-05-08 01:57:18 -0700 | [diff] [blame] | 130 |   return NextPacket(kNoPacketTimeoutMs) == nullptr; | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 131 | } | 
 | 132 |  | 
 | 133 | int TestClient::GetError() { | 
 | 134 |   return socket_->GetError(); | 
 | 135 | } | 
 | 136 |  | 
 | 137 | int TestClient::SetOption(Socket::Option opt, int value) { | 
 | 138 |   return socket_->SetOption(opt, value); | 
 | 139 | } | 
 | 140 |  | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 141 | void TestClient::OnPacket(AsyncPacketSocket* socket, const char* buf, | 
 | 142 |                           size_t size, const SocketAddress& remote_addr, | 
 | 143 |                           const PacketTime& packet_time) { | 
 | 144 |   CritScope cs(&crit_); | 
| nisse | 32f2505 | 2017-05-08 01:57:18 -0700 | [diff] [blame] | 145 |   packets_.push_back(MakeUnique<Packet>(remote_addr, buf, size, packet_time)); | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 146 | } | 
 | 147 |  | 
 | 148 | void TestClient::OnReadyToSend(AsyncPacketSocket* socket) { | 
| Taylor Brandstetter | e753641 | 2016-09-09 13:16:15 -0700 | [diff] [blame] | 149 |   ++ready_to_send_count_; | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 150 | } | 
 | 151 |  | 
| Stefan Holmer | 9131efd | 2016-05-23 18:19:26 +0200 | [diff] [blame] | 152 | TestClient::Packet::Packet(const SocketAddress& a, | 
 | 153 |                            const char* b, | 
 | 154 |                            size_t s, | 
 | 155 |                            const PacketTime& packet_time) | 
 | 156 |     : addr(a), buf(0), size(s), packet_time(packet_time) { | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 157 |   buf = new char[size]; | 
 | 158 |   memcpy(buf, b, size); | 
 | 159 | } | 
 | 160 |  | 
 | 161 | TestClient::Packet::Packet(const Packet& p) | 
| Stefan Holmer | 9131efd | 2016-05-23 18:19:26 +0200 | [diff] [blame] | 162 |     : addr(p.addr), buf(0), size(p.size), packet_time(p.packet_time) { | 
| henrike@webrtc.org | f048872 | 2014-05-13 18:00:26 +0000 | [diff] [blame] | 163 |   buf = new char[size]; | 
 | 164 |   memcpy(buf, p.buf, size); | 
 | 165 | } | 
 | 166 |  | 
 | 167 | TestClient::Packet::~Packet() { | 
 | 168 |   delete[] buf; | 
 | 169 | } | 
 | 170 |  | 
 | 171 | }  // namespace rtc |