| // 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/third_party/quiche/src/quic/core/quic_dispatcher.h" |
| |
| #include <string> |
| #include <utility> |
| |
| #include "net/third_party/quiche/src/quic/core/chlo_extractor.h" |
| #include "net/third_party/quiche/src/quic/core/crypto/crypto_protocol.h" |
| #include "net/third_party/quiche/src/quic/core/crypto/quic_random.h" |
| #include "net/third_party/quiche/src/quic/core/quic_error_codes.h" |
| #include "net/third_party/quiche/src/quic/core/quic_time_wait_list_manager.h" |
| #include "net/third_party/quiche/src/quic/core/quic_types.h" |
| #include "net/third_party/quiche/src/quic/core/quic_utils.h" |
| #include "net/third_party/quiche/src/quic/core/quic_versions.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_bug_tracker.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_flag_utils.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_flags.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_logging.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_ptr_util.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_stack_trace.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_string_piece.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_text_utils.h" |
| |
| namespace quic { |
| |
| typedef QuicBufferedPacketStore::BufferedPacket BufferedPacket; |
| typedef QuicBufferedPacketStore::BufferedPacketList BufferedPacketList; |
| typedef QuicBufferedPacketStore::EnqueuePacketResult EnqueuePacketResult; |
| |
| namespace { |
| |
| // An alarm that informs the QuicDispatcher to delete old sessions. |
| class DeleteSessionsAlarm : public QuicAlarm::Delegate { |
| public: |
| explicit DeleteSessionsAlarm(QuicDispatcher* dispatcher) |
| : dispatcher_(dispatcher) {} |
| DeleteSessionsAlarm(const DeleteSessionsAlarm&) = delete; |
| DeleteSessionsAlarm& operator=(const DeleteSessionsAlarm&) = delete; |
| |
| void OnAlarm() override { dispatcher_->DeleteSessions(); } |
| |
| private: |
| // Not owned. |
| QuicDispatcher* dispatcher_; |
| }; |
| |
| // Collects packets serialized by a QuicPacketCreator in order |
| // to be handed off to the time wait list manager. |
| class PacketCollector : public QuicPacketCreator::DelegateInterface, |
| public QuicStreamFrameDataProducer { |
| public: |
| explicit PacketCollector(QuicBufferAllocator* allocator) |
| : send_buffer_(allocator) {} |
| ~PacketCollector() override = default; |
| |
| // QuicPacketCreator::DelegateInterface methods: |
| void OnSerializedPacket(SerializedPacket* serialized_packet) override { |
| // Make a copy of the serialized packet to send later. |
| packets_.emplace_back( |
| new QuicEncryptedPacket(CopyBuffer(*serialized_packet), |
| serialized_packet->encrypted_length, true)); |
| serialized_packet->encrypted_buffer = nullptr; |
| DeleteFrames(&(serialized_packet->retransmittable_frames)); |
| serialized_packet->retransmittable_frames.clear(); |
| } |
| |
| char* GetPacketBuffer() override { |
| // Let QuicPacketCreator to serialize packets on stack buffer. |
| return nullptr; |
| } |
| |
| void OnUnrecoverableError(QuicErrorCode /*error*/, |
| const std::string& /*error_details*/) override {} |
| |
| // QuicStreamFrameDataProducer |
| WriteStreamDataResult WriteStreamData(QuicStreamId /*id*/, |
| QuicStreamOffset offset, |
| QuicByteCount data_length, |
| QuicDataWriter* writer) override { |
| if (send_buffer_.WriteStreamData(offset, data_length, writer)) { |
| return WRITE_SUCCESS; |
| } |
| return WRITE_FAILED; |
| } |
| bool WriteCryptoData(EncryptionLevel /*level*/, |
| QuicStreamOffset offset, |
| QuicByteCount data_length, |
| QuicDataWriter* writer) override { |
| return send_buffer_.WriteStreamData(offset, data_length, writer); |
| } |
| |
| std::vector<std::unique_ptr<QuicEncryptedPacket>>* packets() { |
| return &packets_; |
| } |
| |
| private: |
| std::vector<std::unique_ptr<QuicEncryptedPacket>> packets_; |
| // This is only needed until the packets are encrypted. Once packets are |
| // encrypted, the stream data is no longer required. |
| QuicStreamSendBuffer send_buffer_; |
| }; |
| |
| // Helper for statelessly closing connections by generating the |
| // correct termination packets and adding the connection to the time wait |
| // list manager. |
| class StatelessConnectionTerminator { |
| public: |
| StatelessConnectionTerminator(QuicConnectionId server_connection_id, |
| const ParsedQuicVersion version, |
| QuicConnectionHelperInterface* helper, |
| QuicTimeWaitListManager* time_wait_list_manager) |
| : server_connection_id_(server_connection_id), |
| framer_(ParsedQuicVersionVector{version}, |
| /*unused*/ QuicTime::Zero(), |
| Perspective::IS_SERVER, |
| /*unused*/ kQuicDefaultConnectionIdLength), |
| collector_(helper->GetStreamSendBufferAllocator()), |
| creator_(server_connection_id, &framer_, &collector_), |
| time_wait_list_manager_(time_wait_list_manager) { |
| framer_.set_data_producer(&collector_); |
| } |
| |
| ~StatelessConnectionTerminator() { |
| // Clear framer's producer. |
| framer_.set_data_producer(nullptr); |
| } |
| |
| // Generates a packet containing a CONNECTION_CLOSE frame specifying |
| // |error_code| and |error_details| and add the connection to time wait. |
| void CloseConnection(QuicErrorCode error_code, |
| const std::string& error_details, |
| bool ietf_quic) { |
| QuicConnectionCloseFrame* frame = |
| new QuicConnectionCloseFrame(error_code, error_details); |
| if (VersionHasIetfQuicFrames(framer_.transport_version())) { |
| frame->close_type = IETF_QUIC_TRANSPORT_CONNECTION_CLOSE; |
| } |
| |
| if (!creator_.AddSavedFrame(QuicFrame(frame), NOT_RETRANSMISSION)) { |
| QUIC_BUG << "Unable to add frame to an empty packet"; |
| delete frame; |
| return; |
| } |
| creator_.Flush(); |
| DCHECK_EQ(1u, collector_.packets()->size()); |
| time_wait_list_manager_->AddConnectionIdToTimeWait( |
| server_connection_id_, ietf_quic, |
| QuicTimeWaitListManager::SEND_TERMINATION_PACKETS, |
| quic::ENCRYPTION_INITIAL, collector_.packets()); |
| } |
| |
| private: |
| QuicConnectionId server_connection_id_; |
| QuicFramer framer_; |
| // Set as the visitor of |creator_| to collect any generated packets. |
| PacketCollector collector_; |
| QuicPacketCreator creator_; |
| QuicTimeWaitListManager* time_wait_list_manager_; |
| }; |
| |
| // Class which extracts the ALPN from a CHLO packet. |
| class ChloAlpnExtractor : public ChloExtractor::Delegate { |
| public: |
| void OnChlo(QuicTransportVersion /*version*/, |
| QuicConnectionId /*server_connection_id*/, |
| const CryptoHandshakeMessage& chlo) override { |
| QuicStringPiece alpn_value; |
| if (chlo.GetStringPiece(kALPN, &alpn_value)) { |
| alpn_ = std::string(alpn_value); |
| } |
| } |
| |
| std::string&& ConsumeAlpn() { return std::move(alpn_); } |
| |
| private: |
| std::string alpn_; |
| }; |
| |
| } // namespace |
| |
| QuicDispatcher::QuicDispatcher( |
| const QuicConfig* config, |
| const QuicCryptoServerConfig* crypto_config, |
| QuicVersionManager* version_manager, |
| std::unique_ptr<QuicConnectionHelperInterface> helper, |
| std::unique_ptr<QuicCryptoServerStream::Helper> session_helper, |
| std::unique_ptr<QuicAlarmFactory> alarm_factory, |
| uint8_t expected_server_connection_id_length) |
| : config_(config), |
| crypto_config_(crypto_config), |
| compressed_certs_cache_( |
| QuicCompressedCertsCache::kQuicCompressedCertsCacheSize), |
| helper_(std::move(helper)), |
| session_helper_(std::move(session_helper)), |
| alarm_factory_(std::move(alarm_factory)), |
| delete_sessions_alarm_( |
| alarm_factory_->CreateAlarm(new DeleteSessionsAlarm(this))), |
| buffered_packets_(this, helper_->GetClock(), alarm_factory_.get()), |
| version_manager_(version_manager), |
| last_error_(QUIC_NO_ERROR), |
| new_sessions_allowed_per_event_loop_(0u), |
| accept_new_connections_(true), |
| allow_short_initial_server_connection_ids_(false), |
| expected_server_connection_id_length_( |
| expected_server_connection_id_length), |
| should_update_expected_server_connection_id_length_(false) {} |
| |
| QuicDispatcher::~QuicDispatcher() { |
| session_map_.clear(); |
| closed_session_list_.clear(); |
| } |
| |
| void QuicDispatcher::InitializeWithWriter(QuicPacketWriter* writer) { |
| DCHECK(writer_ == nullptr); |
| writer_.reset(writer); |
| time_wait_list_manager_.reset(CreateQuicTimeWaitListManager()); |
| } |
| |
| void QuicDispatcher::ProcessPacket(const QuicSocketAddress& self_address, |
| const QuicSocketAddress& peer_address, |
| const QuicReceivedPacket& packet) { |
| QUIC_DVLOG(2) << "Dispatcher received encrypted " << packet.length() |
| << " bytes:" << std::endl |
| << QuicTextUtils::HexDump( |
| QuicStringPiece(packet.data(), packet.length())); |
| ReceivedPacketInfo packet_info(self_address, peer_address, packet); |
| std::string detailed_error; |
| const QuicErrorCode error = QuicFramer::ProcessPacketDispatcher( |
| packet, expected_server_connection_id_length_, &packet_info.form, |
| &packet_info.version_flag, &packet_info.version_label, |
| &packet_info.destination_connection_id, &packet_info.source_connection_id, |
| &detailed_error); |
| if (error != QUIC_NO_ERROR) { |
| // Packet has framing error. |
| SetLastError(error); |
| QUIC_DLOG(ERROR) << detailed_error; |
| return; |
| } |
| packet_info.version = ParseQuicVersionLabel(packet_info.version_label); |
| if (packet_info.destination_connection_id.length() != |
| expected_server_connection_id_length_ && |
| !should_update_expected_server_connection_id_length_ && |
| !QuicUtils::VariableLengthConnectionIdAllowedForVersion( |
| packet_info.version.transport_version)) { |
| SetLastError(QUIC_INVALID_PACKET_HEADER); |
| QUIC_DLOG(ERROR) << "Invalid Connection Id Length"; |
| return; |
| } |
| if (should_update_expected_server_connection_id_length_) { |
| expected_server_connection_id_length_ = |
| packet_info.destination_connection_id.length(); |
| } |
| |
| if (MaybeDispatchPacket(packet_info)) { |
| // Packet has been dropped or successfully dispatched, stop processing. |
| return; |
| } |
| ProcessHeader(&packet_info); |
| } |
| |
| QuicConnectionId QuicDispatcher::MaybeReplaceServerConnectionId( |
| QuicConnectionId server_connection_id, |
| ParsedQuicVersion version) { |
| if (server_connection_id.length() == expected_server_connection_id_length_) { |
| return server_connection_id; |
| } |
| DCHECK(QuicUtils::VariableLengthConnectionIdAllowedForVersion( |
| version.transport_version)); |
| auto it = connection_id_map_.find(server_connection_id); |
| if (it != connection_id_map_.end()) { |
| return it->second; |
| } |
| QuicConnectionId new_connection_id = |
| session_helper_->GenerateConnectionIdForReject(version.transport_version, |
| server_connection_id); |
| DCHECK_EQ(expected_server_connection_id_length_, new_connection_id.length()); |
| // TODO(dschinazi) Prevent connection_id_map_ from growing indefinitely |
| // before we ship a version that supports variable length connection IDs |
| // to production. |
| connection_id_map_.insert( |
| std::make_pair(server_connection_id, new_connection_id)); |
| QUIC_DLOG(INFO) << "Replacing incoming connection ID " << server_connection_id |
| << " with " << new_connection_id; |
| return new_connection_id; |
| } |
| |
| bool QuicDispatcher::MaybeDispatchPacket( |
| const ReceivedPacketInfo& packet_info) { |
| // Port zero is only allowed for unidirectional UDP, so is disallowed by QUIC. |
| // Given that we can't even send a reply rejecting the packet, just drop the |
| // packet. |
| if (packet_info.peer_address.port() == 0) { |
| return true; |
| } |
| |
| QuicConnectionId server_connection_id = packet_info.destination_connection_id; |
| |
| // The IETF spec requires the client to generate an initial server |
| // connection ID that is at least 64 bits long. After that initial |
| // connection ID, the dispatcher picks a new one of its expected length. |
| // Therefore we should never receive a connection ID that is smaller |
| // than 64 bits and smaller than what we expect. |
| if (server_connection_id.length() < kQuicMinimumInitialConnectionIdLength && |
| server_connection_id.length() < expected_server_connection_id_length_ && |
| !allow_short_initial_server_connection_ids_) { |
| DCHECK(packet_info.version_flag); |
| DCHECK(QuicUtils::VariableLengthConnectionIdAllowedForVersion( |
| packet_info.version.transport_version)); |
| QUIC_DLOG(INFO) << "Packet with short destination connection ID " |
| << server_connection_id << " expected " |
| << static_cast<int>(expected_server_connection_id_length_); |
| if (!GetQuicReloadableFlag(quic_drop_invalid_small_initial_connection_id)) { |
| // Add this connection_id to the time-wait state, to safely reject |
| // future packets. |
| QUIC_DLOG(INFO) << "Adding connection ID " << server_connection_id |
| << " to time-wait list."; |
| StatelesslyTerminateConnection( |
| server_connection_id, packet_info.form, packet_info.version_flag, |
| packet_info.version, QUIC_HANDSHAKE_FAILED, "Reject connection", |
| quic::QuicTimeWaitListManager::SEND_STATELESS_RESET); |
| |
| DCHECK(time_wait_list_manager_->IsConnectionIdInTimeWait( |
| server_connection_id)); |
| time_wait_list_manager_->ProcessPacket( |
| packet_info.self_address, packet_info.peer_address, |
| server_connection_id, packet_info.form, GetPerPacketContext()); |
| |
| buffered_packets_.DiscardPackets(server_connection_id); |
| } else { |
| QUIC_RELOADABLE_FLAG_COUNT(quic_drop_invalid_small_initial_connection_id); |
| // Drop the packet silently. |
| } |
| return true; |
| } |
| |
| // Packets with connection IDs for active connections are processed |
| // immediately. |
| auto it = session_map_.find(server_connection_id); |
| if (it != session_map_.end()) { |
| DCHECK(!buffered_packets_.HasBufferedPackets(server_connection_id)); |
| it->second->ProcessUdpPacket(packet_info.self_address, |
| packet_info.peer_address, packet_info.packet); |
| return true; |
| } else { |
| // We did not find the connection ID, check if we've replaced it. |
| QuicConnectionId replaced_connection_id = MaybeReplaceServerConnectionId( |
| server_connection_id, packet_info.version); |
| if (replaced_connection_id != server_connection_id) { |
| // Search for the replacement. |
| auto it2 = session_map_.find(replaced_connection_id); |
| if (it2 != session_map_.end()) { |
| DCHECK(!buffered_packets_.HasBufferedPackets(replaced_connection_id)); |
| it2->second->ProcessUdpPacket(packet_info.self_address, |
| packet_info.peer_address, |
| packet_info.packet); |
| return true; |
| } |
| } |
| } |
| |
| if (buffered_packets_.HasChloForConnection(server_connection_id)) { |
| BufferEarlyPacket(packet_info); |
| return true; |
| } |
| |
| if (OnFailedToDispatchPacket(packet_info)) { |
| return true; |
| } |
| |
| if (time_wait_list_manager_->IsConnectionIdInTimeWait(server_connection_id)) { |
| // This connection ID is already in time-wait state. |
| time_wait_list_manager_->ProcessPacket( |
| packet_info.self_address, packet_info.peer_address, |
| packet_info.destination_connection_id, packet_info.form, |
| GetPerPacketContext()); |
| return true; |
| } |
| |
| // The packet has an unknown connection ID. |
| |
| // Unless the packet provides a version, assume that we can continue |
| // processing using our preferred version. |
| if (packet_info.version_flag) { |
| if (!IsSupportedVersion(packet_info.version)) { |
| if (ShouldCreateSessionForUnknownVersion(packet_info.version_label)) { |
| return false; |
| } |
| if (!crypto_config()->validate_chlo_size() || |
| packet_info.packet.length() >= kMinPacketSizeForVersionNegotiation) { |
| // Since the version is not supported, send a version negotiation |
| // packet and stop processing the current packet. |
| QuicConnectionId client_connection_id = |
| packet_info.source_connection_id; |
| time_wait_list_manager()->SendVersionNegotiationPacket( |
| server_connection_id, client_connection_id, |
| packet_info.form != GOOGLE_QUIC_PACKET, GetSupportedVersions(), |
| packet_info.self_address, packet_info.peer_address, |
| GetPerPacketContext()); |
| } |
| return true; |
| } |
| } |
| |
| return false; |
| } |
| |
| void QuicDispatcher::ProcessHeader(ReceivedPacketInfo* packet_info) { |
| QuicConnectionId server_connection_id = |
| packet_info->destination_connection_id; |
| // Packet's connection ID is unknown. Apply the validity checks. |
| // TODO(wub): Determine the fate completely in ValidityChecks, then call |
| // ProcessUnauthenticatedHeaderFate in one place. |
| QuicPacketFate fate = ValidityChecks(*packet_info); |
| ChloAlpnExtractor alpn_extractor; |
| switch (fate) { |
| case kFateProcess: { |
| if (packet_info->version.handshake_protocol == PROTOCOL_TLS1_3) { |
| // TODO(nharper): Support buffering non-ClientHello packets when using |
| // TLS. |
| ProcessChlo(/*alpn=*/"", packet_info); |
| break; |
| } |
| ParsedQuicVersionVector chlo_extractor_versions; |
| if (!GetQuicRestartFlag( |
| quic_dispatcher_hands_chlo_extractor_one_version)) { |
| chlo_extractor_versions = GetSupportedVersions(); |
| } else { |
| QUIC_RESTART_FLAG_COUNT( |
| quic_dispatcher_hands_chlo_extractor_one_version); |
| chlo_extractor_versions = {packet_info->version}; |
| // TODO(dschinazi) once we deprecate |
| // quic_dispatcher_hands_chlo_extractor_one_version, we should change |
| // ChloExtractor::Extract to only take one version. |
| } |
| if (GetQuicFlag(FLAGS_quic_allow_chlo_buffering) && |
| !ChloExtractor::Extract(packet_info->packet, chlo_extractor_versions, |
| config_->create_session_tag_indicators(), |
| &alpn_extractor, |
| server_connection_id.length())) { |
| // Buffer non-CHLO packets. |
| BufferEarlyPacket(*packet_info); |
| break; |
| } |
| ProcessChlo(alpn_extractor.ConsumeAlpn(), packet_info); |
| } break; |
| case kFateTimeWait: |
| // Add this connection_id to the time-wait state, to safely reject |
| // future packets. |
| QUIC_DLOG(INFO) << "Adding connection ID " << server_connection_id |
| << " to time-wait list."; |
| QUIC_CODE_COUNT(quic_reject_fate_time_wait); |
| StatelesslyTerminateConnection( |
| server_connection_id, packet_info->form, packet_info->version_flag, |
| packet_info->version, QUIC_HANDSHAKE_FAILED, "Reject connection", |
| quic::QuicTimeWaitListManager::SEND_STATELESS_RESET); |
| |
| DCHECK(time_wait_list_manager_->IsConnectionIdInTimeWait( |
| server_connection_id)); |
| time_wait_list_manager_->ProcessPacket( |
| packet_info->self_address, packet_info->peer_address, |
| server_connection_id, packet_info->form, GetPerPacketContext()); |
| |
| buffered_packets_.DiscardPackets(server_connection_id); |
| break; |
| case kFateDrop: |
| break; |
| } |
| } |
| |
| QuicDispatcher::QuicPacketFate QuicDispatcher::ValidityChecks( |
| const ReceivedPacketInfo& packet_info) { |
| // To have all the checks work properly without tears, insert any new check |
| // into the framework of this method in the section for checks that return the |
| // check's fate value. The sections for checks must be ordered with the |
| // highest priority fate first. |
| |
| // All packets within a connection sent by a client before receiving a |
| // response from the server are required to have the version negotiation flag |
| // set. Since this may be a client continuing a connection we lost track of |
| // via server restart, send a rejection to fast-fail the connection. |
| if (!packet_info.version_flag) { |
| QUIC_DLOG(INFO) |
| << "Packet without version arrived for unknown connection ID " |
| << packet_info.destination_connection_id; |
| if (GetQuicReloadableFlag(quic_reject_unprocessable_packets_statelessly)) { |
| QUIC_RELOADABLE_FLAG_COUNT(quic_reject_unprocessable_packets_statelessly); |
| MaybeResetPacketsWithNoVersion(packet_info); |
| return kFateDrop; |
| } |
| return kFateTimeWait; |
| } |
| |
| // Let the connection parse and validate packet number. |
| return kFateProcess; |
| } |
| |
| void QuicDispatcher::CleanUpSession(SessionMap::iterator it, |
| QuicConnection* connection, |
| ConnectionCloseSource /*source*/) { |
| write_blocked_list_.erase(connection); |
| QuicTimeWaitListManager::TimeWaitAction action = |
| QuicTimeWaitListManager::SEND_STATELESS_RESET; |
| if (connection->termination_packets() != nullptr && |
| !connection->termination_packets()->empty()) { |
| action = QuicTimeWaitListManager::SEND_TERMINATION_PACKETS; |
| } else { |
| if (!connection->IsHandshakeConfirmed()) { |
| if (!VersionHasIetfInvariantHeader(connection->transport_version())) { |
| QUIC_CODE_COUNT(gquic_add_to_time_wait_list_with_handshake_failed); |
| } else { |
| QUIC_CODE_COUNT(quic_v44_add_to_time_wait_list_with_handshake_failed); |
| } |
| action = QuicTimeWaitListManager::SEND_TERMINATION_PACKETS; |
| // This serializes a connection close termination packet with error code |
| // QUIC_HANDSHAKE_FAILED and adds the connection to the time wait list. |
| StatelesslyTerminateConnection( |
| connection->connection_id(), |
| VersionHasIetfInvariantHeader(connection->transport_version()) |
| ? IETF_QUIC_LONG_HEADER_PACKET |
| : GOOGLE_QUIC_PACKET, |
| /*version_flag=*/true, connection->version(), QUIC_HANDSHAKE_FAILED, |
| "Connection is closed by server before handshake confirmed", |
| // Although it is our intention to send termination packets, the |
| // |action| argument is not used by this call to |
| // StatelesslyTerminateConnection(). |
| action); |
| session_map_.erase(it); |
| return; |
| } |
| QUIC_CODE_COUNT(quic_v44_add_to_time_wait_list_with_stateless_reset); |
| } |
| time_wait_list_manager_->AddConnectionIdToTimeWait( |
| it->first, VersionHasIetfInvariantHeader(connection->transport_version()), |
| action, connection->encryption_level(), |
| connection->termination_packets()); |
| session_map_.erase(it); |
| } |
| |
| void QuicDispatcher::StopAcceptingNewConnections() { |
| accept_new_connections_ = false; |
| } |
| |
| std::unique_ptr<QuicPerPacketContext> QuicDispatcher::GetPerPacketContext() |
| const { |
| return nullptr; |
| } |
| |
| void QuicDispatcher::DeleteSessions() { |
| if (!write_blocked_list_.empty()) { |
| for (const std::unique_ptr<QuicSession>& session : closed_session_list_) { |
| if (write_blocked_list_.erase(session->connection()) != 0) { |
| QUIC_BUG << "QuicConnection was in WriteBlockedList before destruction"; |
| } |
| } |
| } |
| closed_session_list_.clear(); |
| } |
| |
| void QuicDispatcher::OnCanWrite() { |
| // The socket is now writable. |
| writer_->SetWritable(); |
| |
| // Move every blocked writer in |write_blocked_list_| to a temporary list. |
| const size_t num_blocked_writers_before = write_blocked_list_.size(); |
| WriteBlockedList temp_list; |
| temp_list.swap(write_blocked_list_); |
| DCHECK(write_blocked_list_.empty()); |
| |
| // Give each blocked writer a chance to write what they indended to write. |
| // If they are blocked again, they will call |OnWriteBlocked| to add |
| // themselves back into |write_blocked_list_|. |
| while (!temp_list.empty()) { |
| QuicBlockedWriterInterface* blocked_writer = temp_list.begin()->first; |
| temp_list.erase(temp_list.begin()); |
| blocked_writer->OnBlockedWriterCanWrite(); |
| } |
| const size_t num_blocked_writers_after = write_blocked_list_.size(); |
| if (num_blocked_writers_after != 0) { |
| if (num_blocked_writers_before == num_blocked_writers_after) { |
| QUIC_CODE_COUNT(quic_zero_progress_on_can_write); |
| } else { |
| QUIC_CODE_COUNT(quic_blocked_again_on_can_write); |
| } |
| } |
| } |
| |
| bool QuicDispatcher::HasPendingWrites() const { |
| return !write_blocked_list_.empty(); |
| } |
| |
| void QuicDispatcher::Shutdown() { |
| while (!session_map_.empty()) { |
| QuicSession* session = session_map_.begin()->second.get(); |
| session->connection()->CloseConnection( |
| QUIC_PEER_GOING_AWAY, "Server shutdown imminent", |
| ConnectionCloseBehavior::SEND_CONNECTION_CLOSE_PACKET); |
| // Validate that the session removes itself from the session map on close. |
| DCHECK(session_map_.empty() || |
| session_map_.begin()->second.get() != session); |
| } |
| DeleteSessions(); |
| } |
| |
| void QuicDispatcher::OnConnectionClosed(QuicConnectionId server_connection_id, |
| QuicErrorCode error, |
| const std::string& error_details, |
| ConnectionCloseSource source) { |
| auto it = session_map_.find(server_connection_id); |
| if (it == session_map_.end()) { |
| QUIC_BUG << "ConnectionId " << server_connection_id |
| << " does not exist in the session map. Error: " |
| << QuicErrorCodeToString(error); |
| QUIC_BUG << QuicStackTrace(); |
| return; |
| } |
| |
| QUIC_DLOG_IF(INFO, error != QUIC_NO_ERROR) |
| << "Closing connection (" << server_connection_id |
| << ") due to error: " << QuicErrorCodeToString(error) |
| << ", with details: " << error_details; |
| |
| QuicConnection* connection = it->second->connection(); |
| if (ShouldDestroySessionAsynchronously()) { |
| // Set up alarm to fire immediately to bring destruction of this session |
| // out of current call stack. |
| if (closed_session_list_.empty()) { |
| delete_sessions_alarm_->Update(helper()->GetClock()->ApproximateNow(), |
| QuicTime::Delta::Zero()); |
| } |
| closed_session_list_.push_back(std::move(it->second)); |
| } |
| CleanUpSession(it, connection, source); |
| } |
| |
| void QuicDispatcher::OnWriteBlocked( |
| QuicBlockedWriterInterface* blocked_writer) { |
| if (!blocked_writer->IsWriterBlocked()) { |
| // It is a programming error if this ever happens. When we are sure it is |
| // not happening, replace it with a DCHECK. |
| QUIC_BUG |
| << "Tried to add writer into blocked list when it shouldn't be added"; |
| // Return without adding the connection to the blocked list, to avoid |
| // infinite loops in OnCanWrite. |
| return; |
| } |
| |
| write_blocked_list_.insert(std::make_pair(blocked_writer, true)); |
| } |
| |
| void QuicDispatcher::OnRstStreamReceived(const QuicRstStreamFrame& /*frame*/) {} |
| |
| void QuicDispatcher::OnStopSendingReceived( |
| const QuicStopSendingFrame& /*frame*/) {} |
| |
| void QuicDispatcher::OnConnectionAddedToTimeWaitList( |
| QuicConnectionId server_connection_id) { |
| QUIC_DLOG(INFO) << "Connection " << server_connection_id |
| << " added to time wait list."; |
| } |
| |
| void QuicDispatcher::StatelesslyTerminateConnection( |
| QuicConnectionId server_connection_id, |
| PacketHeaderFormat format, |
| bool version_flag, |
| ParsedQuicVersion version, |
| QuicErrorCode error_code, |
| const std::string& error_details, |
| QuicTimeWaitListManager::TimeWaitAction action) { |
| if (format != IETF_QUIC_LONG_HEADER_PACKET && !version_flag) { |
| QUIC_DVLOG(1) << "Statelessly terminating " << server_connection_id |
| << " based on a non-ietf-long packet, action:" << action |
| << ", error_code:" << error_code |
| << ", error_details:" << error_details; |
| time_wait_list_manager_->AddConnectionIdToTimeWait( |
| server_connection_id, format != GOOGLE_QUIC_PACKET, action, |
| ENCRYPTION_INITIAL, nullptr); |
| return; |
| } |
| |
| // If the version is known and supported by framer, send a connection close. |
| if (IsSupportedVersion(version)) { |
| QUIC_DVLOG(1) |
| << "Statelessly terminating " << server_connection_id |
| << " based on an ietf-long packet, which has a supported version:" |
| << version << ", error_code:" << error_code |
| << ", error_details:" << error_details; |
| |
| StatelessConnectionTerminator terminator(server_connection_id, version, |
| helper_.get(), |
| time_wait_list_manager_.get()); |
| // This also adds the connection to time wait list. |
| terminator.CloseConnection(error_code, error_details, |
| format != GOOGLE_QUIC_PACKET); |
| return; |
| } |
| |
| QUIC_DVLOG(1) |
| << "Statelessly terminating " << server_connection_id |
| << " based on an ietf-long packet, which has an unsupported version:" |
| << version << ", error_code:" << error_code |
| << ", error_details:" << error_details; |
| // Version is unknown or unsupported by framer, send a version negotiation |
| // with an empty version list, which can be understood by the client. |
| std::vector<std::unique_ptr<QuicEncryptedPacket>> termination_packets; |
| termination_packets.push_back(QuicFramer::BuildVersionNegotiationPacket( |
| server_connection_id, EmptyQuicConnectionId(), |
| /*ietf_quic=*/format != GOOGLE_QUIC_PACKET, |
| /*versions=*/{})); |
| time_wait_list_manager()->AddConnectionIdToTimeWait( |
| server_connection_id, /*ietf_quic=*/format != GOOGLE_QUIC_PACKET, |
| QuicTimeWaitListManager::SEND_TERMINATION_PACKETS, ENCRYPTION_INITIAL, |
| &termination_packets); |
| } |
| |
| bool QuicDispatcher::ShouldCreateSessionForUnknownVersion( |
| QuicVersionLabel /*version_label*/) { |
| return false; |
| } |
| |
| void QuicDispatcher::OnExpiredPackets( |
| QuicConnectionId server_connection_id, |
| BufferedPacketList early_arrived_packets) { |
| QUIC_CODE_COUNT(quic_reject_buffered_packets_expired); |
| StatelesslyTerminateConnection( |
| server_connection_id, |
| early_arrived_packets.ietf_quic ? IETF_QUIC_LONG_HEADER_PACKET |
| : GOOGLE_QUIC_PACKET, |
| /*version_flag=*/true, early_arrived_packets.version, |
| QUIC_HANDSHAKE_FAILED, "Packets buffered for too long", |
| quic::QuicTimeWaitListManager::SEND_STATELESS_RESET); |
| } |
| |
| void QuicDispatcher::ProcessBufferedChlos(size_t max_connections_to_create) { |
| // Reset the counter before starting creating connections. |
| new_sessions_allowed_per_event_loop_ = max_connections_to_create; |
| for (; new_sessions_allowed_per_event_loop_ > 0; |
| --new_sessions_allowed_per_event_loop_) { |
| QuicConnectionId server_connection_id; |
| BufferedPacketList packet_list = |
| buffered_packets_.DeliverPacketsForNextConnection( |
| &server_connection_id); |
| const std::list<BufferedPacket>& packets = packet_list.buffered_packets; |
| if (packets.empty()) { |
| return; |
| } |
| QuicConnectionId original_connection_id = server_connection_id; |
| server_connection_id = MaybeReplaceServerConnectionId(server_connection_id, |
| packet_list.version); |
| QuicSession* session = |
| CreateQuicSession(server_connection_id, packets.front().peer_address, |
| packet_list.alpn, packet_list.version); |
| if (original_connection_id != server_connection_id) { |
| session->connection()->AddIncomingConnectionId(original_connection_id); |
| session->connection()->InstallInitialCrypters(original_connection_id); |
| } |
| QUIC_DLOG(INFO) << "Created new session for " << server_connection_id; |
| |
| DCHECK(session_map_.find(server_connection_id) == session_map_.end()) |
| << "Tried to add session map existing entry " << server_connection_id; |
| |
| session_map_.insert( |
| std::make_pair(server_connection_id, QuicWrapUnique(session))); |
| DeliverPacketsToSession(packets, session); |
| } |
| } |
| |
| bool QuicDispatcher::HasChlosBuffered() const { |
| return buffered_packets_.HasChlosBuffered(); |
| } |
| |
| bool QuicDispatcher::ShouldCreateOrBufferPacketForConnection( |
| const ReceivedPacketInfo& packet_info) { |
| QUIC_VLOG(1) << "Received packet from new connection " |
| << packet_info.destination_connection_id; |
| return true; |
| } |
| |
| // Return true if there is any packet buffered in the store. |
| bool QuicDispatcher::HasBufferedPackets(QuicConnectionId server_connection_id) { |
| return buffered_packets_.HasBufferedPackets(server_connection_id); |
| } |
| |
| void QuicDispatcher::OnBufferPacketFailure( |
| EnqueuePacketResult result, |
| QuicConnectionId server_connection_id) { |
| QUIC_DLOG(INFO) << "Fail to buffer packet on connection " |
| << server_connection_id << " because of " << result; |
| } |
| |
| QuicTimeWaitListManager* QuicDispatcher::CreateQuicTimeWaitListManager() { |
| return new QuicTimeWaitListManager(writer_.get(), this, helper_->GetClock(), |
| alarm_factory_.get()); |
| } |
| |
| void QuicDispatcher::BufferEarlyPacket(const ReceivedPacketInfo& packet_info) { |
| bool is_new_connection = !buffered_packets_.HasBufferedPackets( |
| packet_info.destination_connection_id); |
| if (is_new_connection && |
| !ShouldCreateOrBufferPacketForConnection(packet_info)) { |
| return; |
| } |
| |
| EnqueuePacketResult rs = buffered_packets_.EnqueuePacket( |
| packet_info.destination_connection_id, |
| packet_info.form != GOOGLE_QUIC_PACKET, packet_info.packet, |
| packet_info.self_address, packet_info.peer_address, /*is_chlo=*/false, |
| /*alpn=*/"", packet_info.version); |
| if (rs != EnqueuePacketResult::SUCCESS) { |
| OnBufferPacketFailure(rs, packet_info.destination_connection_id); |
| } |
| } |
| |
| void QuicDispatcher::ProcessChlo(const std::string& alpn, |
| ReceivedPacketInfo* packet_info) { |
| if (!accept_new_connections_) { |
| // Don't any create new connection. |
| QUIC_CODE_COUNT(quic_reject_stop_accepting_new_connections); |
| StatelesslyTerminateConnection( |
| packet_info->destination_connection_id, packet_info->form, |
| /*version_flag=*/true, packet_info->version, QUIC_HANDSHAKE_FAILED, |
| "Stop accepting new connections", |
| quic::QuicTimeWaitListManager::SEND_STATELESS_RESET); |
| // Time wait list will reject the packet correspondingly. |
| time_wait_list_manager()->ProcessPacket( |
| packet_info->self_address, packet_info->peer_address, |
| packet_info->destination_connection_id, packet_info->form, |
| GetPerPacketContext()); |
| return; |
| } |
| if (!buffered_packets_.HasBufferedPackets( |
| packet_info->destination_connection_id) && |
| !ShouldCreateOrBufferPacketForConnection(*packet_info)) { |
| return; |
| } |
| if (GetQuicFlag(FLAGS_quic_allow_chlo_buffering) && |
| new_sessions_allowed_per_event_loop_ <= 0) { |
| // Can't create new session any more. Wait till next event loop. |
| QUIC_BUG_IF(buffered_packets_.HasChloForConnection( |
| packet_info->destination_connection_id)); |
| EnqueuePacketResult rs = buffered_packets_.EnqueuePacket( |
| packet_info->destination_connection_id, |
| packet_info->form != GOOGLE_QUIC_PACKET, packet_info->packet, |
| packet_info->self_address, packet_info->peer_address, |
| /*is_chlo=*/true, alpn, packet_info->version); |
| if (rs != EnqueuePacketResult::SUCCESS) { |
| OnBufferPacketFailure(rs, packet_info->destination_connection_id); |
| } |
| return; |
| } |
| |
| QuicConnectionId original_connection_id = |
| packet_info->destination_connection_id; |
| packet_info->destination_connection_id = MaybeReplaceServerConnectionId( |
| original_connection_id, packet_info->version); |
| // Creates a new session and process all buffered packets for this connection. |
| QuicSession* session = |
| CreateQuicSession(packet_info->destination_connection_id, |
| packet_info->peer_address, alpn, packet_info->version); |
| if (original_connection_id != packet_info->destination_connection_id) { |
| session->connection()->AddIncomingConnectionId(original_connection_id); |
| session->connection()->InstallInitialCrypters(original_connection_id); |
| } |
| QUIC_DLOG(INFO) << "Created new session for " |
| << packet_info->destination_connection_id; |
| |
| DCHECK(session_map_.find(packet_info->destination_connection_id) == |
| session_map_.end()) |
| << "Tried to add session map existing entry " |
| << packet_info->destination_connection_id; |
| |
| session_map_.insert(std::make_pair(packet_info->destination_connection_id, |
| QuicWrapUnique(session))); |
| std::list<BufferedPacket> packets = |
| buffered_packets_.DeliverPackets(packet_info->destination_connection_id) |
| .buffered_packets; |
| // Process CHLO at first. |
| session->ProcessUdpPacket(packet_info->self_address, |
| packet_info->peer_address, packet_info->packet); |
| // Deliver queued-up packets in the same order as they arrived. |
| // Do this even when flag is off because there might be still some packets |
| // buffered in the store before flag is turned off. |
| DeliverPacketsToSession(packets, session); |
| --new_sessions_allowed_per_event_loop_; |
| } |
| |
| bool QuicDispatcher::ShouldDestroySessionAsynchronously() { |
| return true; |
| } |
| |
| void QuicDispatcher::SetLastError(QuicErrorCode error) { |
| last_error_ = error; |
| } |
| |
| bool QuicDispatcher::OnFailedToDispatchPacket( |
| const ReceivedPacketInfo& /*packet_info*/) { |
| return false; |
| } |
| |
| const QuicTransportVersionVector& |
| QuicDispatcher::GetSupportedTransportVersions() { |
| return version_manager_->GetSupportedTransportVersions(); |
| } |
| |
| const ParsedQuicVersionVector& QuicDispatcher::GetSupportedVersions() { |
| return version_manager_->GetSupportedVersions(); |
| } |
| |
| void QuicDispatcher::DeliverPacketsToSession( |
| const std::list<BufferedPacket>& packets, |
| QuicSession* session) { |
| for (const BufferedPacket& packet : packets) { |
| session->ProcessUdpPacket(packet.self_address, packet.peer_address, |
| *(packet.packet)); |
| } |
| } |
| |
| bool QuicDispatcher::IsSupportedVersion(const ParsedQuicVersion version) { |
| for (const ParsedQuicVersion& supported_version : |
| version_manager_->GetSupportedVersions()) { |
| if (version == supported_version) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| void QuicDispatcher::MaybeResetPacketsWithNoVersion( |
| const ReceivedPacketInfo& packet_info) { |
| DCHECK(!packet_info.version_flag); |
| const size_t MinValidPacketLength = |
| kPacketHeaderTypeSize + expected_server_connection_id_length_ + |
| PACKET_1BYTE_PACKET_NUMBER + /*payload size=*/1 + /*tag size=*/12; |
| if (packet_info.packet.length() < MinValidPacketLength) { |
| // The packet size is too small. |
| QUIC_CODE_COUNT(drop_too_small_packets); |
| return; |
| } |
| // TODO(fayang): Consider rate limiting reset packets if reset packet size > |
| // packet_length. |
| |
| time_wait_list_manager()->SendPublicReset( |
| packet_info.self_address, packet_info.peer_address, |
| packet_info.destination_connection_id, |
| packet_info.form != GOOGLE_QUIC_PACKET, GetPerPacketContext()); |
| } |
| |
| } // namespace quic |