Source: lib/media/segment_reference.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.media.InitSegmentReference');
  7. goog.provide('shaka.media.SegmentReference');
  8. goog.require('goog.asserts');
  9. goog.require('shaka.log');
  10. goog.require('shaka.util.ArrayUtils');
  11. goog.require('shaka.util.BufferUtils');
  12. /**
  13. * Creates an InitSegmentReference, which provides the location to an
  14. * initialization segment.
  15. *
  16. * @export
  17. */
  18. shaka.media.InitSegmentReference = class {
  19. /**
  20. * @param {function():!Array.<string>} uris A function that creates the URIs
  21. * of the resource containing the segment.
  22. * @param {number} startByte The offset from the start of the resource to the
  23. * start of the segment.
  24. * @param {?number} endByte The offset from the start of the resource
  25. * to the end of the segment, inclusive. A value of null indicates that the
  26. * segment extends to the end of the resource.
  27. * @param {null|shaka.extern.MediaQualityInfo=} mediaQuality Information about
  28. * the quality of the media associated with this init segment.
  29. * @param {(null|number)=} timescale
  30. * @param {(null|BufferSource)=} segmentData
  31. * @param {?shaka.extern.aesKey=} aesKey
  32. * The segment's AES-128-CBC full segment encryption key and iv.
  33. */
  34. constructor(uris, startByte, endByte, mediaQuality = null, timescale = null,
  35. segmentData = null, aesKey = null) {
  36. /** @type {function():!Array.<string>} */
  37. this.getUris = uris;
  38. /** @const {number} */
  39. this.startByte = startByte;
  40. /** @const {?number} */
  41. this.endByte = endByte;
  42. /** @const {shaka.extern.MediaQualityInfo|null} */
  43. this.mediaQuality = mediaQuality;
  44. /** @type {number|null} */
  45. this.timescale = timescale;
  46. /** @type {BufferSource|null} */
  47. this.segmentData = segmentData;
  48. /** @type {?shaka.extern.aesKey} */
  49. this.aesKey = aesKey;
  50. /** @type {?string} */
  51. this.codecs = null;
  52. /** @type {?string} */
  53. this.mimeType = null;
  54. }
  55. /**
  56. * Returns the offset from the start of the resource to the
  57. * start of the segment.
  58. *
  59. * @return {number}
  60. * @export
  61. */
  62. getStartByte() {
  63. return this.startByte;
  64. }
  65. /**
  66. * Returns the offset from the start of the resource to the end of the
  67. * segment, inclusive. A value of null indicates that the segment extends
  68. * to the end of the resource.
  69. *
  70. * @return {?number}
  71. * @export
  72. */
  73. getEndByte() {
  74. return this.endByte;
  75. }
  76. /**
  77. * Returns the size of the init segment.
  78. * @return {?number}
  79. */
  80. getSize() {
  81. if (this.endByte) {
  82. return this.endByte - this.startByte;
  83. } else {
  84. return null;
  85. }
  86. }
  87. /**
  88. * Returns media quality information for the segments associated with
  89. * this init segment.
  90. *
  91. * @return {?shaka.extern.MediaQualityInfo}
  92. */
  93. getMediaQuality() {
  94. return this.mediaQuality;
  95. }
  96. /**
  97. * Return the segment data.
  98. *
  99. * @return {?BufferSource}
  100. */
  101. getSegmentData() {
  102. return this.segmentData;
  103. }
  104. /**
  105. * Check if two initSegmentReference have all the same values.
  106. * @param {?shaka.media.InitSegmentReference} reference1
  107. * @param {?shaka.media.InitSegmentReference} reference2
  108. * @return {boolean}
  109. */
  110. static equal(reference1, reference2) {
  111. const ArrayUtils = shaka.util.ArrayUtils;
  112. const BufferUtils = shaka.util.BufferUtils;
  113. if (reference1 === reference2) {
  114. return true;
  115. } else if (!reference1 || !reference2) {
  116. return reference1 == reference2;
  117. } else {
  118. return reference1.getStartByte() == reference2.getStartByte() &&
  119. reference1.getEndByte() == reference2.getEndByte() &&
  120. ArrayUtils.equal(
  121. reference1.getUris().sort(), reference2.getUris().sort()) &&
  122. BufferUtils.equal(reference1.getSegmentData(),
  123. reference2.getSegmentData());
  124. }
  125. }
  126. };
  127. /**
  128. * SegmentReference provides the start time, end time, and location to a media
  129. * segment.
  130. *
  131. * @export
  132. */
  133. shaka.media.SegmentReference = class {
  134. /**
  135. * @param {number} startTime The segment's start time in seconds.
  136. * @param {number} endTime The segment's end time in seconds. The segment
  137. * ends the instant before this time, so |endTime| must be strictly greater
  138. * than |startTime|.
  139. * @param {function():!Array.<string>} uris
  140. * A function that creates the URIs of the resource containing the segment.
  141. * @param {number} startByte The offset from the start of the resource to the
  142. * start of the segment.
  143. * @param {?number} endByte The offset from the start of the resource to the
  144. * end of the segment, inclusive. A value of null indicates that the
  145. * segment extends to the end of the resource.
  146. * @param {shaka.media.InitSegmentReference} initSegmentReference
  147. * The segment's initialization segment metadata, or null if the segments
  148. * are self-initializing.
  149. * @param {number} timestampOffset
  150. * The amount of time, in seconds, that must be added to the segment's
  151. * internal timestamps to align it to the presentation timeline.
  152. * <br>
  153. * For DASH, this value should equal the Period start time minus the first
  154. * presentation timestamp of the first frame/sample in the Period. For
  155. * example, for MP4 based streams, this value should equal Period start
  156. * minus the first segment's tfdt box's 'baseMediaDecodeTime' field (after
  157. * it has been converted to seconds).
  158. * <br>
  159. * For HLS, this value should be the start time of the most recent
  160. * discontinuity, or 0 if there is no preceding discontinuity. Only used
  161. * in segments mode.
  162. * @param {number} appendWindowStart
  163. * The start of the append window for this reference, relative to the
  164. * presentation. Any content from before this time will be removed by
  165. * MediaSource.
  166. * @param {number} appendWindowEnd
  167. * The end of the append window for this reference, relative to the
  168. * presentation. Any content from after this time will be removed by
  169. * MediaSource.
  170. * @param {!Array.<!shaka.media.SegmentReference>=} partialReferences
  171. * A list of SegmentReferences for the partial segments.
  172. * @param {?string=} tilesLayout
  173. * The value is a grid-item-dimension consisting of two positive decimal
  174. * integers in the format: column-x-row ('4x3'). It describes the
  175. * arrangement of Images in a Grid. The minimum valid LAYOUT is '1x1'.
  176. * @param {?number=} tileDuration
  177. * The explicit duration of an individual tile within the tiles grid.
  178. * If not provided, the duration should be automatically calculated based on
  179. * the duration of the reference.
  180. * @param {?number=} syncTime
  181. * A time value, expressed in seconds since 1970, which is used to
  182. * synchronize between streams. Both produced and consumed by the HLS
  183. * parser. Other components should not need this value.
  184. * @param {shaka.media.SegmentReference.Status=} status
  185. * The segment status is used to indicate that a segment does not exist or is
  186. * not available.
  187. * @param {?shaka.extern.aesKey=} aesKey
  188. * The segment's AES-128-CBC full segment encryption key and iv.
  189. * @param {boolean=} allPartialSegments
  190. * Indicate if the segment has all partial segments
  191. */
  192. constructor(
  193. startTime, endTime, uris, startByte, endByte, initSegmentReference,
  194. timestampOffset, appendWindowStart, appendWindowEnd,
  195. partialReferences = [], tilesLayout = '', tileDuration = null,
  196. syncTime = null, status = shaka.media.SegmentReference.Status.AVAILABLE,
  197. aesKey = null, allPartialSegments = false) {
  198. // A preload hinted Partial Segment has the same startTime and endTime.
  199. goog.asserts.assert(startTime <= endTime,
  200. 'startTime must be less than or equal to endTime');
  201. goog.asserts.assert((endByte == null) || (startByte < endByte),
  202. 'startByte must be < endByte');
  203. /** @type {number} */
  204. this.startTime = startTime;
  205. /** @type {number} */
  206. this.endTime = endTime;
  207. /**
  208. * The "true" end time of the segment, without considering the period end
  209. * time. This is necessary for thumbnail segments, where timing requires us
  210. * to know the original segment duration as described in the manifest.
  211. * @type {number}
  212. */
  213. this.trueEndTime = endTime;
  214. /** @type {function():!Array.<string>} */
  215. this.getUrisInner = uris;
  216. /** @const {number} */
  217. this.startByte = startByte;
  218. /** @const {?number} */
  219. this.endByte = endByte;
  220. /** @type {shaka.media.InitSegmentReference} */
  221. this.initSegmentReference = initSegmentReference;
  222. /** @type {number} */
  223. this.timestampOffset = timestampOffset;
  224. /** @type {number} */
  225. this.appendWindowStart = appendWindowStart;
  226. /** @type {number} */
  227. this.appendWindowEnd = appendWindowEnd;
  228. /** @type {!Array.<!shaka.media.SegmentReference>} */
  229. this.partialReferences = partialReferences;
  230. /** @type {?string} */
  231. this.tilesLayout = tilesLayout;
  232. /** @type {?number} */
  233. this.tileDuration = tileDuration;
  234. /**
  235. * A time value, expressed in seconds since 1970, which is used to
  236. * synchronize between streams. Both produced and consumed by the HLS
  237. * parser. Other components should not need this value.
  238. *
  239. * @type {?number}
  240. */
  241. this.syncTime = syncTime;
  242. /** @type {shaka.media.SegmentReference.Status} */
  243. this.status = status;
  244. /** @type {boolean} */
  245. this.preload = false;
  246. /** @type {boolean} */
  247. this.independent = true;
  248. /** @type {boolean} */
  249. this.byterangeOptimization = false;
  250. /** @type {?shaka.extern.aesKey} */
  251. this.aesKey = aesKey;
  252. /** @type {?shaka.media.SegmentReference.ThumbnailSprite} */
  253. this.thumbnailSprite = null;
  254. /** @type {number} */
  255. this.discontinuitySequence = -1;
  256. /** @type {boolean} */
  257. this.allPartialSegments = allPartialSegments;
  258. /** @type {boolean} */
  259. this.partial = false;
  260. /** @type {boolean} */
  261. this.lastPartial = false;
  262. for (const partial of this.partialReferences) {
  263. partial.markAsPartial();
  264. }
  265. if (this.allPartialSegments && this.partialReferences.length) {
  266. const lastPartial =
  267. this.partialReferences[this.partialReferences.length - 1];
  268. lastPartial.markAsLastPartial();
  269. }
  270. /** @type {?string} */
  271. this.codecs = null;
  272. /** @type {?string} */
  273. this.mimeType = null;
  274. /** @type {BufferSource|null} */
  275. this.segmentData = null;
  276. }
  277. /**
  278. * Creates and returns the URIs of the resource containing the segment.
  279. *
  280. * @return {!Array.<string>}
  281. * @export
  282. */
  283. getUris() {
  284. return this.getUrisInner();
  285. }
  286. /**
  287. * Returns the segment's start time in seconds.
  288. *
  289. * @return {number}
  290. * @export
  291. */
  292. getStartTime() {
  293. return this.startTime;
  294. }
  295. /**
  296. * Returns the segment's end time in seconds.
  297. *
  298. * @return {number}
  299. * @export
  300. */
  301. getEndTime() {
  302. return this.endTime;
  303. }
  304. /**
  305. * Returns the offset from the start of the resource to the
  306. * start of the segment.
  307. *
  308. * @return {number}
  309. * @export
  310. */
  311. getStartByte() {
  312. return this.startByte;
  313. }
  314. /**
  315. * Returns the offset from the start of the resource to the end of the
  316. * segment, inclusive. A value of null indicates that the segment extends to
  317. * the end of the resource.
  318. *
  319. * @return {?number}
  320. * @export
  321. */
  322. getEndByte() {
  323. return this.endByte;
  324. }
  325. /**
  326. * Returns the size of the segment.
  327. * @return {?number}
  328. */
  329. getSize() {
  330. if (this.endByte) {
  331. return this.endByte - this.startByte;
  332. } else {
  333. return null;
  334. }
  335. }
  336. /**
  337. * Returns true if it contains partial SegmentReferences.
  338. * @return {boolean}
  339. */
  340. hasPartialSegments() {
  341. return this.partialReferences.length > 0;
  342. }
  343. /**
  344. * Returns true if it contains all partial SegmentReferences.
  345. * @return {boolean}
  346. */
  347. hasAllPartialSegments() {
  348. return this.allPartialSegments;
  349. }
  350. /**
  351. * Returns the segment's tiles layout. Only defined in image segments.
  352. *
  353. * @return {?string}
  354. * @export
  355. */
  356. getTilesLayout() {
  357. return this.tilesLayout;
  358. }
  359. /**
  360. * Returns the segment's explicit tile duration.
  361. * Only defined in image segments.
  362. *
  363. * @return {?number}
  364. * @export
  365. */
  366. getTileDuration() {
  367. return this.tileDuration;
  368. }
  369. /**
  370. * Returns the segment's status.
  371. *
  372. * @return {shaka.media.SegmentReference.Status}
  373. * @export
  374. */
  375. getStatus() {
  376. return this.status;
  377. }
  378. /**
  379. * Mark the reference as unavailable.
  380. *
  381. * @export
  382. */
  383. markAsUnavailable() {
  384. this.status = shaka.media.SegmentReference.Status.UNAVAILABLE;
  385. }
  386. /**
  387. * Mark the reference as preload.
  388. *
  389. * @export
  390. */
  391. markAsPreload() {
  392. this.preload = true;
  393. }
  394. /**
  395. * Returns true if the segment is preloaded.
  396. *
  397. * @return {boolean}
  398. * @export
  399. */
  400. isPreload() {
  401. return this.preload;
  402. }
  403. /**
  404. * Mark the reference as non-independent.
  405. *
  406. * @export
  407. */
  408. markAsNonIndependent() {
  409. this.independent = false;
  410. }
  411. /**
  412. * Returns true if the segment is independent.
  413. *
  414. * @return {boolean}
  415. * @export
  416. */
  417. isIndependent() {
  418. return this.independent;
  419. }
  420. /**
  421. * Mark the reference as partial.
  422. *
  423. * @export
  424. */
  425. markAsPartial() {
  426. this.partial = true;
  427. }
  428. /**
  429. * Returns true if the segment is partial.
  430. *
  431. * @return {boolean}
  432. * @export
  433. */
  434. isPartial() {
  435. return this.partial;
  436. }
  437. /**
  438. * Mark the reference as being the last part of the full segment
  439. *
  440. * @export
  441. */
  442. markAsLastPartial() {
  443. this.lastPartial = true;
  444. }
  445. /**
  446. * Returns true if reference as being the last part of the full segment.
  447. *
  448. * @return {boolean}
  449. * @export
  450. */
  451. isLastPartial() {
  452. return this.lastPartial;
  453. }
  454. /**
  455. * Mark the reference as byterange optimization.
  456. *
  457. * The "byterange optimization" means that it is playable using MP4 low
  458. * latency streaming with chunked data.
  459. *
  460. * @export
  461. */
  462. markAsByterangeOptimization() {
  463. this.byterangeOptimization = true;
  464. }
  465. /**
  466. * Returns true if the segment has a byterange optimization.
  467. *
  468. * @return {boolean}
  469. * @export
  470. */
  471. hasByterangeOptimization() {
  472. return this.byterangeOptimization;
  473. }
  474. /**
  475. * Set the segment's thumbnail sprite.
  476. *
  477. * @param {shaka.media.SegmentReference.ThumbnailSprite} thumbnailSprite
  478. * @export
  479. */
  480. setThumbnailSprite(thumbnailSprite) {
  481. this.thumbnailSprite = thumbnailSprite;
  482. }
  483. /**
  484. * Returns the segment's thumbnail sprite.
  485. *
  486. * @return {?shaka.media.SegmentReference.ThumbnailSprite}
  487. * @export
  488. */
  489. getThumbnailSprite() {
  490. return this.thumbnailSprite;
  491. }
  492. /**
  493. * Offset the segment reference by a fixed amount.
  494. *
  495. * @param {number} offset The amount to add to the segment's start and end
  496. * times.
  497. * @export
  498. */
  499. offset(offset) {
  500. this.startTime += offset;
  501. this.endTime += offset;
  502. this.trueEndTime += offset;
  503. for (const partial of this.partialReferences) {
  504. partial.startTime += offset;
  505. partial.endTime += offset;
  506. partial.trueEndTime += offset;
  507. }
  508. }
  509. /**
  510. * Sync this segment against a particular sync time that will serve as "0" in
  511. * the presentation timeline.
  512. *
  513. * @param {number} lowestSyncTime
  514. * @export
  515. */
  516. syncAgainst(lowestSyncTime) {
  517. if (this.syncTime == null) {
  518. shaka.log.alwaysError('Sync attempted without sync time!');
  519. return;
  520. }
  521. const desiredStart = this.syncTime - lowestSyncTime;
  522. const offset = desiredStart - this.startTime;
  523. if (Math.abs(offset) >= 0.001) {
  524. this.offset(offset);
  525. }
  526. }
  527. /**
  528. * Set the segment data.
  529. *
  530. * @param {!BufferSource} segmentData
  531. * @export
  532. */
  533. setSegmentData(segmentData) {
  534. this.segmentData = segmentData;
  535. }
  536. /**
  537. * Return the segment data.
  538. *
  539. * @return {?BufferSource}
  540. * @export
  541. */
  542. getSegmentData() {
  543. return this.segmentData;
  544. }
  545. /**
  546. * Updates the init segment reference and propagates the update to all partial
  547. * references.
  548. * @param {shaka.media.InitSegmentReference} initSegmentReference
  549. */
  550. updateInitSegmentReference(initSegmentReference) {
  551. this.initSegmentReference = initSegmentReference;
  552. for (const partialReference of this.partialReferences) {
  553. partialReference.updateInitSegmentReference(initSegmentReference);
  554. }
  555. }
  556. };
  557. /**
  558. * Rather than using booleans to communicate what the state of the reference,
  559. * we have this enum.
  560. *
  561. * @enum {number}
  562. * @export
  563. */
  564. shaka.media.SegmentReference.Status = {
  565. AVAILABLE: 0,
  566. UNAVAILABLE: 1,
  567. MISSING: 2,
  568. };
  569. /**
  570. * A convenient typedef for when either type of reference is acceptable.
  571. *
  572. * @typedef {shaka.media.InitSegmentReference|shaka.media.SegmentReference}
  573. */
  574. shaka.media.AnySegmentReference;
  575. /**
  576. * @typedef {{
  577. * height: number,
  578. * positionX: number,
  579. * positionY: number,
  580. * width: number
  581. * }}
  582. *
  583. * @property {number} height
  584. * The thumbnail height in px.
  585. * @property {number} positionX
  586. * The thumbnail left position in px.
  587. * @property {number} positionY
  588. * The thumbnail top position in px.
  589. * @property {number} width
  590. * The thumbnail width in px.
  591. * @export
  592. */
  593. shaka.media.SegmentReference.ThumbnailSprite;