Home Reference Source

src/controller/gap-controller.ts

  1. import type { BufferInfo } from '../utils/buffer-helper';
  2. import { BufferHelper } from '../utils/buffer-helper';
  3. import { ErrorTypes, ErrorDetails } from '../errors';
  4. import { Events } from '../events';
  5. import { logger } from '../utils/logger';
  6. import type Hls from '../hls';
  7. import type { HlsConfig } from '../config';
  8. import type { FragmentTracker } from './fragment-tracker';
  9. import { Fragment } from '../loader/fragment';
  10.  
  11. export const STALL_MINIMUM_DURATION_MS = 250;
  12. export const MAX_START_GAP_JUMP = 2.0;
  13. export const SKIP_BUFFER_HOLE_STEP_SECONDS = 0.1;
  14. export const SKIP_BUFFER_RANGE_START = 0.05;
  15.  
  16. export default class GapController {
  17. private config: HlsConfig;
  18. private media: HTMLMediaElement;
  19. private fragmentTracker: FragmentTracker;
  20. private hls: Hls;
  21. private nudgeRetry: number = 0;
  22. private stallReported: boolean = false;
  23. private stalled: number | null = null;
  24. private moved: boolean = false;
  25. private seeking: boolean = false;
  26.  
  27. constructor(config, media, fragmentTracker, hls) {
  28. this.config = config;
  29. this.media = media;
  30. this.fragmentTracker = fragmentTracker;
  31. this.hls = hls;
  32. }
  33.  
  34. public destroy() {
  35. // @ts-ignore
  36. this.hls = this.fragmentTracker = this.media = null;
  37. }
  38.  
  39. /**
  40. * Checks if the playhead is stuck within a gap, and if so, attempts to free it.
  41. * A gap is an unbuffered range between two buffered ranges (or the start and the first buffered range).
  42. *
  43. * @param {number} lastCurrentTime Previously read playhead position
  44. */
  45. public poll(lastCurrentTime: number) {
  46. const { config, media, stalled } = this;
  47. const { currentTime, seeking } = media;
  48. const seeked = this.seeking && !seeking;
  49. const beginSeek = !this.seeking && seeking;
  50.  
  51. this.seeking = seeking;
  52.  
  53. // The playhead is moving, no-op
  54. if (currentTime !== lastCurrentTime) {
  55. this.moved = true;
  56. if (stalled !== null) {
  57. // The playhead is now moving, but was previously stalled
  58. if (this.stallReported) {
  59. const stalledDuration = self.performance.now() - stalled;
  60. logger.warn(
  61. `playback not stuck anymore @${currentTime}, after ${Math.round(
  62. stalledDuration
  63. )}ms`
  64. );
  65. this.stallReported = false;
  66. }
  67. this.stalled = null;
  68. this.nudgeRetry = 0;
  69. }
  70. return;
  71. }
  72.  
  73. // Clear stalled state when beginning or finishing seeking so that we don't report stalls coming out of a seek
  74. if (beginSeek || seeked) {
  75. this.stalled = null;
  76. }
  77.  
  78. // The playhead should not be moving
  79. if (
  80. media.paused ||
  81. media.ended ||
  82. media.playbackRate === 0 ||
  83. !BufferHelper.getBuffered(media).length
  84. ) {
  85. return;
  86. }
  87.  
  88. const bufferInfo = BufferHelper.bufferInfo(media, currentTime, 0);
  89. const isBuffered = bufferInfo.len > 0;
  90. const nextStart = bufferInfo.nextStart || 0;
  91.  
  92. // There is no playable buffer (seeked, waiting for buffer)
  93. if (!isBuffered && !nextStart) {
  94. return;
  95. }
  96.  
  97. if (seeking) {
  98. // Waiting for seeking in a buffered range to complete
  99. const hasEnoughBuffer = bufferInfo.len > MAX_START_GAP_JUMP;
  100. // Next buffered range is too far ahead to jump to while still seeking
  101. const noBufferGap =
  102. !nextStart ||
  103. (nextStart - currentTime > MAX_START_GAP_JUMP &&
  104. !this.fragmentTracker.getPartialFragment(currentTime));
  105. if (hasEnoughBuffer || noBufferGap) {
  106. return;
  107. }
  108. // Reset moved state when seeking to a point in or before a gap
  109. this.moved = false;
  110. }
  111.  
  112. // Skip start gaps if we haven't played, but the last poll detected the start of a stall
  113. // The addition poll gives the browser a chance to jump the gap for us
  114. if (!this.moved && this.stalled !== null) {
  115. // Jump start gaps within jump threshold
  116. const startJump =
  117. Math.max(nextStart, bufferInfo.start || 0) - currentTime;
  118.  
  119. // When joining a live stream with audio tracks, account for live playlist window sliding by allowing
  120. // a larger jump over start gaps caused by the audio-stream-controller buffering a start fragment
  121. // that begins over 1 target duration after the video start position.
  122. const level = this.hls.levels
  123. ? this.hls.levels[this.hls.currentLevel]
  124. : null;
  125. const isLive = level?.details?.live;
  126. const maxStartGapJump = isLive
  127. ? level!.details!.targetduration * 2
  128. : MAX_START_GAP_JUMP;
  129. if (startJump > 0 && startJump <= maxStartGapJump) {
  130. this._trySkipBufferHole(null);
  131. return;
  132. }
  133. }
  134.  
  135. // Start tracking stall time
  136. const tnow = self.performance.now();
  137. if (stalled === null) {
  138. this.stalled = tnow;
  139. return;
  140. }
  141.  
  142. const stalledDuration = tnow - stalled;
  143. if (!seeking && stalledDuration >= STALL_MINIMUM_DURATION_MS) {
  144. // Report stalling after trying to fix
  145. this._reportStall(bufferInfo.len);
  146. }
  147.  
  148. const bufferedWithHoles = BufferHelper.bufferInfo(
  149. media,
  150. currentTime,
  151. config.maxBufferHole
  152. );
  153. this._tryFixBufferStall(bufferedWithHoles, stalledDuration);
  154. }
  155.  
  156. /**
  157. * Detects and attempts to fix known buffer stalling issues.
  158. * @param bufferInfo - The properties of the current buffer.
  159. * @param stalledDurationMs - The amount of time Hls.js has been stalling for.
  160. * @private
  161. */
  162. private _tryFixBufferStall(
  163. bufferInfo: BufferInfo,
  164. stalledDurationMs: number
  165. ) {
  166. const { config, fragmentTracker, media } = this;
  167. const currentTime = media.currentTime;
  168.  
  169. const partial = fragmentTracker.getPartialFragment(currentTime);
  170. if (partial) {
  171. // Try to skip over the buffer hole caused by a partial fragment
  172. // This method isn't limited by the size of the gap between buffered ranges
  173. const targetTime = this._trySkipBufferHole(partial);
  174. // we return here in this case, meaning
  175. // the branch below only executes when we don't handle a partial fragment
  176. if (targetTime) {
  177. return;
  178. }
  179. }
  180.  
  181. // if we haven't had to skip over a buffer hole of a partial fragment
  182. // we may just have to "nudge" the playlist as the browser decoding/rendering engine
  183. // needs to cross some sort of threshold covering all source-buffers content
  184. // to start playing properly.
  185. if (
  186. bufferInfo.len > config.maxBufferHole &&
  187. stalledDurationMs > config.highBufferWatchdogPeriod * 1000
  188. ) {
  189. logger.warn('Trying to nudge playhead over buffer-hole');
  190. // Try to nudge currentTime over a buffer hole if we've been stalling for the configured amount of seconds
  191. // We only try to jump the hole if it's under the configured size
  192. // Reset stalled so to rearm watchdog timer
  193. this.stalled = null;
  194. this._tryNudgeBuffer();
  195. }
  196. }
  197.  
  198. /**
  199. * Triggers a BUFFER_STALLED_ERROR event, but only once per stall period.
  200. * @param bufferLen - The playhead distance from the end of the current buffer segment.
  201. * @private
  202. */
  203. private _reportStall(bufferLen) {
  204. const { hls, media, stallReported } = this;
  205. if (!stallReported) {
  206. // Report stalled error once
  207. this.stallReported = true;
  208. logger.warn(
  209. `Playback stalling at @${media.currentTime} due to low buffer (buffer=${bufferLen})`
  210. );
  211. hls.trigger(Events.ERROR, {
  212. type: ErrorTypes.MEDIA_ERROR,
  213. details: ErrorDetails.BUFFER_STALLED_ERROR,
  214. fatal: false,
  215. buffer: bufferLen,
  216. });
  217. }
  218. }
  219.  
  220. /**
  221. * Attempts to fix buffer stalls by jumping over known gaps caused by partial fragments
  222. * @param partial - The partial fragment found at the current time (where playback is stalling).
  223. * @private
  224. */
  225. private _trySkipBufferHole(partial: Fragment | null): number {
  226. const { config, hls, media } = this;
  227. const currentTime = media.currentTime;
  228. let lastEndTime = 0;
  229. // Check if currentTime is between unbuffered regions of partial fragments
  230. const buffered = BufferHelper.getBuffered(media);
  231. for (let i = 0; i < buffered.length; i++) {
  232. const startTime = buffered.start(i);
  233. if (
  234. currentTime + config.maxBufferHole >= lastEndTime &&
  235. currentTime < startTime
  236. ) {
  237. const targetTime = Math.max(
  238. startTime + SKIP_BUFFER_RANGE_START,
  239. media.currentTime + SKIP_BUFFER_HOLE_STEP_SECONDS
  240. );
  241. logger.warn(
  242. `skipping hole, adjusting currentTime from ${currentTime} to ${targetTime}`
  243. );
  244. this.moved = true;
  245. this.stalled = null;
  246. media.currentTime = targetTime;
  247. if (partial) {
  248. hls.trigger(Events.ERROR, {
  249. type: ErrorTypes.MEDIA_ERROR,
  250. details: ErrorDetails.BUFFER_SEEK_OVER_HOLE,
  251. fatal: false,
  252. reason: `fragment loaded with buffer holes, seeking from ${currentTime} to ${targetTime}`,
  253. frag: partial,
  254. });
  255. }
  256. return targetTime;
  257. }
  258. lastEndTime = buffered.end(i);
  259. }
  260. return 0;
  261. }
  262.  
  263. /**
  264. * Attempts to fix buffer stalls by advancing the mediaElement's current time by a small amount.
  265. * @private
  266. */
  267. private _tryNudgeBuffer() {
  268. const { config, hls, media } = this;
  269. const currentTime = media.currentTime;
  270. const nudgeRetry = (this.nudgeRetry || 0) + 1;
  271. this.nudgeRetry = nudgeRetry;
  272.  
  273. if (nudgeRetry < config.nudgeMaxRetry) {
  274. const targetTime = currentTime + nudgeRetry * config.nudgeOffset;
  275. // playback stalled in buffered area ... let's nudge currentTime to try to overcome this
  276. logger.warn(`Nudging 'currentTime' from ${currentTime} to ${targetTime}`);
  277. media.currentTime = targetTime;
  278. hls.trigger(Events.ERROR, {
  279. type: ErrorTypes.MEDIA_ERROR,
  280. details: ErrorDetails.BUFFER_NUDGE_ON_STALL,
  281. fatal: false,
  282. });
  283. } else {
  284. logger.error(
  285. `Playhead still not moving while enough data buffered @${currentTime} after ${config.nudgeMaxRetry} nudges`
  286. );
  287. hls.trigger(Events.ERROR, {
  288. type: ErrorTypes.MEDIA_ERROR,
  289. details: ErrorDetails.BUFFER_STALLED_ERROR,
  290. fatal: true,
  291. });
  292. }
  293. }
  294. }