Home Reference Source

src/controller/latency-controller.ts

  1. import { LevelDetails } from '../loader/level-details';
  2. import { ErrorDetails } from '../errors';
  3. import { Events } from '../events';
  4. import type {
  5. ErrorData,
  6. LevelUpdatedData,
  7. MediaAttachingData,
  8. } from '../types/events';
  9. import { logger } from '../utils/logger';
  10. import type { ComponentAPI } from '../types/component-api';
  11. import type Hls from '../hls';
  12. import type { HlsConfig } from '../config';
  13.  
  14. export default class LatencyController implements ComponentAPI {
  15. private hls: Hls;
  16. private readonly config: HlsConfig;
  17. private media: HTMLMediaElement | null = null;
  18. private levelDetails: LevelDetails | null = null;
  19. private currentTime: number = 0;
  20. private stallCount: number = 0;
  21. private _latency: number | null = null;
  22. private timeupdateHandler = () => this.timeupdate();
  23.  
  24. constructor(hls: Hls) {
  25. this.hls = hls;
  26. this.config = hls.config;
  27. this.registerListeners();
  28. }
  29.  
  30. get latency(): number {
  31. return this._latency || 0;
  32. }
  33.  
  34. get maxLatency(): number {
  35. const { config, levelDetails } = this;
  36. if (config.liveMaxLatencyDuration !== undefined) {
  37. return config.liveMaxLatencyDuration;
  38. }
  39. return levelDetails
  40. ? config.liveMaxLatencyDurationCount * levelDetails.targetduration
  41. : 0;
  42. }
  43.  
  44. get targetLatency(): number | null {
  45. const { levelDetails } = this;
  46. if (levelDetails === null) {
  47. return null;
  48. }
  49. const { holdBack, partHoldBack, targetduration } = levelDetails;
  50. const { liveSyncDuration, liveSyncDurationCount, lowLatencyMode } =
  51. this.config;
  52. const userConfig = this.hls.userConfig;
  53. let targetLatency = lowLatencyMode ? partHoldBack || holdBack : holdBack;
  54. if (
  55. userConfig.liveSyncDuration ||
  56. userConfig.liveSyncDurationCount ||
  57. targetLatency === 0
  58. ) {
  59. targetLatency =
  60. liveSyncDuration !== undefined
  61. ? liveSyncDuration
  62. : liveSyncDurationCount * targetduration;
  63. }
  64. const maxLiveSyncOnStallIncrease = targetduration;
  65. const liveSyncOnStallIncrease = 1.0;
  66. return (
  67. targetLatency +
  68. Math.min(
  69. this.stallCount * liveSyncOnStallIncrease,
  70. maxLiveSyncOnStallIncrease
  71. )
  72. );
  73. }
  74.  
  75. get liveSyncPosition(): number | null {
  76. const liveEdge = this.estimateLiveEdge();
  77. const targetLatency = this.targetLatency;
  78. const levelDetails = this.levelDetails;
  79. if (liveEdge === null || targetLatency === null || levelDetails === null) {
  80. return null;
  81. }
  82. const edge = levelDetails.edge;
  83. const syncPosition = liveEdge - targetLatency - this.edgeStalled;
  84. const min = edge - levelDetails.totalduration;
  85. const max =
  86. edge -
  87. ((this.config.lowLatencyMode && levelDetails.partTarget) ||
  88. levelDetails.targetduration);
  89. return Math.min(Math.max(min, syncPosition), max);
  90. }
  91.  
  92. get drift(): number {
  93. const { levelDetails } = this;
  94. if (levelDetails === null) {
  95. return 1;
  96. }
  97. return levelDetails.drift;
  98. }
  99.  
  100. get edgeStalled(): number {
  101. const { levelDetails } = this;
  102. if (levelDetails === null) {
  103. return 0;
  104. }
  105. const maxLevelUpdateAge =
  106. ((this.config.lowLatencyMode && levelDetails.partTarget) ||
  107. levelDetails.targetduration) * 3;
  108. return Math.max(levelDetails.age - maxLevelUpdateAge, 0);
  109. }
  110.  
  111. private get forwardBufferLength(): number {
  112. const { media, levelDetails } = this;
  113. if (!media || !levelDetails) {
  114. return 0;
  115. }
  116. const bufferedRanges = media.buffered.length;
  117. return bufferedRanges
  118. ? media.buffered.end(bufferedRanges - 1)
  119. : levelDetails.edge - this.currentTime;
  120. }
  121.  
  122. public destroy(): void {
  123. this.unregisterListeners();
  124. this.onMediaDetaching();
  125. this.levelDetails = null;
  126. // @ts-ignore
  127. this.hls = this.timeupdateHandler = null;
  128. }
  129.  
  130. private registerListeners() {
  131. this.hls.on(Events.MEDIA_ATTACHED, this.onMediaAttached, this);
  132. this.hls.on(Events.MEDIA_DETACHING, this.onMediaDetaching, this);
  133. this.hls.on(Events.MANIFEST_LOADING, this.onManifestLoading, this);
  134. this.hls.on(Events.LEVEL_UPDATED, this.onLevelUpdated, this);
  135. this.hls.on(Events.ERROR, this.onError, this);
  136. }
  137.  
  138. private unregisterListeners() {
  139. this.hls.off(Events.MEDIA_ATTACHED, this.onMediaAttached);
  140. this.hls.off(Events.MEDIA_DETACHING, this.onMediaDetaching);
  141. this.hls.off(Events.MANIFEST_LOADING, this.onManifestLoading);
  142. this.hls.off(Events.LEVEL_UPDATED, this.onLevelUpdated);
  143. this.hls.off(Events.ERROR, this.onError);
  144. }
  145.  
  146. private onMediaAttached(
  147. event: Events.MEDIA_ATTACHED,
  148. data: MediaAttachingData
  149. ) {
  150. this.media = data.media;
  151. this.media.addEventListener('timeupdate', this.timeupdateHandler);
  152. }
  153.  
  154. private onMediaDetaching() {
  155. if (this.media) {
  156. this.media.removeEventListener('timeupdate', this.timeupdateHandler);
  157. this.media = null;
  158. }
  159. }
  160.  
  161. private onManifestLoading() {
  162. this.levelDetails = null;
  163. this._latency = null;
  164. this.stallCount = 0;
  165. }
  166.  
  167. private onLevelUpdated(
  168. event: Events.LEVEL_UPDATED,
  169. { details }: LevelUpdatedData
  170. ) {
  171. this.levelDetails = details;
  172. if (details.advanced) {
  173. this.timeupdate();
  174. }
  175. if (!details.live && this.media) {
  176. this.media.removeEventListener('timeupdate', this.timeupdateHandler);
  177. }
  178. }
  179.  
  180. private onError(event: Events.ERROR, data: ErrorData) {
  181. if (data.details !== ErrorDetails.BUFFER_STALLED_ERROR) {
  182. return;
  183. }
  184. this.stallCount++;
  185. logger.warn(
  186. '[playback-rate-controller]: Stall detected, adjusting target latency'
  187. );
  188. }
  189.  
  190. private timeupdate() {
  191. const { media, levelDetails } = this;
  192. if (!media || !levelDetails) {
  193. return;
  194. }
  195. this.currentTime = media.currentTime;
  196.  
  197. const latency = this.computeLatency();
  198. if (latency === null) {
  199. return;
  200. }
  201. this._latency = latency;
  202.  
  203. // Adapt playbackRate to meet target latency in low-latency mode
  204. const { lowLatencyMode, maxLiveSyncPlaybackRate } = this.config;
  205. if (!lowLatencyMode || maxLiveSyncPlaybackRate === 1) {
  206. return;
  207. }
  208. const targetLatency = this.targetLatency;
  209. if (targetLatency === null) {
  210. return;
  211. }
  212. const distanceFromTarget = latency - targetLatency;
  213. // Only adjust playbackRate when within one target duration of targetLatency
  214. // and more than one second from under-buffering.
  215. // Playback further than one target duration from target can be considered DVR playback.
  216. const liveMinLatencyDuration = Math.min(
  217. this.maxLatency,
  218. targetLatency + levelDetails.targetduration
  219. );
  220. const inLiveRange = distanceFromTarget < liveMinLatencyDuration;
  221. if (
  222. levelDetails.live &&
  223. inLiveRange &&
  224. distanceFromTarget > 0.05 &&
  225. this.forwardBufferLength > 1
  226. ) {
  227. const max = Math.min(2, Math.max(1.0, maxLiveSyncPlaybackRate));
  228. const rate =
  229. Math.round(
  230. (2 / (1 + Math.exp(-0.75 * distanceFromTarget - this.edgeStalled))) *
  231. 20
  232. ) / 20;
  233. media.playbackRate = Math.min(max, Math.max(1, rate));
  234. } else if (media.playbackRate !== 1 && media.playbackRate !== 0) {
  235. media.playbackRate = 1;
  236. }
  237. }
  238.  
  239. private estimateLiveEdge(): number | null {
  240. const { levelDetails } = this;
  241. if (levelDetails === null) {
  242. return null;
  243. }
  244. return levelDetails.edge + levelDetails.age;
  245. }
  246.  
  247. private computeLatency(): number | null {
  248. const liveEdge = this.estimateLiveEdge();
  249. if (liveEdge === null) {
  250. return null;
  251. }
  252. return liveEdge - this.currentTime;
  253. }
  254. }