1 // Copyright 2013 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.
5 #ifndef NET_SPDY_WRITE_BLOCKED_LIST_H_
6 #define NET_SPDY_WRITE_BLOCKED_LIST_H_
11 #include "base/containers/hash_tables.h"
12 #include "base/logging.h"
13 #include "net/spdy/spdy_protocol.h"
18 class WriteBlockedListPeer
;
21 const int kHighestPriority
= 0;
22 const int kLowestPriority
= 7;
24 template <typename IdType
>
25 class WriteBlockedList
{
27 // 0(1) size lookup. 0(1) insert at front or back.
28 typedef std::deque
<IdType
> BlockedList
;
29 typedef typename
BlockedList::iterator iterator
;
31 explicit WriteBlockedList(bool use_stream_to_priority_map
)
32 : use_stream_to_priority_(use_stream_to_priority_map
) {}
34 static SpdyPriority
ClampPriority(SpdyPriority priority
) {
35 if (priority
< kHighestPriority
) {
36 LOG(DFATAL
) << "Invalid priority: " << static_cast<int>(priority
);
37 return kHighestPriority
;
39 if (priority
> kLowestPriority
) {
40 LOG(DFATAL
) << "Invalid priority: " << static_cast<int>(priority
);
41 return kLowestPriority
;
46 // Returns the priority of the highest priority list with sessions on it.
47 SpdyPriority
GetHighestPriorityWriteBlockedList() const {
48 for (SpdyPriority i
= 0; i
<= kLowestPriority
; ++i
) {
49 if (write_blocked_lists_
[i
].size() > 0)
52 LOG(DFATAL
) << "No blocked streams";
53 return kHighestPriority
;
56 IdType
PopFront(SpdyPriority priority
) {
57 priority
= ClampPriority(priority
);
58 DCHECK(!write_blocked_lists_
[priority
].empty());
59 IdType stream_id
= write_blocked_lists_
[priority
].front();
60 write_blocked_lists_
[priority
].pop_front();
61 if (use_stream_to_priority_
) {
62 stream_to_priority_
.erase(stream_id
);
67 bool HasWriteBlockedStreamsGreaterThanPriority(SpdyPriority priority
) const {
68 priority
= ClampPriority(priority
);
69 for (SpdyPriority i
= kHighestPriority
; i
< priority
; ++i
) {
70 if (!write_blocked_lists_
[i
].empty()) {
77 bool HasWriteBlockedStreams() const {
78 for (SpdyPriority i
= kHighestPriority
; i
<= kLowestPriority
; ++i
) {
79 if (!write_blocked_lists_
[i
].empty()) {
86 void PushBack(IdType stream_id
, SpdyPriority priority
) {
87 priority
= ClampPriority(priority
);
88 DVLOG(2) << "Adding stream " << stream_id
<< " at priority "
89 << static_cast<int>(priority
);
90 bool should_insert_stream
= true;
91 if (use_stream_to_priority_
) {
92 typename
StreamToPriorityMap::iterator iter
=
93 stream_to_priority_
.find(stream_id
);
94 if (iter
!= stream_to_priority_
.end()) {
95 DVLOG(1) << "Stream " << stream_id
<< " already in write blocked list.";
96 if (iter
->second
== priority
) {
97 // The stream is already in the write blocked list for the priority.
98 should_insert_stream
= false;
100 // The stream is in a write blocked list for a different priority.
102 RemoveStreamFromWriteBlockedList(stream_id
, iter
->second
);
106 if (should_insert_stream
) {
107 stream_to_priority_
[stream_id
] = priority
;
108 write_blocked_lists_
[priority
].push_back(stream_id
);
111 write_blocked_lists_
[priority
].push_back(stream_id
);
115 bool RemoveStreamFromWriteBlockedList(IdType stream_id
,
116 SpdyPriority priority
) {
117 if (use_stream_to_priority_
) {
118 typename
StreamToPriorityMap::iterator iter
=
119 stream_to_priority_
.find(stream_id
);
120 if (iter
== stream_to_priority_
.end()) {
121 // The stream is not present in the write blocked list.
123 } else if (iter
->second
== priority
) {
124 stream_to_priority_
.erase(iter
);
126 // The stream is not present at the specified priority level.
130 // We shouldn't really add a stream_id to a list multiple times,
131 // but under some conditions it does happen. Doing a check in PushBack
132 // would be too costly, so instead we check here to eliminate duplicates.
134 iterator it
= std::find(write_blocked_lists_
[priority
].begin(),
135 write_blocked_lists_
[priority
].end(), stream_id
);
136 while (it
!= write_blocked_lists_
[priority
].end()) {
138 iterator next_it
= write_blocked_lists_
[priority
].erase(it
);
139 it
= std::find(next_it
, write_blocked_lists_
[priority
].end(), stream_id
);
144 void UpdateStreamPriorityInWriteBlockedList(IdType stream_id
,
145 SpdyPriority old_priority
,
146 SpdyPriority new_priority
) {
147 if (old_priority
== new_priority
) {
150 bool found
= RemoveStreamFromWriteBlockedList(stream_id
, old_priority
);
152 PushBack(stream_id
, new_priority
);
156 size_t NumBlockedStreams() const {
157 size_t num_blocked_streams
= 0;
158 for (SpdyPriority i
= kHighestPriority
; i
<= kLowestPriority
; ++i
) {
159 num_blocked_streams
+= write_blocked_lists_
[i
].size();
161 return num_blocked_streams
;
164 bool avoids_inserting_duplicates() const { return use_stream_to_priority_
; }
167 friend class net::test::WriteBlockedListPeer
;
169 typedef base::hash_map
<IdType
, SpdyPriority
> StreamToPriorityMap
;
171 BlockedList write_blocked_lists_
[kLowestPriority
+ 1];
172 StreamToPriorityMap stream_to_priority_
;
173 bool use_stream_to_priority_
;
178 #endif // NET_SPDY_WRITE_BLOCKED_LIST_H_