OLD | NEW |
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 Loading... |
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 Loading... |
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 Loading... |
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 |
OLD | NEW |