blob: 54c3b57862bfa0ceb488166c6cac57762b0f0fe6 [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"
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
21namespace quic {
22namespace simulator {
23
24// Takes a SHA-1 hash of the name and converts it into five 32-bit integers.
25static 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
41QuicSocketAddress 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
57QuicEndpointBase::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
72QuicEndpointBase::~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());
wub7e3fe6b2019-12-06 10:10:54 -080079 QuicRecordTrace(identifier, trace_visitor_->trace()->SerializeAsString());
vasilvv4db9be52019-10-22 15:17:21 -070080 }
81}
82
83void QuicEndpointBase::DropNextIncomingPacket() {
84 drop_next_packet_ = true;
85}
86
87void QuicEndpointBase::RecordTrace() {
88 trace_visitor_ = std::make_unique<QuicTraceVisitor>(connection_.get());
89 connection_->set_debug_visitor(trace_visitor_.get());
90}
91
92void 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
107UnconstrainedPortInterface* QuicEndpointBase::GetRxPort() {
108 return this;
109}
110
111void 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
116void 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
125QuicEndpointBase::Writer::Writer(QuicEndpointBase* endpoint)
126 : endpoint_(endpoint), is_blocked_(false) {}
127
128QuicEndpointBase::Writer::~Writer() {}
129
130WriteResult 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
161bool QuicEndpointBase::Writer::IsWriteBlocked() const {
162 return is_blocked_;
163}
164
165void QuicEndpointBase::Writer::SetWritable() {
166 is_blocked_ = false;
167}
168
169QuicByteCount QuicEndpointBase::Writer::GetMaxPacketSize(
170 const QuicSocketAddress& /*peer_address*/) const {
171 return kMaxOutgoingPacketSize;
172}
173
174bool QuicEndpointBase::Writer::SupportsReleaseTime() const {
175 return false;
176}
177
178bool QuicEndpointBase::Writer::IsBatchMode() const {
179 return false;
180}
181
182char* QuicEndpointBase::Writer::GetNextWriteLocation(
183 const QuicIpAddress& /*self_address*/,
184 const QuicSocketAddress& /*peer_address*/) {
185 return nullptr;
186}
187
188WriteResult QuicEndpointBase::Writer::Flush() {
189 return WriteResult(WRITE_STATUS_OK, 0);
190}
191
192QuicEndpointMultiplexer::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
201QuicEndpointMultiplexer::~QuicEndpointMultiplexer() {}
202
203void 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}
211UnconstrainedPortInterface* QuicEndpointMultiplexer::GetRxPort() {
212 return this;
213}
214void 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