Index: net/quic/quic_framer.cc |
diff --git a/net/quic/quic_framer.cc b/net/quic/quic_framer.cc |
deleted file mode 100644 |
index 7f4ae71736bda615cbb275129ba46b9e83666559..0000000000000000000000000000000000000000 |
--- a/net/quic/quic_framer.cc |
+++ /dev/null |
@@ -1,2715 +0,0 @@ |
-// Copyright (c) 2012 The Chromium Authors. All rights reserved. |
-// Use of this source code is governed by a BSD-style license that can be |
-// found in the LICENSE file. |
- |
-#include "net/quic/quic_framer.h" |
- |
-#include <cstdint> |
-#include <memory> |
-#include <vector> |
- |
-#include "base/compiler_specific.h" |
-#include "base/logging.h" |
-#include "base/stl_util.h" |
-#include "net/quic/crypto/crypto_framer.h" |
-#include "net/quic/crypto/crypto_handshake_message.h" |
-#include "net/quic/crypto/crypto_protocol.h" |
-#include "net/quic/crypto/quic_decrypter.h" |
-#include "net/quic/crypto/quic_encrypter.h" |
-#include "net/quic/quic_bug_tracker.h" |
-#include "net/quic/quic_data_reader.h" |
-#include "net/quic/quic_data_writer.h" |
-#include "net/quic/quic_flags.h" |
-#include "net/quic/quic_socket_address_coder.h" |
-#include "net/quic/quic_utils.h" |
- |
-using base::StringPiece; |
-using std::map; |
-using std::max; |
-using std::min; |
-using std::numeric_limits; |
-using std::string; |
-using std::vector; |
-#define PREDICT_FALSE(x) (x) |
- |
-namespace net { |
- |
-namespace { |
- |
-// Mask to select the lowest 48 bits of a packet number. |
-const QuicPacketNumber k6ByteSequenceNumberMask = UINT64_C(0x0000FFFFFFFFFFFF); |
-const QuicPacketNumber k4ByteSequenceNumberMask = UINT64_C(0x00000000FFFFFFFF); |
-const QuicPacketNumber k2ByteSequenceNumberMask = UINT64_C(0x000000000000FFFF); |
-const QuicPacketNumber k1ByteSequenceNumberMask = UINT64_C(0x00000000000000FF); |
- |
-// Number of bits the packet number length bits are shifted from the right |
-// edge of the public header. |
-const uint8_t kPublicHeaderSequenceNumberShift = 4; |
- |
-// New Frame Types, QUIC v. >= 10: |
-// There are two interpretations for the Frame Type byte in the QUIC protocol, |
-// resulting in two Frame Types: Special Frame Types and Regular Frame Types. |
-// |
-// Regular Frame Types use the Frame Type byte simply. Currently defined |
-// Regular Frame Types are: |
-// Padding : 0b 00000000 (0x00) |
-// ResetStream : 0b 00000001 (0x01) |
-// ConnectionClose : 0b 00000010 (0x02) |
-// GoAway : 0b 00000011 (0x03) |
-// WindowUpdate : 0b 00000100 (0x04) |
-// Blocked : 0b 00000101 (0x05) |
-// |
-// Special Frame Types encode both a Frame Type and corresponding flags |
-// all in the Frame Type byte. Currently defined Special Frame Types are: |
-// Stream : 0b 1xxxxxxx |
-// Ack : 0b 01xxxxxx |
-// |
-// Semantics of the flag bits above (the x bits) depends on the frame type. |
- |
-// Masks to determine if the frame type is a special use |
-// and for specific special frame types. |
-const uint8_t kQuicFrameTypeSpecialMask = 0xE0; // 0b 11100000 |
-const uint8_t kQuicFrameTypeStreamMask = 0x80; |
-const uint8_t kQuicFrameTypeAckMask = 0x40; |
- |
-// Stream frame relative shifts and masks for interpreting the stream flags. |
-// StreamID may be 1, 2, 3, or 4 bytes. |
-const uint8_t kQuicStreamIdShift = 2; |
-const uint8_t kQuicStreamIDLengthMask = 0x03; |
- |
-// Offset may be 0, 2, 3, 4, 5, 6, 7, 8 bytes. |
-const uint8_t kQuicStreamOffsetShift = 3; |
-const uint8_t kQuicStreamOffsetMask = 0x07; |
- |
-// Data length may be 0 or 2 bytes. |
-const uint8_t kQuicStreamDataLengthShift = 1; |
-const uint8_t kQuicStreamDataLengthMask = 0x01; |
- |
-// Fin bit may be set or not. |
-const uint8_t kQuicStreamFinShift = 1; |
-const uint8_t kQuicStreamFinMask = 0x01; |
- |
-// packet number size shift used in AckFrames. |
-const uint8_t kQuicSequenceNumberLengthShift = 2; |
- |
-// Acks may be truncated. |
-const uint8_t kQuicAckTruncatedShift = 1; |
-const uint8_t kQuicAckTruncatedMask = 0x01; |
- |
-// Acks may not have any nacks. |
-const uint8_t kQuicHasNacksMask = 0x01; |
-// Acks may have only one ack block. |
-const uint8_t kQuicHasMultipleAckBlocksMask = 0x01; |
-const uint8_t kQuicHasMultipleAckBlocksShift = 1; |
- |
-// Returns the absolute value of the difference between |a| and |b|. |
-QuicPacketNumber Delta(QuicPacketNumber a, QuicPacketNumber b) { |
- // Since these are unsigned numbers, we can't just return abs(a - b) |
- if (a < b) { |
- return b - a; |
- } |
- return a - b; |
-} |
- |
-QuicPacketNumber ClosestTo(QuicPacketNumber target, |
- QuicPacketNumber a, |
- QuicPacketNumber b) { |
- return (Delta(target, a) < Delta(target, b)) ? a : b; |
-} |
- |
-QuicPacketNumberLength ReadSequenceNumberLength(uint8_t flags) { |
- switch (flags & PACKET_FLAGS_6BYTE_PACKET) { |
- case PACKET_FLAGS_6BYTE_PACKET: |
- return PACKET_6BYTE_PACKET_NUMBER; |
- case PACKET_FLAGS_4BYTE_PACKET: |
- return PACKET_4BYTE_PACKET_NUMBER; |
- case PACKET_FLAGS_2BYTE_PACKET: |
- return PACKET_2BYTE_PACKET_NUMBER; |
- case PACKET_FLAGS_1BYTE_PACKET: |
- return PACKET_1BYTE_PACKET_NUMBER; |
- default: |
- QUIC_BUG << "Unreachable case statement."; |
- return PACKET_6BYTE_PACKET_NUMBER; |
- } |
-} |
- |
-} // namespace |
- |
-QuicFramer::QuicFramer(const QuicVersionVector& supported_versions, |
- QuicTime creation_time, |
- Perspective perspective) |
- : visitor_(nullptr), |
- entropy_calculator_(nullptr), |
- error_(QUIC_NO_ERROR), |
- last_packet_number_(0), |
- last_path_id_(kInvalidPathId), |
- last_serialized_connection_id_(0), |
- supported_versions_(supported_versions), |
- decrypter_level_(ENCRYPTION_NONE), |
- alternative_decrypter_level_(ENCRYPTION_NONE), |
- alternative_decrypter_latch_(false), |
- perspective_(perspective), |
- validate_flags_(true), |
- creation_time_(creation_time), |
- last_timestamp_(QuicTime::Delta::Zero()) { |
- DCHECK(!supported_versions.empty()); |
- quic_version_ = supported_versions_[0]; |
- decrypter_.reset(QuicDecrypter::Create(kNULL)); |
- encrypter_[ENCRYPTION_NONE].reset(QuicEncrypter::Create(kNULL)); |
-} |
- |
-QuicFramer::~QuicFramer() {} |
- |
-// static |
-size_t QuicFramer::GetMinStreamFrameSize(QuicStreamId stream_id, |
- QuicStreamOffset offset, |
- bool last_frame_in_packet) { |
- return kQuicFrameTypeSize + GetStreamIdSize(stream_id) + |
- GetStreamOffsetSize(offset) + |
- (last_frame_in_packet ? 0 : kQuicStreamPayloadLengthSize); |
-} |
- |
-// static |
-size_t QuicFramer::GetMinAckFrameSize( |
- QuicVersion version, |
- QuicPacketNumberLength largest_observed_length) { |
- size_t min_size = kQuicFrameTypeSize + largest_observed_length + |
- kQuicDeltaTimeLargestObservedSize; |
- if (version <= QUIC_VERSION_33) { |
- return min_size + kQuicEntropyHashSize; |
- } |
- return min_size + kQuicNumTimestampsSize; |
-} |
- |
-// static |
-size_t QuicFramer::GetStopWaitingFrameSize( |
- QuicVersion version, |
- QuicPacketNumberLength packet_number_length) { |
- size_t min_size = kQuicFrameTypeSize + packet_number_length; |
- if (version <= QUIC_VERSION_33) { |
- return min_size + kQuicEntropyHashSize; |
- } |
- return min_size; |
-} |
- |
-// static |
-size_t QuicFramer::GetRstStreamFrameSize() { |
- return kQuicFrameTypeSize + kQuicMaxStreamIdSize + kQuicMaxStreamOffsetSize + |
- kQuicErrorCodeSize; |
-} |
- |
-// static |
-size_t QuicFramer::GetMinConnectionCloseFrameSize() { |
- return kQuicFrameTypeSize + kQuicErrorCodeSize + kQuicErrorDetailsLengthSize; |
-} |
- |
-// static |
-size_t QuicFramer::GetMinGoAwayFrameSize() { |
- return kQuicFrameTypeSize + kQuicErrorCodeSize + kQuicErrorDetailsLengthSize + |
- kQuicMaxStreamIdSize; |
-} |
- |
-// static |
-size_t QuicFramer::GetWindowUpdateFrameSize() { |
- return kQuicFrameTypeSize + kQuicMaxStreamIdSize + kQuicMaxStreamOffsetSize; |
-} |
- |
-// static |
-size_t QuicFramer::GetBlockedFrameSize() { |
- return kQuicFrameTypeSize + kQuicMaxStreamIdSize; |
-} |
- |
-// static |
-size_t QuicFramer::GetPathCloseFrameSize() { |
- return kQuicFrameTypeSize + kQuicPathIdSize; |
-} |
- |
-// static |
-size_t QuicFramer::GetStreamIdSize(QuicStreamId stream_id) { |
- // Sizes are 1 through 4 bytes. |
- for (int i = 1; i <= 4; ++i) { |
- stream_id >>= 8; |
- if (stream_id == 0) { |
- return i; |
- } |
- } |
- QUIC_BUG << "Failed to determine StreamIDSize."; |
- return 4; |
-} |
- |
-// static |
-size_t QuicFramer::GetStreamOffsetSize(QuicStreamOffset offset) { |
- // 0 is a special case. |
- if (offset == 0) { |
- return 0; |
- } |
- // 2 through 8 are the remaining sizes. |
- offset >>= 8; |
- for (int i = 2; i <= 8; ++i) { |
- offset >>= 8; |
- if (offset == 0) { |
- return i; |
- } |
- } |
- QUIC_BUG << "Failed to determine StreamOffsetSize."; |
- return 8; |
-} |
- |
-// static |
-size_t QuicFramer::GetVersionNegotiationPacketSize(size_t number_versions) { |
- return kPublicFlagsSize + PACKET_8BYTE_CONNECTION_ID + |
- number_versions * kQuicVersionSize; |
-} |
- |
-bool QuicFramer::IsSupportedVersion(const QuicVersion version) const { |
- for (size_t i = 0; i < supported_versions_.size(); ++i) { |
- if (version == supported_versions_[i]) { |
- return true; |
- } |
- } |
- return false; |
-} |
- |
-size_t QuicFramer::GetSerializedFrameLength( |
- const QuicFrame& frame, |
- size_t free_bytes, |
- bool first_frame, |
- bool last_frame, |
- QuicPacketNumberLength packet_number_length) { |
- // Prevent a rare crash reported in b/19458523. |
- if ((frame.type == STREAM_FRAME || frame.type == ACK_FRAME) && |
- frame.stream_frame == nullptr) { |
- QUIC_BUG << "Cannot compute the length of a null frame. " |
- << "type:" << frame.type << "free_bytes:" << free_bytes |
- << " first_frame:" << first_frame << " last_frame:" << last_frame |
- << " seq num length:" << packet_number_length; |
- set_error(QUIC_INTERNAL_ERROR); |
- visitor_->OnError(this); |
- return 0; |
- } |
- if (frame.type == PADDING_FRAME) { |
- if (frame.padding_frame.num_padding_bytes == -1) { |
- // Full padding to the end of the packet. |
- return free_bytes; |
- } else { |
- // Lite padding. |
- return free_bytes < |
- static_cast<size_t>(frame.padding_frame.num_padding_bytes) |
- ? free_bytes |
- : frame.padding_frame.num_padding_bytes; |
- } |
- } |
- |
- size_t frame_len = |
- ComputeFrameLength(frame, last_frame, packet_number_length); |
- if (frame_len <= free_bytes) { |
- // Frame fits within packet. Note that acks may be truncated. |
- return frame_len; |
- } |
- // Only truncate the first frame in a packet, so if subsequent ones go |
- // over, stop including more frames. |
- if (!first_frame) { |
- return 0; |
- } |
- bool can_truncate = |
- frame.type == ACK_FRAME && |
- free_bytes >= |
- GetMinAckFrameSize(quic_version_, PACKET_6BYTE_PACKET_NUMBER); |
- if (can_truncate) { |
- // Truncate the frame so the packet will not exceed kMaxPacketSize. |
- // Note that we may not use every byte of the writer in this case. |
- DVLOG(1) << "Truncating large frame, free bytes: " << free_bytes; |
- return free_bytes; |
- } |
- return 0; |
-} |
- |
-QuicFramer::AckFrameInfo::AckFrameInfo() : max_delta(0) {} |
- |
-QuicFramer::AckFrameInfo::AckFrameInfo(const AckFrameInfo& other) = default; |
- |
-QuicFramer::AckFrameInfo::~AckFrameInfo() {} |
- |
-QuicFramer::AckBlock::AckBlock(uint8_t gap, QuicPacketNumber length) |
- : gap(gap), length(length) {} |
- |
-QuicFramer::AckBlock::AckBlock(const AckBlock& other) = default; |
- |
-QuicFramer::AckBlock::~AckBlock() {} |
- |
-QuicFramer::NewAckFrameInfo::NewAckFrameInfo() |
- : max_block_length(0), first_block_length(0), num_ack_blocks(0) {} |
- |
-QuicFramer::NewAckFrameInfo::NewAckFrameInfo(const NewAckFrameInfo& other) = |
- default; |
- |
-QuicFramer::NewAckFrameInfo::~NewAckFrameInfo() {} |
- |
-// static |
-QuicPacketEntropyHash QuicFramer::GetPacketEntropyHash( |
- const QuicPacketHeader& header) { |
- return header.entropy_flag << (header.packet_number % 8); |
-} |
- |
-size_t QuicFramer::BuildDataPacket(const QuicPacketHeader& header, |
- const QuicFrames& frames, |
- char* buffer, |
- size_t packet_length) { |
- QuicDataWriter writer(packet_length, buffer); |
- if (!AppendPacketHeader(header, &writer)) { |
- QUIC_BUG << "AppendPacketHeader failed"; |
- return 0; |
- } |
- |
- size_t i = 0; |
- for (const QuicFrame& frame : frames) { |
- // Determine if we should write stream frame length in header. |
- const bool no_stream_frame_length = i == frames.size() - 1; |
- if (!AppendTypeByte(frame, no_stream_frame_length, &writer)) { |
- QUIC_BUG << "AppendTypeByte failed"; |
- return 0; |
- } |
- |
- switch (frame.type) { |
- case PADDING_FRAME: |
- writer.WritePadding(); |
- break; |
- case STREAM_FRAME: |
- if (!AppendStreamFrame(*frame.stream_frame, no_stream_frame_length, |
- &writer)) { |
- QUIC_BUG << "AppendStreamFrame failed"; |
- return 0; |
- } |
- break; |
- case ACK_FRAME: |
- if (quic_version_ <= QUIC_VERSION_33) { |
- if (!AppendAckFrameAndTypeByte(header, *frame.ack_frame, &writer)) { |
- QUIC_BUG << "AppendAckFrameAndTypeByte failed" |
- << " header: " << header |
- << " ack_fame: " << *frame.ack_frame; |
- return 0; |
- } |
- } else { |
- if (!AppendNewAckFrameAndTypeByte(*frame.ack_frame, &writer)) { |
- QUIC_BUG << "AppendNewAckFrameAndTypeByte failed"; |
- return 0; |
- } |
- } |
- break; |
- case STOP_WAITING_FRAME: |
- if (!AppendStopWaitingFrame(header, *frame.stop_waiting_frame, |
- &writer)) { |
- QUIC_BUG << "AppendStopWaitingFrame failed"; |
- return 0; |
- } |
- break; |
- case MTU_DISCOVERY_FRAME: |
- // MTU discovery frames are serialized as ping frames. |
- case PING_FRAME: |
- // Ping has no payload. |
- break; |
- case RST_STREAM_FRAME: |
- if (!AppendRstStreamFrame(*frame.rst_stream_frame, &writer)) { |
- QUIC_BUG << "AppendRstStreamFrame failed"; |
- return 0; |
- } |
- break; |
- case CONNECTION_CLOSE_FRAME: |
- if (!AppendConnectionCloseFrame(*frame.connection_close_frame, |
- &writer)) { |
- QUIC_BUG << "AppendConnectionCloseFrame failed"; |
- return 0; |
- } |
- break; |
- case GOAWAY_FRAME: |
- if (!AppendGoAwayFrame(*frame.goaway_frame, &writer)) { |
- QUIC_BUG << "AppendGoAwayFrame failed"; |
- return 0; |
- } |
- break; |
- case WINDOW_UPDATE_FRAME: |
- if (!AppendWindowUpdateFrame(*frame.window_update_frame, &writer)) { |
- QUIC_BUG << "AppendWindowUpdateFrame failed"; |
- return 0; |
- } |
- break; |
- case BLOCKED_FRAME: |
- if (!AppendBlockedFrame(*frame.blocked_frame, &writer)) { |
- QUIC_BUG << "AppendBlockedFrame failed"; |
- return 0; |
- } |
- break; |
- case PATH_CLOSE_FRAME: |
- if (!AppendPathCloseFrame(*frame.path_close_frame, &writer)) { |
- QUIC_BUG << "AppendPathCloseFrame failed"; |
- return 0; |
- } |
- break; |
- default: |
- RaiseError(QUIC_INVALID_FRAME_DATA); |
- QUIC_BUG << "QUIC_INVALID_FRAME_DATA"; |
- return 0; |
- } |
- ++i; |
- } |
- |
- return writer.length(); |
-} |
- |
-// static |
-QuicEncryptedPacket* QuicFramer::BuildPublicResetPacket( |
- const QuicPublicResetPacket& packet) { |
- DCHECK(packet.public_header.reset_flag); |
- |
- CryptoHandshakeMessage reset; |
- reset.set_tag(kPRST); |
- reset.SetValue(kRNON, packet.nonce_proof); |
- reset.SetValue(kRSEQ, packet.rejected_packet_number); |
- if (!packet.client_address.address().empty()) { |
- // packet.client_address is non-empty. |
- QuicSocketAddressCoder address_coder(packet.client_address); |
- string serialized_address = address_coder.Encode(); |
- if (serialized_address.empty()) { |
- return nullptr; |
- } |
- reset.SetStringPiece(kCADR, serialized_address); |
- } |
- const QuicData& reset_serialized = reset.GetSerialized(); |
- |
- size_t len = |
- kPublicFlagsSize + PACKET_8BYTE_CONNECTION_ID + reset_serialized.length(); |
- std::unique_ptr<char[]> buffer(new char[len]); |
- QuicDataWriter writer(len, buffer.get()); |
- |
- uint8_t flags = static_cast<uint8_t>(PACKET_PUBLIC_FLAGS_RST | |
- PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID); |
- if (FLAGS_quic_use_old_public_reset_packets) { |
- // TODO(rch): Remove this QUIC_VERSION_32 is retired. |
- flags |= static_cast<uint8_t>(PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID_OLD); |
- } |
- if (!writer.WriteUInt8(flags)) { |
- return nullptr; |
- } |
- |
- if (!writer.WriteUInt64(packet.public_header.connection_id)) { |
- return nullptr; |
- } |
- |
- if (!writer.WriteBytes(reset_serialized.data(), reset_serialized.length())) { |
- return nullptr; |
- } |
- |
- return new QuicEncryptedPacket(buffer.release(), len, true); |
-} |
- |
-// static |
-QuicEncryptedPacket* QuicFramer::BuildVersionNegotiationPacket( |
- QuicConnectionId connection_id, |
- const QuicVersionVector& versions) { |
- DCHECK(!versions.empty()); |
- size_t len = GetVersionNegotiationPacketSize(versions.size()); |
- std::unique_ptr<char[]> buffer(new char[len]); |
- QuicDataWriter writer(len, buffer.get()); |
- |
- uint8_t flags = static_cast<uint8_t>( |
- PACKET_PUBLIC_FLAGS_VERSION | PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID | |
- // TODO(rch): Remove this QUIC_VERSION_32 is retired. |
- PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID_OLD); |
- if (!writer.WriteUInt8(flags)) { |
- return nullptr; |
- } |
- |
- if (!writer.WriteUInt64(connection_id)) { |
- return nullptr; |
- } |
- |
- for (QuicVersion version : versions) { |
- if (!writer.WriteUInt32(QuicVersionToQuicTag(version))) { |
- return nullptr; |
- } |
- } |
- |
- return new QuicEncryptedPacket(buffer.release(), len, true); |
-} |
- |
-bool QuicFramer::ProcessPacket(const QuicEncryptedPacket& packet) { |
- QuicDataReader reader(packet.data(), packet.length()); |
- |
- visitor_->OnPacket(); |
- |
- // First parse the public header. |
- QuicPacketPublicHeader public_header; |
- if (!ProcessPublicHeader(&reader, &public_header)) { |
- DLOG(WARNING) << "Unable to process public header."; |
- DCHECK_NE("", detailed_error_); |
- return RaiseError(QUIC_INVALID_PACKET_HEADER); |
- } |
- |
- if (!visitor_->OnUnauthenticatedPublicHeader(public_header)) { |
- // The visitor suppresses further processing of the packet. |
- return true; |
- } |
- |
- if (perspective_ == Perspective::IS_SERVER && public_header.version_flag && |
- public_header.versions[0] != quic_version_) { |
- if (!visitor_->OnProtocolVersionMismatch(public_header.versions[0])) { |
- return true; |
- } |
- } |
- |
- bool rv; |
- if (perspective_ == Perspective::IS_CLIENT && public_header.version_flag) { |
- rv = ProcessVersionNegotiationPacket(&reader, &public_header); |
- } else if (public_header.reset_flag) { |
- rv = ProcessPublicResetPacket(&reader, public_header); |
- } else if (packet.length() <= kMaxPacketSize) { |
- // The optimized decryption algorithm implementations run faster when |
- // operating on aligned memory. |
- // |
- // TODO(rtenneti): Change the default 64 alignas value (used the default |
- // value from CACHELINE_SIZE). |
- ALIGNAS(64) char buffer[kMaxPacketSize]; |
- rv = ProcessDataPacket(&reader, public_header, packet, buffer, |
- kMaxPacketSize); |
- } else { |
- std::unique_ptr<char[]> large_buffer(new char[packet.length()]); |
- rv = ProcessDataPacket(&reader, public_header, packet, large_buffer.get(), |
- packet.length()); |
- QUIC_BUG_IF(rv) << "QUIC should never successfully process packets larger" |
- << "than kMaxPacketSize. packet size:" << packet.length(); |
- } |
- |
- return rv; |
-} |
- |
-bool QuicFramer::ProcessVersionNegotiationPacket( |
- QuicDataReader* reader, |
- QuicPacketPublicHeader* public_header) { |
- DCHECK_EQ(Perspective::IS_CLIENT, perspective_); |
- // Try reading at least once to raise error if the packet is invalid. |
- do { |
- QuicTag version; |
- if (!reader->ReadBytes(&version, kQuicVersionSize)) { |
- set_detailed_error("Unable to read supported version in negotiation."); |
- return RaiseError(QUIC_INVALID_VERSION_NEGOTIATION_PACKET); |
- } |
- public_header->versions.push_back(QuicTagToQuicVersion(version)); |
- } while (!reader->IsDoneReading()); |
- |
- visitor_->OnVersionNegotiationPacket(*public_header); |
- return true; |
-} |
- |
-bool QuicFramer::ProcessDataPacket(QuicDataReader* encrypted_reader, |
- const QuicPacketPublicHeader& public_header, |
- const QuicEncryptedPacket& packet, |
- char* decrypted_buffer, |
- size_t buffer_length) { |
- QuicPacketHeader header(public_header); |
- if (!ProcessUnauthenticatedHeader(encrypted_reader, &header)) { |
- DLOG(WARNING) << "Unable to process packet header. Stopping parsing."; |
- return false; |
- } |
- |
- size_t decrypted_length = 0; |
- if (!DecryptPayload(encrypted_reader, header, packet, decrypted_buffer, |
- buffer_length, &decrypted_length)) { |
- set_detailed_error("Unable to decrypt payload."); |
- return RaiseError(QUIC_DECRYPTION_FAILURE); |
- } |
- |
- QuicDataReader reader(decrypted_buffer, decrypted_length); |
- if (quic_version_ <= QUIC_VERSION_33) { |
- if (!ProcessAuthenticatedHeader(&reader, &header)) { |
- DLOG(WARNING) << "Unable to process packet header. Stopping parsing."; |
- return false; |
- } |
- } |
- |
- // Set the last packet number after we have decrypted the packet |
- // so we are confident is not attacker controlled. |
- SetLastPacketNumber(header); |
- |
- if (!visitor_->OnPacketHeader(header)) { |
- // The visitor suppresses further processing of the packet. |
- return true; |
- } |
- |
- if (packet.length() > kMaxPacketSize) { |
- // If the packet has gotten this far, it should not be too large. |
- QUIC_BUG << "Packet too large:" << packet.length(); |
- return RaiseError(QUIC_PACKET_TOO_LARGE); |
- } |
- |
- DCHECK(!header.fec_flag); |
- // Handle the payload. |
- if (!ProcessFrameData(&reader, header)) { |
- DCHECK_NE(QUIC_NO_ERROR, error_); // ProcessFrameData sets the error. |
- DLOG(WARNING) << "Unable to process frame data."; |
- return false; |
- } |
- |
- visitor_->OnPacketComplete(); |
- return true; |
-} |
- |
-bool QuicFramer::ProcessPublicResetPacket( |
- QuicDataReader* reader, |
- const QuicPacketPublicHeader& public_header) { |
- QuicPublicResetPacket packet(public_header); |
- |
- std::unique_ptr<CryptoHandshakeMessage> reset( |
- CryptoFramer::ParseMessage(reader->ReadRemainingPayload())); |
- if (!reset.get()) { |
- set_detailed_error("Unable to read reset message."); |
- return RaiseError(QUIC_INVALID_PUBLIC_RST_PACKET); |
- } |
- if (reset->tag() != kPRST) { |
- set_detailed_error("Incorrect message tag."); |
- return RaiseError(QUIC_INVALID_PUBLIC_RST_PACKET); |
- } |
- |
- if (reset->GetUint64(kRNON, &packet.nonce_proof) != QUIC_NO_ERROR) { |
- set_detailed_error("Unable to read nonce proof."); |
- return RaiseError(QUIC_INVALID_PUBLIC_RST_PACKET); |
- } |
- // TODO(satyamshekhar): validate nonce to protect against DoS. |
- |
- StringPiece address; |
- if (reset->GetStringPiece(kCADR, &address)) { |
- QuicSocketAddressCoder address_coder; |
- if (address_coder.Decode(address.data(), address.length())) { |
- packet.client_address = |
- IPEndPoint(address_coder.ip(), address_coder.port()); |
- } |
- } |
- |
- visitor_->OnPublicResetPacket(packet); |
- return true; |
-} |
- |
-bool QuicFramer::AppendPacketHeader(const QuicPacketHeader& header, |
- QuicDataWriter* writer) { |
- DVLOG(1) << "Appending header: " << header; |
- uint8_t public_flags = 0; |
- if (header.public_header.reset_flag) { |
- public_flags |= PACKET_PUBLIC_FLAGS_RST; |
- } |
- if (header.public_header.version_flag) { |
- public_flags |= PACKET_PUBLIC_FLAGS_VERSION; |
- } |
- if (header.public_header.multipath_flag) { |
- public_flags |= PACKET_PUBLIC_FLAGS_MULTIPATH; |
- } |
- |
- public_flags |= |
- GetSequenceNumberFlags(header.public_header.packet_number_length) |
- << kPublicHeaderSequenceNumberShift; |
- |
- if (header.public_header.nonce != nullptr) { |
- DCHECK_EQ(Perspective::IS_SERVER, perspective_); |
- public_flags |= PACKET_PUBLIC_FLAGS_NONCE; |
- } |
- |
- switch (header.public_header.connection_id_length) { |
- case PACKET_0BYTE_CONNECTION_ID: |
- if (!writer->WriteUInt8(public_flags | |
- PACKET_PUBLIC_FLAGS_0BYTE_CONNECTION_ID)) { |
- return false; |
- } |
- break; |
- case PACKET_8BYTE_CONNECTION_ID: |
- if (quic_version_ > QUIC_VERSION_32) { |
- public_flags |= PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID; |
- if (perspective_ == Perspective::IS_CLIENT) { |
- // TODO(rch): Fix this when v33 flags are supported by middle boxes. |
- public_flags |= PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID_OLD; |
- } |
- |
- } else { |
- public_flags |= PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID_OLD; |
- } |
- if (!writer->WriteUInt8(public_flags) || |
- !writer->WriteUInt64(header.public_header.connection_id)) { |
- return false; |
- } |
- break; |
- } |
- last_serialized_connection_id_ = header.public_header.connection_id; |
- |
- if (header.public_header.version_flag) { |
- DCHECK_EQ(Perspective::IS_CLIENT, perspective_); |
- QuicTag tag = QuicVersionToQuicTag(quic_version_); |
- writer->WriteUInt32(tag); |
- DVLOG(1) << "version = " << quic_version_ << ", tag = '" |
- << QuicUtils::TagToString(tag) << "'"; |
- } |
- |
- if (header.public_header.multipath_flag && |
- !writer->WriteUInt8(header.path_id)) { |
- return false; |
- } |
- |
- if (header.public_header.nonce != nullptr && |
- !writer->WriteBytes(header.public_header.nonce, |
- kDiversificationNonceSize)) { |
- return false; |
- } |
- |
- if (!AppendPacketSequenceNumber(header.public_header.packet_number_length, |
- header.packet_number, writer)) { |
- return false; |
- } |
- if (quic_version_ > QUIC_VERSION_33) { |
- return true; |
- } |
- |
- uint8_t private_flags = 0; |
- if (header.entropy_flag) { |
- private_flags |= PACKET_PRIVATE_FLAGS_ENTROPY; |
- } |
- if (!writer->WriteUInt8(private_flags)) { |
- return false; |
- } |
- |
- return true; |
-} |
- |
-const QuicTime::Delta QuicFramer::CalculateTimestampFromWire( |
- uint32_t time_delta_us) { |
- // The new time_delta might have wrapped to the next epoch, or it |
- // might have reverse wrapped to the previous epoch, or it might |
- // remain in the same epoch. Select the time closest to the previous |
- // time. |
- // |
- // epoch_delta is the delta between epochs. A delta is 4 bytes of |
- // microseconds. |
- const uint64_t epoch_delta = UINT64_C(1) << 32; |
- uint64_t epoch = last_timestamp_.ToMicroseconds() & ~(epoch_delta - 1); |
- // Wrapping is safe here because a wrapped value will not be ClosestTo below. |
- uint64_t prev_epoch = epoch - epoch_delta; |
- uint64_t next_epoch = epoch + epoch_delta; |
- |
- uint64_t time = ClosestTo( |
- last_timestamp_.ToMicroseconds(), epoch + time_delta_us, |
- ClosestTo(last_timestamp_.ToMicroseconds(), prev_epoch + time_delta_us, |
- next_epoch + time_delta_us)); |
- |
- return QuicTime::Delta::FromMicroseconds(time); |
-} |
- |
-bool QuicFramer::IsValidPath(QuicPathId path_id, |
- QuicPacketNumber* last_packet_number) { |
- if (ContainsKey(closed_paths_, path_id)) { |
- // Path is closed. |
- return false; |
- } |
- |
- if (path_id == last_path_id_) { |
- *last_packet_number = last_packet_number_; |
- return true; |
- } |
- |
- if (ContainsKey(last_packet_numbers_, path_id)) { |
- *last_packet_number = last_packet_numbers_[path_id]; |
- } else { |
- *last_packet_number = 0; |
- } |
- |
- return true; |
-} |
- |
-void QuicFramer::SetLastPacketNumber(const QuicPacketHeader& header) { |
- if (header.public_header.multipath_flag && header.path_id != last_path_id_) { |
- if (last_path_id_ != kInvalidPathId) { |
- // Save current last packet number before changing path. |
- last_packet_numbers_[last_path_id_] = last_packet_number_; |
- } |
- // Change path. |
- last_path_id_ = header.path_id; |
- } |
- last_packet_number_ = header.packet_number; |
-} |
- |
-void QuicFramer::OnPathClosed(QuicPathId path_id) { |
- closed_paths_.insert(path_id); |
- last_packet_numbers_.erase(path_id); |
-} |
- |
-QuicPacketNumber QuicFramer::CalculatePacketNumberFromWire( |
- QuicPacketNumberLength packet_number_length, |
- QuicPacketNumber last_packet_number, |
- QuicPacketNumber packet_number) const { |
- // The new packet number might have wrapped to the next epoch, or |
- // it might have reverse wrapped to the previous epoch, or it might |
- // remain in the same epoch. Select the packet number closest to the |
- // next expected packet number, the previous packet number plus 1. |
- |
- // epoch_delta is the delta between epochs the packet number was serialized |
- // with, so the correct value is likely the same epoch as the last sequence |
- // number or an adjacent epoch. |
- const QuicPacketNumber epoch_delta = UINT64_C(1) |
- << (8 * packet_number_length); |
- QuicPacketNumber next_packet_number = last_packet_number + 1; |
- QuicPacketNumber epoch = last_packet_number & ~(epoch_delta - 1); |
- QuicPacketNumber prev_epoch = epoch - epoch_delta; |
- QuicPacketNumber next_epoch = epoch + epoch_delta; |
- |
- return ClosestTo(next_packet_number, epoch + packet_number, |
- ClosestTo(next_packet_number, prev_epoch + packet_number, |
- next_epoch + packet_number)); |
-} |
- |
-bool QuicFramer::ProcessPublicHeader(QuicDataReader* reader, |
- QuicPacketPublicHeader* public_header) { |
- uint8_t public_flags; |
- if (!reader->ReadBytes(&public_flags, 1)) { |
- set_detailed_error("Unable to read public flags."); |
- return false; |
- } |
- |
- public_header->multipath_flag = |
- (public_flags & PACKET_PUBLIC_FLAGS_MULTIPATH) != 0; |
- public_header->reset_flag = (public_flags & PACKET_PUBLIC_FLAGS_RST) != 0; |
- public_header->version_flag = |
- (public_flags & PACKET_PUBLIC_FLAGS_VERSION) != 0; |
- |
- if (validate_flags_ && !public_header->version_flag && |
- public_flags > PACKET_PUBLIC_FLAGS_MAX) { |
- set_detailed_error("Illegal public flags value."); |
- return false; |
- } |
- |
- if (public_header->reset_flag && public_header->version_flag) { |
- set_detailed_error("Got version flag in reset packet"); |
- return false; |
- } |
- |
- switch (public_flags & PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID) { |
- case PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID: |
- if (!reader->ReadUInt64(&public_header->connection_id)) { |
- set_detailed_error("Unable to read ConnectionId."); |
- return false; |
- } |
- public_header->connection_id_length = PACKET_8BYTE_CONNECTION_ID; |
- break; |
- case PACKET_PUBLIC_FLAGS_0BYTE_CONNECTION_ID: |
- public_header->connection_id_length = PACKET_0BYTE_CONNECTION_ID; |
- public_header->connection_id = last_serialized_connection_id_; |
- break; |
- } |
- |
- public_header->packet_number_length = ReadSequenceNumberLength( |
- public_flags >> kPublicHeaderSequenceNumberShift); |
- |
- // Read the version only if the packet is from the client. |
- // version flag from the server means version negotiation packet. |
- if (public_header->version_flag && perspective_ == Perspective::IS_SERVER) { |
- QuicTag version_tag; |
- if (!reader->ReadUInt32(&version_tag)) { |
- set_detailed_error("Unable to read protocol version."); |
- return false; |
- } |
- |
- // If the version from the new packet is the same as the version of this |
- // framer, then the public flags should be set to something we understand. |
- // If not, this raises an error. |
- last_version_tag_ = version_tag; |
- QuicVersion version = QuicTagToQuicVersion(version_tag); |
- if (version == quic_version_ && public_flags > PACKET_PUBLIC_FLAGS_MAX) { |
- set_detailed_error("Illegal public flags value."); |
- return false; |
- } |
- public_header->versions.push_back(version); |
- } |
- |
- // A nonce should only be present in packets from the server to the client, |
- // which are neither version negotiation nor public reset packets |
- // and only for versions after QUIC_VERSION_32. Earlier versions will |
- // set this bit when indicating an 8-byte connection ID, which should |
- // not be interpreted as indicating a nonce is present. |
- if (quic_version_ > QUIC_VERSION_32 && |
- public_flags & PACKET_PUBLIC_FLAGS_NONCE && |
- !(public_flags & PACKET_PUBLIC_FLAGS_VERSION) && |
- !(public_flags & PACKET_PUBLIC_FLAGS_RST) && |
- // The nonce flag from a client is ignored and is assumed to be an older |
- // client indicating an eight-byte connection ID. |
- perspective_ == Perspective::IS_CLIENT) { |
- if (!reader->ReadBytes(reinterpret_cast<uint8_t*>(last_nonce_), |
- sizeof(last_nonce_))) { |
- set_detailed_error("Unable to read nonce."); |
- return false; |
- } |
- public_header->nonce = &last_nonce_; |
- } else { |
- public_header->nonce = nullptr; |
- } |
- |
- return true; |
-} |
- |
-// static |
-QuicPacketNumberLength QuicFramer::GetMinSequenceNumberLength( |
- QuicPacketNumber packet_number) { |
- if (packet_number < 1 << (PACKET_1BYTE_PACKET_NUMBER * 8)) { |
- return PACKET_1BYTE_PACKET_NUMBER; |
- } else if (packet_number < 1 << (PACKET_2BYTE_PACKET_NUMBER * 8)) { |
- return PACKET_2BYTE_PACKET_NUMBER; |
- } else if (packet_number < UINT64_C(1) << (PACKET_4BYTE_PACKET_NUMBER * 8)) { |
- return PACKET_4BYTE_PACKET_NUMBER; |
- } else { |
- return PACKET_6BYTE_PACKET_NUMBER; |
- } |
-} |
- |
-// static |
-uint8_t QuicFramer::GetSequenceNumberFlags( |
- QuicPacketNumberLength packet_number_length) { |
- switch (packet_number_length) { |
- case PACKET_1BYTE_PACKET_NUMBER: |
- return PACKET_FLAGS_1BYTE_PACKET; |
- case PACKET_2BYTE_PACKET_NUMBER: |
- return PACKET_FLAGS_2BYTE_PACKET; |
- case PACKET_4BYTE_PACKET_NUMBER: |
- return PACKET_FLAGS_4BYTE_PACKET; |
- case PACKET_6BYTE_PACKET_NUMBER: |
- return PACKET_FLAGS_6BYTE_PACKET; |
- default: |
- QUIC_BUG << "Unreachable case statement."; |
- return PACKET_FLAGS_6BYTE_PACKET; |
- } |
-} |
- |
-// static |
-QuicFramer::AckFrameInfo QuicFramer::GetAckFrameInfo( |
- const QuicAckFrame& frame) { |
- AckFrameInfo ack_info; |
- if (frame.packets.Empty()) { |
- return ack_info; |
- } |
- DCHECK_GE(frame.largest_observed, frame.packets.Max()); |
- if (FLAGS_quic_use_packet_number_queue_intervals) { |
- QuicPacketNumber last_largest_missing = 0; |
- for (auto itr = frame.packets.begin_intervals(); |
- itr != frame.packets.end_intervals(); ++itr) { |
- const Interval<QuicPacketNumber>& interval = *itr; |
- for (QuicPacketNumber interval_start = interval.min(); |
- interval_start < interval.max(); |
- interval_start += (1ull + numeric_limits<uint8_t>::max())) { |
- uint8_t cur_range_length = |
- interval.max() - interval_start > numeric_limits<uint8_t>::max() |
- ? numeric_limits<uint8_t>::max() |
- : (interval.max() - interval_start) - 1; |
- ack_info.nack_ranges[interval_start] = cur_range_length; |
- } |
- ack_info.max_delta = max(ack_info.max_delta, |
- last_largest_missing == 0 |
- ? QuicPacketNumber{0} |
- : (interval.min() - last_largest_missing)); |
- last_largest_missing = interval.max() - 1; |
- } |
- // Include the range to the largest observed. |
- ack_info.max_delta = |
- max(ack_info.max_delta, frame.largest_observed - last_largest_missing); |
- } else { |
- size_t cur_range_length = 0; |
- PacketNumberQueue::const_iterator iter = frame.packets.begin(); |
- QuicPacketNumber last_missing = *iter; |
- ++iter; |
- for (; iter != frame.packets.end(); ++iter) { |
- if (cur_range_length < numeric_limits<uint8_t>::max() && |
- *iter == (last_missing + 1)) { |
- ++cur_range_length; |
- } else { |
- ack_info.nack_ranges[last_missing - cur_range_length] = |
- static_cast<uint8_t>(cur_range_length); |
- cur_range_length = 0; |
- } |
- ack_info.max_delta = max(ack_info.max_delta, *iter - last_missing); |
- last_missing = *iter; |
- } |
- // Include the last nack range. |
- ack_info.nack_ranges[last_missing - cur_range_length] = |
- static_cast<uint8_t>(cur_range_length); |
- // Include the range to the largest observed. |
- ack_info.max_delta = |
- max(ack_info.max_delta, frame.largest_observed - last_missing); |
- } |
- return ack_info; |
-} |
- |
-// static |
-QuicFramer::NewAckFrameInfo QuicFramer::GetNewAckFrameInfo( |
- const QuicAckFrame& frame, |
- bool construct_blocks) { |
- NewAckFrameInfo new_ack_info; |
- if (frame.packets.Empty()) { |
- return new_ack_info; |
- } |
- if (!construct_blocks) { |
- // The first block is the last interval. It isn't encoded with the |
- // gap-length encoding, so skip it. |
- new_ack_info.first_block_length = frame.packets.LastIntervalLength(); |
- auto itr = frame.packets.rbegin_intervals(); |
- QuicPacketNumber previous_start = itr->min(); |
- new_ack_info.max_block_length = itr->Length(); |
- ++itr; |
- |
- // Don't do any more work after getting information for 256 ACK blocks; any |
- // more can't be encoded anyway. |
- for (; itr != frame.packets.rend_intervals() && |
- new_ack_info.num_ack_blocks < numeric_limits<uint8_t>::max(); |
- previous_start = itr->min(), ++itr) { |
- const auto& interval = *itr; |
- const QuicPacketNumber total_gap = previous_start - interval.max(); |
- new_ack_info.num_ack_blocks += |
- (total_gap + numeric_limits<uint8_t>::max() - 1) / |
- numeric_limits<uint8_t>::max(); |
- new_ack_info.max_block_length = |
- max(new_ack_info.max_block_length, interval.Length()); |
- } |
- } else { |
- QuicPacketNumber cur_range_length = 1; |
- PacketNumberQueue::const_iterator iter = frame.packets.begin(); |
- QuicPacketNumber last_received = *iter; |
- ++iter; |
- for (; iter != frame.packets.end(); ++iter) { |
- if (*iter == (last_received + 1)) { |
- ++cur_range_length; |
- } else { |
- size_t total_gap = *iter - last_received - 1; |
- size_t num_blocks = static_cast<size_t>(ceil( |
- static_cast<double>(total_gap) / numeric_limits<uint8_t>::max())); |
- uint8_t last_gap = static_cast<uint8_t>( |
- total_gap - (num_blocks - 1) * numeric_limits<uint8_t>::max()); |
- for (size_t i = 0; i < num_blocks; ++i) { |
- if (i == 0) { |
- new_ack_info.ack_blocks.push_back( |
- AckBlock(last_gap, cur_range_length)); |
- } else { |
- // Add an ack block of length 0 because there are more than 255 |
- // missing packets in a row. |
- new_ack_info.ack_blocks.push_back( |
- AckBlock(numeric_limits<uint8_t>::max(), 0)); |
- } |
- } |
- new_ack_info.max_block_length = |
- max(new_ack_info.max_block_length, cur_range_length); |
- cur_range_length = 1; |
- } |
- last_received = *iter; |
- } |
- new_ack_info.first_block_length = cur_range_length; |
- new_ack_info.max_block_length = |
- max(new_ack_info.max_block_length, new_ack_info.first_block_length); |
- new_ack_info.num_ack_blocks = new_ack_info.ack_blocks.size(); |
- } |
- return new_ack_info; |
-} |
- |
-bool QuicFramer::ProcessUnauthenticatedHeader(QuicDataReader* encrypted_reader, |
- QuicPacketHeader* header) { |
- header->path_id = kDefaultPathId; |
- if (header->public_header.multipath_flag && |
- !ProcessPathId(encrypted_reader, &header->path_id)) { |
- set_detailed_error("Unable to read path id."); |
- return RaiseError(QUIC_INVALID_PACKET_HEADER); |
- } |
- |
- QuicPacketNumber last_packet_number = last_packet_number_; |
- if (header->public_header.multipath_flag && |
- !IsValidPath(header->path_id, &last_packet_number)) { |
- // Stop processing because path is closed. |
- return false; |
- } |
- |
- if (!ProcessPacketSequenceNumber( |
- encrypted_reader, header->public_header.packet_number_length, |
- last_packet_number, &header->packet_number)) { |
- set_detailed_error("Unable to read packet number."); |
- return RaiseError(QUIC_INVALID_PACKET_HEADER); |
- } |
- |
- if (header->packet_number == 0u) { |
- set_detailed_error("packet numbers cannot be 0."); |
- return RaiseError(QUIC_INVALID_PACKET_HEADER); |
- } |
- |
- if (!visitor_->OnUnauthenticatedHeader(*header)) { |
- return false; |
- } |
- return true; |
-} |
- |
-bool QuicFramer::ProcessAuthenticatedHeader(QuicDataReader* reader, |
- QuicPacketHeader* header) { |
- uint8_t private_flags; |
- if (!reader->ReadBytes(&private_flags, 1)) { |
- set_detailed_error("Unable to read private flags."); |
- return RaiseError(QUIC_INVALID_PACKET_HEADER); |
- } |
- |
- if (quic_version_ > QUIC_VERSION_31) { |
- if (private_flags > PACKET_PRIVATE_FLAGS_MAX_VERSION_32) { |
- set_detailed_error("Illegal private flags value."); |
- return RaiseError(QUIC_INVALID_PACKET_HEADER); |
- } |
- } else { |
- if (private_flags > PACKET_PRIVATE_FLAGS_MAX) { |
- set_detailed_error("Illegal private flags value."); |
- return RaiseError(QUIC_INVALID_PACKET_HEADER); |
- } |
- } |
- |
- header->entropy_flag = (private_flags & PACKET_PRIVATE_FLAGS_ENTROPY) != 0; |
- header->fec_flag = (private_flags & PACKET_PRIVATE_FLAGS_FEC) != 0; |
- |
- if ((private_flags & PACKET_PRIVATE_FLAGS_FEC_GROUP) != 0) { |
- uint8_t first_fec_protected_packet_offset; |
- if (!reader->ReadBytes(&first_fec_protected_packet_offset, 1)) { |
- set_detailed_error("Unable to read first fec protected packet offset."); |
- return RaiseError(QUIC_INVALID_PACKET_HEADER); |
- } |
- if (first_fec_protected_packet_offset >= header->packet_number) { |
- set_detailed_error( |
- "First fec protected packet offset must be less " |
- "than the packet number."); |
- return RaiseError(QUIC_INVALID_PACKET_HEADER); |
- } |
- } |
- |
- header->entropy_hash = GetPacketEntropyHash(*header); |
- return true; |
-} |
- |
-bool QuicFramer::ProcessPathId(QuicDataReader* reader, QuicPathId* path_id) { |
- if (!reader->ReadBytes(path_id, 1)) { |
- return false; |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessPacketSequenceNumber( |
- QuicDataReader* reader, |
- QuicPacketNumberLength packet_number_length, |
- QuicPacketNumber last_packet_number, |
- QuicPacketNumber* packet_number) { |
- QuicPacketNumber wire_packet_number = 0u; |
- if (!reader->ReadBytes(&wire_packet_number, packet_number_length)) { |
- return false; |
- } |
- |
- // TODO(ianswett): Explore the usefulness of trying multiple packet numbers |
- // in case the first guess is incorrect. |
- *packet_number = CalculatePacketNumberFromWire( |
- packet_number_length, last_packet_number, wire_packet_number); |
- return true; |
-} |
- |
-bool QuicFramer::ProcessFrameData(QuicDataReader* reader, |
- const QuicPacketHeader& header) { |
- if (reader->IsDoneReading()) { |
- set_detailed_error("Packet has no frames."); |
- return RaiseError(QUIC_MISSING_PAYLOAD); |
- } |
- while (!reader->IsDoneReading()) { |
- uint8_t frame_type; |
- if (!reader->ReadBytes(&frame_type, 1)) { |
- set_detailed_error("Unable to read frame type."); |
- return RaiseError(QUIC_INVALID_FRAME_DATA); |
- } |
- |
- if (frame_type & kQuicFrameTypeSpecialMask) { |
- // Stream Frame |
- if (frame_type & kQuicFrameTypeStreamMask) { |
- QuicStreamFrame frame; |
- if (!ProcessStreamFrame(reader, frame_type, &frame)) { |
- return RaiseError(QUIC_INVALID_STREAM_DATA); |
- } |
- if (!visitor_->OnStreamFrame(frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- |
- // Ack Frame |
- if (frame_type & kQuicFrameTypeAckMask) { |
- QuicAckFrame frame; |
- if (quic_version_ <= QUIC_VERSION_33) { |
- if (!ProcessAckFrame(reader, frame_type, &frame)) { |
- return RaiseError(QUIC_INVALID_ACK_DATA); |
- } |
- } else { |
- if (!ProcessNewAckFrame(reader, frame_type, &frame)) { |
- return RaiseError(QUIC_INVALID_ACK_DATA); |
- } |
- } |
- if (!visitor_->OnAckFrame(frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- |
- // This was a special frame type that did not match any |
- // of the known ones. Error. |
- set_detailed_error("Illegal frame type."); |
- DLOG(WARNING) << "Illegal frame type: " << static_cast<int>(frame_type); |
- return RaiseError(QUIC_INVALID_FRAME_DATA); |
- } |
- |
- switch (frame_type) { |
- case PADDING_FRAME: { |
- QuicPaddingFrame frame(reader->BytesRemaining()); |
- if (!visitor_->OnPaddingFrame(frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- } |
- // We're done with the packet. |
- return true; |
- } |
- |
- case RST_STREAM_FRAME: { |
- QuicRstStreamFrame frame; |
- if (!ProcessRstStreamFrame(reader, &frame)) { |
- return RaiseError(QUIC_INVALID_RST_STREAM_DATA); |
- } |
- if (!visitor_->OnRstStreamFrame(frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- |
- case CONNECTION_CLOSE_FRAME: { |
- QuicConnectionCloseFrame frame; |
- if (!ProcessConnectionCloseFrame(reader, &frame)) { |
- return RaiseError(QUIC_INVALID_CONNECTION_CLOSE_DATA); |
- } |
- |
- if (!visitor_->OnConnectionCloseFrame(frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- |
- case GOAWAY_FRAME: { |
- QuicGoAwayFrame goaway_frame; |
- if (!ProcessGoAwayFrame(reader, &goaway_frame)) { |
- return RaiseError(QUIC_INVALID_GOAWAY_DATA); |
- } |
- if (!visitor_->OnGoAwayFrame(goaway_frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- |
- case WINDOW_UPDATE_FRAME: { |
- QuicWindowUpdateFrame window_update_frame; |
- if (!ProcessWindowUpdateFrame(reader, &window_update_frame)) { |
- return RaiseError(QUIC_INVALID_WINDOW_UPDATE_DATA); |
- } |
- if (!visitor_->OnWindowUpdateFrame(window_update_frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- |
- case BLOCKED_FRAME: { |
- QuicBlockedFrame blocked_frame; |
- if (!ProcessBlockedFrame(reader, &blocked_frame)) { |
- return RaiseError(QUIC_INVALID_BLOCKED_DATA); |
- } |
- if (!visitor_->OnBlockedFrame(blocked_frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- |
- case STOP_WAITING_FRAME: { |
- QuicStopWaitingFrame stop_waiting_frame; |
- if (!ProcessStopWaitingFrame(reader, header, &stop_waiting_frame)) { |
- return RaiseError(QUIC_INVALID_STOP_WAITING_DATA); |
- } |
- if (!visitor_->OnStopWaitingFrame(stop_waiting_frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- case PING_FRAME: { |
- // Ping has no payload. |
- QuicPingFrame ping_frame; |
- if (!visitor_->OnPingFrame(ping_frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- case PATH_CLOSE_FRAME: { |
- QuicPathCloseFrame path_close_frame; |
- if (!ProcessPathCloseFrame(reader, &path_close_frame)) { |
- return RaiseError(QUIC_INVALID_PATH_CLOSE_DATA); |
- } |
- if (!visitor_->OnPathCloseFrame(path_close_frame)) { |
- DVLOG(1) << "Visitor asked to stop further processing."; |
- // Returning true since there was no parsing error. |
- return true; |
- } |
- continue; |
- } |
- |
- default: |
- set_detailed_error("Illegal frame type."); |
- DLOG(WARNING) << "Illegal frame type: " << static_cast<int>(frame_type); |
- return RaiseError(QUIC_INVALID_FRAME_DATA); |
- } |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessStreamFrame(QuicDataReader* reader, |
- uint8_t frame_type, |
- QuicStreamFrame* frame) { |
- uint8_t stream_flags = frame_type; |
- |
- stream_flags &= ~kQuicFrameTypeStreamMask; |
- |
- // Read from right to left: StreamID, Offset, Data Length, Fin. |
- const uint8_t stream_id_length = (stream_flags & kQuicStreamIDLengthMask) + 1; |
- stream_flags >>= kQuicStreamIdShift; |
- |
- uint8_t offset_length = (stream_flags & kQuicStreamOffsetMask); |
- // There is no encoding for 1 byte, only 0 and 2 through 8. |
- if (offset_length > 0) { |
- offset_length += 1; |
- } |
- stream_flags >>= kQuicStreamOffsetShift; |
- |
- bool has_data_length = |
- (stream_flags & kQuicStreamDataLengthMask) == kQuicStreamDataLengthMask; |
- stream_flags >>= kQuicStreamDataLengthShift; |
- |
- frame->fin = (stream_flags & kQuicStreamFinMask) == kQuicStreamFinShift; |
- |
- frame->stream_id = 0; |
- if (!reader->ReadBytes(&frame->stream_id, stream_id_length)) { |
- set_detailed_error("Unable to read stream_id."); |
- return false; |
- } |
- |
- frame->offset = 0; |
- if (!reader->ReadBytes(&frame->offset, offset_length)) { |
- set_detailed_error("Unable to read offset."); |
- return false; |
- } |
- |
- // TODO(ianswett): Don't use StringPiece as an intermediary. |
- StringPiece data; |
- if (has_data_length) { |
- if (!reader->ReadStringPiece16(&data)) { |
- set_detailed_error("Unable to read frame data."); |
- return false; |
- } |
- } else { |
- if (!reader->ReadStringPiece(&data, reader->BytesRemaining())) { |
- set_detailed_error("Unable to read frame data."); |
- return false; |
- } |
- } |
- frame->data_buffer = data.data(); |
- frame->data_length = static_cast<uint16_t>(data.length()); |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessAckFrame(QuicDataReader* reader, |
- uint8_t frame_type, |
- QuicAckFrame* ack_frame) { |
- // Determine the three lengths from the frame type: largest observed length, |
- // missing packet number length, and missing range length. |
- const QuicPacketNumberLength missing_packet_number_length = |
- ReadSequenceNumberLength(frame_type); |
- frame_type >>= kQuicSequenceNumberLengthShift; |
- const QuicPacketNumberLength largest_observed_packet_number_length = |
- ReadSequenceNumberLength(frame_type); |
- frame_type >>= kQuicSequenceNumberLengthShift; |
- ack_frame->is_truncated = frame_type & kQuicAckTruncatedMask; |
- frame_type >>= kQuicAckTruncatedShift; |
- bool has_nacks = frame_type & kQuicHasNacksMask; |
- |
- if (!reader->ReadBytes(&ack_frame->entropy_hash, 1)) { |
- set_detailed_error("Unable to read entropy hash for received packets."); |
- return false; |
- } |
- |
- if (!reader->ReadBytes(&ack_frame->largest_observed, |
- largest_observed_packet_number_length)) { |
- set_detailed_error("Unable to read largest observed."); |
- return false; |
- } |
- |
- uint64_t ack_delay_time_us; |
- if (!reader->ReadUFloat16(&ack_delay_time_us)) { |
- set_detailed_error("Unable to read ack delay time."); |
- return false; |
- } |
- |
- if (ack_delay_time_us == kUFloat16MaxValue) { |
- ack_frame->ack_delay_time = QuicTime::Delta::Infinite(); |
- } else { |
- ack_frame->ack_delay_time = |
- QuicTime::Delta::FromMicroseconds(ack_delay_time_us); |
- } |
- |
- if (!ProcessTimestampsInAckFrame(reader, ack_frame)) { |
- return false; |
- } |
- |
- if (!has_nacks) { |
- return true; |
- } |
- |
- uint8_t num_missing_ranges; |
- if (!reader->ReadBytes(&num_missing_ranges, 1)) { |
- set_detailed_error("Unable to read num missing packet ranges."); |
- return false; |
- } |
- |
- QuicPacketNumber last_packet_number = ack_frame->largest_observed; |
- for (size_t i = 0; i < num_missing_ranges; ++i) { |
- QuicPacketNumber missing_delta = 0; |
- if (!reader->ReadBytes(&missing_delta, missing_packet_number_length)) { |
- set_detailed_error("Unable to read missing packet number delta."); |
- return false; |
- } |
- last_packet_number -= missing_delta; |
- QuicPacketNumber range_length = 0; |
- if (!reader->ReadBytes(&range_length, PACKET_1BYTE_PACKET_NUMBER)) { |
- set_detailed_error("Unable to read missing packet number range."); |
- return false; |
- } |
- ack_frame->packets.Add(last_packet_number - range_length, |
- last_packet_number + 1); |
- // Subtract an extra 1 to ensure ranges are represented efficiently and |
- // can't overlap by 1 packet number. This allows a missing_delta of 0 |
- // to represent an adjacent nack range. |
- last_packet_number -= (range_length + 1); |
- } |
- |
- if (quic_version_ > QUIC_VERSION_31) { |
- return true; |
- } |
- |
- // Parse the revived packets list. |
- // TODO(ianswett): Change the ack frame so it only expresses one revived. |
- uint8_t num_revived_packets; |
- if (!reader->ReadBytes(&num_revived_packets, 1)) { |
- set_detailed_error("Unable to read num revived packets."); |
- return false; |
- } |
- |
- for (size_t i = 0; i < num_revived_packets; ++i) { |
- QuicPacketNumber revived_packet = 0; |
- if (!reader->ReadBytes(&revived_packet, |
- largest_observed_packet_number_length)) { |
- set_detailed_error("Unable to read revived packet."); |
- return false; |
- } |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessNewAckFrame(QuicDataReader* reader, |
- uint8_t frame_type, |
- QuicAckFrame* ack_frame) { |
- // Determine the two lengths from the frame type: largest acked length, |
- // ack block length. |
- const QuicPacketNumberLength ack_block_length = |
- ReadSequenceNumberLength(frame_type); |
- frame_type >>= kQuicSequenceNumberLengthShift; |
- const QuicPacketNumberLength largest_acked_length = |
- ReadSequenceNumberLength(frame_type); |
- frame_type >>= kQuicSequenceNumberLengthShift; |
- frame_type >>= kQuicHasMultipleAckBlocksShift; |
- bool has_ack_blocks = frame_type & kQuicHasMultipleAckBlocksMask; |
- ack_frame->missing = false; |
- |
- if (!reader->ReadBytes(&ack_frame->largest_observed, largest_acked_length)) { |
- set_detailed_error("Unable to read largest acked."); |
- return false; |
- } |
- |
- uint64_t ack_delay_time_us; |
- if (!reader->ReadUFloat16(&ack_delay_time_us)) { |
- set_detailed_error("Unable to read ack delay time."); |
- return false; |
- } |
- |
- if (ack_delay_time_us == kUFloat16MaxValue) { |
- ack_frame->ack_delay_time = QuicTime::Delta::Infinite(); |
- } else { |
- ack_frame->ack_delay_time = |
- QuicTime::Delta::FromMicroseconds(ack_delay_time_us); |
- } |
- |
- uint8_t num_ack_blocks = 0; |
- if (has_ack_blocks) { |
- if (!reader->ReadBytes(&num_ack_blocks, 1)) { |
- set_detailed_error("Unable to read num of ack blocks."); |
- return false; |
- } |
- } |
- |
- size_t first_block_length = 0; |
- if (!reader->ReadBytes(&first_block_length, ack_block_length)) { |
- set_detailed_error("Unable to read first ack block length."); |
- return false; |
- } |
- QuicPacketNumber first_received = |
- ack_frame->largest_observed + 1 - first_block_length; |
- ack_frame->packets.Add(first_received, ack_frame->largest_observed + 1); |
- |
- if (num_ack_blocks > 0) { |
- for (size_t i = 0; i < num_ack_blocks; ++i) { |
- size_t gap = 0; |
- if (!reader->ReadBytes(&gap, PACKET_1BYTE_PACKET_NUMBER)) { |
- set_detailed_error("Unable to read gap to next ack block."); |
- return false; |
- } |
- size_t current_block_length = 0; |
- if (!reader->ReadBytes(¤t_block_length, ack_block_length)) { |
- set_detailed_error("Unable to ack block length."); |
- return false; |
- } |
- first_received -= (gap + current_block_length); |
- if (current_block_length > 0) { |
- ack_frame->packets.Add(first_received, |
- first_received + current_block_length); |
- } |
- } |
- } |
- |
- if (!ProcessTimestampsInAckFrame(reader, ack_frame)) { |
- return false; |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessTimestampsInAckFrame(QuicDataReader* reader, |
- QuicAckFrame* ack_frame) { |
- if (ack_frame->is_truncated) { |
- return true; |
- } |
- uint8_t num_received_packets; |
- if (!reader->ReadBytes(&num_received_packets, 1)) { |
- set_detailed_error("Unable to read num received packets."); |
- return false; |
- } |
- |
- if (num_received_packets > 0) { |
- uint8_t delta_from_largest_observed; |
- if (!reader->ReadBytes(&delta_from_largest_observed, |
- PACKET_1BYTE_PACKET_NUMBER)) { |
- set_detailed_error("Unable to read sequence delta in received packets."); |
- return false; |
- } |
- QuicPacketNumber seq_num = |
- ack_frame->largest_observed - delta_from_largest_observed; |
- |
- // Time delta from the framer creation. |
- uint32_t time_delta_us; |
- if (!reader->ReadBytes(&time_delta_us, sizeof(time_delta_us))) { |
- set_detailed_error("Unable to read time delta in received packets."); |
- return false; |
- } |
- |
- last_timestamp_ = CalculateTimestampFromWire(time_delta_us); |
- |
- ack_frame->received_packet_times.reserve(num_received_packets); |
- ack_frame->received_packet_times.push_back( |
- std::make_pair(seq_num, creation_time_ + last_timestamp_)); |
- |
- for (uint8_t i = 1; i < num_received_packets; ++i) { |
- if (!reader->ReadBytes(&delta_from_largest_observed, |
- PACKET_1BYTE_PACKET_NUMBER)) { |
- set_detailed_error( |
- "Unable to read sequence delta in received packets."); |
- return false; |
- } |
- seq_num = ack_frame->largest_observed - delta_from_largest_observed; |
- |
- // Time delta from the previous timestamp. |
- uint64_t incremental_time_delta_us; |
- if (!reader->ReadUFloat16(&incremental_time_delta_us)) { |
- set_detailed_error( |
- "Unable to read incremental time delta in received packets."); |
- return false; |
- } |
- |
- last_timestamp_ = last_timestamp_ + QuicTime::Delta::FromMicroseconds( |
- incremental_time_delta_us); |
- ack_frame->received_packet_times.push_back( |
- std::make_pair(seq_num, creation_time_ + last_timestamp_)); |
- } |
- } |
- return true; |
-} |
- |
-bool QuicFramer::ProcessStopWaitingFrame(QuicDataReader* reader, |
- const QuicPacketHeader& header, |
- QuicStopWaitingFrame* stop_waiting) { |
- if (quic_version_ <= QUIC_VERSION_33) { |
- if (!reader->ReadBytes(&stop_waiting->entropy_hash, 1)) { |
- set_detailed_error("Unable to read entropy hash for sent packets."); |
- return false; |
- } |
- } |
- |
- QuicPacketNumber least_unacked_delta = 0; |
- if (!reader->ReadBytes(&least_unacked_delta, |
- header.public_header.packet_number_length)) { |
- set_detailed_error("Unable to read least unacked delta."); |
- return false; |
- } |
- DCHECK_GE(header.packet_number, least_unacked_delta); |
- stop_waiting->least_unacked = header.packet_number - least_unacked_delta; |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessRstStreamFrame(QuicDataReader* reader, |
- QuicRstStreamFrame* frame) { |
- if (!reader->ReadUInt32(&frame->stream_id)) { |
- set_detailed_error("Unable to read stream_id."); |
- return false; |
- } |
- |
- if (!reader->ReadUInt64(&frame->byte_offset)) { |
- set_detailed_error("Unable to read rst stream sent byte offset."); |
- return false; |
- } |
- |
- uint32_t error_code; |
- if (!reader->ReadUInt32(&error_code)) { |
- set_detailed_error("Unable to read rst stream error code."); |
- return false; |
- } |
- |
- if (error_code >= QUIC_STREAM_LAST_ERROR) { |
- // Ignore invalid stream error code if any. |
- error_code = QUIC_STREAM_LAST_ERROR; |
- } |
- |
- frame->error_code = static_cast<QuicRstStreamErrorCode>(error_code); |
- return true; |
-} |
- |
-bool QuicFramer::ProcessConnectionCloseFrame(QuicDataReader* reader, |
- QuicConnectionCloseFrame* frame) { |
- uint32_t error_code; |
- if (!reader->ReadUInt32(&error_code)) { |
- set_detailed_error("Unable to read connection close error code."); |
- return false; |
- } |
- |
- if (error_code >= QUIC_LAST_ERROR) { |
- // Ignore invalid QUIC error code if any. |
- error_code = QUIC_LAST_ERROR; |
- } |
- |
- frame->error_code = static_cast<QuicErrorCode>(error_code); |
- |
- StringPiece error_details; |
- if (!reader->ReadStringPiece16(&error_details)) { |
- set_detailed_error("Unable to read connection close error details."); |
- return false; |
- } |
- frame->error_details = error_details.as_string(); |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessGoAwayFrame(QuicDataReader* reader, |
- QuicGoAwayFrame* frame) { |
- uint32_t error_code; |
- if (!reader->ReadUInt32(&error_code)) { |
- set_detailed_error("Unable to read go away error code."); |
- return false; |
- } |
- |
- if (error_code >= QUIC_LAST_ERROR) { |
- // Ignore invalid QUIC error code if any. |
- error_code = QUIC_LAST_ERROR; |
- } |
- frame->error_code = static_cast<QuicErrorCode>(error_code); |
- |
- uint32_t stream_id; |
- if (!reader->ReadUInt32(&stream_id)) { |
- set_detailed_error("Unable to read last good stream id."); |
- return false; |
- } |
- frame->last_good_stream_id = static_cast<QuicStreamId>(stream_id); |
- |
- StringPiece reason_phrase; |
- if (!reader->ReadStringPiece16(&reason_phrase)) { |
- set_detailed_error("Unable to read goaway reason."); |
- return false; |
- } |
- frame->reason_phrase = reason_phrase.as_string(); |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessWindowUpdateFrame(QuicDataReader* reader, |
- QuicWindowUpdateFrame* frame) { |
- if (!reader->ReadUInt32(&frame->stream_id)) { |
- set_detailed_error("Unable to read stream_id."); |
- return false; |
- } |
- |
- if (!reader->ReadUInt64(&frame->byte_offset)) { |
- set_detailed_error("Unable to read window byte_offset."); |
- return false; |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessBlockedFrame(QuicDataReader* reader, |
- QuicBlockedFrame* frame) { |
- if (!reader->ReadUInt32(&frame->stream_id)) { |
- set_detailed_error("Unable to read stream_id."); |
- return false; |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::ProcessPathCloseFrame(QuicDataReader* reader, |
- QuicPathCloseFrame* frame) { |
- if (!reader->ReadBytes(&frame->path_id, 1)) { |
- set_detailed_error("Unable to read path_id."); |
- return false; |
- } |
- |
- return true; |
-} |
- |
-// static |
-StringPiece QuicFramer::GetAssociatedDataFromEncryptedPacket( |
- QuicVersion version, |
- const QuicEncryptedPacket& encrypted, |
- QuicConnectionIdLength connection_id_length, |
- bool includes_version, |
- bool includes_path_id, |
- bool includes_diversification_nonce, |
- QuicPacketNumberLength packet_number_length) { |
- // TODO(ianswett): This is identical to QuicData::AssociatedData. |
- return StringPiece( |
- encrypted.data(), |
- GetStartOfEncryptedData(version, connection_id_length, includes_version, |
- includes_path_id, includes_diversification_nonce, |
- packet_number_length)); |
-} |
- |
-void QuicFramer::SetDecrypter(EncryptionLevel level, QuicDecrypter* decrypter) { |
- DCHECK(alternative_decrypter_.get() == nullptr); |
- DCHECK_GE(level, decrypter_level_); |
- decrypter_.reset(decrypter); |
- decrypter_level_ = level; |
-} |
- |
-void QuicFramer::SetAlternativeDecrypter(EncryptionLevel level, |
- QuicDecrypter* decrypter, |
- bool latch_once_used) { |
- alternative_decrypter_.reset(decrypter); |
- alternative_decrypter_level_ = level; |
- alternative_decrypter_latch_ = latch_once_used; |
-} |
- |
-const QuicDecrypter* QuicFramer::decrypter() const { |
- return decrypter_.get(); |
-} |
- |
-const QuicDecrypter* QuicFramer::alternative_decrypter() const { |
- return alternative_decrypter_.get(); |
-} |
- |
-void QuicFramer::SetEncrypter(EncryptionLevel level, QuicEncrypter* encrypter) { |
- DCHECK_GE(level, 0); |
- DCHECK_LT(level, NUM_ENCRYPTION_LEVELS); |
- encrypter_[level].reset(encrypter); |
-} |
- |
-size_t QuicFramer::EncryptInPlace(EncryptionLevel level, |
- QuicPathId path_id, |
- QuicPacketNumber packet_number, |
- size_t ad_len, |
- size_t total_len, |
- size_t buffer_len, |
- char* buffer) { |
- size_t output_length = 0; |
- if (!encrypter_[level]->EncryptPacket( |
- path_id, packet_number, |
- StringPiece(buffer, ad_len), // Associated data |
- StringPiece(buffer + ad_len, total_len - ad_len), // Plaintext |
- buffer + ad_len, // Destination buffer |
- &output_length, buffer_len - ad_len)) { |
- RaiseError(QUIC_ENCRYPTION_FAILURE); |
- return 0; |
- } |
- |
- return ad_len + output_length; |
-} |
- |
-size_t QuicFramer::EncryptPayload(EncryptionLevel level, |
- QuicPathId path_id, |
- QuicPacketNumber packet_number, |
- const QuicPacket& packet, |
- char* buffer, |
- size_t buffer_len) { |
- DCHECK(encrypter_[level].get() != nullptr); |
- |
- StringPiece associated_data = packet.AssociatedData(quic_version_); |
- // Copy in the header, because the encrypter only populates the encrypted |
- // plaintext content. |
- const size_t ad_len = associated_data.length(); |
- memmove(buffer, associated_data.data(), ad_len); |
- // Encrypt the plaintext into the buffer. |
- size_t output_length = 0; |
- if (!encrypter_[level]->EncryptPacket(path_id, packet_number, associated_data, |
- packet.Plaintext(quic_version_), |
- buffer + ad_len, &output_length, |
- buffer_len - ad_len)) { |
- RaiseError(QUIC_ENCRYPTION_FAILURE); |
- return 0; |
- } |
- |
- return ad_len + output_length; |
-} |
- |
-size_t QuicFramer::GetMaxPlaintextSize(size_t ciphertext_size) { |
- // In order to keep the code simple, we don't have the current encryption |
- // level to hand. Both the NullEncrypter and AES-GCM have a tag length of 12. |
- size_t min_plaintext_size = ciphertext_size; |
- |
- for (int i = ENCRYPTION_NONE; i < NUM_ENCRYPTION_LEVELS; i++) { |
- if (encrypter_[i].get() != nullptr) { |
- size_t size = encrypter_[i]->GetMaxPlaintextSize(ciphertext_size); |
- if (size < min_plaintext_size) { |
- min_plaintext_size = size; |
- } |
- } |
- } |
- |
- return min_plaintext_size; |
-} |
- |
-bool QuicFramer::DecryptPayload(QuicDataReader* encrypted_reader, |
- const QuicPacketHeader& header, |
- const QuicEncryptedPacket& packet, |
- char* decrypted_buffer, |
- size_t buffer_length, |
- size_t* decrypted_length) { |
- StringPiece encrypted = encrypted_reader->ReadRemainingPayload(); |
- DCHECK(decrypter_.get() != nullptr); |
- StringPiece associated_data = GetAssociatedDataFromEncryptedPacket( |
- quic_version_, packet, header.public_header.connection_id_length, |
- header.public_header.version_flag, header.public_header.multipath_flag, |
- header.public_header.nonce != nullptr, |
- header.public_header.packet_number_length); |
- |
- bool success = decrypter_->DecryptPacket( |
- header.path_id, header.packet_number, associated_data, encrypted, |
- decrypted_buffer, decrypted_length, buffer_length); |
- if (success) { |
- visitor_->OnDecryptedPacket(decrypter_level_); |
- } else if (alternative_decrypter_.get() != nullptr) { |
- if (header.public_header.nonce != nullptr) { |
- DCHECK_EQ(perspective_, Perspective::IS_CLIENT); |
- alternative_decrypter_->SetDiversificationNonce( |
- *header.public_header.nonce); |
- } |
- bool try_alternative_decryption = true; |
- if (alternative_decrypter_level_ == ENCRYPTION_INITIAL) { |
- if (perspective_ == Perspective::IS_CLIENT && |
- quic_version_ > QUIC_VERSION_32) { |
- if (header.public_header.nonce == nullptr) { |
- // Can not use INITIAL decryption without a diversification nonce. |
- try_alternative_decryption = false; |
- } |
- } else { |
- DCHECK(header.public_header.nonce == nullptr); |
- } |
- } |
- |
- if (try_alternative_decryption) { |
- success = alternative_decrypter_->DecryptPacket( |
- header.path_id, header.packet_number, associated_data, encrypted, |
- decrypted_buffer, decrypted_length, buffer_length); |
- } |
- if (success) { |
- visitor_->OnDecryptedPacket(alternative_decrypter_level_); |
- if (alternative_decrypter_latch_) { |
- // Switch to the alternative decrypter and latch so that we cannot |
- // switch back. |
- decrypter_.reset(alternative_decrypter_.release()); |
- decrypter_level_ = alternative_decrypter_level_; |
- alternative_decrypter_level_ = ENCRYPTION_NONE; |
- } else { |
- // Switch the alternative decrypter so that we use it first next time. |
- decrypter_.swap(alternative_decrypter_); |
- EncryptionLevel level = alternative_decrypter_level_; |
- alternative_decrypter_level_ = decrypter_level_; |
- decrypter_level_ = level; |
- } |
- } |
- } |
- |
- if (!success) { |
- DLOG(WARNING) << "DecryptPacket failed for packet_number:" |
- << header.packet_number; |
- return false; |
- } |
- |
- return true; |
-} |
- |
-size_t QuicFramer::GetAckFrameTimeStampSize(const QuicAckFrame& ack) { |
- if (ack.received_packet_times.empty()) { |
- return 0; |
- } |
- |
- return 5 + 3 * (ack.received_packet_times.size() - 1); |
-} |
- |
-size_t QuicFramer::GetAckFrameSize( |
- const QuicAckFrame& ack, |
- QuicPacketNumberLength packet_number_length) { |
- size_t ack_size = 0; |
- if (quic_version_ <= QUIC_VERSION_33) { |
- AckFrameInfo ack_info = GetAckFrameInfo(ack); |
- QuicPacketNumberLength largest_observed_length = |
- GetMinSequenceNumberLength(ack.largest_observed); |
- QuicPacketNumberLength missing_packet_number_length = |
- GetMinSequenceNumberLength(ack_info.max_delta); |
- |
- ack_size = GetMinAckFrameSize(quic_version_, largest_observed_length); |
- if (!ack_info.nack_ranges.empty()) { |
- ack_size += kNumberOfNackRangesSize; |
- if (quic_version_ <= QUIC_VERSION_31) { |
- ack_size += kNumberOfRevivedPacketsSize; |
- } |
- ack_size += min(ack_info.nack_ranges.size(), kMaxNackRanges) * |
- (missing_packet_number_length + PACKET_1BYTE_PACKET_NUMBER); |
- } |
- |
- // In version 23, if the ack will be truncated due to too many nack ranges, |
- // then do not include the number of timestamps (1 byte). |
- if (ack_info.nack_ranges.size() <= kMaxNackRanges) { |
- // 1 byte for the number of timestamps. |
- ack_size += 1; |
- ack_size += GetAckFrameTimeStampSize(ack); |
- } |
- |
- return ack_size; |
- } |
- |
- NewAckFrameInfo ack_info = |
- GetNewAckFrameInfo(ack, !FLAGS_quic_use_packet_number_queue_intervals); |
- QuicPacketNumberLength largest_acked_length = |
- GetMinSequenceNumberLength(ack.largest_observed); |
- QuicPacketNumberLength ack_block_length = |
- GetMinSequenceNumberLength(ack_info.max_block_length); |
- |
- ack_size = GetMinAckFrameSize(quic_version_, largest_acked_length); |
- // First ack block length. |
- ack_size += ack_block_length; |
- if (ack_info.num_ack_blocks != 0) { |
- ack_size += kNumberOfAckBlocksSize; |
- ack_size += min(ack_info.num_ack_blocks, kMaxAckBlocks) * |
- (ack_block_length + PACKET_1BYTE_PACKET_NUMBER); |
- } |
- |
- // Include timestamps. |
- ack_size += GetAckFrameTimeStampSize(ack); |
- |
- return ack_size; |
-} |
- |
-size_t QuicFramer::ComputeFrameLength( |
- const QuicFrame& frame, |
- bool last_frame_in_packet, |
- QuicPacketNumberLength packet_number_length) { |
- switch (frame.type) { |
- case STREAM_FRAME: |
- return GetMinStreamFrameSize(frame.stream_frame->stream_id, |
- frame.stream_frame->offset, |
- last_frame_in_packet) + |
- frame.stream_frame->data_length; |
- case ACK_FRAME: { |
- return GetAckFrameSize(*frame.ack_frame, packet_number_length); |
- } |
- case STOP_WAITING_FRAME: |
- return GetStopWaitingFrameSize(quic_version_, packet_number_length); |
- case MTU_DISCOVERY_FRAME: |
- // MTU discovery frames are serialized as ping frames. |
- case PING_FRAME: |
- // Ping has no payload. |
- return kQuicFrameTypeSize; |
- case RST_STREAM_FRAME: |
- return GetRstStreamFrameSize(); |
- case CONNECTION_CLOSE_FRAME: |
- return GetMinConnectionCloseFrameSize() + |
- frame.connection_close_frame->error_details.size(); |
- case GOAWAY_FRAME: |
- return GetMinGoAwayFrameSize() + frame.goaway_frame->reason_phrase.size(); |
- case WINDOW_UPDATE_FRAME: |
- return GetWindowUpdateFrameSize(); |
- case BLOCKED_FRAME: |
- return GetBlockedFrameSize(); |
- case PATH_CLOSE_FRAME: |
- return GetPathCloseFrameSize(); |
- case PADDING_FRAME: |
- DCHECK(false); |
- return 0; |
- case NUM_FRAME_TYPES: |
- DCHECK(false); |
- return 0; |
- } |
- |
- // Not reachable, but some Chrome compilers can't figure that out. *sigh* |
- DCHECK(false); |
- return 0; |
-} |
- |
-bool QuicFramer::AppendTypeByte(const QuicFrame& frame, |
- bool no_stream_frame_length, |
- QuicDataWriter* writer) { |
- uint8_t type_byte = 0; |
- switch (frame.type) { |
- case STREAM_FRAME: { |
- if (frame.stream_frame == nullptr) { |
- QUIC_BUG << "Failed to append STREAM frame with no stream_frame."; |
- } |
- // Fin bit. |
- type_byte |= frame.stream_frame->fin ? kQuicStreamFinMask : 0; |
- |
- // Data Length bit. |
- type_byte <<= kQuicStreamDataLengthShift; |
- type_byte |= no_stream_frame_length ? 0 : kQuicStreamDataLengthMask; |
- |
- // Offset 3 bits. |
- type_byte <<= kQuicStreamOffsetShift; |
- const size_t offset_len = GetStreamOffsetSize(frame.stream_frame->offset); |
- if (offset_len > 0) { |
- type_byte |= offset_len - 1; |
- } |
- |
- // stream id 2 bits. |
- type_byte <<= kQuicStreamIdShift; |
- type_byte |= GetStreamIdSize(frame.stream_frame->stream_id) - 1; |
- type_byte |= kQuicFrameTypeStreamMask; // Set Stream Frame Type to 1. |
- break; |
- } |
- case ACK_FRAME: |
- return true; |
- case MTU_DISCOVERY_FRAME: |
- type_byte = static_cast<uint8_t>(PING_FRAME); |
- break; |
- default: |
- type_byte = static_cast<uint8_t>(frame.type); |
- break; |
- } |
- |
- return writer->WriteUInt8(type_byte); |
-} |
- |
-// static |
-bool QuicFramer::AppendPacketSequenceNumber( |
- QuicPacketNumberLength packet_number_length, |
- QuicPacketNumber packet_number, |
- QuicDataWriter* writer) { |
- // Ensure the entire packet number can be written. |
- if (writer->capacity() - writer->length() < |
- static_cast<size_t>(packet_number_length)) { |
- return false; |
- } |
- switch (packet_number_length) { |
- case PACKET_1BYTE_PACKET_NUMBER: |
- return writer->WriteUInt8(packet_number & k1ByteSequenceNumberMask); |
- break; |
- case PACKET_2BYTE_PACKET_NUMBER: |
- return writer->WriteUInt16(packet_number & k2ByteSequenceNumberMask); |
- break; |
- case PACKET_4BYTE_PACKET_NUMBER: |
- return writer->WriteUInt32(packet_number & k4ByteSequenceNumberMask); |
- break; |
- case PACKET_6BYTE_PACKET_NUMBER: |
- return writer->WriteUInt48(packet_number & k6ByteSequenceNumberMask); |
- break; |
- default: |
- DCHECK(false) << "packet_number_length: " << packet_number_length; |
- return false; |
- } |
-} |
- |
-// static |
-bool QuicFramer::AppendAckBlock(uint8_t gap, |
- QuicPacketNumberLength length_length, |
- QuicPacketNumber length, |
- QuicDataWriter* writer) { |
- return AppendPacketSequenceNumber(PACKET_1BYTE_PACKET_NUMBER, gap, writer) && |
- AppendPacketSequenceNumber(length_length, length, writer); |
-} |
- |
-bool QuicFramer::AppendStreamFrame(const QuicStreamFrame& frame, |
- bool no_stream_frame_length, |
- QuicDataWriter* writer) { |
- if (!writer->WriteBytes(&frame.stream_id, GetStreamIdSize(frame.stream_id))) { |
- QUIC_BUG << "Writing stream id size failed."; |
- return false; |
- } |
- if (!writer->WriteBytes(&frame.offset, GetStreamOffsetSize(frame.offset))) { |
- QUIC_BUG << "Writing offset size failed."; |
- return false; |
- } |
- if (!no_stream_frame_length) { |
- if ((frame.data_length > numeric_limits<uint16_t>::max()) || |
- !writer->WriteUInt16(static_cast<uint16_t>(frame.data_length))) { |
- QUIC_BUG << "Writing stream frame length failed"; |
- return false; |
- } |
- } |
- |
- if (!writer->WriteBytes(frame.data_buffer, frame.data_length)) { |
- QUIC_BUG << "Writing frame data failed."; |
- return false; |
- } |
- return true; |
-} |
- |
-void QuicFramer::set_version(const QuicVersion version) { |
- DCHECK(IsSupportedVersion(version)) << QuicVersionToString(version); |
- quic_version_ = version; |
-} |
- |
-bool QuicFramer::AppendAckFrameAndTypeByte(const QuicPacketHeader& header, |
- const QuicAckFrame& frame, |
- QuicDataWriter* writer) { |
- AckFrameInfo ack_info = GetAckFrameInfo(frame); |
- QuicPacketNumber ack_largest_observed = frame.largest_observed; |
- QuicPacketNumberLength largest_observed_length = |
- GetMinSequenceNumberLength(ack_largest_observed); |
- QuicPacketNumberLength missing_packet_number_length = |
- GetMinSequenceNumberLength(ack_info.max_delta); |
- // Determine whether we need to truncate ranges. |
- size_t available_range_bytes = |
- writer->capacity() - writer->length() - kNumberOfNackRangesSize - |
- GetMinAckFrameSize(quic_version_, largest_observed_length); |
- if (quic_version_ <= QUIC_VERSION_31) { |
- available_range_bytes -= kNumberOfRevivedPacketsSize; |
- } |
- size_t max_num_ranges = |
- available_range_bytes / |
- (missing_packet_number_length + PACKET_1BYTE_PACKET_NUMBER); |
- max_num_ranges = min(kMaxNackRanges, max_num_ranges); |
- bool truncated = ack_info.nack_ranges.size() > max_num_ranges; |
- DVLOG_IF(1, truncated) << "Truncating ack from " |
- << ack_info.nack_ranges.size() << " ranges to " |
- << max_num_ranges; |
- // Write out the type byte by setting the low order bits and doing shifts |
- // to make room for the next bit flags to be set. |
- // Whether there are any nacks. |
- uint8_t type_byte = ack_info.nack_ranges.empty() ? 0 : kQuicHasNacksMask; |
- |
- // truncating bit. |
- type_byte <<= kQuicAckTruncatedShift; |
- type_byte |= truncated ? kQuicAckTruncatedMask : 0; |
- |
- // Largest observed packet number length. |
- type_byte <<= kQuicSequenceNumberLengthShift; |
- type_byte |= GetSequenceNumberFlags(largest_observed_length); |
- |
- // Missing packet number length. |
- type_byte <<= kQuicSequenceNumberLengthShift; |
- type_byte |= GetSequenceNumberFlags(missing_packet_number_length); |
- |
- type_byte |= kQuicFrameTypeAckMask; |
- |
- if (!writer->WriteUInt8(type_byte)) { |
- QUIC_BUG << "type byte failed"; |
- return false; |
- } |
- |
- QuicPacketEntropyHash ack_entropy_hash = frame.entropy_hash; |
- NackRangeMap::reverse_iterator ack_iter = ack_info.nack_ranges.rbegin(); |
- if (truncated) { |
- // Skip the nack ranges which the truncated ack won't include and set |
- // a correct largest observed for the truncated ack. |
- for (size_t i = 1; i < (ack_info.nack_ranges.size() - max_num_ranges); |
- ++i) { |
- ++ack_iter; |
- } |
- // If the last range is followed by acks, include them. |
- // If the last range is followed by another range, specify the end of the |
- // range as the largest_observed. |
- ack_largest_observed = ack_iter->first - 1; |
- // Also update the entropy so it matches the largest observed. |
- ack_entropy_hash = entropy_calculator_->EntropyHash(ack_largest_observed); |
- ++ack_iter; |
- } |
- |
- if (!writer->WriteUInt8(ack_entropy_hash)) { |
- QUIC_BUG << "hash failed."; |
- return false; |
- } |
- |
- if (!AppendPacketSequenceNumber(largest_observed_length, ack_largest_observed, |
- writer)) { |
- QUIC_BUG << "AppendPacketSequenceNumber failed. " |
- << "largest_observed_length: " << largest_observed_length |
- << " ack_largest_observed: " << ack_largest_observed; |
- return false; |
- } |
- |
- uint64_t ack_delay_time_us = kUFloat16MaxValue; |
- if (!frame.ack_delay_time.IsInfinite()) { |
- DCHECK_LE(0u, frame.ack_delay_time.ToMicroseconds()); |
- ack_delay_time_us = frame.ack_delay_time.ToMicroseconds(); |
- } |
- |
- if (!writer->WriteUFloat16(ack_delay_time_us)) { |
- QUIC_BUG << "ack delay time failed."; |
- return false; |
- } |
- |
- // Timestamp goes at the end of the required fields. |
- if (!truncated) { |
- if (!AppendTimestampToAckFrame(frame, writer)) { |
- QUIC_BUG << "AppendTimestampToAckFrame failed"; |
- return false; |
- } |
- } |
- |
- if (ack_info.nack_ranges.empty()) { |
- return true; |
- } |
- |
- const uint8_t num_missing_ranges = |
- static_cast<uint8_t>(min(ack_info.nack_ranges.size(), max_num_ranges)); |
- if (!writer->WriteBytes(&num_missing_ranges, 1)) { |
- QUIC_BUG << "num_missing_ranges failed: " |
- << static_cast<uint32_t>(num_missing_ranges); |
- return false; |
- } |
- |
- int num_ranges_written = 0; |
- QuicPacketNumber last_sequence_written = ack_largest_observed; |
- for (; ack_iter != ack_info.nack_ranges.rend(); ++ack_iter) { |
- // Calculate the delta to the last number in the range. |
- QuicPacketNumber missing_delta = |
- last_sequence_written - (ack_iter->first + ack_iter->second); |
- if (!AppendPacketSequenceNumber(missing_packet_number_length, missing_delta, |
- writer)) { |
- QUIC_BUG << "AppendPacketSequenceNumber failed: " |
- << "missing_packet_number_length: " |
- << missing_packet_number_length << " missing_delta " |
- << missing_delta; |
- return false; |
- } |
- if (!AppendPacketSequenceNumber(PACKET_1BYTE_PACKET_NUMBER, |
- ack_iter->second, writer)) { |
- QUIC_BUG << "AppendPacketSequenceNumber failed"; |
- return false; |
- } |
- // Subtract 1 so a missing_delta of 0 means an adjacent range. |
- last_sequence_written = ack_iter->first - 1; |
- ++num_ranges_written; |
- } |
- DCHECK_EQ(num_missing_ranges, num_ranges_written); |
- |
- if (quic_version_ > QUIC_VERSION_31) { |
- return true; |
- } |
- |
- // Append revived packets. |
- // FEC is not supported. |
- uint8_t num_revived_packets = 0; |
- if (!writer->WriteBytes(&num_revived_packets, 1)) { |
- QUIC_BUG << "num_revived_packets failed: " << num_revived_packets; |
- return false; |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::AppendNewAckFrameAndTypeByte(const QuicAckFrame& frame, |
- QuicDataWriter* writer) { |
- const bool new_ack_info_construct_blocks = |
- !FLAGS_quic_use_packet_number_queue_intervals; |
- const NewAckFrameInfo new_ack_info = |
- GetNewAckFrameInfo(frame, new_ack_info_construct_blocks); |
- QuicPacketNumber largest_acked = frame.largest_observed; |
- QuicPacketNumberLength largest_acked_length = |
- GetMinSequenceNumberLength(largest_acked); |
- QuicPacketNumberLength ack_block_length = |
- GetMinSequenceNumberLength(new_ack_info.max_block_length); |
- // Calculate available bytes for timestamps and ack blocks. |
- int32_t available_timestamp_and_ack_block_bytes = |
- writer->capacity() - writer->length() - ack_block_length - |
- GetMinAckFrameSize(quic_version_, largest_acked_length) - |
- (new_ack_info.num_ack_blocks != 0 ? kNumberOfAckBlocksSize : 0); |
- DCHECK_LE(0, available_timestamp_and_ack_block_bytes); |
- |
- // Write out the type byte by setting the low order bits and doing shifts |
- // to make room for the next bit flags to be set. |
- // Whether there are multiple ack blocks. |
- uint8_t type_byte = |
- new_ack_info.num_ack_blocks == 0 ? 0 : kQuicHasMultipleAckBlocksMask; |
- type_byte <<= kQuicHasMultipleAckBlocksShift; |
- |
- // Largest acked length. |
- type_byte <<= kQuicSequenceNumberLengthShift; |
- type_byte |= GetSequenceNumberFlags(largest_acked_length); |
- |
- // Ack block length. |
- type_byte <<= kQuicSequenceNumberLengthShift; |
- type_byte |= GetSequenceNumberFlags(ack_block_length); |
- |
- type_byte |= kQuicFrameTypeAckMask; |
- |
- if (!writer->WriteUInt8(type_byte)) { |
- return false; |
- } |
- |
- // Largest acked. |
- if (!AppendPacketSequenceNumber(largest_acked_length, largest_acked, |
- writer)) { |
- return false; |
- } |
- |
- // Largest acked delta time. |
- uint64_t ack_delay_time_us = kUFloat16MaxValue; |
- if (!frame.ack_delay_time.IsInfinite()) { |
- DCHECK_LE(0u, frame.ack_delay_time.ToMicroseconds()); |
- ack_delay_time_us = frame.ack_delay_time.ToMicroseconds(); |
- } |
- if (!writer->WriteUFloat16(ack_delay_time_us)) { |
- return false; |
- } |
- |
- size_t max_num_ack_blocks = available_timestamp_and_ack_block_bytes / |
- (ack_block_length + PACKET_1BYTE_PACKET_NUMBER); |
- |
- // Number of ack blocks. |
- size_t num_ack_blocks = min(new_ack_info.num_ack_blocks, max_num_ack_blocks); |
- if (num_ack_blocks > numeric_limits<uint8_t>::max()) { |
- num_ack_blocks = numeric_limits<uint8_t>::max(); |
- } |
- |
- if (num_ack_blocks > 0) { |
- if (!writer->WriteBytes(&num_ack_blocks, 1)) { |
- return false; |
- } |
- } |
- |
- // First ack block length. |
- if (!AppendPacketSequenceNumber(ack_block_length, |
- new_ack_info.first_block_length, writer)) { |
- return false; |
- } |
- |
- // Ack blocks. |
- if (num_ack_blocks > 0) { |
- size_t num_ack_blocks_written = 0; |
- if (!new_ack_info_construct_blocks) { |
- // Append, in descending order from the largest ACKed packet, a series of |
- // ACK blocks that represents the successfully acknoweldged packets. Each |
- // appended gap/block length represents a descending delta from the |
- // previous block. i.e.: |
- // |--- length ---|--- gap ---|--- length ---|--- gap ---|--- largest ---| |
- // For gaps larger than can be represented by a single encoded gap, a 0 |
- // length gap of the maximum is used, i.e.: |
- // |--- length ---|--- gap ---|- 0 -|--- gap ---|--- largest ---| |
- auto itr = frame.packets.rbegin_intervals(); |
- QuicPacketNumber previous_start = itr->min(); |
- ++itr; |
- |
- for (; itr != frame.packets.rend_intervals() && |
- num_ack_blocks_written < num_ack_blocks; |
- previous_start = itr->min(), ++itr) { |
- const auto& interval = *itr; |
- const QuicPacketNumber total_gap = previous_start - interval.max(); |
- const size_t num_encoded_gaps = |
- (total_gap + numeric_limits<uint8_t>::max() - 1) / |
- numeric_limits<uint8_t>::max(); |
- DCHECK_GT(num_encoded_gaps, 0u); |
- |
- // Append empty ACK blocks because the gap is longer than a single gap. |
- for (size_t i = 1; |
- i < num_encoded_gaps && num_ack_blocks_written < num_ack_blocks; |
- ++i) { |
- if (!AppendAckBlock(numeric_limits<uint8_t>::max(), ack_block_length, |
- 0, writer)) { |
- return false; |
- } |
- ++num_ack_blocks_written; |
- } |
- if (num_ack_blocks_written >= num_ack_blocks) { |
- if (PREDICT_FALSE(num_ack_blocks_written != num_ack_blocks)) { |
- QUIC_BUG << "Wrote " << num_ack_blocks_written |
- << ", expected to write " << num_ack_blocks; |
- } |
- break; |
- } |
- |
- const uint8_t last_gap = |
- total_gap - (num_encoded_gaps - 1) * numeric_limits<uint8_t>::max(); |
- // Append the final ACK block with a non-empty size. |
- if (!AppendAckBlock(last_gap, ack_block_length, interval.Length(), |
- writer)) { |
- return false; |
- } |
- ++num_ack_blocks_written; |
- } |
- } else { |
- DCHECK_EQ(new_ack_info.num_ack_blocks, new_ack_info.ack_blocks.size()); |
- vector<AckBlock>::const_reverse_iterator iter = |
- new_ack_info.ack_blocks.rbegin(); |
- for (; iter != new_ack_info.ack_blocks.rend(); ++iter) { |
- if (!AppendPacketSequenceNumber(PACKET_1BYTE_PACKET_NUMBER, iter->gap, |
- writer)) { |
- return false; |
- } |
- if (!AppendPacketSequenceNumber(ack_block_length, iter->length, |
- writer)) { |
- return false; |
- } |
- if (++num_ack_blocks_written == num_ack_blocks) { |
- break; |
- } |
- } |
- } |
- DCHECK_EQ(num_ack_blocks, num_ack_blocks_written); |
- } |
- |
- // Timestamps. |
- // If we don't have enough available space to append all the timestamps, don't |
- // append any of them. |
- if (writer->capacity() - writer->length() >= |
- GetAckFrameTimeStampSize(frame)) { |
- if (!AppendTimestampToAckFrame(frame, writer)) { |
- return false; |
- } |
- } else { |
- uint8_t num_received_packets = 0; |
- if (!writer->WriteBytes(&num_received_packets, 1)) { |
- return false; |
- } |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::AppendTimestampToAckFrame(const QuicAckFrame& frame, |
- QuicDataWriter* writer) { |
- DCHECK_GE(numeric_limits<uint8_t>::max(), frame.received_packet_times.size()); |
- // num_received_packets is only 1 byte. |
- if (frame.received_packet_times.size() > numeric_limits<uint8_t>::max()) { |
- return false; |
- } |
- |
- uint8_t num_received_packets = frame.received_packet_times.size(); |
- if (!writer->WriteBytes(&num_received_packets, 1)) { |
- return false; |
- } |
- if (num_received_packets == 0) { |
- return true; |
- } |
- |
- PacketTimeVector::const_iterator it = frame.received_packet_times.begin(); |
- QuicPacketNumber packet_number = it->first; |
- QuicPacketNumber delta_from_largest_observed = |
- frame.largest_observed - packet_number; |
- |
- DCHECK_GE(numeric_limits<uint8_t>::max(), delta_from_largest_observed); |
- if (delta_from_largest_observed > numeric_limits<uint8_t>::max()) { |
- return false; |
- } |
- |
- if (!writer->WriteUInt8(delta_from_largest_observed & |
- k1ByteSequenceNumberMask)) { |
- return false; |
- } |
- |
- // Use the lowest 4 bytes of the time delta from the creation_time_. |
- const uint64_t time_epoch_delta_us = UINT64_C(1) << 32; |
- uint32_t time_delta_us = |
- static_cast<uint32_t>((it->second - creation_time_).ToMicroseconds() & |
- (time_epoch_delta_us - 1)); |
- if (!writer->WriteBytes(&time_delta_us, sizeof(time_delta_us))) { |
- return false; |
- } |
- |
- QuicTime prev_time = it->second; |
- |
- for (++it; it != frame.received_packet_times.end(); ++it) { |
- packet_number = it->first; |
- delta_from_largest_observed = frame.largest_observed - packet_number; |
- |
- if (delta_from_largest_observed > numeric_limits<uint8_t>::max()) { |
- return false; |
- } |
- |
- if (!writer->WriteUInt8(delta_from_largest_observed & |
- k1ByteSequenceNumberMask)) { |
- return false; |
- } |
- |
- uint64_t frame_time_delta_us = (it->second - prev_time).ToMicroseconds(); |
- prev_time = it->second; |
- if (!writer->WriteUFloat16(frame_time_delta_us)) { |
- return false; |
- } |
- } |
- return true; |
-} |
- |
-bool QuicFramer::AppendStopWaitingFrame(const QuicPacketHeader& header, |
- const QuicStopWaitingFrame& frame, |
- QuicDataWriter* writer) { |
- DCHECK_GE(header.packet_number, frame.least_unacked); |
- const QuicPacketNumber least_unacked_delta = |
- header.packet_number - frame.least_unacked; |
- const QuicPacketNumber length_shift = |
- header.public_header.packet_number_length * 8; |
- if (quic_version_ <= QUIC_VERSION_33) { |
- if (!writer->WriteUInt8(frame.entropy_hash)) { |
- QUIC_BUG << " hash failed"; |
- return false; |
- } |
- } |
- |
- if (least_unacked_delta >> length_shift > 0) { |
- QUIC_BUG << "packet_number_length " |
- << header.public_header.packet_number_length |
- << " is too small for least_unacked_delta: " << least_unacked_delta |
- << " packet_number:" << header.packet_number |
- << " least_unacked:" << frame.least_unacked |
- << " version:" << quic_version_; |
- return false; |
- } |
- if (!AppendPacketSequenceNumber(header.public_header.packet_number_length, |
- least_unacked_delta, writer)) { |
- QUIC_BUG << " seq failed: " << header.public_header.packet_number_length; |
- return false; |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::AppendRstStreamFrame(const QuicRstStreamFrame& frame, |
- QuicDataWriter* writer) { |
- if (!writer->WriteUInt32(frame.stream_id)) { |
- return false; |
- } |
- |
- if (!writer->WriteUInt64(frame.byte_offset)) { |
- return false; |
- } |
- |
- uint32_t error_code = static_cast<uint32_t>(frame.error_code); |
- if (!writer->WriteUInt32(error_code)) { |
- return false; |
- } |
- |
- return true; |
-} |
- |
-bool QuicFramer::AppendConnectionCloseFrame( |
- const QuicConnectionCloseFrame& frame, |
- QuicDataWriter* writer) { |
- uint32_t error_code = static_cast<uint32_t>(frame.error_code); |
- if (!writer->WriteUInt32(error_code)) { |
- return false; |
- } |
- if (!writer->WriteStringPiece16(frame.error_details)) { |
- return false; |
- } |
- return true; |
-} |
- |
-bool QuicFramer::AppendGoAwayFrame(const QuicGoAwayFrame& frame, |
- QuicDataWriter* writer) { |
- uint32_t error_code = static_cast<uint32_t>(frame.error_code); |
- if (!writer->WriteUInt32(error_code)) { |
- return false; |
- } |
- uint32_t stream_id = static_cast<uint32_t>(frame.last_good_stream_id); |
- if (!writer->WriteUInt32(stream_id)) { |
- return false; |
- } |
- if (!writer->WriteStringPiece16(frame.reason_phrase)) { |
- return false; |
- } |
- return true; |
-} |
- |
-bool QuicFramer::AppendWindowUpdateFrame(const QuicWindowUpdateFrame& frame, |
- QuicDataWriter* writer) { |
- uint32_t stream_id = static_cast<uint32_t>(frame.stream_id); |
- if (!writer->WriteUInt32(stream_id)) { |
- return false; |
- } |
- if (!writer->WriteUInt64(frame.byte_offset)) { |
- return false; |
- } |
- return true; |
-} |
- |
-bool QuicFramer::AppendBlockedFrame(const QuicBlockedFrame& frame, |
- QuicDataWriter* writer) { |
- uint32_t stream_id = static_cast<uint32_t>(frame.stream_id); |
- if (!writer->WriteUInt32(stream_id)) { |
- return false; |
- } |
- return true; |
-} |
- |
-bool QuicFramer::AppendPathCloseFrame(const QuicPathCloseFrame& frame, |
- QuicDataWriter* writer) { |
- uint8_t path_id = static_cast<uint8_t>(frame.path_id); |
- if (!writer->WriteUInt8(path_id)) { |
- return false; |
- } |
- return true; |
-} |
- |
-bool QuicFramer::RaiseError(QuicErrorCode error) { |
- DVLOG(1) << "Error: " << QuicUtils::ErrorToString(error) |
- << " detail: " << detailed_error_; |
- set_error(error); |
- visitor_->OnError(this); |
- return false; |
-} |
- |
-} // namespace net |