vasilvv | 4db9be5 | 2019-10-22 15:17:21 -0700 | [diff] [blame] | 1 | // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
| 5 | #include "net/third_party/quiche/src/quic/test_tools/simulator/quic_endpoint_base.h" |
| 6 | |
| 7 | #include <memory> |
| 8 | #include <utility> |
| 9 | |
| 10 | #include "net/third_party/quiche/src/quic/core/crypto/crypto_handshake_message.h" |
| 11 | #include "net/third_party/quiche/src/quic/core/crypto/crypto_protocol.h" |
| 12 | #include "net/third_party/quiche/src/quic/core/quic_connection.h" |
| 13 | #include "net/third_party/quiche/src/quic/core/quic_data_writer.h" |
vasilvv | 4db9be5 | 2019-10-22 15:17:21 -0700 | [diff] [blame] | 14 | #include "net/third_party/quiche/src/quic/platform/api/quic_test_output.h" |
vasilvv | 4db9be5 | 2019-10-22 15:17:21 -0700 | [diff] [blame] | 15 | #include "net/third_party/quiche/src/quic/test_tools/quic_connection_peer.h" |
| 16 | #include "net/third_party/quiche/src/quic/test_tools/quic_test_utils.h" |
| 17 | #include "net/third_party/quiche/src/quic/test_tools/simulator/simulator.h" |
QUICHE team | 6dcf6ab | 2019-12-11 10:10:51 -0800 | [diff] [blame] | 18 | #include "net/third_party/quiche/src/common/platform/api/quiche_str_cat.h" |
vasilvv | 4db9be5 | 2019-10-22 15:17:21 -0700 | [diff] [blame] | 19 | |
| 20 | namespace quic { |
| 21 | namespace simulator { |
| 22 | |
| 23 | // Takes a SHA-1 hash of the name and converts it into five 32-bit integers. |
| 24 | static std::vector<uint32_t> HashNameIntoFive32BitIntegers(std::string name) { |
| 25 | const std::string hash = test::Sha1Hash(name); |
| 26 | |
| 27 | std::vector<uint32_t> output; |
| 28 | uint32_t current_number = 0; |
| 29 | for (size_t i = 0; i < hash.size(); i++) { |
| 30 | current_number = (current_number << 8) + hash[i]; |
| 31 | if (i % 4 == 3) { |
| 32 | output.push_back(i); |
| 33 | current_number = 0; |
| 34 | } |
| 35 | } |
| 36 | |
| 37 | return output; |
| 38 | } |
| 39 | |
| 40 | QuicSocketAddress GetAddressFromName(std::string name) { |
| 41 | const std::vector<uint32_t> hash = HashNameIntoFive32BitIntegers(name); |
| 42 | |
| 43 | // Generate a random port between 1025 and 65535. |
| 44 | const uint16_t port = 1025 + hash[0] % (65535 - 1025 + 1); |
| 45 | |
| 46 | // Generate a random 10.x.x.x address, where x is between 1 and 254. |
| 47 | std::string ip_address{"\xa\0\0\0", 4}; |
| 48 | for (size_t i = 1; i < 4; i++) { |
| 49 | ip_address[i] = 1 + hash[i] % 254; |
| 50 | } |
| 51 | QuicIpAddress host; |
| 52 | host.FromPackedString(ip_address.c_str(), ip_address.length()); |
| 53 | return QuicSocketAddress(host, port); |
| 54 | } |
| 55 | |
| 56 | QuicEndpointBase::QuicEndpointBase(Simulator* simulator, |
| 57 | std::string name, |
| 58 | std::string peer_name) |
| 59 | : Endpoint(simulator, name), |
| 60 | peer_name_(peer_name), |
| 61 | writer_(this), |
| 62 | nic_tx_queue_(simulator, |
QUICHE team | 6dcf6ab | 2019-12-11 10:10:51 -0800 | [diff] [blame] | 63 | quiche::QuicheStringPrintf("%s (TX Queue)", name.c_str()), |
vasilvv | 4db9be5 | 2019-10-22 15:17:21 -0700 | [diff] [blame] | 64 | kMaxOutgoingPacketSize * kTxQueueSize), |
| 65 | connection_(nullptr), |
| 66 | write_blocked_count_(0), |
| 67 | drop_next_packet_(false) { |
| 68 | nic_tx_queue_.set_listener_interface(this); |
| 69 | } |
| 70 | |
| 71 | QuicEndpointBase::~QuicEndpointBase() { |
| 72 | if (trace_visitor_ != nullptr) { |
| 73 | const char* perspective_prefix = |
| 74 | connection_->perspective() == Perspective::IS_CLIENT ? "C" : "S"; |
| 75 | |
QUICHE team | 6dcf6ab | 2019-12-11 10:10:51 -0800 | [diff] [blame] | 76 | std::string identifier = quiche::QuicheStrCat( |
| 77 | perspective_prefix, connection_->connection_id().ToString()); |
wub | 7e3fe6b | 2019-12-06 10:10:54 -0800 | [diff] [blame] | 78 | QuicRecordTrace(identifier, trace_visitor_->trace()->SerializeAsString()); |
vasilvv | 4db9be5 | 2019-10-22 15:17:21 -0700 | [diff] [blame] | 79 | } |
| 80 | } |
| 81 | |
| 82 | void QuicEndpointBase::DropNextIncomingPacket() { |
| 83 | drop_next_packet_ = true; |
| 84 | } |
| 85 | |
| 86 | void QuicEndpointBase::RecordTrace() { |
| 87 | trace_visitor_ = std::make_unique<QuicTraceVisitor>(connection_.get()); |
| 88 | connection_->set_debug_visitor(trace_visitor_.get()); |
| 89 | } |
| 90 | |
| 91 | void QuicEndpointBase::AcceptPacket(std::unique_ptr<Packet> packet) { |
| 92 | if (packet->destination != name_) { |
| 93 | return; |
| 94 | } |
| 95 | if (drop_next_packet_) { |
| 96 | drop_next_packet_ = false; |
| 97 | return; |
| 98 | } |
| 99 | |
| 100 | QuicReceivedPacket received_packet(packet->contents.data(), |
| 101 | packet->contents.size(), clock_->Now()); |
| 102 | connection_->ProcessUdpPacket(connection_->self_address(), |
| 103 | connection_->peer_address(), received_packet); |
| 104 | } |
| 105 | |
| 106 | UnconstrainedPortInterface* QuicEndpointBase::GetRxPort() { |
| 107 | return this; |
| 108 | } |
| 109 | |
| 110 | void QuicEndpointBase::SetTxPort(ConstrainedPortInterface* port) { |
| 111 | // Any egress done by the endpoint is actually handled by a queue on an NIC. |
| 112 | nic_tx_queue_.set_tx_port(port); |
| 113 | } |
| 114 | |
| 115 | void QuicEndpointBase::OnPacketDequeued() { |
| 116 | if (writer_.IsWriteBlocked() && |
| 117 | (nic_tx_queue_.capacity() - nic_tx_queue_.bytes_queued()) >= |
| 118 | kMaxOutgoingPacketSize) { |
| 119 | writer_.SetWritable(); |
| 120 | connection_->OnCanWrite(); |
| 121 | } |
| 122 | } |
| 123 | |
| 124 | QuicEndpointBase::Writer::Writer(QuicEndpointBase* endpoint) |
| 125 | : endpoint_(endpoint), is_blocked_(false) {} |
| 126 | |
| 127 | QuicEndpointBase::Writer::~Writer() {} |
| 128 | |
| 129 | WriteResult QuicEndpointBase::Writer::WritePacket( |
| 130 | const char* buffer, |
| 131 | size_t buf_len, |
| 132 | const QuicIpAddress& /*self_address*/, |
| 133 | const QuicSocketAddress& /*peer_address*/, |
| 134 | PerPacketOptions* options) { |
| 135 | DCHECK(!IsWriteBlocked()); |
| 136 | DCHECK(options == nullptr); |
| 137 | DCHECK(buf_len <= kMaxOutgoingPacketSize); |
| 138 | |
| 139 | // Instead of losing a packet, become write-blocked when the egress queue is |
| 140 | // full. |
| 141 | if (endpoint_->nic_tx_queue_.packets_queued() > kTxQueueSize) { |
| 142 | is_blocked_ = true; |
| 143 | endpoint_->write_blocked_count_++; |
| 144 | return WriteResult(WRITE_STATUS_BLOCKED, 0); |
| 145 | } |
| 146 | |
| 147 | auto packet = std::make_unique<Packet>(); |
| 148 | packet->source = endpoint_->name(); |
| 149 | packet->destination = endpoint_->peer_name_; |
| 150 | packet->tx_timestamp = endpoint_->clock_->Now(); |
| 151 | |
| 152 | packet->contents = std::string(buffer, buf_len); |
| 153 | packet->size = buf_len; |
| 154 | |
| 155 | endpoint_->nic_tx_queue_.AcceptPacket(std::move(packet)); |
| 156 | |
| 157 | return WriteResult(WRITE_STATUS_OK, buf_len); |
| 158 | } |
| 159 | |
| 160 | bool QuicEndpointBase::Writer::IsWriteBlocked() const { |
| 161 | return is_blocked_; |
| 162 | } |
| 163 | |
| 164 | void QuicEndpointBase::Writer::SetWritable() { |
| 165 | is_blocked_ = false; |
| 166 | } |
| 167 | |
| 168 | QuicByteCount QuicEndpointBase::Writer::GetMaxPacketSize( |
| 169 | const QuicSocketAddress& /*peer_address*/) const { |
| 170 | return kMaxOutgoingPacketSize; |
| 171 | } |
| 172 | |
| 173 | bool QuicEndpointBase::Writer::SupportsReleaseTime() const { |
| 174 | return false; |
| 175 | } |
| 176 | |
| 177 | bool QuicEndpointBase::Writer::IsBatchMode() const { |
| 178 | return false; |
| 179 | } |
| 180 | |
| 181 | char* QuicEndpointBase::Writer::GetNextWriteLocation( |
| 182 | const QuicIpAddress& /*self_address*/, |
| 183 | const QuicSocketAddress& /*peer_address*/) { |
| 184 | return nullptr; |
| 185 | } |
| 186 | |
| 187 | WriteResult QuicEndpointBase::Writer::Flush() { |
| 188 | return WriteResult(WRITE_STATUS_OK, 0); |
| 189 | } |
| 190 | |
| 191 | QuicEndpointMultiplexer::QuicEndpointMultiplexer( |
| 192 | std::string name, |
| 193 | const std::vector<QuicEndpointBase*>& endpoints) |
| 194 | : Endpoint((*endpoints.begin())->simulator(), name) { |
| 195 | for (QuicEndpointBase* endpoint : endpoints) { |
| 196 | mapping_.insert(std::make_pair(endpoint->name(), endpoint)); |
| 197 | } |
| 198 | } |
| 199 | |
| 200 | QuicEndpointMultiplexer::~QuicEndpointMultiplexer() {} |
| 201 | |
| 202 | void QuicEndpointMultiplexer::AcceptPacket(std::unique_ptr<Packet> packet) { |
| 203 | auto key_value_pair_it = mapping_.find(packet->destination); |
| 204 | if (key_value_pair_it == mapping_.end()) { |
| 205 | return; |
| 206 | } |
| 207 | |
| 208 | key_value_pair_it->second->GetRxPort()->AcceptPacket(std::move(packet)); |
| 209 | } |
| 210 | UnconstrainedPortInterface* QuicEndpointMultiplexer::GetRxPort() { |
| 211 | return this; |
| 212 | } |
| 213 | void QuicEndpointMultiplexer::SetTxPort(ConstrainedPortInterface* port) { |
| 214 | for (auto& key_value_pair : mapping_) { |
| 215 | key_value_pair.second->SetTxPort(port); |
| 216 | } |
| 217 | } |
| 218 | |
| 219 | } // namespace simulator |
| 220 | } // namespace quic |