blob: 21815d8fe31525ee6572b9decb34a6592a113a4a [file] [log] [blame]
vasilvv4db9be52019-10-22 15:17:21 -07001// 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"
vasilvv4db9be52019-10-22 15:17:21 -070014#include "net/third_party/quiche/src/quic/platform/api/quic_test_output.h"
vasilvv4db9be52019-10-22 15:17:21 -070015#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 team6dcf6ab2019-12-11 10:10:51 -080018#include "net/third_party/quiche/src/common/platform/api/quiche_str_cat.h"
vasilvv4db9be52019-10-22 15:17:21 -070019
20namespace quic {
21namespace simulator {
22
23// Takes a SHA-1 hash of the name and converts it into five 32-bit integers.
24static 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
40QuicSocketAddress 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
56QuicEndpointBase::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 team6dcf6ab2019-12-11 10:10:51 -080063 quiche::QuicheStringPrintf("%s (TX Queue)", name.c_str()),
vasilvv4db9be52019-10-22 15:17:21 -070064 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
71QuicEndpointBase::~QuicEndpointBase() {
72 if (trace_visitor_ != nullptr) {
73 const char* perspective_prefix =
74 connection_->perspective() == Perspective::IS_CLIENT ? "C" : "S";
75
QUICHE team6dcf6ab2019-12-11 10:10:51 -080076 std::string identifier = quiche::QuicheStrCat(
77 perspective_prefix, connection_->connection_id().ToString());
wub7e3fe6b2019-12-06 10:10:54 -080078 QuicRecordTrace(identifier, trace_visitor_->trace()->SerializeAsString());
vasilvv4db9be52019-10-22 15:17:21 -070079 }
80}
81
82void QuicEndpointBase::DropNextIncomingPacket() {
83 drop_next_packet_ = true;
84}
85
86void QuicEndpointBase::RecordTrace() {
87 trace_visitor_ = std::make_unique<QuicTraceVisitor>(connection_.get());
88 connection_->set_debug_visitor(trace_visitor_.get());
89}
90
91void 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
106UnconstrainedPortInterface* QuicEndpointBase::GetRxPort() {
107 return this;
108}
109
110void 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
115void 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
124QuicEndpointBase::Writer::Writer(QuicEndpointBase* endpoint)
125 : endpoint_(endpoint), is_blocked_(false) {}
126
127QuicEndpointBase::Writer::~Writer() {}
128
129WriteResult 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
160bool QuicEndpointBase::Writer::IsWriteBlocked() const {
161 return is_blocked_;
162}
163
164void QuicEndpointBase::Writer::SetWritable() {
165 is_blocked_ = false;
166}
167
168QuicByteCount QuicEndpointBase::Writer::GetMaxPacketSize(
169 const QuicSocketAddress& /*peer_address*/) const {
170 return kMaxOutgoingPacketSize;
171}
172
173bool QuicEndpointBase::Writer::SupportsReleaseTime() const {
174 return false;
175}
176
177bool QuicEndpointBase::Writer::IsBatchMode() const {
178 return false;
179}
180
181char* QuicEndpointBase::Writer::GetNextWriteLocation(
182 const QuicIpAddress& /*self_address*/,
183 const QuicSocketAddress& /*peer_address*/) {
184 return nullptr;
185}
186
187WriteResult QuicEndpointBase::Writer::Flush() {
188 return WriteResult(WRITE_STATUS_OK, 0);
189}
190
191QuicEndpointMultiplexer::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
200QuicEndpointMultiplexer::~QuicEndpointMultiplexer() {}
201
202void 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}
210UnconstrainedPortInterface* QuicEndpointMultiplexer::GetRxPort() {
211 return this;
212}
213void 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