Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1006)

Unified Diff: media/cast/congestion_control/congestion_control.cc

Issue 326783002: Cast: Updated congestion control (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: merge Created 6 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: media/cast/congestion_control/congestion_control.cc
diff --git a/media/cast/congestion_control/congestion_control.cc b/media/cast/congestion_control/congestion_control.cc
index 39d68b39f01e23b5b590b9a3cf8fa89333ae9c06..a6986798d4f50e96fe67757a73b56e28b2c6928c 100644
--- a/media/cast/congestion_control/congestion_control.cc
+++ b/media/cast/congestion_control/congestion_control.cc
@@ -11,112 +11,176 @@
namespace media {
namespace cast {
-static const int64 kCongestionControlMinChangeIntervalMs = 10;
-static const int64 kCongestionControlMaxChangeIntervalMs = 100;
+// This means that we *try* to keep our buffer 90% empty.
Alpha Left Google 2014/06/10 22:20:39 Where is this "buffer"? Please elaborate. I think
hubbe 2014/06/11 00:04:15 Done.
+// If it is less full, we increase the bandwidth, if it is more
+// we decrease the bandwidth. Making this smaller makes the
+// congestion control more aggressive.
+static const double kTargetEmptyBufferFraction = 0.9;
-// At 10 ms RTT TCP Reno would ramp 1500 * 8 * 100 = 1200 Kbit/s.
-// NACK is sent after a maximum of 10 ms.
-static const int kCongestionControlMaxBitrateIncreasePerMillisecond = 1200;
+// This is the size of our history. Larger values makes the
+// congestion control adapt slower.
+static const size_t kHistorySize = 100;
-static const int64 kMaxElapsedTimeMs = kCongestionControlMaxChangeIntervalMs;
+CongestionControl::FrameStats::FrameStats() : frame_size(0) {
+}
CongestionControl::CongestionControl(base::TickClock* clock,
- float congestion_control_back_off,
uint32 max_bitrate_configured,
uint32 min_bitrate_configured,
- uint32 start_bitrate)
+ size_t max_unacked_frames)
: clock_(clock),
- congestion_control_back_off_(congestion_control_back_off),
max_bitrate_configured_(max_bitrate_configured),
min_bitrate_configured_(min_bitrate_configured),
- bitrate_(start_bitrate) {
- DCHECK_GT(congestion_control_back_off, 0.0f) << "Invalid config";
- DCHECK_LT(congestion_control_back_off, 1.0f) << "Invalid config";
+ last_frame_stats_(static_cast<uint32>(-1)),
+ last_acked_frame_(static_cast<uint32>(-1)),
+ last_encoded_frame_(static_cast<uint32>(-1)),
+ history_size_(max_unacked_frames + kHistorySize),
+ acked_bits_in_history_(0) {
DCHECK_GE(max_bitrate_configured, min_bitrate_configured) << "Invalid config";
- DCHECK_GE(max_bitrate_configured, start_bitrate) << "Invalid config";
- DCHECK_GE(start_bitrate, min_bitrate_configured) << "Invalid config";
+ frame_stats_.resize(2);
+ base::TimeTicks now = clock->NowTicks();
+ frame_stats_[0].ack_time = now;
+ frame_stats_[0].sent_time = now;
+ frame_stats_[1].ack_time = now;
+ DCHECK(!frame_stats_[0].ack_time.is_null());
}
CongestionControl::~CongestionControl() {}
-bool CongestionControl::OnAck(base::TimeDelta rtt, uint32* new_bitrate) {
- base::TimeTicks now = clock_->NowTicks();
+void CongestionControl::UpdateRTT(base::TimeDelta rtt) {
+ rtt_ = base::TimeDelta::FromSecondsD(
+ (rtt_.InSecondsF() * 7 + rtt.InSecondsF()) / 8);
+}
+
+// Calculate how much "dead air" there is between two frames.
+base::TimeDelta CongestionControl::DeadTime(const FrameStats& a,
+ const FrameStats& b) {
+ if (b.sent_time > a.ack_time) {
+ return b.sent_time - a.ack_time;
+ } else {
+ return base::TimeDelta();
+ }
+}
+
+double CongestionControl::CalculateSafeBitrate() {
+ double transmit_time =
+ (GetFrameStats(last_acked_frame_)->ack_time -
+ frame_stats_.front().sent_time - dead_time_in_history_).InSecondsF();
+
+ if (acked_bits_in_history_ == 0 || transmit_time <= 0.0) {
+ return min_bitrate_configured_;
+ }
+ return acked_bits_in_history_ / transmit_time;
Alpha Left Google 2014/06/10 22:20:39 What happens if transmit_time is very close to zer
hubbe 2014/06/11 00:04:14 It's pretty unlikely, except possibly for the firs
Alpha Left Google 2014/06/11 01:16:28 Okay.
+}
+
+CongestionControl::FrameStats* CongestionControl::GetFrameStats(
+ uint32 frame_id) {
+ int32 offset = static_cast<int32>(frame_id - last_frame_stats_);
+ DCHECK_LT(offset, 100);
Alpha Left Google 2014/06/10 22:20:39 Should be kHistorySize instead of 100.
hubbe 2014/06/11 00:04:15 Done.
+ if (offset > 0) {
+ frame_stats_.resize(frame_stats_.size() + offset);
+ last_frame_stats_ += offset;
+ offset = 0;
+ }
+ while (frame_stats_.size() > history_size_) {
+ DCHECK_GT(frame_stats_.size(), 1UL);
+ DCHECK(!frame_stats_[0].ack_time.is_null());
+ acked_bits_in_history_ -= frame_stats_[0].frame_size;
+ dead_time_in_history_ -= DeadTime(frame_stats_[0], frame_stats_[1]);
+ DCHECK_GE(acked_bits_in_history_, 0UL);
+ VLOG(2) << "DT: " << dead_time_in_history_.InSecondsF();
+ DCHECK_GE(dead_time_in_history_.InSecondsF(), 0.0);
+ frame_stats_.pop_front();
+ }
+ offset += frame_stats_.size() - 1;
+ if (offset < 0) {
+ return NULL;
+ }
+ DCHECK_LT(offset, static_cast<int32>(frame_stats_.size()));
Alpha Left Google 2014/06/10 22:20:38 This should be an if statement to prevent out of b
hubbe 2014/06/11 00:04:15 I don't think so. If an out-of-bounds indexing hap
Alpha Left Google 2014/06/11 01:16:28 If |offset| is trivial then I think CHECK would be
hubbe 2014/06/11 07:11:47 Done.
+ return &frame_stats_[offset];
+}
- // First feedback?
- if (time_last_increase_.is_null()) {
- time_last_increase_ = now;
- time_last_decrease_ = now;
- return false;
+void CongestionControl::AckFrame(uint32 frame_id, base::TimeTicks when) {
+ FrameStats* frame_stats = GetFrameStats(last_acked_frame_);
+ while (IsNewerFrameId(frame_id, last_acked_frame_)) {
+ FrameStats* last_frame_stats = frame_stats;
+ last_acked_frame_++;
+ frame_stats = GetFrameStats(last_acked_frame_);
+ DCHECK(frame_stats);
+ frame_stats->ack_time = when;
+ acked_bits_in_history_ += frame_stats->frame_size;
+ dead_time_in_history_ += DeadTime(*last_frame_stats, *frame_stats);
}
- // Are we at the max bitrate?
- if (max_bitrate_configured_ == bitrate_)
- return false;
-
- // Make sure RTT is never less than 1 ms.
- rtt = std::max(rtt, base::TimeDelta::FromMilliseconds(1));
-
- base::TimeDelta elapsed_time =
- std::min(now - time_last_increase_,
- base::TimeDelta::FromMilliseconds(kMaxElapsedTimeMs));
- base::TimeDelta change_interval = std::max(
- rtt,
- base::TimeDelta::FromMilliseconds(kCongestionControlMinChangeIntervalMs));
- change_interval = std::min(
- change_interval,
- base::TimeDelta::FromMilliseconds(kCongestionControlMaxChangeIntervalMs));
-
- // Have enough time have passed?
- if (elapsed_time < change_interval)
- return false;
-
- time_last_increase_ = now;
-
- // One packet per RTT multiplied by the elapsed time fraction.
- // 1500 * 8 * (1000 / rtt_ms) * (elapsed_time_ms / 1000) =>
- // 1500 * 8 * elapsed_time_ms / rtt_ms.
- uint32 bitrate_increase =
- (1500 * 8 * elapsed_time.InMilliseconds()) / rtt.InMilliseconds();
- uint32 max_bitrate_increase =
- kCongestionControlMaxBitrateIncreasePerMillisecond *
- elapsed_time.InMilliseconds();
- bitrate_increase = std::min(max_bitrate_increase, bitrate_increase);
- *new_bitrate = std::min(bitrate_increase + bitrate_, max_bitrate_configured_);
- bitrate_ = *new_bitrate;
- return true;
}
-bool CongestionControl::OnNack(base::TimeDelta rtt, uint32* new_bitrate) {
- base::TimeTicks now = clock_->NowTicks();
+void CongestionControl::SendFrameToTransport(uint32 frame_id,
+ size_t frame_size,
+ base::TimeTicks when) {
+ last_encoded_frame_ = frame_id;
+ FrameStats* frame_stats = GetFrameStats(frame_id);
+ DCHECK(frame_stats);
+ frame_stats->frame_size = frame_size;
+ frame_stats->sent_time = when;
+}
+
+base::TimeTicks CongestionControl::EstimatedAckTime(uint32 frame_id,
+ double bitrate) {
+ FrameStats* frame_stats = GetFrameStats(frame_id);
+ DCHECK(frame_stats);
+ if (frame_stats->ack_time.is_null()) {
+ DCHECK(frame_stats->frame_size) << "frame_id: " << frame_id;
+ base::TimeTicks ret = EstimatedSendingTime(frame_id, bitrate);
+ ret += base::TimeDelta::FromSecondsD(frame_stats->frame_size / bitrate);
+ ret += rtt_;
+ base::TimeTicks now = clock_->NowTicks();
+ if (ret < now) {
+ // Compromise: We estimated that this frame should have been acked by
+ // now, but it has in fact not been acked. We assume that some of the
+ // information has been sent and received, but not all of it. For now
+ // we simply assume half.
+ return now + (now - ret) / 2;
Alpha Left Google 2014/06/10 22:20:39 I don't think this is doing what the comments sugg
hubbe 2014/06/11 00:04:15 Changed the comment instead. On 2014/06/10 22:20:
+ } else {
+ return ret;
+ }
+ } else {
+ return frame_stats->ack_time;
+ }
+}
- // First feedback?
- if (time_last_decrease_.is_null()) {
- time_last_increase_ = now;
- time_last_decrease_ = now;
- return false;
+base::TimeTicks CongestionControl::EstimatedSendingTime(uint32 frame_id,
+ double bitrate) {
+ FrameStats* frame_stats = GetFrameStats(frame_id);
Alpha Left Google 2014/06/10 22:20:39 We should be able to get this from the pacer, why
hubbe 2014/06/11 00:04:15 Sending time is when we start sending data for a f
+ DCHECK(frame_stats);
+ base::TimeTicks ret = EstimatedAckTime(frame_id - 1, bitrate) - rtt_;
+ if (frame_stats->sent_time.is_null()) {
Alpha Left Google 2014/06/10 22:20:38 How is this possible?
hubbe 2014/06/11 00:04:15 It hasn't been sent yet. Note: "sent time" is when
+ // Not sent yet, but we can't start sending it in the past.
+ return std::max(ret, clock_->NowTicks());
+ } else {
+ return std::max(ret, frame_stats->sent_time);
Alpha Left Google 2014/06/10 22:20:39 This assumes there is a black box that sends the n
hubbe 2014/06/11 00:04:15 In simulation, it works even with high latencies.
}
- base::TimeDelta elapsed_time =
- std::min(now - time_last_decrease_,
- base::TimeDelta::FromMilliseconds(kMaxElapsedTimeMs));
- base::TimeDelta change_interval = std::max(
- rtt,
- base::TimeDelta::FromMilliseconds(kCongestionControlMinChangeIntervalMs));
- change_interval = std::min(
- change_interval,
- base::TimeDelta::FromMilliseconds(kCongestionControlMaxChangeIntervalMs));
-
- // Have enough time have passed?
- if (elapsed_time < change_interval)
- return false;
-
- time_last_decrease_ = now;
- time_last_increase_ = now;
-
- *new_bitrate =
- std::max(static_cast<uint32>(bitrate_ * congestion_control_back_off_),
- min_bitrate_configured_);
-
- bitrate_ = *new_bitrate;
- return true;
+}
+
+uint32 CongestionControl::GetBitrate(base::TimeTicks playout_time,
+ base::TimeDelta playout_delay) {
+ double safe_bitrate = CalculateSafeBitrate();
+ // Estimate when we might start sending the next frame.
+ base::TimeDelta time_to_catch_up =
+ playout_time -
+ EstimatedSendingTime(last_encoded_frame_ + 1, safe_bitrate);
+
+ double empty_buffer_fraction =
+ time_to_catch_up.InSecondsF() / playout_delay.InSecondsF();
+ empty_buffer_fraction = std::min(empty_buffer_fraction, 1.0);
+ empty_buffer_fraction = std::max(empty_buffer_fraction, 0.0);
+
+ uint32 bits_per_second = static_cast<uint32>(
+ safe_bitrate * empty_buffer_fraction / kTargetEmptyBufferFraction);
+ VLOG(3) << " FBR:" << (bits_per_second / 1E6)
+ << " EBF:" << empty_buffer_fraction
+ << " SBR:" << (safe_bitrate / 1E6);
+ bits_per_second = std::max(bits_per_second, min_bitrate_configured_);
+ bits_per_second = std::min(bits_per_second, max_bitrate_configured_);
+ return bits_per_second;
}
} // namespace cast

Powered by Google App Engine
This is Rietveld 408576698