blob: 3207e59558b96aa3037de406fd3896b278c88cb3 [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/quic_stream_id_manager.h"
5
6#include <cstdint>
vasilvv872e7a32019-03-12 16:42:44 -07007#include <string>
QUICHE teama6ef0a62019-03-07 20:34:33 -05008#include <utility>
9
renjietang315428e2020-01-27 10:18:12 -080010#include "net/third_party/quiche/src/quic/core/quic_constants.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050011#include "net/third_party/quiche/src/quic/core/quic_utils.h"
dschinazi38cc1ee2020-02-28 14:33:58 -080012#include "net/third_party/quiche/src/quic/core/quic_versions.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050013#include "net/third_party/quiche/src/quic/platform/api/quic_expect_bug.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050014#include "net/third_party/quiche/src/quic/platform/api/quic_test.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050015#include "net/third_party/quiche/src/quic/test_tools/quic_stream_id_manager_peer.h"
dmcardlecf0bfcf2019-12-13 08:08:21 -080016#include "net/third_party/quiche/src/common/platform/api/quiche_str_cat.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050017
18using testing::_;
QUICHE teama6ef0a62019-03-07 20:34:33 -050019using testing::StrictMock;
20
21namespace quic {
22namespace test {
23namespace {
24
rcha8b56e42019-09-20 10:41:48 -070025class MockDelegate : public QuicStreamIdManager::DelegateInterface {
QUICHE teama6ef0a62019-03-07 20:34:33 -050026 public:
rcha8b56e42019-09-20 10:41:48 -070027 MOCK_METHOD2(SendMaxStreams,
28 void(QuicStreamCount stream_count, bool unidirectional));
QUICHE teama6ef0a62019-03-07 20:34:33 -050029};
30
rch9301d3c2019-09-20 14:30:48 -070031struct TestParams {
dschinazi38cc1ee2020-02-28 14:33:58 -080032 TestParams(ParsedQuicVersion version,
33 Perspective perspective,
34 bool is_unidirectional)
35 : version(version),
36 perspective(perspective),
37 is_unidirectional(is_unidirectional) {}
rch9301d3c2019-09-20 14:30:48 -070038
dschinazi38cc1ee2020-02-28 14:33:58 -080039 ParsedQuicVersion version;
rch9301d3c2019-09-20 14:30:48 -070040 Perspective perspective;
41 bool is_unidirectional;
42};
43
44// Used by ::testing::PrintToStringParamName().
45std::string PrintToString(const TestParams& p) {
dmcardlecf0bfcf2019-12-13 08:08:21 -080046 return quiche::QuicheStrCat(
dschinazi38cc1ee2020-02-28 14:33:58 -080047 ParsedQuicVersionToString(p.version), "_",
rch9301d3c2019-09-20 14:30:48 -070048 (p.perspective == Perspective::IS_CLIENT ? "Client" : "Server"),
49 (p.is_unidirectional ? "Unidirectional" : "Bidirectional"));
50}
51
52std::vector<TestParams> GetTestParams() {
53 std::vector<TestParams> params;
dschinazi38cc1ee2020-02-28 14:33:58 -080054 for (const ParsedQuicVersion& version : AllSupportedVersions()) {
55 if (!version.HasIetfQuicFrames()) {
56 continue;
57 }
58 for (Perspective perspective :
59 {Perspective::IS_CLIENT, Perspective::IS_SERVER}) {
60 for (bool is_unidirectional : {true, false}) {
61 params.push_back(TestParams(version, perspective, is_unidirectional));
62 }
rch9301d3c2019-09-20 14:30:48 -070063 }
64 }
65 return params;
66}
67
68class QuicStreamIdManagerTest : public QuicTestWithParam<TestParams> {
QUICHE teama6ef0a62019-03-07 20:34:33 -050069 protected:
rch9301d3c2019-09-20 14:30:48 -070070 QuicStreamIdManagerTest()
rcha8b56e42019-09-20 10:41:48 -070071 : stream_id_manager_(&delegate_,
rch9301d3c2019-09-20 14:30:48 -070072 IsUnidirectional(),
73 perspective(),
74 transport_version(),
renjietang315428e2020-01-27 10:18:12 -080075 0,
rch9301d3c2019-09-20 14:30:48 -070076 kDefaultMaxStreamsPerConnection) {
77 DCHECK(VersionHasIetfQuicFrames(transport_version()));
78 }
QUICHE teama6ef0a62019-03-07 20:34:33 -050079
dschinazi38cc1ee2020-02-28 14:33:58 -080080 QuicTransportVersion transport_version() const {
81 return GetParam().version.transport_version;
82 }
QUICHE teama6ef0a62019-03-07 20:34:33 -050083
rch9301d3c2019-09-20 14:30:48 -070084 // Returns the stream ID for the Nth incoming stream (created by the peer)
85 // of the corresponding directionality of this manager.
86 QuicStreamId GetNthIncomingStreamId(int n) {
87 return kV99StreamIdIncrement * n +
88 (IsUnidirectional()
89 ? QuicUtils::GetFirstUnidirectionalStreamId(
90 transport_version(),
91 QuicUtils::InvertPerspective(perspective()))
92 : QuicUtils::GetFirstBidirectionalStreamId(
93 transport_version(),
94 QuicUtils::InvertPerspective(perspective())));
QUICHE teama6ef0a62019-03-07 20:34:33 -050095 }
96
rch9301d3c2019-09-20 14:30:48 -070097 bool IsUnidirectional() { return GetParam().is_unidirectional; }
98 Perspective perspective() { return GetParam().perspective; }
fkastenholz3c4eabf2019-04-22 07:49:59 -070099
rcha8b56e42019-09-20 10:41:48 -0700100 StrictMock<MockDelegate> delegate_;
101 QuicStreamIdManager stream_id_manager_;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500102};
103
dschinazi142051a2019-09-18 18:17:29 -0700104INSTANTIATE_TEST_SUITE_P(Tests,
rch9301d3c2019-09-20 14:30:48 -0700105 QuicStreamIdManagerTest,
106 ::testing::ValuesIn(GetTestParams()),
dschinazi142051a2019-09-18 18:17:29 -0700107 ::testing::PrintToStringParamName());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500108
rch9301d3c2019-09-20 14:30:48 -0700109TEST_P(QuicStreamIdManagerTest, Initialization) {
renjietang315428e2020-01-27 10:18:12 -0800110 EXPECT_EQ(0u, stream_id_manager_.outgoing_max_streams());
fkastenholz3c4eabf2019-04-22 07:49:59 -0700111
QUICHE teama6ef0a62019-03-07 20:34:33 -0500112 EXPECT_EQ(kDefaultMaxStreamsPerConnection,
rcha8b56e42019-09-20 10:41:48 -0700113 stream_id_manager_.incoming_actual_max_streams());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500114 EXPECT_EQ(kDefaultMaxStreamsPerConnection,
rcha8b56e42019-09-20 10:41:48 -0700115 stream_id_manager_.incoming_advertised_max_streams());
fkastenholz3c4eabf2019-04-22 07:49:59 -0700116 EXPECT_EQ(kDefaultMaxStreamsPerConnection,
rcha8b56e42019-09-20 10:41:48 -0700117 stream_id_manager_.incoming_initial_max_open_streams());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500118
119 // The window for advertising updates to the MAX STREAM ID is half the number
120 // of streams allowed.
rch9301d3c2019-09-20 14:30:48 -0700121 EXPECT_EQ(kDefaultMaxStreamsPerConnection / 2,
rcha8b56e42019-09-20 10:41:48 -0700122 stream_id_manager_.max_streams_window());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500123}
124
125// This test checks that the stream advertisement window is set to 1
126// if the number of stream ids is 1. This is a special case in the code.
rch9301d3c2019-09-20 14:30:48 -0700127TEST_P(QuicStreamIdManagerTest, CheckMaxStreamsWindowForSingleStream) {
rcha8b56e42019-09-20 10:41:48 -0700128 stream_id_manager_.SetMaxOpenIncomingStreams(1);
129 EXPECT_EQ(1u, stream_id_manager_.incoming_initial_max_open_streams());
130 EXPECT_EQ(1u, stream_id_manager_.incoming_actual_max_streams());
rcha8b56e42019-09-20 10:41:48 -0700131 EXPECT_EQ(1u, stream_id_manager_.max_streams_window());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500132}
133
rch9301d3c2019-09-20 14:30:48 -0700134TEST_P(QuicStreamIdManagerTest, CheckMaxStreamsBadValuesOverMaxFailsOutgoing) {
renjietangb885a122020-03-20 08:24:16 -0700135 QuicStreamCount implementation_max = QuicUtils::GetMaxStreamCount();
fkastenholz3c4eabf2019-04-22 07:49:59 -0700136 // Ensure that the limit is less than the implementation maximum.
rcha8b56e42019-09-20 10:41:48 -0700137 EXPECT_LT(stream_id_manager_.outgoing_max_streams(), implementation_max);
fkastenholz3c4eabf2019-04-22 07:49:59 -0700138
renjietangab9039a2020-03-30 14:53:19 -0700139 EXPECT_TRUE(
140 stream_id_manager_.MaybeAllowNewOutgoingStreams(implementation_max + 1));
fkastenholz3c4eabf2019-04-22 07:49:59 -0700141 // Should be pegged at the max.
rcha8b56e42019-09-20 10:41:48 -0700142 EXPECT_EQ(implementation_max, stream_id_manager_.outgoing_max_streams());
fkastenholz3c4eabf2019-04-22 07:49:59 -0700143}
144
fkastenholz3c4eabf2019-04-22 07:49:59 -0700145// Check the case of the stream count in a STREAMS_BLOCKED frame is less than
146// the count most recently advertised in a MAX_STREAMS frame. This should cause
147// a MAX_STREAMS frame with the most recently advertised count to be sent.
rch9301d3c2019-09-20 14:30:48 -0700148TEST_P(QuicStreamIdManagerTest, ProcessStreamsBlockedOk) {
fkastenholz3c4eabf2019-04-22 07:49:59 -0700149 QuicStreamCount stream_count =
rcha8b56e42019-09-20 10:41:48 -0700150 stream_id_manager_.incoming_initial_max_open_streams();
rch9301d3c2019-09-20 14:30:48 -0700151 QuicStreamsBlockedFrame frame(0, stream_count - 1, IsUnidirectional());
152 EXPECT_CALL(delegate_, SendMaxStreams(stream_count, IsUnidirectional()));
renjietang7133ea92020-04-01 12:49:36 -0700153 std::string error_details;
154 EXPECT_TRUE(stream_id_manager_.OnStreamsBlockedFrame(frame, &error_details));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500155}
156
fkastenholz3c4eabf2019-04-22 07:49:59 -0700157// Check the case of the stream count in a STREAMS_BLOCKED frame is equal to the
158// count most recently advertised in a MAX_STREAMS frame. No MAX_STREAMS
159// should be generated.
rch9301d3c2019-09-20 14:30:48 -0700160TEST_P(QuicStreamIdManagerTest, ProcessStreamsBlockedNoOp) {
fkastenholz3c4eabf2019-04-22 07:49:59 -0700161 QuicStreamCount stream_count =
rcha8b56e42019-09-20 10:41:48 -0700162 stream_id_manager_.incoming_initial_max_open_streams();
rch9301d3c2019-09-20 14:30:48 -0700163 QuicStreamsBlockedFrame frame(0, stream_count, IsUnidirectional());
rcha8b56e42019-09-20 10:41:48 -0700164 EXPECT_CALL(delegate_, SendMaxStreams(_, _)).Times(0);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500165}
166
fkastenholz3c4eabf2019-04-22 07:49:59 -0700167// Check the case of the stream count in a STREAMS_BLOCKED frame is greater than
168// the count most recently advertised in a MAX_STREAMS frame. Expect a
QUICHE teama6ef0a62019-03-07 20:34:33 -0500169// connection close with an error.
rch9301d3c2019-09-20 14:30:48 -0700170TEST_P(QuicStreamIdManagerTest, ProcessStreamsBlockedTooBig) {
rcha8b56e42019-09-20 10:41:48 -0700171 EXPECT_CALL(delegate_, SendMaxStreams(_, _)).Times(0);
fkastenholz3c4eabf2019-04-22 07:49:59 -0700172 QuicStreamCount stream_count =
rcha8b56e42019-09-20 10:41:48 -0700173 stream_id_manager_.incoming_initial_max_open_streams() + 1;
rch9301d3c2019-09-20 14:30:48 -0700174 QuicStreamsBlockedFrame frame(0, stream_count, IsUnidirectional());
renjietang7133ea92020-04-01 12:49:36 -0700175 std::string error_details;
176 EXPECT_FALSE(stream_id_manager_.OnStreamsBlockedFrame(frame, &error_details));
177 EXPECT_EQ(
178 error_details,
179 "StreamsBlockedFrame's stream count 101 exceeds incoming max stream 100");
QUICHE teama6ef0a62019-03-07 20:34:33 -0500180}
181
182// Same basic tests as above, but calls
183// QuicStreamIdManager::MaybeIncreaseLargestPeerStreamId directly, avoiding the
184// call chain. The intent is that if there is a problem, the following tests
185// will point to either the stream ID manager or the call chain. They also
186// provide specific, small scale, tests of a public QuicStreamIdManager method.
187// First test make sure that streams with ids below the limit are accepted.
rch9301d3c2019-09-20 14:30:48 -0700188TEST_P(QuicStreamIdManagerTest, IsIncomingStreamIdValidBelowLimit) {
189 QuicStreamId stream_id = GetNthIncomingStreamId(
190 stream_id_manager_.incoming_actual_max_streams() - 2);
renjietang7133ea92020-04-01 12:49:36 -0700191 EXPECT_TRUE(
192 stream_id_manager_.MaybeIncreaseLargestPeerStreamId(stream_id, nullptr));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500193}
194
195// Accept a stream with an ID that equals the limit.
rch9301d3c2019-09-20 14:30:48 -0700196TEST_P(QuicStreamIdManagerTest, IsIncomingStreamIdValidAtLimit) {
197 QuicStreamId stream_id = GetNthIncomingStreamId(
198 stream_id_manager_.incoming_actual_max_streams() - 1);
renjietang7133ea92020-04-01 12:49:36 -0700199 EXPECT_TRUE(
200 stream_id_manager_.MaybeIncreaseLargestPeerStreamId(stream_id, nullptr));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500201}
202
203// Close the connection if the id exceeds the limit.
rch9301d3c2019-09-20 14:30:48 -0700204TEST_P(QuicStreamIdManagerTest, IsIncomingStreamIdInValidAboveLimit) {
205 QuicStreamId stream_id =
206 GetNthIncomingStreamId(stream_id_manager_.incoming_actual_max_streams());
renjietang7133ea92020-04-01 12:49:36 -0700207 std::string error_details;
208 EXPECT_FALSE(stream_id_manager_.MaybeIncreaseLargestPeerStreamId(
209 stream_id, &error_details));
210 EXPECT_EQ(error_details,
211 quiche::QuicheStrCat("Stream id ", stream_id,
212 " would exceed stream count limit 100"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500213}
214
rch9301d3c2019-09-20 14:30:48 -0700215TEST_P(QuicStreamIdManagerTest, OnStreamsBlockedFrame) {
fkastenholz3c4eabf2019-04-22 07:49:59 -0700216 // Get the current maximum allowed incoming stream count.
217 QuicStreamCount advertised_stream_count =
rcha8b56e42019-09-20 10:41:48 -0700218 stream_id_manager_.incoming_advertised_max_streams();
219
fkastenholz3c4eabf2019-04-22 07:49:59 -0700220 QuicStreamsBlockedFrame frame;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500221
rch9301d3c2019-09-20 14:30:48 -0700222 frame.unidirectional = IsUnidirectional();
fkastenholz3c4eabf2019-04-22 07:49:59 -0700223
224 // If the peer is saying it's blocked on the stream count that
QUICHE teama6ef0a62019-03-07 20:34:33 -0500225 // we've advertised, it's a noop since the peer has the correct information.
fkastenholz3c4eabf2019-04-22 07:49:59 -0700226 frame.stream_count = advertised_stream_count;
renjietang7133ea92020-04-01 12:49:36 -0700227 std::string error_details;
228 EXPECT_TRUE(stream_id_manager_.OnStreamsBlockedFrame(frame, &error_details));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500229
fkastenholz3c4eabf2019-04-22 07:49:59 -0700230 // If the peer is saying it's blocked on a stream count that is larger
QUICHE teama6ef0a62019-03-07 20:34:33 -0500231 // than what we've advertised, the connection should get closed.
fkastenholz3c4eabf2019-04-22 07:49:59 -0700232 frame.stream_count = advertised_stream_count + 1;
renjietang7133ea92020-04-01 12:49:36 -0700233 EXPECT_FALSE(stream_id_manager_.OnStreamsBlockedFrame(frame, &error_details));
234 EXPECT_EQ(
235 error_details,
236 "StreamsBlockedFrame's stream count 101 exceeds incoming max stream 100");
QUICHE teama6ef0a62019-03-07 20:34:33 -0500237
fkastenholz3c4eabf2019-04-22 07:49:59 -0700238 // If the peer is saying it's blocked on a count that is less than
239 // our actual count, we send a MAX_STREAMS frame and update
QUICHE teama6ef0a62019-03-07 20:34:33 -0500240 // the advertised value.
241 // First, need to bump up the actual max so there is room for the MAX
fkastenholz3c4eabf2019-04-22 07:49:59 -0700242 // STREAMS frame to send a larger ID.
243 QuicStreamCount actual_stream_count =
rcha8b56e42019-09-20 10:41:48 -0700244 stream_id_manager_.incoming_actual_max_streams();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500245
fkastenholz3c4eabf2019-04-22 07:49:59 -0700246 // Closing a stream will result in the ability to initiate one more
247 // stream
rcha8b56e42019-09-20 10:41:48 -0700248 stream_id_manager_.OnStreamClosed(
249 QuicStreamIdManagerPeer::GetFirstIncomingStreamId(&stream_id_manager_));
fkastenholzd035fc32019-04-23 12:24:02 -0700250 EXPECT_EQ(actual_stream_count + 1u,
rcha8b56e42019-09-20 10:41:48 -0700251 stream_id_manager_.incoming_actual_max_streams());
252 EXPECT_EQ(stream_id_manager_.incoming_actual_max_streams(),
253 stream_id_manager_.incoming_advertised_max_streams() + 1u);
fkastenholz3c4eabf2019-04-22 07:49:59 -0700254
255 // Now simulate receiving a STREAMS_BLOCKED frame...
256 // Changing the actual maximum, above, forces a MAX_STREAMS frame to be
257 // sent, so the logic for that (SendMaxStreamsFrame(), etc) is tested.
258
259 // The STREAMS_BLOCKED frame contains the previous advertised count,
260 // not the one that the peer would have received as a result of the
261 // MAX_STREAMS sent earler.
262 frame.stream_count = advertised_stream_count;
263
rcha8b56e42019-09-20 10:41:48 -0700264 EXPECT_CALL(delegate_,
265 SendMaxStreams(stream_id_manager_.incoming_actual_max_streams(),
rch9301d3c2019-09-20 14:30:48 -0700266 IsUnidirectional()));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500267
renjietang7133ea92020-04-01 12:49:36 -0700268 EXPECT_TRUE(stream_id_manager_.OnStreamsBlockedFrame(frame, &error_details));
fkastenholz3c4eabf2019-04-22 07:49:59 -0700269 // Check that the saved frame is correct.
rcha8b56e42019-09-20 10:41:48 -0700270 EXPECT_EQ(stream_id_manager_.incoming_actual_max_streams(),
271 stream_id_manager_.incoming_advertised_max_streams());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500272}
273
rch9301d3c2019-09-20 14:30:48 -0700274TEST_P(QuicStreamIdManagerTest, GetNextOutgoingStream) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500275 // Number of streams we can open and the first one we should get when
276 // opening...
renjietang7fc869e2019-08-14 11:02:42 -0700277 size_t number_of_streams = kDefaultMaxStreamsPerConnection;
fkastenholz56055be2019-09-17 11:17:37 -0700278
renjietangab9039a2020-03-30 14:53:19 -0700279 EXPECT_TRUE(
280 stream_id_manager_.MaybeAllowNewOutgoingStreams(number_of_streams));
rcha8b56e42019-09-20 10:41:48 -0700281
QUICHE teama6ef0a62019-03-07 20:34:33 -0500282 QuicStreamId stream_id =
rch9301d3c2019-09-20 14:30:48 -0700283 IsUnidirectional()
284 ? QuicUtils::GetFirstUnidirectionalStreamId(
285 transport_version(), stream_id_manager_.perspective())
286 : QuicUtils::GetFirstBidirectionalStreamId(
287 transport_version(), stream_id_manager_.perspective());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500288
rcha8b56e42019-09-20 10:41:48 -0700289 EXPECT_EQ(number_of_streams, stream_id_manager_.outgoing_max_streams());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500290 while (number_of_streams) {
rcha8b56e42019-09-20 10:41:48 -0700291 EXPECT_TRUE(stream_id_manager_.CanOpenNextOutgoingStream());
292 EXPECT_EQ(stream_id, stream_id_manager_.GetNextOutgoingStreamId());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500293 stream_id += kV99StreamIdIncrement;
294 number_of_streams--;
295 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500296
297 // If we try to check that the next outgoing stream id is available it should
fayang769172b2020-03-19 14:27:39 -0700298 // fail.
rcha8b56e42019-09-20 10:41:48 -0700299 EXPECT_FALSE(stream_id_manager_.CanOpenNextOutgoingStream());
300
QUICHE teama6ef0a62019-03-07 20:34:33 -0500301 // If we try to get the next id (above the limit), it should cause a quic-bug.
302 EXPECT_QUIC_BUG(
rcha8b56e42019-09-20 10:41:48 -0700303 stream_id_manager_.GetNextOutgoingStreamId(),
fkastenholz3c4eabf2019-04-22 07:49:59 -0700304 "Attempt to allocate a new outgoing stream that would exceed the limit");
QUICHE teama6ef0a62019-03-07 20:34:33 -0500305}
306
rch9301d3c2019-09-20 14:30:48 -0700307TEST_P(QuicStreamIdManagerTest, MaybeIncreaseLargestPeerStreamId) {
308 QuicStreamId max_stream_id = GetNthIncomingStreamId(
309 stream_id_manager_.incoming_actual_max_streams() - 1);
renjietang7133ea92020-04-01 12:49:36 -0700310 EXPECT_TRUE(stream_id_manager_.MaybeIncreaseLargestPeerStreamId(max_stream_id,
311 nullptr));
fkastenholz3c4eabf2019-04-22 07:49:59 -0700312
rch9301d3c2019-09-20 14:30:48 -0700313 QuicStreamId first_stream_id = GetNthIncomingStreamId(0);
renjietang7133ea92020-04-01 12:49:36 -0700314 EXPECT_TRUE(stream_id_manager_.MaybeIncreaseLargestPeerStreamId(
315 first_stream_id, nullptr));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500316 // A bad stream ID results in a closed connection.
renjietang7133ea92020-04-01 12:49:36 -0700317 std::string error_details;
rcha8b56e42019-09-20 10:41:48 -0700318 EXPECT_FALSE(stream_id_manager_.MaybeIncreaseLargestPeerStreamId(
renjietang7133ea92020-04-01 12:49:36 -0700319 max_stream_id + kV99StreamIdIncrement, &error_details));
320 EXPECT_EQ(
321 error_details,
322 quiche::QuicheStrCat("Stream id ", max_stream_id + kV99StreamIdIncrement,
323 " would exceed stream count limit 100"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500324}
325
rch9301d3c2019-09-20 14:30:48 -0700326TEST_P(QuicStreamIdManagerTest, MaxStreamsWindow) {
fkastenholz3c4eabf2019-04-22 07:49:59 -0700327 // Test that a MAX_STREAMS frame is generated when the peer has less than
328 // |max_streams_window_| streams left that it can initiate.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500329
fkastenholz3c4eabf2019-04-22 07:49:59 -0700330 // First, open, and then close, max_streams_window_ streams. This will
331 // max_streams_window_ streams available for the peer -- no MAX_STREAMS
QUICHE teama6ef0a62019-03-07 20:34:33 -0500332 // should be sent. The -1 is because the check in
fkastenholz3c4eabf2019-04-22 07:49:59 -0700333 // QuicStreamIdManager::MaybeSendMaxStreamsFrame sends a MAX_STREAMS if the
334 // number of available streams at the peer is <= |max_streams_window_|
rcha8b56e42019-09-20 10:41:48 -0700335 int stream_count = stream_id_manager_.max_streams_window() - 1;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500336
337 // Should not get a control-frame transmission since the peer should have
338 // "plenty" of stream IDs to use.
rcha8b56e42019-09-20 10:41:48 -0700339 EXPECT_CALL(delegate_, SendMaxStreams(_, _)).Times(0);
fkastenholz3c4eabf2019-04-22 07:49:59 -0700340
341 // Get the first incoming stream ID to try and allocate.
rch9301d3c2019-09-20 14:30:48 -0700342 QuicStreamId stream_id = GetNthIncomingStreamId(0);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500343 size_t old_available_incoming_streams =
rcha8b56e42019-09-20 10:41:48 -0700344 stream_id_manager_.available_incoming_streams();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500345 while (stream_count) {
renjietang7133ea92020-04-01 12:49:36 -0700346 EXPECT_TRUE(stream_id_manager_.MaybeIncreaseLargestPeerStreamId(stream_id,
347 nullptr));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500348
fkastenholz3c4eabf2019-04-22 07:49:59 -0700349 // This node should think that the peer believes it has one fewer
350 // stream it can create.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500351 old_available_incoming_streams--;
352 EXPECT_EQ(old_available_incoming_streams,
rcha8b56e42019-09-20 10:41:48 -0700353 stream_id_manager_.available_incoming_streams());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500354
355 stream_count--;
356 stream_id += kV99StreamIdIncrement;
357 }
358
fkastenholz3c4eabf2019-04-22 07:49:59 -0700359 // Now close them, still should get no MAX_STREAMS
rcha8b56e42019-09-20 10:41:48 -0700360 stream_count = stream_id_manager_.max_streams_window();
rch9301d3c2019-09-20 14:30:48 -0700361 stream_id = GetNthIncomingStreamId(0);
fkastenholz3c4eabf2019-04-22 07:49:59 -0700362 QuicStreamCount expected_actual_max =
rcha8b56e42019-09-20 10:41:48 -0700363 stream_id_manager_.incoming_actual_max_streams();
fkastenholz3c4eabf2019-04-22 07:49:59 -0700364 QuicStreamCount expected_advertised_max_streams =
rcha8b56e42019-09-20 10:41:48 -0700365 stream_id_manager_.incoming_advertised_max_streams();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500366 while (stream_count) {
rcha8b56e42019-09-20 10:41:48 -0700367 stream_id_manager_.OnStreamClosed(stream_id);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500368 stream_count--;
369 stream_id += kV99StreamIdIncrement;
fkastenholz3c4eabf2019-04-22 07:49:59 -0700370 expected_actual_max++;
371 EXPECT_EQ(expected_actual_max,
rcha8b56e42019-09-20 10:41:48 -0700372 stream_id_manager_.incoming_actual_max_streams());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500373 // Advertised maximum should remain the same.
fkastenholz3c4eabf2019-04-22 07:49:59 -0700374 EXPECT_EQ(expected_advertised_max_streams,
rcha8b56e42019-09-20 10:41:48 -0700375 stream_id_manager_.incoming_advertised_max_streams());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500376 }
377
378 // This should not change.
379 EXPECT_EQ(old_available_incoming_streams,
rcha8b56e42019-09-20 10:41:48 -0700380 stream_id_manager_.available_incoming_streams());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500381
fkastenholz3c4eabf2019-04-22 07:49:59 -0700382 // Now whenever we close a stream we should get a MAX_STREAMS frame.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500383 // Above code closed all the open streams, so we have to open/close
rcha8b56e42019-09-20 10:41:48 -0700384 // EXPECT_CALL(delegate_,
385 // SendMaxStreams(stream_id_manager_.incoming_actual_max_streams(),
rch9301d3c2019-09-20 14:30:48 -0700386 // IsUnidirectional()));
387 EXPECT_CALL(delegate_, SendMaxStreams(_, IsUnidirectional()));
renjietang7133ea92020-04-01 12:49:36 -0700388 EXPECT_TRUE(
389 stream_id_manager_.MaybeIncreaseLargestPeerStreamId(stream_id, nullptr));
rcha8b56e42019-09-20 10:41:48 -0700390 stream_id_manager_.OnStreamClosed(stream_id);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500391}
392
rch9301d3c2019-09-20 14:30:48 -0700393TEST_P(QuicStreamIdManagerTest, StreamsBlockedEdgeConditions) {
fkastenholz3c4eabf2019-04-22 07:49:59 -0700394 QuicStreamsBlockedFrame frame;
rch9301d3c2019-09-20 14:30:48 -0700395 frame.unidirectional = IsUnidirectional();
fkastenholz3c4eabf2019-04-22 07:49:59 -0700396
397 // Check that receipt of a STREAMS BLOCKED with stream-count = 0 does nothing
398 // when max_allowed_incoming_streams is 0.
rcha8b56e42019-09-20 10:41:48 -0700399 EXPECT_CALL(delegate_, SendMaxStreams(_, _)).Times(0);
rcha8b56e42019-09-20 10:41:48 -0700400 stream_id_manager_.SetMaxOpenIncomingStreams(0);
fkastenholz3c4eabf2019-04-22 07:49:59 -0700401 frame.stream_count = 0;
renjietang7133ea92020-04-01 12:49:36 -0700402 std::string error_details;
403 EXPECT_TRUE(stream_id_manager_.OnStreamsBlockedFrame(frame, &error_details));
fkastenholz3c4eabf2019-04-22 07:49:59 -0700404
405 // Check that receipt of a STREAMS BLOCKED with stream-count = 0 invokes a
406 // MAX STREAMS, count = 123, when the MaxOpen... is set to 123.
rch9301d3c2019-09-20 14:30:48 -0700407 EXPECT_CALL(delegate_, SendMaxStreams(123u, IsUnidirectional()));
rcha8b56e42019-09-20 10:41:48 -0700408 stream_id_manager_.SetMaxOpenIncomingStreams(123);
fkastenholz3c4eabf2019-04-22 07:49:59 -0700409 frame.stream_count = 0;
renjietang7133ea92020-04-01 12:49:36 -0700410 EXPECT_TRUE(stream_id_manager_.OnStreamsBlockedFrame(frame, &error_details));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500411}
412
fkastenholz3c4eabf2019-04-22 07:49:59 -0700413// Test that a MAX_STREAMS frame is generated when half the stream ids become
QUICHE teama6ef0a62019-03-07 20:34:33 -0500414// available. This has a useful side effect of testing that when streams are
415// closed, the number of available stream ids increases.
rch9301d3c2019-09-20 14:30:48 -0700416TEST_P(QuicStreamIdManagerTest, MaxStreamsSlidingWindow) {
fkastenholz3c4eabf2019-04-22 07:49:59 -0700417 QuicStreamCount first_advert =
rcha8b56e42019-09-20 10:41:48 -0700418 stream_id_manager_.incoming_advertised_max_streams();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500419
fkastenholz3c4eabf2019-04-22 07:49:59 -0700420 // Open/close enough streams to shrink the window without causing a MAX
421 // STREAMS to be generated. The window will open (and a MAX STREAMS generated)
422 // when max_streams_window() stream IDs have been made available. The loop
QUICHE teama6ef0a62019-03-07 20:34:33 -0500423 // will make that many stream IDs available, so the last CloseStream should
rcha8b56e42019-09-20 10:41:48 -0700424
fkastenholz3c4eabf2019-04-22 07:49:59 -0700425 // cause a MAX STREAMS frame to be generated.
rcha8b56e42019-09-20 10:41:48 -0700426 int i = static_cast<int>(stream_id_manager_.max_streams_window());
fkastenholz3c4eabf2019-04-22 07:49:59 -0700427 QuicStreamId id =
rcha8b56e42019-09-20 10:41:48 -0700428 QuicStreamIdManagerPeer::GetFirstIncomingStreamId(&stream_id_manager_);
429 EXPECT_CALL(
430 delegate_,
431 SendMaxStreams(first_advert + stream_id_manager_.max_streams_window(),
rch9301d3c2019-09-20 14:30:48 -0700432 IsUnidirectional()));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500433 while (i) {
renjietang7133ea92020-04-01 12:49:36 -0700434 EXPECT_TRUE(
435 stream_id_manager_.MaybeIncreaseLargestPeerStreamId(id, nullptr));
rcha8b56e42019-09-20 10:41:48 -0700436 stream_id_manager_.OnStreamClosed(id);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500437 i--;
438 id += kV99StreamIdIncrement;
439 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500440}
441
rch9301d3c2019-09-20 14:30:48 -0700442TEST_P(QuicStreamIdManagerTest, NewStreamDoesNotExceedLimit) {
renjietangab9039a2020-03-30 14:53:19 -0700443 EXPECT_TRUE(stream_id_manager_.MaybeAllowNewOutgoingStreams(100));
fkastenholz56055be2019-09-17 11:17:37 -0700444
rcha8b56e42019-09-20 10:41:48 -0700445 size_t stream_count = stream_id_manager_.outgoing_max_streams();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500446 EXPECT_NE(0u, stream_count);
rcha8b56e42019-09-20 10:41:48 -0700447
QUICHE teama6ef0a62019-03-07 20:34:33 -0500448 while (stream_count) {
rcha8b56e42019-09-20 10:41:48 -0700449 EXPECT_TRUE(stream_id_manager_.CanOpenNextOutgoingStream());
450 stream_id_manager_.GetNextOutgoingStreamId();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500451 stream_count--;
452 }
fkastenholz3c4eabf2019-04-22 07:49:59 -0700453
rcha8b56e42019-09-20 10:41:48 -0700454 EXPECT_EQ(stream_id_manager_.outgoing_stream_count(),
455 stream_id_manager_.outgoing_max_streams());
fayang769172b2020-03-19 14:27:39 -0700456 // Create another, it should fail.
rcha8b56e42019-09-20 10:41:48 -0700457 EXPECT_FALSE(stream_id_manager_.CanOpenNextOutgoingStream());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500458}
459
rch9301d3c2019-09-20 14:30:48 -0700460TEST_P(QuicStreamIdManagerTest, AvailableStreams) {
renjietang7133ea92020-04-01 12:49:36 -0700461 stream_id_manager_.MaybeIncreaseLargestPeerStreamId(GetNthIncomingStreamId(3),
462 nullptr);
rch9301d3c2019-09-20 14:30:48 -0700463
464 EXPECT_TRUE(stream_id_manager_.IsAvailableStream(GetNthIncomingStreamId(1)));
465 EXPECT_TRUE(stream_id_manager_.IsAvailableStream(GetNthIncomingStreamId(2)));
466 EXPECT_FALSE(stream_id_manager_.IsAvailableStream(GetNthIncomingStreamId(3)));
467 EXPECT_TRUE(stream_id_manager_.IsAvailableStream(GetNthIncomingStreamId(4)));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500468}
469
470// Tests that if MaybeIncreaseLargestPeerStreamId is given an extremely
471// large stream ID (larger than the limit) it is rejected.
472// This is a regression for Chromium bugs 909987 and 910040
rch9301d3c2019-09-20 14:30:48 -0700473TEST_P(QuicStreamIdManagerTest, ExtremeMaybeIncreaseLargestPeerStreamId) {
474 QuicStreamId too_big_stream_id = GetNthIncomingStreamId(
475 stream_id_manager_.incoming_actual_max_streams() + 20);
nharper46833c32019-05-15 21:33:05 -0700476
renjietang7133ea92020-04-01 12:49:36 -0700477 std::string error_details;
478 EXPECT_FALSE(stream_id_manager_.MaybeIncreaseLargestPeerStreamId(
479 too_big_stream_id, &error_details));
480 EXPECT_EQ(error_details,
481 quiche::QuicheStrCat("Stream id ", too_big_stream_id,
482 " would exceed stream count limit 100"));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500483}
484
QUICHE teama6ef0a62019-03-07 20:34:33 -0500485} // namespace
486} // namespace test
487} // namespace quic