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