Source: lib/abr/ewma_bandwidth_estimator.js

  1. /**
  2. * @license
  3. * Copyright 2016 Google Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. goog.provide('shaka.abr.EwmaBandwidthEstimator');
  18. goog.require('shaka.abr.Ewma');
  19. /**
  20. * Tracks bandwidth samples and estimates available bandwidth.
  21. * Based on the minimum of two exponentially-weighted moving averages with
  22. * different half-lives.
  23. *
  24. * @constructor
  25. * @struct
  26. */
  27. shaka.abr.EwmaBandwidthEstimator = function() {
  28. /**
  29. * A fast-moving average.
  30. * Half of the estimate is based on the last 2 seconds of sample history.
  31. * @private {!shaka.abr.Ewma}
  32. */
  33. this.fast_ = new shaka.abr.Ewma(2);
  34. /**
  35. * A slow-moving average.
  36. * Half of the estimate is based on the last 5 seconds of sample history.
  37. * @private {!shaka.abr.Ewma}
  38. */
  39. this.slow_ = new shaka.abr.Ewma(5);
  40. /**
  41. * Number of bytes sampled.
  42. * @private {number}
  43. */
  44. this.bytesSampled_ = 0;
  45. /**
  46. * Minimum number of bytes sampled before we trust the estimate. If we have
  47. * not sampled much data, our estimate may not be accurate enough to trust.
  48. * If bytesSampled_ is less than minTotalBytes_, we use defaultEstimate_.
  49. * This specific value is based on experimentation.
  50. *
  51. * @private {number}
  52. * @const
  53. */
  54. this.minTotalBytes_ = 128e3; // 128kB
  55. /**
  56. * Minimum number of bytes, under which samples are discarded. Our models do
  57. * not include latency information, so connection startup time (time to first
  58. * byte) is considered part of the download time. Because of this, we should
  59. * ignore very small downloads which would cause our estimate to be too low.
  60. * This specific value is based on experimentation.
  61. *
  62. * @private {number}
  63. * @const
  64. */
  65. this.minBytes_ = 16e3; // 16kB
  66. };
  67. /**
  68. * Takes a bandwidth sample.
  69. *
  70. * @param {number} durationMs The amount of time, in milliseconds, for a
  71. * particular request.
  72. * @param {number} numBytes The total number of bytes transferred in that
  73. * request.
  74. */
  75. shaka.abr.EwmaBandwidthEstimator.prototype.sample = function(
  76. durationMs, numBytes) {
  77. if (numBytes < this.minBytes_) {
  78. return;
  79. }
  80. let bandwidth = 8000 * numBytes / durationMs;
  81. let weight = durationMs / 1000;
  82. this.bytesSampled_ += numBytes;
  83. this.fast_.sample(weight, bandwidth);
  84. this.slow_.sample(weight, bandwidth);
  85. };
  86. /**
  87. * Gets the current bandwidth estimate.
  88. *
  89. * @param {number} defaultEstimate
  90. * @return {number} The bandwidth estimate in bits per second.
  91. */
  92. shaka.abr.EwmaBandwidthEstimator.prototype.getBandwidthEstimate =
  93. function(defaultEstimate) {
  94. if (this.bytesSampled_ < this.minTotalBytes_) {
  95. return defaultEstimate;
  96. }
  97. // Take the minimum of these two estimates. This should have the effect of
  98. // adapting down quickly, but up more slowly.
  99. return Math.min(this.fast_.getEstimate(), this.slow_.getEstimate());
  100. };
  101. /**
  102. * @return {boolean} True if there is enough data to produce a meaningful
  103. * estimate.
  104. */
  105. shaka.abr.EwmaBandwidthEstimator.prototype.hasGoodEstimate = function() {
  106. return this.bytesSampled_ >= this.minTotalBytes_;
  107. };