QUICHE team | a6ef0a6 | 2019-03-07 20:34:33 -0500 | [diff] [blame] | 1 | // Copyright (c) 2018 The Chromium Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | #include "net/third_party/quiche/src/quic/core/legacy_quic_stream_id_manager.h" |
| 5 | |
| 6 | #include "net/third_party/quiche/src/quic/core/quic_session.h" |
| 7 | #include "net/third_party/quiche/src/quic/core/quic_utils.h" |
| 8 | #include "net/third_party/quiche/src/quic/platform/api/quic_map_util.h" |
| 9 | |
| 10 | namespace quic { |
| 11 | |
| 12 | #define ENDPOINT \ |
| 13 | (session_->perspective() == Perspective::IS_SERVER ? "Server: " : "Client: ") |
| 14 | |
| 15 | LegacyQuicStreamIdManager::LegacyQuicStreamIdManager( |
| 16 | QuicSession* session, |
| 17 | size_t max_open_outgoing_streams, |
| 18 | size_t max_open_incoming_streams) |
| 19 | : session_(session), |
| 20 | max_open_outgoing_streams_(max_open_outgoing_streams), |
| 21 | max_open_incoming_streams_(max_open_incoming_streams), |
nharper | 46833c3 | 2019-05-15 21:33:05 -0700 | [diff] [blame] | 22 | next_outgoing_stream_id_(QuicUtils::GetFirstBidirectionalStreamId( |
| 23 | session->connection()->transport_version(), |
| 24 | session->perspective())), |
QUICHE team | a6ef0a6 | 2019-03-07 20:34:33 -0500 | [diff] [blame] | 25 | largest_peer_created_stream_id_( |
| 26 | session->perspective() == Perspective::IS_SERVER |
nharper | 46833c3 | 2019-05-15 21:33:05 -0700 | [diff] [blame] | 27 | ? (QuicVersionUsesCryptoFrames( |
| 28 | session->connection()->transport_version()) |
| 29 | ? QuicUtils::GetFirstBidirectionalStreamId( |
| 30 | session->connection()->transport_version(), |
| 31 | Perspective::IS_CLIENT) |
| 32 | : QuicUtils::GetCryptoStreamId( |
| 33 | session->connection()->transport_version())) |
QUICHE team | a6ef0a6 | 2019-03-07 20:34:33 -0500 | [diff] [blame] | 34 | : QuicUtils::GetInvalidStreamId( |
| 35 | session->connection()->transport_version())) {} |
| 36 | |
| 37 | LegacyQuicStreamIdManager::~LegacyQuicStreamIdManager() { |
| 38 | QUIC_LOG_IF(WARNING, |
| 39 | session_->num_locally_closed_incoming_streams_highest_offset() > |
| 40 | max_open_incoming_streams_) |
| 41 | << "Surprisingly high number of locally closed peer initiated streams" |
| 42 | "still waiting for final byte offset: " |
| 43 | << session_->num_locally_closed_incoming_streams_highest_offset(); |
| 44 | QUIC_LOG_IF(WARNING, |
| 45 | session_->GetNumLocallyClosedOutgoingStreamsHighestOffset() > |
| 46 | max_open_outgoing_streams_) |
| 47 | << "Surprisingly high number of locally closed self initiated streams" |
| 48 | "still waiting for final byte offset: " |
| 49 | << session_->GetNumLocallyClosedOutgoingStreamsHighestOffset(); |
| 50 | } |
| 51 | |
| 52 | bool LegacyQuicStreamIdManager::CanOpenNextOutgoingStream( |
| 53 | size_t current_num_open_outgoing_streams) const { |
| 54 | if (current_num_open_outgoing_streams >= max_open_outgoing_streams_) { |
| 55 | QUIC_DLOG(INFO) << "Failed to create a new outgoing stream. " |
| 56 | << "Already " << current_num_open_outgoing_streams |
| 57 | << " open."; |
| 58 | return false; |
| 59 | } |
| 60 | return true; |
| 61 | } |
| 62 | |
| 63 | bool LegacyQuicStreamIdManager::CanOpenIncomingStream( |
| 64 | size_t current_num_open_incoming_streams) const { |
| 65 | // Check if the new number of open streams would cause the number of |
| 66 | // open streams to exceed the limit. |
| 67 | return current_num_open_incoming_streams < max_open_incoming_streams_; |
| 68 | } |
| 69 | |
| 70 | bool LegacyQuicStreamIdManager::MaybeIncreaseLargestPeerStreamId( |
| 71 | const QuicStreamId stream_id) { |
| 72 | available_streams_.erase(stream_id); |
| 73 | |
| 74 | if (largest_peer_created_stream_id_ != |
| 75 | QuicUtils::GetInvalidStreamId( |
| 76 | session_->connection()->transport_version()) && |
| 77 | stream_id <= largest_peer_created_stream_id_) { |
| 78 | return true; |
| 79 | } |
| 80 | |
| 81 | // Check if the new number of available streams would cause the number of |
| 82 | // available streams to exceed the limit. Note that the peer can create |
| 83 | // only alternately-numbered streams. |
| 84 | size_t additional_available_streams = |
| 85 | (stream_id - largest_peer_created_stream_id_) / 2 - 1; |
| 86 | size_t new_num_available_streams = |
| 87 | GetNumAvailableStreams() + additional_available_streams; |
| 88 | if (new_num_available_streams > MaxAvailableStreams()) { |
| 89 | QUIC_DLOG(INFO) << ENDPOINT |
| 90 | << "Failed to create a new incoming stream with id:" |
| 91 | << stream_id << ". There are already " |
| 92 | << GetNumAvailableStreams() |
| 93 | << " streams available, which would become " |
| 94 | << new_num_available_streams << ", which exceeds the limit " |
| 95 | << MaxAvailableStreams() << "."; |
| 96 | session_->connection()->CloseConnection( |
| 97 | QUIC_TOO_MANY_AVAILABLE_STREAMS, |
| 98 | QuicStrCat(new_num_available_streams, " above ", MaxAvailableStreams()), |
| 99 | ConnectionCloseBehavior::SEND_CONNECTION_CLOSE_PACKET); |
| 100 | return false; |
| 101 | } |
| 102 | for (QuicStreamId id = largest_peer_created_stream_id_ + 2; id < stream_id; |
| 103 | id += 2) { |
| 104 | available_streams_.insert(id); |
| 105 | } |
| 106 | largest_peer_created_stream_id_ = stream_id; |
| 107 | |
| 108 | return true; |
| 109 | } |
| 110 | |
| 111 | QuicStreamId LegacyQuicStreamIdManager::GetNextOutgoingStreamId() { |
| 112 | QuicStreamId id = next_outgoing_stream_id_; |
| 113 | next_outgoing_stream_id_ += 2; |
| 114 | return id; |
| 115 | } |
| 116 | |
| 117 | bool LegacyQuicStreamIdManager::IsAvailableStream(QuicStreamId id) const { |
| 118 | if (!IsIncomingStream(id)) { |
| 119 | // Stream IDs under next_ougoing_stream_id_ are either open or previously |
| 120 | // open but now closed. |
| 121 | return id >= next_outgoing_stream_id_; |
| 122 | } |
| 123 | // For peer created streams, we also need to consider available streams. |
| 124 | return largest_peer_created_stream_id_ == |
| 125 | QuicUtils::GetInvalidStreamId( |
| 126 | session_->connection()->transport_version()) || |
| 127 | id > largest_peer_created_stream_id_ || |
| 128 | QuicContainsKey(available_streams_, id); |
| 129 | } |
| 130 | |
| 131 | bool LegacyQuicStreamIdManager::IsIncomingStream(QuicStreamId id) const { |
| 132 | return id % 2 != next_outgoing_stream_id_ % 2; |
| 133 | } |
| 134 | |
| 135 | size_t LegacyQuicStreamIdManager::GetNumAvailableStreams() const { |
| 136 | return available_streams_.size(); |
| 137 | } |
| 138 | |
| 139 | size_t LegacyQuicStreamIdManager::MaxAvailableStreams() const { |
| 140 | return max_open_incoming_streams_ * kMaxAvailableStreamsMultiplier; |
| 141 | } |
| 142 | |
| 143 | } // namespace quic |