blob: 4a2b4f09147fa27b0d37605929592ec55af496c2 [file] [log] [blame]
fayang89b9af52019-07-23 07:52:49 -07001// Copyright (c) 2019 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
5#ifndef QUICHE_SPDY_CORE_HTTP2_PRIORITY_WRITE_SCHEDULER_H_
6#define QUICHE_SPDY_CORE_HTTP2_PRIORITY_WRITE_SCHEDULER_H_
7
8#include <cstdint>
9#include <deque>
10#include <map>
11#include <memory>
12#include <queue>
13#include <set>
bnc44712912019-08-15 18:58:14 -070014#include <string>
fayang89b9af52019-07-23 07:52:49 -070015#include <tuple>
16#include <unordered_map>
17#include <utility>
18#include <vector>
19
20#include "net/third_party/quiche/src/spdy/core/spdy_intrusive_list.h"
21#include "net/third_party/quiche/src/spdy/core/spdy_protocol.h"
22#include "net/third_party/quiche/src/spdy/core/write_scheduler.h"
23#include "net/third_party/quiche/src/spdy/platform/api/spdy_bug_tracker.h"
24#include "net/third_party/quiche/src/spdy/platform/api/spdy_containers.h"
25#include "net/third_party/quiche/src/spdy/platform/api/spdy_logging.h"
26#include "net/third_party/quiche/src/spdy/platform/api/spdy_map_util.h"
fayang89b9af52019-07-23 07:52:49 -070027#include "net/third_party/quiche/src/spdy/platform/api/spdy_string_utils.h"
28
29namespace spdy {
30
31namespace test {
32template <typename StreamIdType>
33class Http2PriorityWriteSchedulerPeer;
34}
35
36// This data structure implements the HTTP/2 stream priority tree defined in
37// section 5.3 of RFC 7540:
38// http://tools.ietf.org/html/rfc7540#section-5.3
39//
40// Streams can be added and removed, and dependencies between them defined.
41// Streams constitute a tree rooted at stream ID 0: each stream has a single
42// parent stream, and 0 or more child streams. Individual streams can be
43// marked as ready to read/write, and then the whole structure can be queried
44// to pick the next stream to read/write out of those that are ready.
45template <typename StreamIdType>
46class Http2PriorityWriteScheduler : public WriteScheduler<StreamIdType> {
47 public:
48 using typename WriteScheduler<StreamIdType>::StreamPrecedenceType;
49
50 Http2PriorityWriteScheduler();
51 Http2PriorityWriteScheduler(const Http2PriorityWriteScheduler&) = delete;
52 Http2PriorityWriteScheduler& operator=(const Http2PriorityWriteScheduler&) =
53 delete;
54
55 // WriteScheduler methods
56 void RegisterStream(StreamIdType stream_id,
57 const StreamPrecedenceType& precedence) override;
58 void UnregisterStream(StreamIdType stream_id) override;
59 bool StreamRegistered(StreamIdType stream_id) const override;
60 StreamPrecedenceType GetStreamPrecedence(
61 StreamIdType stream_id) const override;
62 void UpdateStreamPrecedence(StreamIdType stream_id,
63 const StreamPrecedenceType& precedence) override;
64 std::vector<StreamIdType> GetStreamChildren(
65 StreamIdType stream_id) const override;
66 void RecordStreamEventTime(StreamIdType stream_id,
67 int64_t now_in_usec) override;
68 int64_t GetLatestEventWithPrecedence(StreamIdType stream_id) const override;
69 bool ShouldYield(StreamIdType stream_id) const override;
70 void MarkStreamReady(StreamIdType stream_id, bool add_to_front) override;
71 void MarkStreamNotReady(StreamIdType stream_id) override;
72 bool HasReadyStreams() const override;
73 StreamIdType PopNextReadyStream() override;
74 std::tuple<StreamIdType, StreamPrecedenceType>
75 PopNextReadyStreamAndPrecedence() override;
76 size_t NumReadyStreams() const override;
77 bool IsStreamReady(StreamIdType stream_id) const override;
fayang753002d2019-07-24 11:54:43 -070078 size_t NumRegisteredStreams() const override;
bnc44712912019-08-15 18:58:14 -070079 std::string DebugString() const override;
fayang89b9af52019-07-23 07:52:49 -070080
fayang89b9af52019-07-23 07:52:49 -070081 private:
82 friend class test::Http2PriorityWriteSchedulerPeer<StreamIdType>;
83
84 struct StreamInfo;
85 typedef SpdyInlinedVector<StreamInfo*, 4> StreamInfoVector;
86
87 struct StreamInfo : public SpdyIntrusiveLink<StreamInfo> {
88 // ID for this stream.
89 StreamIdType id;
90 // StreamInfo for parent stream.
91 StreamInfo* parent = nullptr;
92 // Weights can range between 1 and 256 (inclusive).
93 int weight = kHttp2DefaultStreamWeight;
94 // The total weight of this stream's direct descendants.
95 int total_child_weights = 0;
96 // Pointers to StreamInfos for children, if any.
97 StreamInfoVector children;
98 // Whether the stream is ready for writing. The stream is present in
99 // scheduling_queue_ iff true.
100 bool ready = false;
101 // The scheduling priority of this stream. Streams with higher priority
102 // values are scheduled first.
103 // TODO(mpw): rename to avoid confusion with SPDY priorities,
104 // which this is not.
105 float priority = 0;
106 // Ordinal value for this stream, used to ensure round-robin scheduling:
107 // among streams with the same scheduling priority, streams with lower
108 // ordinal are scheduled first.
109 int64_t ordinal = 0;
110 // Time of latest write event for stream of this priority, in microseconds.
111 int64_t last_event_time_usec = 0;
112
fayang86189502019-07-26 08:22:33 -0700113 // Returns true if this stream is ancestor of |other|.
114 bool IsAncestorOf(const StreamInfo& other) const {
fayangb8103922019-08-02 06:16:45 -0700115 for (const StreamInfo* p = other.parent; p != nullptr; p = p->parent) {
116 if (p == this) {
fayang86189502019-07-26 08:22:33 -0700117 return true;
118 }
119 }
120 return false;
121 }
122
fayang89b9af52019-07-23 07:52:49 -0700123 // Whether this stream should be scheduled ahead of another stream.
124 bool SchedulesBefore(const StreamInfo& other) const {
125 return (priority != other.priority) ? priority > other.priority
126 : ordinal < other.ordinal;
127 }
128
129 // Returns the StreamPrecedenceType for this StreamInfo.
130 StreamPrecedenceType ToStreamPrecedence() const {
131 StreamIdType parent_id =
132 parent == nullptr ? kHttp2RootStreamId : parent->id;
133 bool exclusive = parent != nullptr && parent->children.size() == 1;
134 return StreamPrecedenceType(parent_id, weight, exclusive);
135 }
136 };
137
138 static bool Remove(StreamInfoVector* stream_infos,
139 const StreamInfo* stream_info);
140
141 // Returns true iff any direct or transitive parent of the given stream is
142 // currently ready.
143 static bool HasReadyAncestor(const StreamInfo& stream_info);
144
145 // Returns StreamInfo for the given stream, or nullptr if it isn't
146 // registered.
147 const StreamInfo* FindStream(StreamIdType stream_id) const;
148 StreamInfo* FindStream(StreamIdType stream_id);
149
150 // Helpers for UpdateStreamPrecedence().
151 void UpdateStreamParent(StreamInfo* stream_info,
152 StreamIdType parent_id,
153 bool exclusive);
154 void UpdateStreamWeight(StreamInfo* stream_info, int weight);
155
156 // Update all priority values in the subtree rooted at the given stream, not
157 // including the stream itself. If this results in priority value changes for
158 // scheduled streams, those streams are rescheduled to ensure proper ordering
159 // of scheduling_queue_.
160 // TODO(mpw): rename to avoid confusion with SPDY priorities.
161 void UpdatePrioritiesUnder(StreamInfo* stream_info);
162
163 // Inserts stream into scheduling_queue_ at the appropriate location given
164 // its priority and ordinal. Time complexity is O(scheduling_queue.size()).
165 void Schedule(StreamInfo* stream_info);
166
167 // Removes stream from scheduling_queue_.
168 void Unschedule(StreamInfo* stream_info);
169
170 // Return true if all internal invariants hold (useful for unit tests).
171 // Unless there are bugs, this should always return true.
172 bool ValidateInvariantsForTests() const;
173
174 // Returns true if the parent stream has the given stream in its children.
175 bool StreamHasChild(const StreamInfo& parent_info,
176 const StreamInfo* child_info) const;
177
178 // Pointee owned by all_stream_infos_.
179 StreamInfo* root_stream_info_;
180 // Maps from stream IDs to StreamInfo objects.
181 SpdySmallMap<StreamIdType, std::unique_ptr<StreamInfo>, 10> all_stream_infos_;
182 // Queue containing all ready streams, ordered with streams of higher
183 // priority before streams of lower priority, and, among streams of equal
184 // priority, streams with lower ordinal before those with higher
185 // ordinal. Note that not all streams in scheduling_queue_ are eligible to be
186 // picked as the next stream: some may have ancestor stream(s) that are ready
187 // and unblocked. In these situations the occluded child streams are left in
188 // the queue, to reduce churn.
189 SpdyIntrusiveList<StreamInfo> scheduling_queue_;
190 // Ordinal value to assign to next node inserted into scheduling_queue_ when
191 // |add_to_front == true|. Decremented after each assignment.
192 int64_t head_ordinal_ = -1;
193 // Ordinal value to assign to next node inserted into scheduling_queue_ when
194 // |add_to_front == false|. Incremented after each assignment.
195 int64_t tail_ordinal_ = 0;
196};
197
198template <typename StreamIdType>
199Http2PriorityWriteScheduler<StreamIdType>::Http2PriorityWriteScheduler() {
bnc463f2352019-10-10 04:49:34 -0700200 auto root_stream_info = std::make_unique<StreamInfo>();
fayang89b9af52019-07-23 07:52:49 -0700201 root_stream_info_ = root_stream_info.get();
202 root_stream_info->id = kHttp2RootStreamId;
203 root_stream_info->weight = kHttp2DefaultStreamWeight;
204 root_stream_info->parent = nullptr;
205 root_stream_info->priority = 1.0;
206 root_stream_info->ready = false;
207 all_stream_infos_[kHttp2RootStreamId] = std::move(root_stream_info);
208}
209
210template <typename StreamIdType>
fayang89b9af52019-07-23 07:52:49 -0700211bool Http2PriorityWriteScheduler<StreamIdType>::StreamRegistered(
212 StreamIdType stream_id) const {
213 return SpdyContainsKey(all_stream_infos_, stream_id);
214}
215
216template <typename StreamIdType>
217void Http2PriorityWriteScheduler<StreamIdType>::RegisterStream(
218 StreamIdType stream_id,
219 const StreamPrecedenceType& precedence) {
220 // TODO(mpw): uncomment the SPDY_BUG_IF below once all tests
221 // (e.g. SpdyClientDispatcher) modified to pass StreamPrecedence instances
222 // appropriate for protocol version under test.
223 //
224 // SPDY_BUG_IF(precedence.is_spdy3_priority())
225 // << "Expected HTTP/2 stream dependency";
226
227 if (StreamRegistered(stream_id)) {
228 SPDY_BUG << "Stream " << stream_id << " already registered";
229 return;
230 }
231
232 StreamInfo* parent = FindStream(precedence.parent_id());
233 if (parent == nullptr) {
234 // parent_id may legitimately not be registered yet--see b/15676312.
235 SPDY_VLOG(1) << "Parent stream " << precedence.parent_id()
236 << " not registered";
237 parent = root_stream_info_;
238 }
239
bnc463f2352019-10-10 04:49:34 -0700240 auto new_stream_info = std::make_unique<StreamInfo>();
fayang89b9af52019-07-23 07:52:49 -0700241 StreamInfo* new_stream_info_ptr = new_stream_info.get();
242 new_stream_info_ptr->id = stream_id;
243 new_stream_info_ptr->weight = precedence.weight();
244 new_stream_info_ptr->parent = parent;
245 all_stream_infos_[stream_id] = std::move(new_stream_info);
246 if (precedence.is_exclusive()) {
247 // Move the parent's current children below the new stream.
248 using std::swap;
249 swap(new_stream_info_ptr->children, parent->children);
250 new_stream_info_ptr->total_child_weights = parent->total_child_weights;
251 // Update each child's parent.
252 for (StreamInfo* child : new_stream_info_ptr->children) {
253 child->parent = new_stream_info_ptr;
254 }
255 // Clear parent's old child data.
256 DCHECK(parent->children.empty());
257 parent->total_child_weights = 0;
258 }
259 // Add new stream to parent.
260 parent->children.push_back(new_stream_info_ptr);
261 parent->total_child_weights += precedence.weight();
262
263 // Update all priorities under parent, since addition of a stream affects
264 // sibling priorities as well.
265 UpdatePrioritiesUnder(parent);
266
267 // Stream starts with ready == false, so no need to schedule it yet.
268 DCHECK(!new_stream_info_ptr->ready);
269}
270
271template <typename StreamIdType>
272void Http2PriorityWriteScheduler<StreamIdType>::UnregisterStream(
273 StreamIdType stream_id) {
274 if (stream_id == kHttp2RootStreamId) {
275 SPDY_BUG << "Cannot unregister root stream";
276 return;
277 }
278 // Remove the stream from table.
279 auto it = all_stream_infos_.find(stream_id);
280 if (it == all_stream_infos_.end()) {
281 SPDY_BUG << "Stream " << stream_id << " not registered";
282 return;
283 }
284 std::unique_ptr<StreamInfo> stream_info(std::move(it->second));
285 all_stream_infos_.erase(it);
286 // If ready (and hence scheduled), unschedule.
287 if (stream_info->ready) {
288 Unschedule(stream_info.get());
289 }
290
291 StreamInfo* parent = stream_info->parent;
292 // Remove the stream from parent's child list.
293 Remove(&parent->children, stream_info.get());
294 parent->total_child_weights -= stream_info->weight;
295
296 // Move the stream's children to the parent's child list.
297 // Update each child's parent and weight.
298 for (StreamInfo* child : stream_info->children) {
299 child->parent = parent;
300 parent->children.push_back(child);
301 // Divide the removed stream's weight among its children, rounding to the
302 // nearest valid weight.
303 float float_weight = stream_info->weight *
304 static_cast<float>(child->weight) /
305 static_cast<float>(stream_info->total_child_weights);
306 int new_weight = floor(float_weight + 0.5);
307 if (new_weight == 0) {
308 new_weight = 1;
309 }
310 child->weight = new_weight;
311 parent->total_child_weights += child->weight;
312 }
313 UpdatePrioritiesUnder(parent);
314}
315
316template <typename StreamIdType>
317typename Http2PriorityWriteScheduler<StreamIdType>::StreamPrecedenceType
318Http2PriorityWriteScheduler<StreamIdType>::GetStreamPrecedence(
319 StreamIdType stream_id) const {
320 const StreamInfo* stream_info = FindStream(stream_id);
321 if (stream_info == nullptr) {
322 // Unknown streams tolerated due to b/15676312. However, return lowest
323 // weight.
324 SPDY_VLOG(1) << "Stream " << stream_id << " not registered";
325 return StreamPrecedenceType(kHttp2RootStreamId, kHttp2MinStreamWeight,
326 false);
327 }
328 return stream_info->ToStreamPrecedence();
329}
330
331template <typename StreamIdType>
332std::vector<StreamIdType>
333Http2PriorityWriteScheduler<StreamIdType>::GetStreamChildren(
334 StreamIdType stream_id) const {
335 std::vector<StreamIdType> child_vec;
336 const StreamInfo* stream_info = FindStream(stream_id);
337 if (stream_info == nullptr) {
338 SPDY_BUG << "Stream " << stream_id << " not registered";
339 } else {
340 child_vec.reserve(stream_info->children.size());
341 for (StreamInfo* child : stream_info->children) {
342 child_vec.push_back(child->id);
343 }
344 }
345 return child_vec;
346}
347
348template <typename StreamIdType>
349void Http2PriorityWriteScheduler<StreamIdType>::UpdateStreamPrecedence(
350 StreamIdType stream_id,
351 const StreamPrecedenceType& precedence) {
352 // TODO(mpw): uncomment the SPDY_BUG_IF below once all tests
353 // (e.g. SpdyClientDispatcher) modified to pass StreamPrecedence instances
354 // appropriate for protocol version under test.
355 //
356 // SPDY_BUG_IF(precedence.is_spdy3_priority())
357 // << "Expected HTTP/2 stream dependency";
358 if (stream_id == kHttp2RootStreamId) {
359 SPDY_BUG << "Cannot set precedence of root stream";
360 return;
361 }
362
363 StreamInfo* stream_info = FindStream(stream_id);
364 if (stream_info == nullptr) {
365 // TODO(mpw): add to all_stream_infos_ on demand--see b/15676312.
366 SPDY_VLOG(1) << "Stream " << stream_id << " not registered";
367 return;
368 }
369 UpdateStreamParent(stream_info, precedence.parent_id(),
370 precedence.is_exclusive());
371 UpdateStreamWeight(stream_info, precedence.weight());
372}
373
374template <typename StreamIdType>
375void Http2PriorityWriteScheduler<StreamIdType>::UpdateStreamWeight(
376 StreamInfo* stream_info,
377 int weight) {
378 if (weight == stream_info->weight) {
379 return;
380 }
381 if (stream_info->parent != nullptr) {
382 stream_info->parent->total_child_weights += (weight - stream_info->weight);
383 }
384 stream_info->weight = weight;
385
386 // Change in weight also affects sibling priorities.
387 UpdatePrioritiesUnder(stream_info->parent);
388}
389
390template <typename StreamIdType>
391void Http2PriorityWriteScheduler<StreamIdType>::UpdateStreamParent(
392 StreamInfo* stream_info,
393 StreamIdType parent_id,
394 bool exclusive) {
395 if (stream_info->id == parent_id) {
396 SPDY_BUG << "Cannot set stream to be its own parent";
397 return;
398 }
399 StreamInfo* new_parent = FindStream(parent_id);
400 if (new_parent == nullptr) {
401 // parent_id may legitimately not be registered yet--see b/15676312.
402 SPDY_VLOG(1) << "Parent stream " << parent_id << " not registered";
403 return;
404 }
405
fayang6cf80a52019-07-29 07:41:14 -0700406 if (stream_info->parent == new_parent &&
fayang67012992019-07-30 12:02:42 -0700407 (!exclusive || new_parent->children.size() == 1u)) {
fayang6cf80a52019-07-29 07:41:14 -0700408 // If the new parent is already the stream's parent, and exclusivity (if
409 // specified) is already satisfied, we are done.
fayang89b9af52019-07-23 07:52:49 -0700410 return;
411 }
412
413 // Next, check to see if the new parent is currently a descendant
414 // of the stream.
415 StreamInfo* last = new_parent->parent;
416 bool cycle_exists = false;
417 while (last != nullptr) {
418 if (last == stream_info) {
419 cycle_exists = true;
420 break;
421 }
422 last = last->parent;
423 }
424
425 if (cycle_exists) {
426 // The new parent moves to the level of the current stream.
427 UpdateStreamParent(new_parent, stream_info->parent->id, false);
428 }
429
430 // Remove stream from old parent's child list.
431 StreamInfo* old_parent = stream_info->parent;
432 Remove(&old_parent->children, stream_info);
433 old_parent->total_child_weights -= stream_info->weight;
434 UpdatePrioritiesUnder(old_parent);
435
436 if (exclusive) {
437 // Move the new parent's current children below the current stream.
438 for (StreamInfo* child : new_parent->children) {
439 child->parent = stream_info;
440 stream_info->children.push_back(child);
441 }
442 stream_info->total_child_weights += new_parent->total_child_weights;
443 // Clear new parent's old child data.
444 new_parent->children.clear();
445 new_parent->total_child_weights = 0;
446 }
447
448 // Make the change.
449 stream_info->parent = new_parent;
450 new_parent->children.push_back(stream_info);
451 new_parent->total_child_weights += stream_info->weight;
452 UpdatePrioritiesUnder(new_parent);
453}
454
455template <typename StreamIdType>
456void Http2PriorityWriteScheduler<StreamIdType>::RecordStreamEventTime(
457 StreamIdType stream_id,
458 int64_t now_in_usec) {
459 if (stream_id == kHttp2RootStreamId) {
460 SPDY_BUG << "Cannot record event time for root stream";
461 return;
462 }
463 StreamInfo* stream_info = FindStream(stream_id);
464 if (stream_info == nullptr) {
465 SPDY_BUG << "Stream " << stream_id << " not registered";
466 return;
467 }
468 stream_info->last_event_time_usec = now_in_usec;
469}
470
471// O(n) in the number of streams, which isn't great. However, this method will
472// soon be superseded by
473// Http2WeightedWriteScheduler::GetLatestEventWithPrecedence(), for which an
474// efficient implementation is straightforward. Also, this method is only
475// called when calculating idle timeouts, so performance isn't key.
476template <typename StreamIdType>
477int64_t Http2PriorityWriteScheduler<StreamIdType>::GetLatestEventWithPrecedence(
478 StreamIdType stream_id) const {
479 if (stream_id == kHttp2RootStreamId) {
480 SPDY_BUG << "Invalid argument: root stream";
481 return 0;
482 }
483 const StreamInfo* stream_info = FindStream(stream_id);
484 if (stream_info == nullptr) {
485 SPDY_BUG << "Stream " << stream_id << " not registered";
486 return 0;
487 }
488 int64_t last_event_time_usec = 0;
489 for (const auto& kv : all_stream_infos_) {
490 const StreamInfo& other = *kv.second;
491 if (other.priority > stream_info->priority) {
492 last_event_time_usec =
493 std::max(last_event_time_usec, other.last_event_time_usec);
494 }
495 }
496 return last_event_time_usec;
497}
498
499// Worst-case time complexity of O(n*d), where n is scheduling queue length and
500// d is tree depth. In practice, should be much shorter, since loop terminates
501// at first writable stream or |stream_id| (whichever is first).
502template <typename StreamIdType>
503bool Http2PriorityWriteScheduler<StreamIdType>::ShouldYield(
504 StreamIdType stream_id) const {
505 if (stream_id == kHttp2RootStreamId) {
506 SPDY_BUG << "Invalid argument: root stream";
507 return false;
508 }
509 const StreamInfo* stream_info = FindStream(stream_id);
510 if (stream_info == nullptr) {
511 SPDY_BUG << "Stream " << stream_id << " not registered";
512 return false;
513 }
fayang86189502019-07-26 08:22:33 -0700514 if (HasReadyAncestor(*stream_info)) {
515 return true;
516 }
fayang89b9af52019-07-23 07:52:49 -0700517 for (const StreamInfo& scheduled : scheduling_queue_) {
fayang86189502019-07-26 08:22:33 -0700518 if (HasReadyAncestor(scheduled)) {
519 // Skip streams which cannot be scheduled.
520 continue;
521 }
522 if (stream_info->IsAncestorOf(scheduled)) {
523 // Do not yield to descendants.
fayang89b9af52019-07-23 07:52:49 -0700524 return false;
525 }
fayang86189502019-07-26 08:22:33 -0700526 // Yield to streams with higher priorities.
527 return scheduled.SchedulesBefore(*stream_info);
fayang89b9af52019-07-23 07:52:49 -0700528 }
529 return false;
530}
531
532template <typename StreamIdType>
533void Http2PriorityWriteScheduler<StreamIdType>::MarkStreamReady(
534 StreamIdType stream_id,
535 bool add_to_front) {
536 if (stream_id == kHttp2RootStreamId) {
537 SPDY_BUG << "Cannot mark root stream ready";
538 return;
539 }
540 StreamInfo* stream_info = FindStream(stream_id);
541 if (stream_info == nullptr) {
542 SPDY_BUG << "Stream " << stream_id << " not registered";
543 return;
544 }
545 if (stream_info->ready) {
546 return;
547 }
548 stream_info->ordinal = add_to_front ? head_ordinal_-- : tail_ordinal_++;
549 Schedule(stream_info);
550}
551
552template <typename StreamIdType>
553void Http2PriorityWriteScheduler<StreamIdType>::MarkStreamNotReady(
554 StreamIdType stream_id) {
555 if (stream_id == kHttp2RootStreamId) {
556 SPDY_BUG << "Cannot mark root stream unready";
557 return;
558 }
559 StreamInfo* stream_info = FindStream(stream_id);
560 if (stream_info == nullptr) {
561 SPDY_BUG << "Stream " << stream_id << " not registered";
562 return;
563 }
564 if (!stream_info->ready) {
565 return;
566 }
567 Unschedule(stream_info);
568}
569
570template <typename StreamIdType>
571bool Http2PriorityWriteScheduler<StreamIdType>::Remove(
572 StreamInfoVector* stream_infos,
573 const StreamInfo* stream_info) {
574 for (typename StreamInfoVector::iterator it = stream_infos->begin();
575 it != stream_infos->end(); ++it) {
576 if (*it == stream_info) {
577 stream_infos->erase(it);
578 return true;
579 }
580 }
581 return false;
582}
583
584template <typename StreamIdType>
585bool Http2PriorityWriteScheduler<StreamIdType>::HasReadyAncestor(
586 const StreamInfo& stream_info) {
587 for (const StreamInfo* parent = stream_info.parent; parent != nullptr;
588 parent = parent->parent) {
589 if (parent->ready) {
590 return true;
591 }
592 }
593 return false;
594}
595
596template <typename StreamIdType>
597const typename Http2PriorityWriteScheduler<StreamIdType>::StreamInfo*
598Http2PriorityWriteScheduler<StreamIdType>::FindStream(
599 StreamIdType stream_id) const {
600 auto it = all_stream_infos_.find(stream_id);
601 return it == all_stream_infos_.end() ? nullptr : it->second.get();
602}
603
604template <typename StreamIdType>
605typename Http2PriorityWriteScheduler<StreamIdType>::StreamInfo*
606Http2PriorityWriteScheduler<StreamIdType>::FindStream(StreamIdType stream_id) {
607 auto it = all_stream_infos_.find(stream_id);
608 return it == all_stream_infos_.end() ? nullptr : it->second.get();
609}
610
611template <typename StreamIdType>
612void Http2PriorityWriteScheduler<StreamIdType>::UpdatePrioritiesUnder(
613 StreamInfo* stream_info) {
614 for (StreamInfo* child : stream_info->children) {
615 child->priority = stream_info->priority *
616 (static_cast<float>(child->weight) /
617 static_cast<float>(stream_info->total_child_weights));
618 if (child->ready) {
619 // Reposition in scheduling_queue_. Use post-order for scheduling, to
620 // benefit from the fact that children have priority <= parent priority.
621 Unschedule(child);
622 UpdatePrioritiesUnder(child);
623 Schedule(child);
624 } else {
625 UpdatePrioritiesUnder(child);
626 }
627 }
628}
629
630template <typename StreamIdType>
631void Http2PriorityWriteScheduler<StreamIdType>::Schedule(
632 StreamInfo* stream_info) {
633 DCHECK(!stream_info->ready);
634 for (StreamInfo& s : scheduling_queue_) {
635 if (stream_info->SchedulesBefore(s)) {
636 scheduling_queue_.insert(&s, stream_info);
637 stream_info->ready = true;
638 return;
639 }
640 }
641 scheduling_queue_.push_back(stream_info);
642 stream_info->ready = true;
643}
644
645template <typename StreamIdType>
646void Http2PriorityWriteScheduler<StreamIdType>::Unschedule(
647 StreamInfo* stream_info) {
648 DCHECK(stream_info->ready);
649 scheduling_queue_.erase(stream_info);
650 stream_info->ready = false;
651}
652
653template <typename StreamIdType>
654bool Http2PriorityWriteScheduler<StreamIdType>::StreamHasChild(
655 const StreamInfo& parent_info,
656 const StreamInfo* child_info) const {
657 auto found = std::find(parent_info.children.begin(),
658 parent_info.children.end(), child_info);
659 return found != parent_info.children.end();
660}
661
662template <typename StreamIdType>
663bool Http2PriorityWriteScheduler<StreamIdType>::HasReadyStreams() const {
664 return !scheduling_queue_.empty();
665}
666
667template <typename StreamIdType>
668StreamIdType Http2PriorityWriteScheduler<StreamIdType>::PopNextReadyStream() {
669 return std::get<0>(PopNextReadyStreamAndPrecedence());
670}
671
672template <typename StreamIdType>
673std::tuple<
674 StreamIdType,
675 typename Http2PriorityWriteScheduler<StreamIdType>::StreamPrecedenceType>
676Http2PriorityWriteScheduler<StreamIdType>::PopNextReadyStreamAndPrecedence() {
677 for (StreamInfo& stream_info : scheduling_queue_) {
678 if (!HasReadyAncestor(stream_info)) {
679 Unschedule(&stream_info);
680 return std::make_tuple(stream_info.id, stream_info.ToStreamPrecedence());
681 }
682 }
683 SPDY_BUG << "No ready streams";
684 return std::make_tuple(
685 kHttp2RootStreamId,
686 StreamPrecedenceType(kHttp2RootStreamId, kHttp2MinStreamWeight, false));
687}
688
689template <typename StreamIdType>
690size_t Http2PriorityWriteScheduler<StreamIdType>::NumReadyStreams() const {
691 return scheduling_queue_.size();
692}
693
694template <typename StreamIdType>
695bool Http2PriorityWriteScheduler<StreamIdType>::IsStreamReady(
696 StreamIdType stream_id) const {
697 if (stream_id == kHttp2RootStreamId) {
698 SPDY_BUG << "Try to check whether root stream is ready";
699 return false;
700 }
701 const StreamInfo* stream_info = FindStream(stream_id);
702 if (stream_info == nullptr) {
703 SPDY_BUG << "Stream " << stream_id << " not registered";
704 return false;
705 }
706 return stream_info->ready;
707}
708
709template <typename StreamIdType>
fayang753002d2019-07-24 11:54:43 -0700710size_t Http2PriorityWriteScheduler<StreamIdType>::NumRegisteredStreams() const {
711 return all_stream_infos_.size();
712}
713
714template <typename StreamIdType>
bnc44712912019-08-15 18:58:14 -0700715std::string Http2PriorityWriteScheduler<StreamIdType>::DebugString() const {
fayang753002d2019-07-24 11:54:43 -0700716 return SpdyStrCat("Http2PriorityWriteScheduler {num_registered_streams=",
717 NumRegisteredStreams(),
fayang89b9af52019-07-23 07:52:49 -0700718 " num_ready_streams=", NumReadyStreams(), "}");
719}
720
721template <typename StreamIdType>
722bool Http2PriorityWriteScheduler<StreamIdType>::ValidateInvariantsForTests()
723 const {
fayang67012992019-07-30 12:02:42 -0700724 size_t total_streams = 0;
725 size_t streams_visited = 0;
fayang89b9af52019-07-23 07:52:49 -0700726 // Iterate through all streams in the map.
727 for (const auto& kv : all_stream_infos_) {
728 ++total_streams;
729 ++streams_visited;
730 StreamIdType stream_id = kv.first;
731 const StreamInfo& stream_info = *kv.second;
732
733 // Verify each StreamInfo mapped under the proper stream ID.
734 if (stream_id != stream_info.id) {
735 SPDY_DLOG(INFO) << "Stream ID " << stream_id
736 << " maps to StreamInfo with ID " << stream_info.id;
737 return false;
738 }
739
740 // All streams except the root should have a parent, and should appear in
741 // the children of that parent.
742 if (stream_info.id != kHttp2RootStreamId &&
743 !StreamHasChild(*stream_info.parent, &stream_info)) {
744 SPDY_DLOG(INFO) << "Parent stream " << stream_info.parent->id
745 << " is not registered, or does not list stream "
746 << stream_info.id << " as its child.";
747 return false;
748 }
749
750 if (!stream_info.children.empty()) {
751 int total_child_weights = 0;
752 // Iterate through the stream's children.
753 for (StreamInfo* child : stream_info.children) {
754 ++streams_visited;
755 // Each stream in the list should exist and should have this stream
756 // set as its parent.
757 if (!StreamRegistered(child->id) || child->parent != &stream_info) {
758 SPDY_DLOG(INFO) << "Child stream " << child->id
759 << " is not registered, "
760 << "or does not list " << stream_info.id
761 << " as its parent.";
762 return false;
763 }
764 total_child_weights += child->weight;
765 }
766 // Verify that total_child_weights is correct.
767 if (total_child_weights != stream_info.total_child_weights) {
768 SPDY_DLOG(INFO) << "Child weight totals do not agree. For stream "
769 << stream_info.id << " total_child_weights has value "
770 << stream_info.total_child_weights << ", expected "
771 << total_child_weights;
772 return false;
773 }
774 }
775 }
776
fayang753002d2019-07-24 11:54:43 -0700777 // Make sure NumRegisteredStreams() reflects the total number of streams the
778 // map contains.
779 if (total_streams != NumRegisteredStreams()) {
fayang89b9af52019-07-23 07:52:49 -0700780 SPDY_DLOG(INFO) << "Map contains incorrect number of streams.";
781 return false;
782 }
783 // Validate the validation function; we should have visited each stream twice
784 // (except for the root)
fayang753002d2019-07-24 11:54:43 -0700785 DCHECK(streams_visited == 2 * NumRegisteredStreams() - 1);
fayang89b9af52019-07-23 07:52:49 -0700786 return true;
787}
788
789} // namespace spdy
790
791#endif // QUICHE_SPDY_CORE_HTTP2_PRIORITY_WRITE_SCHEDULER_H_