blob: 447f16d84bd7049babe956e89c57a0ddf6fdfdef [file] [log] [blame]
QUICHE teama6ef0a62019-03-07 20:34:33 -05001// 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
10namespace quic {
11
12#define ENDPOINT \
13 (session_->perspective() == Perspective::IS_SERVER ? "Server: " : "Client: ")
14
15LegacyQuicStreamIdManager::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),
nharper46833c32019-05-15 21:33:05 -070022 next_outgoing_stream_id_(QuicUtils::GetFirstBidirectionalStreamId(
23 session->connection()->transport_version(),
24 session->perspective())),
QUICHE teama6ef0a62019-03-07 20:34:33 -050025 largest_peer_created_stream_id_(
26 session->perspective() == Perspective::IS_SERVER
nharper46833c32019-05-15 21:33:05 -070027 ? (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 teama6ef0a62019-03-07 20:34:33 -050034 : QuicUtils::GetInvalidStreamId(
35 session->connection()->transport_version())) {}
36
37LegacyQuicStreamIdManager::~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
52bool 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
63bool 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
70bool 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
111QuicStreamId LegacyQuicStreamIdManager::GetNextOutgoingStreamId() {
112 QuicStreamId id = next_outgoing_stream_id_;
113 next_outgoing_stream_id_ += 2;
114 return id;
115}
116
117bool 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
131bool LegacyQuicStreamIdManager::IsIncomingStream(QuicStreamId id) const {
132 return id % 2 != next_outgoing_stream_id_ % 2;
133}
134
135size_t LegacyQuicStreamIdManager::GetNumAvailableStreams() const {
136 return available_streams_.size();
137}
138
139size_t LegacyQuicStreamIdManager::MaxAvailableStreams() const {
140 return max_open_incoming_streams_ * kMaxAvailableStreamsMultiplier;
141}
142
143} // namespace quic