| OLD | NEW |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 // | 4 // |
| 5 #include "net/quic/quic_write_blocked_list.h" | 5 #include "net/quic/quic_write_blocked_list.h" |
| 6 | 6 |
| 7 #include "net/quic/test_tools/quic_test_utils.h" | 7 #include "net/quic/test_tools/quic_test_utils.h" |
| 8 #include "testing/gtest/include/gtest/gtest.h" | 8 #include "testing/gtest/include/gtest/gtest.h" |
| 9 | 9 |
| 10 namespace net { | 10 namespace net { |
| 11 namespace test { | 11 namespace test { |
| 12 namespace { | 12 namespace { |
| 13 | 13 |
| 14 TEST(QuicWriteBlockedListTest, PriorityOrder) { | 14 TEST(QuicWriteBlockedListTest, PriorityOrder) { |
| 15 QuicWriteBlockedList write_blocked_list; | 15 QuicWriteBlockedList write_blocked_list; |
| 16 | 16 |
| 17 // Mark streams blocked in roughly reverse priority order, and | 17 // Mark streams blocked in roughly reverse priority order, and |
| 18 // verify that streams are sorted. | 18 // verify that streams are sorted. |
| 19 write_blocked_list.PushBack(40, | 19 write_blocked_list.AddStream(40, |
| 20 QuicWriteBlockedList::kLowestPriority); | 20 QuicWriteBlockedList::kLowestPriority); |
| 21 write_blocked_list.PushBack(23, | 21 write_blocked_list.AddStream(23, |
| 22 QuicWriteBlockedList::kHighestPriority); | 22 QuicWriteBlockedList::kHighestPriority); |
| 23 write_blocked_list.PushBack(17, | 23 write_blocked_list.AddStream(17, |
| 24 QuicWriteBlockedList::kHighestPriority); | 24 QuicWriteBlockedList::kHighestPriority); |
| 25 write_blocked_list.PushBack(kHeadersStreamId, | 25 write_blocked_list.AddStream(kHeadersStreamId, |
| 26 QuicWriteBlockedList::kHighestPriority); | 26 QuicWriteBlockedList::kHighestPriority); |
| 27 write_blocked_list.PushBack(kCryptoStreamId, | 27 write_blocked_list.AddStream(kCryptoStreamId, |
| 28 QuicWriteBlockedList::kHighestPriority); | 28 QuicWriteBlockedList::kHighestPriority); |
| 29 | 29 |
| 30 EXPECT_EQ(5u, write_blocked_list.NumBlockedStreams()); | 30 EXPECT_EQ(5u, write_blocked_list.NumBlockedStreams()); |
| 31 EXPECT_TRUE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); | 31 EXPECT_TRUE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); |
| 32 EXPECT_TRUE(write_blocked_list.HasWriteBlockedDataStreams()); | 32 EXPECT_TRUE(write_blocked_list.HasWriteBlockedDataStreams()); |
| 33 // The Crypto stream is highest priority. | 33 // The Crypto stream is highest priority. |
| 34 EXPECT_EQ(kCryptoStreamId, write_blocked_list.PopFront()); | 34 EXPECT_EQ(kCryptoStreamId, write_blocked_list.PopFront()); |
| 35 // Followed by the Headers stream. | 35 // Followed by the Headers stream. |
| 36 EXPECT_EQ(kHeadersStreamId, write_blocked_list.PopFront()); | 36 EXPECT_EQ(kHeadersStreamId, write_blocked_list.PopFront()); |
| 37 // Streams with same priority are popped in the order they were inserted. | 37 // Streams with same priority are popped in the order they were inserted. |
| 38 EXPECT_EQ(23u, write_blocked_list.PopFront()); | 38 EXPECT_EQ(23u, write_blocked_list.PopFront()); |
| 39 EXPECT_EQ(17u, write_blocked_list.PopFront()); | 39 EXPECT_EQ(17u, write_blocked_list.PopFront()); |
| 40 // Low priority stream appears last. | 40 // Low priority stream appears last. |
| 41 EXPECT_EQ(40u, write_blocked_list.PopFront()); | 41 EXPECT_EQ(40u, write_blocked_list.PopFront()); |
| 42 | 42 |
| 43 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); | 43 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); |
| 44 EXPECT_FALSE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); | 44 EXPECT_FALSE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); |
| 45 EXPECT_FALSE(write_blocked_list.HasWriteBlockedDataStreams()); | 45 EXPECT_FALSE(write_blocked_list.HasWriteBlockedDataStreams()); |
| 46 } | 46 } |
| 47 | 47 |
| 48 TEST(QuicWriteBlockedListTest, CryptoStream) { | 48 TEST(QuicWriteBlockedListTest, CryptoStream) { |
| 49 QuicWriteBlockedList write_blocked_list; | 49 QuicWriteBlockedList write_blocked_list; |
| 50 write_blocked_list.PushBack(kCryptoStreamId, | 50 write_blocked_list.AddStream(kCryptoStreamId, |
| 51 QuicWriteBlockedList::kHighestPriority); | 51 QuicWriteBlockedList::kHighestPriority); |
| 52 | 52 |
| 53 EXPECT_EQ(1u, write_blocked_list.NumBlockedStreams()); | 53 EXPECT_EQ(1u, write_blocked_list.NumBlockedStreams()); |
| 54 EXPECT_TRUE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); | 54 EXPECT_TRUE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); |
| 55 EXPECT_EQ(kCryptoStreamId, write_blocked_list.PopFront()); | 55 EXPECT_EQ(kCryptoStreamId, write_blocked_list.PopFront()); |
| 56 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); | 56 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); |
| 57 EXPECT_FALSE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); | 57 EXPECT_FALSE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); |
| 58 } | 58 } |
| 59 | 59 |
| 60 TEST(QuicWriteBlockedListTest, HeadersStream) { | 60 TEST(QuicWriteBlockedListTest, HeadersStream) { |
| 61 QuicWriteBlockedList write_blocked_list; | 61 QuicWriteBlockedList write_blocked_list; |
| 62 write_blocked_list.PushBack(kHeadersStreamId, | 62 write_blocked_list.AddStream(kHeadersStreamId, |
| 63 QuicWriteBlockedList::kHighestPriority); | 63 QuicWriteBlockedList::kHighestPriority); |
| 64 | 64 |
| 65 EXPECT_EQ(1u, write_blocked_list.NumBlockedStreams()); | 65 EXPECT_EQ(1u, write_blocked_list.NumBlockedStreams()); |
| 66 EXPECT_TRUE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); | 66 EXPECT_TRUE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); |
| 67 EXPECT_EQ(kHeadersStreamId, write_blocked_list.PopFront()); | 67 EXPECT_EQ(kHeadersStreamId, write_blocked_list.PopFront()); |
| 68 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); | 68 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); |
| 69 EXPECT_FALSE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); | 69 EXPECT_FALSE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); |
| 70 } | 70 } |
| 71 | 71 |
| 72 TEST(QuicWriteBlockedListTest, VerifyHeadersStream) { | 72 TEST(QuicWriteBlockedListTest, VerifyHeadersStream) { |
| 73 QuicWriteBlockedList write_blocked_list; | 73 QuicWriteBlockedList write_blocked_list; |
| 74 write_blocked_list.PushBack(5, | 74 write_blocked_list.AddStream(5, |
| 75 QuicWriteBlockedList::kHighestPriority); | 75 QuicWriteBlockedList::kHighestPriority); |
| 76 write_blocked_list.PushBack(kHeadersStreamId, | 76 write_blocked_list.AddStream(kHeadersStreamId, |
| 77 QuicWriteBlockedList::kHighestPriority); | 77 QuicWriteBlockedList::kHighestPriority); |
| 78 | 78 |
| 79 EXPECT_EQ(2u, write_blocked_list.NumBlockedStreams()); | 79 EXPECT_EQ(2u, write_blocked_list.NumBlockedStreams()); |
| 80 EXPECT_TRUE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); | 80 EXPECT_TRUE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); |
| 81 EXPECT_TRUE(write_blocked_list.HasWriteBlockedDataStreams()); | 81 EXPECT_TRUE(write_blocked_list.HasWriteBlockedDataStreams()); |
| 82 // In newer QUIC versions, there is a headers stream which is | 82 // In newer QUIC versions, there is a headers stream which is |
| 83 // higher priority than data streams. | 83 // higher priority than data streams. |
| 84 EXPECT_EQ(kHeadersStreamId, write_blocked_list.PopFront()); | 84 EXPECT_EQ(kHeadersStreamId, write_blocked_list.PopFront()); |
| 85 EXPECT_EQ(5u, write_blocked_list.PopFront()); | 85 EXPECT_EQ(5u, write_blocked_list.PopFront()); |
| 86 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); | 86 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); |
| 87 EXPECT_FALSE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); | 87 EXPECT_FALSE(write_blocked_list.HasWriteBlockedCryptoOrHeadersStream()); |
| 88 EXPECT_FALSE(write_blocked_list.HasWriteBlockedDataStreams()); | 88 EXPECT_FALSE(write_blocked_list.HasWriteBlockedDataStreams()); |
| 89 } | 89 } |
| 90 | 90 |
| 91 TEST(QuicWriteBlockedListTest, NoDuplicateEntries) { | 91 TEST(QuicWriteBlockedListTest, NoDuplicateEntries) { |
| 92 // Test that QuicWriteBlockedList doesn't allow duplicate entries. | 92 // Test that QuicWriteBlockedList doesn't allow duplicate entries. |
| 93 QuicWriteBlockedList write_blocked_list; | 93 QuicWriteBlockedList write_blocked_list; |
| 94 | 94 |
| 95 // Try to add a stream to the write blocked list multiple times at the same | 95 // Try to add a stream to the write blocked list multiple times at the same |
| 96 // priority. | 96 // priority. |
| 97 const QuicStreamId kBlockedId = kClientDataStreamId1; | 97 const QuicStreamId kBlockedId = kClientDataStreamId1; |
| 98 write_blocked_list.PushBack(kBlockedId, | 98 write_blocked_list.AddStream(kBlockedId, |
| 99 QuicWriteBlockedList::kHighestPriority); | 99 QuicWriteBlockedList::kHighestPriority); |
| 100 write_blocked_list.PushBack(kBlockedId, | 100 write_blocked_list.AddStream(kBlockedId, |
| 101 QuicWriteBlockedList::kHighestPriority); | 101 QuicWriteBlockedList::kHighestPriority); |
| 102 write_blocked_list.PushBack(kBlockedId, | 102 write_blocked_list.AddStream(kBlockedId, |
| 103 QuicWriteBlockedList::kHighestPriority); | 103 QuicWriteBlockedList::kHighestPriority); |
| 104 | 104 |
| 105 // This should only result in one blocked stream being added. | 105 // This should only result in one blocked stream being added. |
| 106 EXPECT_EQ(1u, write_blocked_list.NumBlockedStreams()); | 106 EXPECT_EQ(1u, write_blocked_list.NumBlockedStreams()); |
| 107 EXPECT_TRUE(write_blocked_list.HasWriteBlockedDataStreams()); | 107 EXPECT_TRUE(write_blocked_list.HasWriteBlockedDataStreams()); |
| 108 | 108 |
| 109 // There should only be one stream to pop off the front. | 109 // There should only be one stream to pop off the front. |
| 110 EXPECT_EQ(kBlockedId, write_blocked_list.PopFront()); | 110 EXPECT_EQ(kBlockedId, write_blocked_list.PopFront()); |
| 111 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); | 111 EXPECT_EQ(0u, write_blocked_list.NumBlockedStreams()); |
| 112 EXPECT_FALSE(write_blocked_list.HasWriteBlockedDataStreams()); | 112 EXPECT_FALSE(write_blocked_list.HasWriteBlockedDataStreams()); |
| 113 } | 113 } |
| 114 | 114 |
| 115 TEST(QuicWriteBlockedListTest, BatchingWrites) { |
| 116 FLAGS_quic_batch_writes = true; |
| 117 QuicWriteBlockedList write_blocked_list; |
| 118 |
| 119 const QuicStreamId id1 = kClientDataStreamId1; |
| 120 const QuicStreamId id2 = kClientDataStreamId2; |
| 121 |
| 122 write_blocked_list.AddStream(id1, QuicWriteBlockedList::kLowestPriority); |
| 123 write_blocked_list.AddStream(id2, QuicWriteBlockedList::kLowestPriority); |
| 124 EXPECT_EQ(2u, write_blocked_list.NumBlockedStreams()); |
| 125 |
| 126 // The first stream we push back should stay at the front until 16k is |
| 127 // written. |
| 128 EXPECT_EQ(id1, write_blocked_list.PopFront()); |
| 129 write_blocked_list.UpdateBytesForStream(id1, 15999); |
| 130 write_blocked_list.AddStream(id1, QuicWriteBlockedList::kLowestPriority); |
| 131 EXPECT_EQ(2u, write_blocked_list.NumBlockedStreams()); |
| 132 EXPECT_EQ(id1, write_blocked_list.PopFront()); |
| 133 |
| 134 // Once 16k is written the first stream will cede to the next. |
| 135 write_blocked_list.UpdateBytesForStream(id1, 1); |
| 136 write_blocked_list.AddStream(id1, QuicWriteBlockedList::kLowestPriority); |
| 137 EXPECT_EQ(2u, write_blocked_list.NumBlockedStreams()); |
| 138 EXPECT_EQ(id2, write_blocked_list.PopFront()); |
| 139 |
| 140 // Set the new stream to have written all but one byte. |
| 141 write_blocked_list.UpdateBytesForStream(id2, 15999); |
| 142 write_blocked_list.AddStream(id2, QuicWriteBlockedList::kLowestPriority); |
| 143 EXPECT_EQ(2u, write_blocked_list.NumBlockedStreams()); |
| 144 |
| 145 // Ensure higher priority streams are popped first. |
| 146 const QuicStreamId id3 = kClientDataStreamId2 + 2; |
| 147 write_blocked_list.AddStream(id3, QuicWriteBlockedList::kHighestPriority); |
| 148 EXPECT_EQ(id3, write_blocked_list.PopFront()); |
| 149 |
| 150 // Higher priority streams will always be popped first, even if using their |
| 151 // byte quota |
| 152 write_blocked_list.UpdateBytesForStream(id1, 20000); |
| 153 write_blocked_list.AddStream(id3, QuicWriteBlockedList::kHighestPriority); |
| 154 EXPECT_EQ(id3, write_blocked_list.PopFront()); |
| 155 |
| 156 // Once the higher priority stream is out of the way, id2 will resume its 16k |
| 157 // write, with only 1 byte remaining of its guaranteed write allocation. |
| 158 EXPECT_EQ(id2, write_blocked_list.PopFront()); |
| 159 write_blocked_list.UpdateBytesForStream(id2, 1); |
| 160 write_blocked_list.AddStream(id2, QuicWriteBlockedList::kLowestPriority); |
| 161 EXPECT_EQ(2u, write_blocked_list.NumBlockedStreams()); |
| 162 EXPECT_EQ(id1, write_blocked_list.PopFront()); |
| 163 } |
| 164 |
| 115 } // namespace | 165 } // namespace |
| 116 } // namespace test | 166 } // namespace test |
| 117 } // namespace net | 167 } // namespace net |
| OLD | NEW |