Source: lib/abr/simple_abr_manager.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.SimpleAbrManager');
  18. goog.require('goog.asserts');
  19. goog.require('shaka.abr.EwmaBandwidthEstimator');
  20. goog.require('shaka.log');
  21. /**
  22. * Creates a new SimpleAbrManager.
  23. *
  24. * @constructor
  25. * @struct
  26. * @implements {shakaExtern.AbrManager}
  27. * @export
  28. */
  29. shaka.abr.SimpleAbrManager = function() {
  30. /** @private {?shakaExtern.AbrManager.SwitchCallback} */
  31. this.switch_ = null;
  32. /** @private {boolean} */
  33. this.enabled_ = false;
  34. /** @private {shaka.abr.EwmaBandwidthEstimator} */
  35. this.bandwidthEstimator_ = new shaka.abr.EwmaBandwidthEstimator();
  36. /**
  37. * The last StreamSets given to us via chooseStreams().
  38. * @private {Object.<string, shakaExtern.StreamSet>}
  39. */
  40. this.streamSetsByType_ = {};
  41. /**
  42. * The last Streams chosen.
  43. * @private {Object.<string, shakaExtern.Stream>}
  44. */
  45. this.streamsByType_ = {};
  46. /** @private {boolean} */
  47. this.startupComplete_ = false;
  48. /**
  49. * The last wall-clock time, in milliseconds, when Streams were chosen via
  50. * chooseStreams() or switch_().
  51. *
  52. * @private {?number}
  53. */
  54. this.lastTimeChosenMs_ = null;
  55. };
  56. /**
  57. * The minimum amount of time that must pass between switches, in milliseconds.
  58. * This keeps us from changing too often and annoying the user.
  59. *
  60. * @const {number}
  61. */
  62. shaka.abr.SimpleAbrManager.SWITCH_INTERVAL_MS = 8000;
  63. /**
  64. * The fraction of the estimated bandwidth which we should try to use when
  65. * upgrading.
  66. *
  67. * @private
  68. * @const {number}
  69. */
  70. shaka.abr.SimpleAbrManager.BANDWIDTH_UPGRADE_TARGET_ = 0.85;
  71. /**
  72. * The largest fraction of the estimated bandwidth we should use. We should
  73. * downgrade to avoid this.
  74. *
  75. * @private
  76. * @const {number}
  77. */
  78. shaka.abr.SimpleAbrManager.BANDWIDTH_DOWNGRADE_TARGET_ = 0.95;
  79. /**
  80. * @override
  81. * @export
  82. */
  83. shaka.abr.SimpleAbrManager.prototype.stop = function() {
  84. this.switch_ = null;
  85. this.enabled_ = false;
  86. this.streamSetsByType_ = {};
  87. this.streamsByType_ = {};
  88. this.lastTimeChosenMs_ = null;
  89. // Don't reset |startupComplete_|: if we've left the startup interval then we
  90. // can start using bandwidth estimates right away if init() is called again.
  91. };
  92. /**
  93. * @override
  94. * @export
  95. */
  96. shaka.abr.SimpleAbrManager.prototype.init = function(switchCallback) {
  97. this.switch_ = switchCallback;
  98. };
  99. /**
  100. * @override
  101. * @export
  102. */
  103. shaka.abr.SimpleAbrManager.prototype.chooseStreams = function(
  104. streamSetsByType) {
  105. // Merge StreamSets. We may have been given a partial list.
  106. for (var type in streamSetsByType) {
  107. this.streamSetsByType_[type] = streamSetsByType[type];
  108. }
  109. // Choose streams for the specific types requested.
  110. var chosen = {};
  111. if ('audio' in streamSetsByType) {
  112. var audioStream = this.chooseAudioStream_();
  113. if (audioStream) {
  114. chosen['audio'] = audioStream;
  115. this.streamsByType_['audio'] = audioStream;
  116. } else {
  117. delete this.streamsByType_['audio'];
  118. }
  119. }
  120. if ('video' in streamSetsByType) {
  121. var videoStream = this.chooseVideoStream_();
  122. if (videoStream) {
  123. chosen['video'] = videoStream;
  124. this.streamsByType_['video'] = videoStream;
  125. } else {
  126. delete this.streamsByType_['video'];
  127. }
  128. }
  129. if ('text' in streamSetsByType) {
  130. // We don't adapt text, so just choose stream 0.
  131. chosen['text'] = streamSetsByType['text'].streams[0];
  132. }
  133. this.lastTimeChosenMs_ = Date.now();
  134. return chosen;
  135. };
  136. /**
  137. * @override
  138. * @export
  139. */
  140. shaka.abr.SimpleAbrManager.prototype.enable = function() {
  141. this.enabled_ = true;
  142. };
  143. /**
  144. * @override
  145. * @export
  146. */
  147. shaka.abr.SimpleAbrManager.prototype.disable = function() {
  148. this.enabled_ = false;
  149. };
  150. /**
  151. * @override
  152. * @export
  153. */
  154. shaka.abr.SimpleAbrManager.prototype.segmentDownloaded = function(
  155. startTimeMs, endTimeMs, numBytes) {
  156. shaka.log.v2('Segment downloaded:',
  157. 'startTimeMs=' + startTimeMs,
  158. 'endTimeMs=' + endTimeMs,
  159. 'numBytes=' + numBytes);
  160. goog.asserts.assert(endTimeMs >= startTimeMs,
  161. 'expected a non-negative duration');
  162. this.bandwidthEstimator_.sample(endTimeMs - startTimeMs, numBytes);
  163. if ((this.lastTimeChosenMs_ != null) && this.enabled_)
  164. this.suggestStreams_();
  165. };
  166. /**
  167. * @override
  168. * @export
  169. */
  170. shaka.abr.SimpleAbrManager.prototype.getBandwidthEstimate = function() {
  171. return this.bandwidthEstimator_.getBandwidthEstimate();
  172. };
  173. /**
  174. * @override
  175. * @export
  176. */
  177. shaka.abr.SimpleAbrManager.prototype.setDefaultEstimate = function(estimate) {
  178. this.bandwidthEstimator_.setDefaultEstimate(estimate);
  179. };
  180. /**
  181. * Calls switch_() with which Streams to switch to.
  182. *
  183. * @private
  184. */
  185. shaka.abr.SimpleAbrManager.prototype.suggestStreams_ = function() {
  186. shaka.log.v2('Suggesting Streams...');
  187. goog.asserts.assert(this.lastTimeChosenMs_ != null,
  188. 'lastTimeChosenMs_ should not be null');
  189. if (!this.startupComplete_) {
  190. // Check if we've got enough data yet.
  191. if (!this.bandwidthEstimator_.hasGoodEstimate()) {
  192. shaka.log.v2('Still waiting for a good estimate...');
  193. return;
  194. }
  195. this.startupComplete_ = true;
  196. } else {
  197. // Check if we've left the switch interval.
  198. var now = Date.now();
  199. var delta = now - this.lastTimeChosenMs_;
  200. if (delta < shaka.abr.SimpleAbrManager.SWITCH_INTERVAL_MS) {
  201. shaka.log.v2('Still within switch interval...');
  202. return;
  203. }
  204. }
  205. var chosen = this.chooseStreams_();
  206. var currentBandwidthKbps =
  207. Math.round(this.bandwidthEstimator_.getBandwidthEstimate() / 1000.0);
  208. shaka.log.debug(
  209. 'Calling switch_(), bandwidth=' + currentBandwidthKbps + ' kbps');
  210. // If any of these chosen streams are already chosen, Player will filter them
  211. // out before passing the choices on to StreamingEngine.
  212. this.switch_(chosen);
  213. };
  214. /**
  215. * Chooses which Streams to switch to.
  216. *
  217. * @return {!Object.<string, !shakaExtern.Stream>}
  218. * @private
  219. */
  220. shaka.abr.SimpleAbrManager.prototype.chooseStreams_ = function() {
  221. var streamsByType = {};
  222. // Choose audio Stream.
  223. var audioStream = this.chooseAudioStream_();
  224. if (audioStream) {
  225. streamsByType['audio'] = audioStream;
  226. this.streamsByType_['audio'] = audioStream;
  227. }
  228. // Choose video Stream.
  229. var videoStream = this.chooseVideoStream_();
  230. if (videoStream) {
  231. streamsByType['video'] = videoStream;
  232. this.streamsByType_['video'] = videoStream;
  233. }
  234. this.lastTimeChosenMs_ = Date.now();
  235. return streamsByType;
  236. };
  237. /**
  238. * Chooses which audio Stream to switch to.
  239. *
  240. * @return {?shakaExtern.Stream}
  241. * @private
  242. */
  243. shaka.abr.SimpleAbrManager.prototype.chooseAudioStream_ = function() {
  244. // Alias.
  245. var SimpleAbrManager = shaka.abr.SimpleAbrManager;
  246. // Get sorted audio Streams.
  247. var audioStreamSet = this.streamSetsByType_['audio'];
  248. if (!audioStreamSet)
  249. return null;
  250. var audioStreams = SimpleAbrManager.sortStreamsByBandwidth_(audioStreamSet);
  251. // Just pick the middle one.
  252. // TODO: Implement better audio adaptation.
  253. return audioStreams[Math.floor(audioStreams.length / 2)];
  254. };
  255. /**
  256. * Chooses which video Stream to switch to.
  257. *
  258. * @return {?shakaExtern.Stream}
  259. * @private
  260. */
  261. shaka.abr.SimpleAbrManager.prototype.chooseVideoStream_ = function() {
  262. // Alias.
  263. var SimpleAbrManager = shaka.abr.SimpleAbrManager;
  264. // Get sorted video Streams.
  265. var videoStreamSet = this.streamSetsByType_['video'];
  266. if (!videoStreamSet)
  267. return null;
  268. var videoStreams = SimpleAbrManager.sortStreamsByBandwidth_(videoStreamSet);
  269. var audioStream = this.streamsByType_['audio'];
  270. var audioBandwidth = (audioStream && audioStream.bandwidth) || 0;
  271. var currentBandwidth = this.bandwidthEstimator_.getBandwidthEstimate();
  272. // Start by assuming that we will use the first Stream.
  273. var chosen = videoStreams[0];
  274. for (var i = 0; i < videoStreams.length; ++i) {
  275. var stream = videoStreams[i];
  276. var nextStream = (i + 1 < videoStreams.length) ?
  277. videoStreams[i + 1] :
  278. {bandwidth: Infinity};
  279. // Ignore Streams which don't have bandwidth information.
  280. if (!stream.bandwidth) continue;
  281. var minBandwidth = (stream.bandwidth + audioBandwidth) /
  282. SimpleAbrManager.BANDWIDTH_DOWNGRADE_TARGET_;
  283. var maxBandwidth = (nextStream.bandwidth + audioBandwidth) /
  284. SimpleAbrManager.BANDWIDTH_UPGRADE_TARGET_;
  285. shaka.log.v2('Bandwidth ranges:',
  286. ((stream.bandwidth + audioBandwidth) / 1e6).toFixed(3),
  287. (minBandwidth / 1e6).toFixed(3),
  288. (maxBandwidth / 1e6).toFixed(3));
  289. if (currentBandwidth >= minBandwidth && currentBandwidth <= maxBandwidth)
  290. chosen = stream;
  291. }
  292. return chosen;
  293. };
  294. /**
  295. * @param {!shakaExtern.StreamSet} streamSet
  296. * @return {!Array.<shakaExtern.Stream>} |streamSet|'s Streams sorted
  297. * in ascending order of bandwidth.
  298. * @private
  299. */
  300. shaka.abr.SimpleAbrManager.sortStreamsByBandwidth_ = function(streamSet) {
  301. return streamSet.streams.slice(0)
  302. .filter(function(s) {
  303. return s.allowedByApplication && s.allowedByKeySystem;
  304. })
  305. .sort(function(s1, s2) { return s1.bandwidth - s2.bandwidth; });
  306. };