| // Copyright (c) 2019 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 "quic/qbone/qbone_session_base.h" | 
 |  | 
 | #include <netinet/icmp6.h> | 
 | #include <netinet/ip6.h> | 
 |  | 
 | #include <utility> | 
 |  | 
 | #include "absl/strings/string_view.h" | 
 | #include "quic/core/quic_buffer_allocator.h" | 
 | #include "quic/core/quic_data_reader.h" | 
 | #include "quic/core/quic_types.h" | 
 | #include "quic/platform/api/quic_exported_stats.h" | 
 | #include "quic/platform/api/quic_logging.h" | 
 | #include "quic/platform/api/quic_testvalue.h" | 
 | #include "quic/qbone/platform/icmp_packet.h" | 
 | #include "quic/qbone/qbone_constants.h" | 
 |  | 
 | DEFINE_QUIC_COMMAND_LINE_FLAG( | 
 |     bool, | 
 |     qbone_close_ephemeral_frames, | 
 |     true, | 
 |     "If true, we'll call CloseStream even when we receive ephemeral frames."); | 
 |  | 
 | namespace quic { | 
 |  | 
 | #define ENDPOINT \ | 
 |   (perspective() == Perspective::IS_SERVER ? "Server: " : "Client: ") | 
 |  | 
 | QboneSessionBase::QboneSessionBase( | 
 |     QuicConnection* connection, | 
 |     Visitor* owner, | 
 |     const QuicConfig& config, | 
 |     const ParsedQuicVersionVector& supported_versions, | 
 |     QbonePacketWriter* writer) | 
 |     : QuicSession(connection, | 
 |                   owner, | 
 |                   config, | 
 |                   supported_versions, | 
 |                   /*num_expected_unidirectional_static_streams = */ 0) { | 
 |   set_writer(writer); | 
 |   const uint32_t max_streams = | 
 |       (std::numeric_limits<uint32_t>::max() / kMaxAvailableStreamsMultiplier) - | 
 |       1; | 
 |   this->config()->SetMaxBidirectionalStreamsToSend(max_streams); | 
 |   if (VersionHasIetfQuicFrames(transport_version())) { | 
 |     this->config()->SetMaxUnidirectionalStreamsToSend(max_streams); | 
 |   } | 
 | } | 
 |  | 
 | QboneSessionBase::~QboneSessionBase() {} | 
 |  | 
 | void QboneSessionBase::Initialize() { | 
 |   crypto_stream_ = CreateCryptoStream(); | 
 |   QuicSession::Initialize(); | 
 | } | 
 |  | 
 | const QuicCryptoStream* QboneSessionBase::GetCryptoStream() const { | 
 |   return crypto_stream_.get(); | 
 | } | 
 |  | 
 | QuicCryptoStream* QboneSessionBase::GetMutableCryptoStream() { | 
 |   return crypto_stream_.get(); | 
 | } | 
 |  | 
 | QuicStream* QboneSessionBase::CreateOutgoingStream() { | 
 |   return ActivateDataStream( | 
 |       CreateDataStream(GetNextOutgoingUnidirectionalStreamId())); | 
 | } | 
 |  | 
 | void QboneSessionBase::OnStreamFrame(const QuicStreamFrame& frame) { | 
 |   if (frame.offset == 0 && frame.fin && frame.data_length > 0) { | 
 |     ++num_ephemeral_packets_; | 
 |     ProcessPacketFromPeer( | 
 |         absl::string_view(frame.data_buffer, frame.data_length)); | 
 |     flow_controller()->AddBytesConsumed(frame.data_length); | 
 |     // TODO(b/147817422): Add a counter for how many streams were actually | 
 |     // closed here. | 
 |     if (GetQuicFlag(FLAGS_qbone_close_ephemeral_frames)) { | 
 |       ResetStream(frame.stream_id, QUIC_STREAM_CANCELLED); | 
 |     } | 
 |     return; | 
 |   } | 
 |   QuicSession::OnStreamFrame(frame); | 
 | } | 
 |  | 
 | void QboneSessionBase::OnMessageReceived(absl::string_view message) { | 
 |   ++num_message_packets_; | 
 |   ProcessPacketFromPeer(message); | 
 | } | 
 |  | 
 | QuicStream* QboneSessionBase::CreateIncomingStream(QuicStreamId id) { | 
 |   return ActivateDataStream(CreateDataStream(id)); | 
 | } | 
 |  | 
 | QuicStream* QboneSessionBase::CreateIncomingStream(PendingStream* /*pending*/) { | 
 |   QUIC_NOTREACHED(); | 
 |   return nullptr; | 
 | } | 
 |  | 
 | bool QboneSessionBase::ShouldKeepConnectionAlive() const { | 
 |   // QBONE connections stay alive until they're explicitly closed. | 
 |   return true; | 
 | } | 
 |  | 
 | std::unique_ptr<QuicStream> QboneSessionBase::CreateDataStream( | 
 |     QuicStreamId id) { | 
 |   if (crypto_stream_ == nullptr || !crypto_stream_->encryption_established()) { | 
 |     // Encryption not active so no stream created | 
 |     return nullptr; | 
 |   } | 
 |  | 
 |   if (IsIncomingStream(id)) { | 
 |     ++num_streamed_packets_; | 
 |     return std::make_unique<QboneReadOnlyStream>(id, this); | 
 |   } | 
 |  | 
 |   return std::make_unique<QboneWriteOnlyStream>(id, this); | 
 | } | 
 |  | 
 | QuicStream* QboneSessionBase::ActivateDataStream( | 
 |     std::unique_ptr<QuicStream> stream) { | 
 |   // Transfer ownership of the data stream to the session via ActivateStream(). | 
 |   QuicStream* raw = stream.get(); | 
 |   if (stream) { | 
 |     // Make QuicSession take ownership of the stream. | 
 |     ActivateStream(std::move(stream)); | 
 |   } | 
 |   return raw; | 
 | } | 
 |  | 
 | void QboneSessionBase::SendPacketToPeer(absl::string_view packet) { | 
 |   if (crypto_stream_ == nullptr) { | 
 |     QUIC_BUG(quic_bug_10987_1) | 
 |         << "Attempting to send packet before encryption established"; | 
 |     return; | 
 |   } | 
 |  | 
 |   if (send_packets_as_messages_) { | 
 |     QuicMemSlice slice(QuicBuffer::Copy( | 
 |         connection()->helper()->GetStreamSendBufferAllocator(), packet)); | 
 |     switch (SendMessage(absl::MakeSpan(&slice, 1), /*flush=*/true).status) { | 
 |       case MESSAGE_STATUS_SUCCESS: | 
 |         break; | 
 |       case MESSAGE_STATUS_TOO_LARGE: { | 
 |         if (packet.size() < sizeof(ip6_hdr)) { | 
 |           QUIC_BUG(quic_bug_10987_2) | 
 |               << "Dropped malformed packet: IPv6 header too short"; | 
 |           break; | 
 |         } | 
 |         auto* header = reinterpret_cast<const ip6_hdr*>(packet.begin()); | 
 |         icmp6_hdr icmp_header{}; | 
 |         icmp_header.icmp6_type = ICMP6_PACKET_TOO_BIG; | 
 |         icmp_header.icmp6_mtu = | 
 |             connection()->GetGuaranteedLargestMessagePayload(); | 
 |  | 
 |         CreateIcmpPacket(header->ip6_dst, header->ip6_src, icmp_header, packet, | 
 |                          [this](absl::string_view icmp_packet) { | 
 |                            writer_->WritePacketToNetwork(icmp_packet.data(), | 
 |                                                          icmp_packet.size()); | 
 |                          }); | 
 |         break; | 
 |       } | 
 |       case MESSAGE_STATUS_ENCRYPTION_NOT_ESTABLISHED: | 
 |         QUIC_BUG(quic_bug_10987_3) | 
 |             << "MESSAGE_STATUS_ENCRYPTION_NOT_ESTABLISHED"; | 
 |         break; | 
 |       case MESSAGE_STATUS_UNSUPPORTED: | 
 |         QUIC_BUG(quic_bug_10987_4) << "MESSAGE_STATUS_UNSUPPORTED"; | 
 |         break; | 
 |       case MESSAGE_STATUS_BLOCKED: | 
 |         QUIC_BUG(quic_bug_10987_5) << "MESSAGE_STATUS_BLOCKED"; | 
 |         break; | 
 |       case MESSAGE_STATUS_INTERNAL_ERROR: | 
 |         QUIC_BUG(quic_bug_10987_6) << "MESSAGE_STATUS_INTERNAL_ERROR"; | 
 |         break; | 
 |     } | 
 |     return; | 
 |   } | 
 |  | 
 |   // QBONE streams are ephemeral. | 
 |   QuicStream* stream = CreateOutgoingStream(); | 
 |   if (!stream) { | 
 |     QUIC_BUG(quic_bug_10987_7) << "Failed to create an outgoing QBONE stream."; | 
 |     return; | 
 |   } | 
 |  | 
 |   QboneWriteOnlyStream* qbone_stream = | 
 |       static_cast<QboneWriteOnlyStream*>(stream); | 
 |   qbone_stream->WritePacketToQuicStream(packet); | 
 | } | 
 |  | 
 | uint64_t QboneSessionBase::GetNumEphemeralPackets() const { | 
 |   return num_ephemeral_packets_; | 
 | } | 
 |  | 
 | uint64_t QboneSessionBase::GetNumStreamedPackets() const { | 
 |   return num_streamed_packets_; | 
 | } | 
 |  | 
 | uint64_t QboneSessionBase::GetNumMessagePackets() const { | 
 |   return num_message_packets_; | 
 | } | 
 |  | 
 | uint64_t QboneSessionBase::GetNumFallbackToStream() const { | 
 |   return num_fallback_to_stream_; | 
 | } | 
 |  | 
 | void QboneSessionBase::set_writer(QbonePacketWriter* writer) { | 
 |   writer_ = writer; | 
 |   quic::AdjustTestValue("quic_QbonePacketWriter", &writer_); | 
 | } | 
 |  | 
 | }  // namespace quic |