Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1049)

Side by Side Diff: net/quic/core/quic_utils.cc

Issue 2516033003: Landing Recent QUIC changes until Mon Nov 14 04:43:50 2016 +0000 (Closed)
Patch Set: Remove unused UpdatePacketGapSentHistogram() function. Created 4 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « net/quic/core/quic_utils.h ('k') | net/quic/core/quic_utils_test.cc » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 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 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "net/quic/core/quic_utils.h" 5 #include "net/quic/core/quic_utils.h"
6 6
7 #include <ctype.h> 7 #include <ctype.h>
8 #include <stdint.h> 8 #include <stdint.h>
9 9
10 #include <algorithm> 10 #include <algorithm>
11 #include <vector> 11 #include <vector>
12 12
13 #include "base/containers/adapters.h" 13 #include "base/containers/adapters.h"
14 #include "base/logging.h" 14 #include "base/logging.h"
15 #include "base/strings/string_number_conversions.h" 15 #include "base/strings/string_number_conversions.h"
16 #include "base/strings/string_split.h" 16 #include "base/strings/string_split.h"
17 #include "base/strings/stringprintf.h" 17 #include "base/strings/stringprintf.h"
18 #include "net/base/ip_address.h" 18 #include "net/base/ip_address.h"
19 #include "net/quic/core/quic_flags.h" 19 #include "net/quic/core/quic_flags.h"
20 #include "net/quic/core/quic_write_blocked_list.h"
21 20
22 using base::StringPiece; 21 using base::StringPiece;
23 using std::string; 22 using std::string;
24 23
25 namespace net { 24 namespace net {
26 namespace { 25 namespace {
27 26
28 // We know that >= GCC 4.8 and Clang have a __uint128_t intrinsic. Other 27 // We know that >= GCC 4.8 and Clang have a __uint128_t intrinsic. Other
29 // compilers don't necessarily, notably MSVC. 28 // compilers don't necessarily, notably MSVC.
30 #if defined(__x86_64__) && \ 29 #if defined(__x86_64__) && \
(...skipping 87 matching lines...) Expand 10 before | Expand all | Expand 10 after
118 UINT64_C(7113472399480571277)); 117 UINT64_C(7113472399480571277));
119 118
120 uint128 hash = IncrementalHash(kOffset, data1, len1); 119 uint128 hash = IncrementalHash(kOffset, data1, len1);
121 if (data2 == nullptr) { 120 if (data2 == nullptr) {
122 return hash; 121 return hash;
123 } 122 }
124 return IncrementalHash(hash, data2, len2); 123 return IncrementalHash(hash, data2, len2);
125 } 124 }
126 125
127 // static 126 // static
128 bool QuicUtils::FindMutualTag(const QuicTagVector& our_tags_vector,
129 const QuicTag* their_tags,
130 size_t num_their_tags,
131 Priority priority,
132 QuicTag* out_result,
133 size_t* out_index) {
134 if (our_tags_vector.empty()) {
135 return false;
136 }
137 const size_t num_our_tags = our_tags_vector.size();
138 const QuicTag* our_tags = &our_tags_vector[0];
139
140 size_t num_priority_tags, num_inferior_tags;
141 const QuicTag* priority_tags;
142 const QuicTag* inferior_tags;
143 if (priority == LOCAL_PRIORITY) {
144 num_priority_tags = num_our_tags;
145 priority_tags = our_tags;
146 num_inferior_tags = num_their_tags;
147 inferior_tags = their_tags;
148 } else {
149 num_priority_tags = num_their_tags;
150 priority_tags = their_tags;
151 num_inferior_tags = num_our_tags;
152 inferior_tags = our_tags;
153 }
154
155 for (size_t i = 0; i < num_priority_tags; i++) {
156 for (size_t j = 0; j < num_inferior_tags; j++) {
157 if (priority_tags[i] == inferior_tags[j]) {
158 *out_result = priority_tags[i];
159 if (out_index) {
160 if (priority == LOCAL_PRIORITY) {
161 *out_index = j;
162 } else {
163 *out_index = i;
164 }
165 }
166 return true;
167 }
168 }
169 }
170
171 return false;
172 }
173
174 // static
175 void QuicUtils::SerializeUint128Short(uint128 v, uint8_t* out) { 127 void QuicUtils::SerializeUint128Short(uint128 v, uint8_t* out) {
176 const uint64_t lo = Uint128Low64(v); 128 const uint64_t lo = Uint128Low64(v);
177 const uint64_t hi = Uint128High64(v); 129 const uint64_t hi = Uint128High64(v);
178 // This assumes that the system is little-endian. 130 // This assumes that the system is little-endian.
179 memcpy(out, &lo, sizeof(lo)); 131 memcpy(out, &lo, sizeof(lo));
180 memcpy(out + sizeof(lo), &hi, sizeof(hi) / 2); 132 memcpy(out + sizeof(lo), &hi, sizeof(hi) / 2);
181 } 133 }
182 134
183 #define RETURN_STRING_LITERAL(x) \ 135 #define RETURN_STRING_LITERAL(x) \
184 case x: \ 136 case x: \
185 return #x; 137 return #x;
186 138
187 // static 139 // static
188 const char* QuicUtils::StreamErrorToString(QuicRstStreamErrorCode error) {
189 switch (error) {
190 RETURN_STRING_LITERAL(QUIC_STREAM_NO_ERROR);
191 RETURN_STRING_LITERAL(QUIC_STREAM_CONNECTION_ERROR);
192 RETURN_STRING_LITERAL(QUIC_ERROR_PROCESSING_STREAM);
193 RETURN_STRING_LITERAL(QUIC_MULTIPLE_TERMINATION_OFFSETS);
194 RETURN_STRING_LITERAL(QUIC_BAD_APPLICATION_PAYLOAD);
195 RETURN_STRING_LITERAL(QUIC_STREAM_PEER_GOING_AWAY);
196 RETURN_STRING_LITERAL(QUIC_STREAM_CANCELLED);
197 RETURN_STRING_LITERAL(QUIC_RST_ACKNOWLEDGEMENT);
198 RETURN_STRING_LITERAL(QUIC_REFUSED_STREAM);
199 RETURN_STRING_LITERAL(QUIC_STREAM_LAST_ERROR);
200 RETURN_STRING_LITERAL(QUIC_INVALID_PROMISE_URL);
201 RETURN_STRING_LITERAL(QUIC_UNAUTHORIZED_PROMISE_URL);
202 RETURN_STRING_LITERAL(QUIC_DUPLICATE_PROMISE_URL);
203 RETURN_STRING_LITERAL(QUIC_PROMISE_VARY_MISMATCH);
204 RETURN_STRING_LITERAL(QUIC_INVALID_PROMISE_METHOD);
205 RETURN_STRING_LITERAL(QUIC_PUSH_STREAM_TIMED_OUT);
206 RETURN_STRING_LITERAL(QUIC_HEADERS_TOO_LARGE);
207 }
208 // Return a default value so that we return this when |error| doesn't match
209 // any of the QuicRstStreamErrorCodes. This can happen when the RstStream
210 // frame sent by the peer (attacker) has invalid error code.
211 return "INVALID_RST_STREAM_ERROR_CODE";
212 }
213
214 // static
215 const char* QuicUtils::ErrorToString(QuicErrorCode error) {
216 switch (error) {
217 RETURN_STRING_LITERAL(QUIC_NO_ERROR);
218 RETURN_STRING_LITERAL(QUIC_INTERNAL_ERROR);
219 RETURN_STRING_LITERAL(QUIC_STREAM_DATA_AFTER_TERMINATION);
220 RETURN_STRING_LITERAL(QUIC_INVALID_PACKET_HEADER);
221 RETURN_STRING_LITERAL(QUIC_INVALID_FRAME_DATA);
222 RETURN_STRING_LITERAL(QUIC_MISSING_PAYLOAD);
223 RETURN_STRING_LITERAL(QUIC_INVALID_FEC_DATA);
224 RETURN_STRING_LITERAL(QUIC_INVALID_STREAM_DATA);
225 RETURN_STRING_LITERAL(QUIC_OVERLAPPING_STREAM_DATA);
226 RETURN_STRING_LITERAL(QUIC_UNENCRYPTED_STREAM_DATA);
227 RETURN_STRING_LITERAL(QUIC_INVALID_RST_STREAM_DATA);
228 RETURN_STRING_LITERAL(QUIC_INVALID_CONNECTION_CLOSE_DATA);
229 RETURN_STRING_LITERAL(QUIC_INVALID_GOAWAY_DATA);
230 RETURN_STRING_LITERAL(QUIC_INVALID_WINDOW_UPDATE_DATA);
231 RETURN_STRING_LITERAL(QUIC_INVALID_BLOCKED_DATA);
232 RETURN_STRING_LITERAL(QUIC_INVALID_STOP_WAITING_DATA);
233 RETURN_STRING_LITERAL(QUIC_INVALID_PATH_CLOSE_DATA);
234 RETURN_STRING_LITERAL(QUIC_INVALID_ACK_DATA);
235 RETURN_STRING_LITERAL(QUIC_INVALID_VERSION_NEGOTIATION_PACKET);
236 RETURN_STRING_LITERAL(QUIC_INVALID_PUBLIC_RST_PACKET);
237 RETURN_STRING_LITERAL(QUIC_DECRYPTION_FAILURE);
238 RETURN_STRING_LITERAL(QUIC_ENCRYPTION_FAILURE);
239 RETURN_STRING_LITERAL(QUIC_PACKET_TOO_LARGE);
240 RETURN_STRING_LITERAL(QUIC_PEER_GOING_AWAY);
241 RETURN_STRING_LITERAL(QUIC_HANDSHAKE_FAILED);
242 RETURN_STRING_LITERAL(QUIC_CRYPTO_TAGS_OUT_OF_ORDER);
243 RETURN_STRING_LITERAL(QUIC_CRYPTO_TOO_MANY_ENTRIES);
244 RETURN_STRING_LITERAL(QUIC_CRYPTO_TOO_MANY_REJECTS);
245 RETURN_STRING_LITERAL(QUIC_CRYPTO_INVALID_VALUE_LENGTH)
246 RETURN_STRING_LITERAL(QUIC_CRYPTO_MESSAGE_AFTER_HANDSHAKE_COMPLETE);
247 RETURN_STRING_LITERAL(QUIC_CRYPTO_INTERNAL_ERROR);
248 RETURN_STRING_LITERAL(QUIC_CRYPTO_VERSION_NOT_SUPPORTED);
249 RETURN_STRING_LITERAL(QUIC_CRYPTO_HANDSHAKE_STATELESS_REJECT);
250 RETURN_STRING_LITERAL(QUIC_CRYPTO_NO_SUPPORT);
251 RETURN_STRING_LITERAL(QUIC_INVALID_CRYPTO_MESSAGE_TYPE);
252 RETURN_STRING_LITERAL(QUIC_INVALID_CRYPTO_MESSAGE_PARAMETER);
253 RETURN_STRING_LITERAL(QUIC_CRYPTO_MESSAGE_PARAMETER_NOT_FOUND);
254 RETURN_STRING_LITERAL(QUIC_CRYPTO_MESSAGE_PARAMETER_NO_OVERLAP);
255 RETURN_STRING_LITERAL(QUIC_CRYPTO_MESSAGE_INDEX_NOT_FOUND);
256 RETURN_STRING_LITERAL(QUIC_UNSUPPORTED_PROOF_DEMAND);
257 RETURN_STRING_LITERAL(QUIC_INVALID_STREAM_ID);
258 RETURN_STRING_LITERAL(QUIC_INVALID_PRIORITY);
259 RETURN_STRING_LITERAL(QUIC_TOO_MANY_OPEN_STREAMS);
260 RETURN_STRING_LITERAL(QUIC_PUBLIC_RESET);
261 RETURN_STRING_LITERAL(QUIC_INVALID_VERSION);
262 RETURN_STRING_LITERAL(QUIC_INVALID_HEADER_ID);
263 RETURN_STRING_LITERAL(QUIC_INVALID_NEGOTIATED_VALUE);
264 RETURN_STRING_LITERAL(QUIC_DECOMPRESSION_FAILURE);
265 RETURN_STRING_LITERAL(QUIC_NETWORK_IDLE_TIMEOUT);
266 RETURN_STRING_LITERAL(QUIC_HANDSHAKE_TIMEOUT);
267 RETURN_STRING_LITERAL(QUIC_ERROR_MIGRATING_ADDRESS);
268 RETURN_STRING_LITERAL(QUIC_ERROR_MIGRATING_PORT);
269 RETURN_STRING_LITERAL(QUIC_PACKET_WRITE_ERROR);
270 RETURN_STRING_LITERAL(QUIC_PACKET_READ_ERROR);
271 RETURN_STRING_LITERAL(QUIC_EMPTY_STREAM_FRAME_NO_FIN);
272 RETURN_STRING_LITERAL(QUIC_INVALID_HEADERS_STREAM_DATA);
273 RETURN_STRING_LITERAL(QUIC_FLOW_CONTROL_RECEIVED_TOO_MUCH_DATA);
274 RETURN_STRING_LITERAL(QUIC_FLOW_CONTROL_SENT_TOO_MUCH_DATA);
275 RETURN_STRING_LITERAL(QUIC_FLOW_CONTROL_INVALID_WINDOW);
276 RETURN_STRING_LITERAL(QUIC_CONNECTION_IP_POOLED);
277 RETURN_STRING_LITERAL(QUIC_PROOF_INVALID);
278 RETURN_STRING_LITERAL(QUIC_CRYPTO_DUPLICATE_TAG);
279 RETURN_STRING_LITERAL(QUIC_CRYPTO_ENCRYPTION_LEVEL_INCORRECT);
280 RETURN_STRING_LITERAL(QUIC_CRYPTO_SERVER_CONFIG_EXPIRED);
281 RETURN_STRING_LITERAL(QUIC_INVALID_CHANNEL_ID_SIGNATURE);
282 RETURN_STRING_LITERAL(QUIC_CRYPTO_SYMMETRIC_KEY_SETUP_FAILED);
283 RETURN_STRING_LITERAL(QUIC_CRYPTO_MESSAGE_WHILE_VALIDATING_CLIENT_HELLO);
284 RETURN_STRING_LITERAL(QUIC_CRYPTO_UPDATE_BEFORE_HANDSHAKE_COMPLETE);
285 RETURN_STRING_LITERAL(QUIC_VERSION_NEGOTIATION_MISMATCH);
286 RETURN_STRING_LITERAL(QUIC_TOO_MANY_OUTSTANDING_SENT_PACKETS);
287 RETURN_STRING_LITERAL(QUIC_TOO_MANY_OUTSTANDING_RECEIVED_PACKETS);
288 RETURN_STRING_LITERAL(QUIC_CONNECTION_CANCELLED);
289 RETURN_STRING_LITERAL(QUIC_BAD_PACKET_LOSS_RATE);
290 RETURN_STRING_LITERAL(QUIC_PUBLIC_RESETS_POST_HANDSHAKE);
291 RETURN_STRING_LITERAL(QUIC_TIMEOUTS_WITH_OPEN_STREAMS);
292 RETURN_STRING_LITERAL(QUIC_FAILED_TO_SERIALIZE_PACKET);
293 RETURN_STRING_LITERAL(QUIC_TOO_MANY_AVAILABLE_STREAMS);
294 RETURN_STRING_LITERAL(QUIC_UNENCRYPTED_FEC_DATA);
295 RETURN_STRING_LITERAL(QUIC_BAD_MULTIPATH_FLAG);
296 RETURN_STRING_LITERAL(QUIC_IP_ADDRESS_CHANGED);
297 RETURN_STRING_LITERAL(QUIC_CONNECTION_MIGRATION_NO_MIGRATABLE_STREAMS);
298 RETURN_STRING_LITERAL(QUIC_CONNECTION_MIGRATION_TOO_MANY_CHANGES);
299 RETURN_STRING_LITERAL(QUIC_CONNECTION_MIGRATION_NO_NEW_NETWORK);
300 RETURN_STRING_LITERAL(QUIC_CONNECTION_MIGRATION_NON_MIGRATABLE_STREAM);
301 RETURN_STRING_LITERAL(QUIC_TOO_MANY_RTOS);
302 RETURN_STRING_LITERAL(QUIC_ATTEMPT_TO_SEND_UNENCRYPTED_STREAM_DATA);
303 RETURN_STRING_LITERAL(QUIC_MAYBE_CORRUPTED_MEMORY);
304 RETURN_STRING_LITERAL(QUIC_CRYPTO_CHLO_TOO_LARGE);
305 RETURN_STRING_LITERAL(QUIC_MULTIPATH_PATH_DOES_NOT_EXIST);
306 RETURN_STRING_LITERAL(QUIC_MULTIPATH_PATH_NOT_ACTIVE);
307 RETURN_STRING_LITERAL(QUIC_TOO_MANY_FRAME_GAPS);
308 RETURN_STRING_LITERAL(QUIC_STREAM_SEQUENCER_INVALID_STATE);
309 RETURN_STRING_LITERAL(QUIC_TOO_MANY_SESSIONS_ON_SERVER);
310 RETURN_STRING_LITERAL(QUIC_LAST_ERROR);
311 // Intentionally have no default case, so we'll break the build
312 // if we add errors and don't put them here.
313 }
314 // Return a default value so that we return this when |error| doesn't match
315 // any of the QuicErrorCodes. This can happen when the ConnectionClose
316 // frame sent by the peer (attacker) has invalid error code.
317 return "INVALID_ERROR_CODE";
318 }
319
320 // static
321 const char* QuicUtils::EncryptionLevelToString(EncryptionLevel level) { 140 const char* QuicUtils::EncryptionLevelToString(EncryptionLevel level) {
322 switch (level) { 141 switch (level) {
323 RETURN_STRING_LITERAL(ENCRYPTION_NONE); 142 RETURN_STRING_LITERAL(ENCRYPTION_NONE);
324 RETURN_STRING_LITERAL(ENCRYPTION_INITIAL); 143 RETURN_STRING_LITERAL(ENCRYPTION_INITIAL);
325 RETURN_STRING_LITERAL(ENCRYPTION_FORWARD_SECURE); 144 RETURN_STRING_LITERAL(ENCRYPTION_FORWARD_SECURE);
326 RETURN_STRING_LITERAL(NUM_ENCRYPTION_LEVELS); 145 RETURN_STRING_LITERAL(NUM_ENCRYPTION_LEVELS);
327 } 146 }
328 return "INVALID_ENCRYPTION_LEVEL"; 147 return "INVALID_ENCRYPTION_LEVEL";
329 } 148 }
330 149
331 // static 150 // static
332 const char* QuicUtils::TransmissionTypeToString(TransmissionType type) { 151 const char* QuicUtils::TransmissionTypeToString(TransmissionType type) {
333 switch (type) { 152 switch (type) {
334 RETURN_STRING_LITERAL(NOT_RETRANSMISSION); 153 RETURN_STRING_LITERAL(NOT_RETRANSMISSION);
335 RETURN_STRING_LITERAL(HANDSHAKE_RETRANSMISSION); 154 RETURN_STRING_LITERAL(HANDSHAKE_RETRANSMISSION);
336 RETURN_STRING_LITERAL(LOSS_RETRANSMISSION); 155 RETURN_STRING_LITERAL(LOSS_RETRANSMISSION);
337 RETURN_STRING_LITERAL(ALL_UNACKED_RETRANSMISSION); 156 RETURN_STRING_LITERAL(ALL_UNACKED_RETRANSMISSION);
338 RETURN_STRING_LITERAL(ALL_INITIAL_RETRANSMISSION); 157 RETURN_STRING_LITERAL(ALL_INITIAL_RETRANSMISSION);
339 RETURN_STRING_LITERAL(RTO_RETRANSMISSION); 158 RETURN_STRING_LITERAL(RTO_RETRANSMISSION);
340 RETURN_STRING_LITERAL(TLP_RETRANSMISSION); 159 RETURN_STRING_LITERAL(TLP_RETRANSMISSION);
341 } 160 }
342 return "INVALID_TRANSMISSION_TYPE"; 161 return "INVALID_TRANSMISSION_TYPE";
343 } 162 }
344 163
345 // static 164 // static
346 string QuicUtils::TagToString(QuicTag tag) {
347 char chars[sizeof tag];
348 bool ascii = true;
349 const QuicTag orig_tag = tag;
350
351 for (size_t i = 0; i < arraysize(chars); i++) {
352 chars[i] = static_cast<char>(tag);
353 if ((chars[i] == 0 || chars[i] == '\xff') && i == arraysize(chars) - 1) {
354 chars[i] = ' ';
355 }
356 if (!isprint(static_cast<unsigned char>(chars[i]))) {
357 ascii = false;
358 break;
359 }
360 tag >>= 8;
361 }
362
363 if (ascii) {
364 return string(chars, sizeof(chars));
365 }
366
367 return base::UintToString(orig_tag);
368 }
369
370 // static
371 QuicTagVector QuicUtils::ParseQuicConnectionOptions( 165 QuicTagVector QuicUtils::ParseQuicConnectionOptions(
372 const std::string& connection_options) { 166 const std::string& connection_options) {
373 QuicTagVector options; 167 QuicTagVector options;
374 // Tokens are expected to be no more than 4 characters long, but we 168 // Tokens are expected to be no more than 4 characters long, but we
375 // handle overflow gracefully. 169 // handle overflow gracefully.
376 for (const base::StringPiece& token : 170 for (const base::StringPiece& token :
377 base::SplitStringPiece(connection_options, ",", base::TRIM_WHITESPACE, 171 base::SplitStringPiece(connection_options, ",", base::TRIM_WHITESPACE,
378 base::SPLIT_WANT_ALL)) { 172 base::SPLIT_WANT_ALL)) {
379 uint32_t option = 0; 173 uint32_t option = 0;
380 for (char token_char : base::Reversed(token)) { 174 for (char token_char : base::Reversed(token)) {
(...skipping 140 matching lines...) Expand 10 before | Expand all | Expand 10 after
521 } 315 }
522 316
523 string QuicUtils::HexEncode(const char* data, size_t length) { 317 string QuicUtils::HexEncode(const char* data, size_t length) {
524 return HexEncode(StringPiece(data, length)); 318 return HexEncode(StringPiece(data, length));
525 } 319 }
526 320
527 string QuicUtils::HexEncode(StringPiece data) { 321 string QuicUtils::HexEncode(StringPiece data) {
528 return ::base::HexEncode(data.data(), data.size()); 322 return ::base::HexEncode(data.data(), data.size());
529 } 323 }
530 324
531 string QuicUtils::HexDecode(const char* data, size_t length) {
532 return HexDecode(StringPiece(data, length));
533 }
534
535 string QuicUtils::HexDecode(StringPiece data) { 325 string QuicUtils::HexDecode(StringPiece data) {
536 if (data.empty()) 326 if (data.empty())
537 return ""; 327 return "";
538 std::vector<uint8_t> v; 328 std::vector<uint8_t> v;
539 if (!base::HexStringToBytes(data.as_string(), &v)) 329 if (!base::HexStringToBytes(data.as_string(), &v))
540 return ""; 330 return "";
541 string out; 331 string out;
542 if (!v.empty()) 332 if (!v.empty())
543 out.assign(reinterpret_cast<const char*>(&v[0]), v.size()); 333 out.assign(reinterpret_cast<const char*>(&v[0]), v.size());
544 return out; 334 return out;
(...skipping 25 matching lines...) Expand all
570 360
571 bytes_remaining -= line_bytes; 361 bytes_remaining -= line_bytes;
572 offset += line_bytes; 362 offset += line_bytes;
573 p += line_bytes; 363 p += line_bytes;
574 s += '\n'; 364 s += '\n';
575 } 365 }
576 return s; 366 return s;
577 } 367 }
578 368
579 } // namespace net 369 } // namespace net
OLDNEW
« no previous file with comments | « net/quic/core/quic_utils.h ('k') | net/quic/core/quic_utils_test.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698