Source: lib/util/mp4_generator.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.util.Mp4Generator');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.util.ManifestParserUtils');
  9. goog.require('shaka.util.Uint8ArrayUtils');
  10. shaka.util.Mp4Generator = class {
  11. /**
  12. * @param {!Array.<shaka.util.Mp4Generator.StreamInfo>} streamInfos
  13. */
  14. constructor(streamInfos) {
  15. shaka.util.Mp4Generator.initStaticProperties_();
  16. /** @private {!Array.<shaka.util.Mp4Generator.StreamInfo>} */
  17. this.streamInfos_ = streamInfos;
  18. }
  19. /**
  20. * Generate a Init Segment (MP4).
  21. *
  22. * @return {!Uint8Array}
  23. */
  24. initSegment() {
  25. const Mp4Generator = shaka.util.Mp4Generator;
  26. const movie = this.moov_();
  27. const length = Mp4Generator.FTYP_.byteLength + movie.byteLength;
  28. const result = new Uint8Array(length);
  29. result.set(Mp4Generator.FTYP_);
  30. result.set(movie, Mp4Generator.FTYP_.byteLength);
  31. return result;
  32. }
  33. /**
  34. * Generate a MOOV box
  35. *
  36. * @return {!Uint8Array}
  37. * @private
  38. */
  39. moov_() {
  40. goog.asserts.assert(this.streamInfos_.length > 0,
  41. 'StreamInfos must have elements');
  42. const Mp4Generator = shaka.util.Mp4Generator;
  43. const trakArrays = [];
  44. for (const streamInfo of this.streamInfos_) {
  45. trakArrays.push(this.trak_(streamInfo));
  46. }
  47. const traks = shaka.util.Uint8ArrayUtils.concat(...trakArrays);
  48. const firstStreamInfo = this.streamInfos_[0];
  49. return Mp4Generator.box('moov',
  50. this.mvhd_(firstStreamInfo),
  51. traks,
  52. this.mvex_(),
  53. this.pssh_(firstStreamInfo));
  54. }
  55. /**
  56. * Generate a MVHD box
  57. *
  58. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  59. * @return {!Uint8Array}
  60. * @private
  61. */
  62. mvhd_(streamInfo) {
  63. const Mp4Generator = shaka.util.Mp4Generator;
  64. const duration = streamInfo.duration * streamInfo.timescale;
  65. const upperWordDuration =
  66. Math.floor(duration / (Mp4Generator.UINT32_MAX_ + 1));
  67. const lowerWordDuration =
  68. Math.floor(duration % (Mp4Generator.UINT32_MAX_ + 1));
  69. const bytes = new Uint8Array([
  70. 0x01, // version 1
  71. 0x00, 0x00, 0x00, // flags
  72. 0x00, 0x00, 0x00, 0x00,
  73. 0x00, 0x00, 0x00, 0x02, // creation_time
  74. 0x00, 0x00, 0x00, 0x00,
  75. 0x00, 0x00, 0x00, 0x03, // modification_time
  76. ...this.breakNumberIntoBytes_(streamInfo.timescale, 4), // timescale
  77. ...this.breakNumberIntoBytes_(upperWordDuration, 4),
  78. ...this.breakNumberIntoBytes_(lowerWordDuration, 4), // duration
  79. 0x00, 0x01, 0x00, 0x00, // 1.0 rate
  80. 0x01, 0x00, // 1.0 volume
  81. 0x00, 0x00, // reserved
  82. 0x00, 0x00, 0x00, 0x00, // reserved
  83. 0x00, 0x00, 0x00, 0x00, // reserved
  84. 0x00, 0x01, 0x00, 0x00,
  85. 0x00, 0x00, 0x00, 0x00,
  86. 0x00, 0x00, 0x00, 0x00,
  87. 0x00, 0x00, 0x00, 0x00,
  88. 0x00, 0x01, 0x00, 0x00,
  89. 0x00, 0x00, 0x00, 0x00,
  90. 0x00, 0x00, 0x00, 0x00,
  91. 0x00, 0x00, 0x00, 0x00,
  92. 0x40, 0x00, 0x00, 0x00, // transformation: unity matrix
  93. 0x00, 0x00, 0x00, 0x00,
  94. 0x00, 0x00, 0x00, 0x00,
  95. 0x00, 0x00, 0x00, 0x00,
  96. 0x00, 0x00, 0x00, 0x00,
  97. 0x00, 0x00, 0x00, 0x00,
  98. 0x00, 0x00, 0x00, 0x00, // pre_defined
  99. 0xff, 0xff, 0xff, 0xff, // next_track_ID
  100. ]);
  101. return Mp4Generator.box('mvhd', bytes);
  102. }
  103. /**
  104. * Generate a TRAK box
  105. *
  106. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  107. * @return {!Uint8Array}
  108. * @private
  109. */
  110. trak_(streamInfo) {
  111. const Mp4Generator = shaka.util.Mp4Generator;
  112. return Mp4Generator.box('trak',
  113. this.tkhd_(streamInfo), this.mdia_(streamInfo));
  114. }
  115. /**
  116. * Generate a TKHD box
  117. *
  118. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  119. * @return {!Uint8Array}
  120. * @private
  121. */
  122. tkhd_(streamInfo) {
  123. const Mp4Generator = shaka.util.Mp4Generator;
  124. const ContentType = shaka.util.ManifestParserUtils.ContentType;
  125. const id = streamInfo.id + 1;
  126. let width = streamInfo.stream.width || 0;
  127. let height = streamInfo.stream.height || 0;
  128. if (streamInfo.type == ContentType.AUDIO) {
  129. width = 0;
  130. height = 0;
  131. }
  132. const duration = streamInfo.duration * streamInfo.timescale;
  133. const upperWordDuration =
  134. Math.floor(duration / (Mp4Generator.UINT32_MAX_ + 1));
  135. const lowerWordDuration =
  136. Math.floor(duration % (Mp4Generator.UINT32_MAX_ + 1));
  137. const bytes = new Uint8Array([
  138. 0x01, // version 1
  139. 0x00, 0x00, 0x07, // flags
  140. 0x00, 0x00, 0x00, 0x00,
  141. 0x00, 0x00, 0x00, 0x02, // creation_time
  142. 0x00, 0x00, 0x00, 0x00,
  143. 0x00, 0x00, 0x00, 0x03, // modification_time
  144. ...this.breakNumberIntoBytes_(id, 4), // track_ID
  145. 0x00, 0x00, 0x00, 0x00, // reserved
  146. ...this.breakNumberIntoBytes_(upperWordDuration, 4),
  147. ...this.breakNumberIntoBytes_(lowerWordDuration, 4), // duration
  148. 0x00, 0x00, 0x00, 0x00,
  149. 0x00, 0x00, 0x00, 0x00, // reserved
  150. 0x00, 0x00, // layer
  151. 0x00, 0x00, // alternate_group
  152. 0x00, 0x00, // non-audio track volume
  153. 0x00, 0x00, // reserved
  154. 0x00, 0x01, 0x00, 0x00,
  155. 0x00, 0x00, 0x00, 0x00,
  156. 0x00, 0x00, 0x00, 0x00,
  157. 0x00, 0x00, 0x00, 0x00,
  158. 0x00, 0x01, 0x00, 0x00,
  159. 0x00, 0x00, 0x00, 0x00,
  160. 0x00, 0x00, 0x00, 0x00,
  161. 0x00, 0x00, 0x00, 0x00,
  162. 0x40, 0x00, 0x00, 0x00, // transformation: unity matrix
  163. ...this.breakNumberIntoBytes_(width, 2),
  164. 0x00, 0x00, // width
  165. ...this.breakNumberIntoBytes_(height, 2),
  166. 0x00, 0x00, // height
  167. ]);
  168. return Mp4Generator.box('tkhd', bytes);
  169. }
  170. /**
  171. * Generate a MDIA box
  172. *
  173. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  174. * @return {!Uint8Array}
  175. * @private
  176. */
  177. mdia_(streamInfo) {
  178. const Mp4Generator = shaka.util.Mp4Generator;
  179. return Mp4Generator.box('mdia', this.mdhd_(streamInfo),
  180. this.hdlr_(streamInfo), this.minf_(streamInfo));
  181. }
  182. /**
  183. * Generate a MDHD box
  184. *
  185. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  186. * @return {!Uint8Array}
  187. * @private
  188. */
  189. mdhd_(streamInfo) {
  190. const Mp4Generator = shaka.util.Mp4Generator;
  191. const duration = streamInfo.duration * streamInfo.timescale;
  192. const upperWordDuration =
  193. Math.floor(duration / (Mp4Generator.UINT32_MAX_ + 1));
  194. const lowerWordDuration =
  195. Math.floor(duration % (Mp4Generator.UINT32_MAX_ + 1));
  196. const language = streamInfo.stream.language;
  197. const languageNumber = ((language.charCodeAt(0) - 0x60) << 10) |
  198. ((language.charCodeAt(1) - 0x60) << 5) |
  199. ((language.charCodeAt(2) - 0x60));
  200. const bytes = new Uint8Array([
  201. 0x01, // version 1
  202. 0x00, 0x00, 0x00, // flags
  203. 0x00, 0x00, 0x00, 0x00,
  204. 0x00, 0x00, 0x00, 0x02, // creation_time
  205. 0x00, 0x00, 0x00, 0x00,
  206. 0x00, 0x00, 0x00, 0x03, // modification_time
  207. ...this.breakNumberIntoBytes_(streamInfo.timescale, 4), // timescale
  208. ...this.breakNumberIntoBytes_(upperWordDuration, 4),
  209. ...this.breakNumberIntoBytes_(lowerWordDuration, 4), // duration
  210. ...this.breakNumberIntoBytes_(languageNumber, 2), // language
  211. 0x00, 0x00,
  212. ]);
  213. return Mp4Generator.box('mdhd', bytes);
  214. }
  215. /**
  216. * Generate a HDLR box
  217. *
  218. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  219. * @return {!Uint8Array}
  220. * @private
  221. */
  222. hdlr_(streamInfo) {
  223. const Mp4Generator = shaka.util.Mp4Generator;
  224. const ContentType = shaka.util.ManifestParserUtils.ContentType;
  225. let bytes = new Uint8Array([]);
  226. switch (streamInfo.type) {
  227. case ContentType.VIDEO:
  228. bytes = Mp4Generator.HDLR_TYPES_.video;
  229. break;
  230. case ContentType.AUDIO:
  231. bytes = Mp4Generator.HDLR_TYPES_.audio;
  232. break;
  233. }
  234. return Mp4Generator.box('hdlr', bytes);
  235. }
  236. /**
  237. * Generate a MINF box
  238. *
  239. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  240. * @return {!Uint8Array}
  241. * @private
  242. */
  243. minf_(streamInfo) {
  244. const Mp4Generator = shaka.util.Mp4Generator;
  245. const ContentType = shaka.util.ManifestParserUtils.ContentType;
  246. switch (streamInfo.type) {
  247. case ContentType.VIDEO:
  248. return Mp4Generator.box(
  249. 'minf', Mp4Generator.box('vmhd', Mp4Generator.VMHD_),
  250. Mp4Generator.DINF_, this.stbl_(streamInfo));
  251. case ContentType.AUDIO:
  252. return Mp4Generator.box(
  253. 'minf', Mp4Generator.box('smhd', Mp4Generator.SMHD_),
  254. Mp4Generator.DINF_, this.stbl_(streamInfo));
  255. }
  256. return new Uint8Array([]);
  257. }
  258. /**
  259. * Generate a STBL box
  260. *
  261. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  262. * @return {!Uint8Array}
  263. * @private
  264. */
  265. stbl_(streamInfo) {
  266. const Mp4Generator = shaka.util.Mp4Generator;
  267. return Mp4Generator.box(
  268. 'stbl',
  269. this.stsd_(streamInfo),
  270. Mp4Generator.box('stts', Mp4Generator.STTS_),
  271. Mp4Generator.box('stsc', Mp4Generator.STSC_),
  272. Mp4Generator.box('stsz', Mp4Generator.STSZ_),
  273. Mp4Generator.box('stco', Mp4Generator.STCO_));
  274. }
  275. /**
  276. * Generate a STSD box
  277. *
  278. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  279. * @return {!Uint8Array}
  280. * @private
  281. */
  282. stsd_(streamInfo) {
  283. const Mp4Generator = shaka.util.Mp4Generator;
  284. const ContentType = shaka.util.ManifestParserUtils.ContentType;
  285. let bytes = new Uint8Array([]);
  286. switch (streamInfo.type) {
  287. case ContentType.VIDEO:
  288. if (streamInfo.codecs.includes('avc1')) {
  289. bytes = this.avc1_(streamInfo);
  290. } else if (streamInfo.codecs.includes('hvc1')) {
  291. bytes = this.hvc1_(streamInfo);
  292. }
  293. break;
  294. case ContentType.AUDIO:
  295. if (streamInfo.codecs.includes('mp3')) {
  296. bytes = this.mp3_(streamInfo);
  297. } else if (streamInfo.codecs.includes('ac-3')) {
  298. bytes = this.ac3_(streamInfo);
  299. } else if (streamInfo.codecs.includes('ec-3')) {
  300. bytes = this.ec3_(streamInfo);
  301. } else if (streamInfo.codecs.includes('opus')) {
  302. bytes = this.opus_(streamInfo);
  303. } else {
  304. bytes = this.mp4a_(streamInfo);
  305. }
  306. break;
  307. }
  308. return Mp4Generator.box('stsd', Mp4Generator.STSD_, bytes);
  309. }
  310. /**
  311. * Generate a AVC1 box
  312. *
  313. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  314. * @return {!Uint8Array}
  315. * @private
  316. */
  317. avc1_(streamInfo) {
  318. const Mp4Generator = shaka.util.Mp4Generator;
  319. const width = streamInfo.stream.width || 0;
  320. const height = streamInfo.stream.height || 0;
  321. let avcCBox;
  322. if (streamInfo.videoConfig.byteLength > 0) {
  323. avcCBox = Mp4Generator.box('avcC', streamInfo.videoConfig);
  324. } else {
  325. avcCBox = Mp4Generator.box('avcC', this.avcC_(streamInfo));
  326. }
  327. const avc1Bytes = new Uint8Array([
  328. 0x00, 0x00, 0x00,
  329. 0x00, 0x00, 0x00, // reserved
  330. 0x00, 0x01, // data_reference_index
  331. 0x00, 0x00, // pre_defined
  332. 0x00, 0x00, // reserved
  333. 0x00, 0x00, 0x00, 0x00,
  334. 0x00, 0x00, 0x00, 0x00,
  335. 0x00, 0x00, 0x00, 0x00, // pre_defined
  336. ...this.breakNumberIntoBytes_(width, 2), // width
  337. ...this.breakNumberIntoBytes_(height, 2), // height
  338. 0x00, 0x48, 0x00, 0x00, // horizresolution
  339. 0x00, 0x48, 0x00, 0x00, // vertresolution
  340. 0x00, 0x00, 0x00, 0x00, // reserved
  341. 0x00, 0x01, // frame_count
  342. 0x13,
  343. 0x76, 0x69, 0x64, 0x65,
  344. 0x6f, 0x6a, 0x73, 0x2d,
  345. 0x63, 0x6f, 0x6e, 0x74,
  346. 0x72, 0x69, 0x62, 0x2d,
  347. 0x68, 0x6c, 0x73, 0x00,
  348. 0x00, 0x00, 0x00, 0x00,
  349. 0x00, 0x00, 0x00, 0x00,
  350. 0x00, 0x00, 0x00, // compressorname
  351. 0x00, 0x18, // depth = 24
  352. 0x11, 0x11, // pre_defined = -1
  353. ]);
  354. let boxName = 'avc1';
  355. const paspBox = this.pasp_(streamInfo);
  356. let sinfBox = new Uint8Array([]);
  357. if (streamInfo.encrypted) {
  358. sinfBox = this.sinf_(streamInfo);
  359. boxName = 'encv';
  360. }
  361. return Mp4Generator.box(boxName, avc1Bytes, avcCBox, paspBox, sinfBox);
  362. }
  363. /**
  364. * Generate a AVCC box
  365. *
  366. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  367. * @return {!Uint8Array}
  368. * @private
  369. */
  370. avcC_(streamInfo) {
  371. const NALUTYPE_SPS = 7;
  372. const NALUTYPE_PPS = 8;
  373. // length = 7 by default (0 SPS and 0 PPS)
  374. let avcCLength = 7;
  375. // First get all SPS and PPS from nalus
  376. const sps = [];
  377. const pps = [];
  378. let AVCProfileIndication = 0;
  379. let AVCLevelIndication = 0;
  380. let profileCompatibility = 0;
  381. for (let i = 0; i < streamInfo.videoNalus.length; i++) {
  382. const naluBytes = this.hexStringToBuffer_(streamInfo.videoNalus[i]);
  383. const naluType = naluBytes[0] & 0x1F;
  384. switch (naluType) {
  385. case NALUTYPE_SPS:
  386. sps.push(naluBytes);
  387. // 2 = sequenceParameterSetLength field length
  388. avcCLength += naluBytes.length + 2;
  389. break;
  390. case NALUTYPE_PPS:
  391. pps.push(naluBytes);
  392. // 2 = pictureParameterSetLength field length
  393. avcCLength += naluBytes.length + 2;
  394. break;
  395. default:
  396. break;
  397. }
  398. }
  399. // Get profile and level from SPS
  400. if (sps.length > 0) {
  401. AVCProfileIndication = sps[0][1];
  402. profileCompatibility = sps[0][2];
  403. AVCLevelIndication = sps[0][3];
  404. }
  405. // Generate avcC buffer
  406. const avcCBytes = new Uint8Array(avcCLength);
  407. let i = 0;
  408. // configurationVersion = 1
  409. avcCBytes[i++] = 1;
  410. avcCBytes[i++] = AVCProfileIndication;
  411. avcCBytes[i++] = profileCompatibility;
  412. avcCBytes[i++] = AVCLevelIndication;
  413. // '11111' + lengthSizeMinusOne = 3
  414. avcCBytes[i++] = 0xFF;
  415. // '111' + numOfSequenceParameterSets
  416. avcCBytes[i++] = 0xE0 | sps.length;
  417. for (let n = 0; n < sps.length; n++) {
  418. avcCBytes[i++] = (sps[n].length & 0xFF00) >> 8;
  419. avcCBytes[i++] = (sps[n].length & 0x00FF);
  420. avcCBytes.set(sps[n], i);
  421. i += sps[n].length;
  422. }
  423. // numOfPictureParameterSets
  424. avcCBytes[i++] = pps.length;
  425. for (let n = 0; n < pps.length; n++) {
  426. avcCBytes[i++] = (pps[n].length & 0xFF00) >> 8;
  427. avcCBytes[i++] = (pps[n].length & 0x00FF);
  428. avcCBytes.set(pps[n], i);
  429. i += pps[n].length;
  430. }
  431. return avcCBytes;
  432. }
  433. /**
  434. * Generate a HVC1 box
  435. *
  436. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  437. * @return {!Uint8Array}
  438. * @private
  439. */
  440. hvc1_(streamInfo) {
  441. const Mp4Generator = shaka.util.Mp4Generator;
  442. const width = streamInfo.stream.width || 0;
  443. const height = streamInfo.stream.height || 0;
  444. let hvcCBox = new Uint8Array([]);
  445. if (streamInfo.videoConfig.byteLength > 0) {
  446. hvcCBox = Mp4Generator.box('hvcC', streamInfo.videoConfig);
  447. }
  448. const hvc1Bytes = new Uint8Array([
  449. 0x00, 0x00, 0x00,
  450. 0x00, 0x00, 0x00, // reserved
  451. 0x00, 0x01, // data_reference_index
  452. 0x00, 0x00, // pre_defined
  453. 0x00, 0x00, // reserved
  454. 0x00, 0x00, 0x00, 0x00,
  455. 0x00, 0x00, 0x00, 0x00,
  456. 0x00, 0x00, 0x00, 0x00, // pre_defined
  457. ...this.breakNumberIntoBytes_(width, 2), // width
  458. ...this.breakNumberIntoBytes_(height, 2), // height
  459. 0x00, 0x48, 0x00, 0x00, // horizresolution
  460. 0x00, 0x48, 0x00, 0x00, // vertresolution
  461. 0x00, 0x00, 0x00, 0x00, // reserved
  462. 0x00, 0x01, // frame_count
  463. 0x13,
  464. 0x76, 0x69, 0x64, 0x65,
  465. 0x6f, 0x6a, 0x73, 0x2d,
  466. 0x63, 0x6f, 0x6e, 0x74,
  467. 0x72, 0x69, 0x62, 0x2d,
  468. 0x68, 0x6c, 0x73, 0x00,
  469. 0x00, 0x00, 0x00, 0x00,
  470. 0x00, 0x00, 0x00, 0x00,
  471. 0x00, 0x00, 0x00, // compressorname
  472. 0x00, 0x18, // depth = 24
  473. 0x11, 0x11, // pre_defined = -1
  474. ]);
  475. let boxName = 'hvc1';
  476. const paspBox = this.pasp_(streamInfo);
  477. let sinfBox = new Uint8Array([]);
  478. if (streamInfo.encrypted) {
  479. sinfBox = this.sinf_(streamInfo);
  480. boxName = 'encv';
  481. }
  482. return Mp4Generator.box(boxName, hvc1Bytes, hvcCBox, paspBox, sinfBox);
  483. }
  484. /**
  485. * Generate a PASP box
  486. *
  487. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  488. * @return {!Uint8Array}
  489. * @private
  490. */
  491. pasp_(streamInfo) {
  492. if (!streamInfo.hSpacing && !streamInfo.vSpacing) {
  493. return new Uint8Array([]);
  494. }
  495. const Mp4Generator = shaka.util.Mp4Generator;
  496. const hSpacing = streamInfo.hSpacing;
  497. const vSpacing = streamInfo.vSpacing;
  498. const bytes = new Uint8Array([
  499. ...this.breakNumberIntoBytes_(hSpacing, 4), // hSpacing
  500. ...this.breakNumberIntoBytes_(vSpacing, 4), // vSpacing
  501. ]);
  502. return Mp4Generator.box('pasp', bytes);
  503. }
  504. /**
  505. * Generate STSD bytes
  506. *
  507. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  508. * @return {!Uint8Array}
  509. * @private
  510. */
  511. audioStsd_(streamInfo) {
  512. const channelsCount = streamInfo.stream.channelsCount || 2;
  513. const audioSamplingRate = streamInfo.stream.audioSamplingRate || 44100;
  514. const bytes = new Uint8Array([
  515. 0x00, 0x00, 0x00, // reserved
  516. 0x00, 0x00, 0x00, // reserved
  517. 0x00, 0x01, // data_reference_index
  518. 0x00, 0x00, 0x00, 0x00,
  519. 0x00, 0x00, 0x00, 0x00, // reserved
  520. 0x00,
  521. channelsCount, // channelcount
  522. 0x00, 0x10, // sampleSize:16bits
  523. 0x00, 0x00, 0x00, 0x00, // reserved2
  524. ...this.breakNumberIntoBytes_(audioSamplingRate, 2), // Sample Rate
  525. 0x00, 0x00,
  526. ]);
  527. return bytes;
  528. }
  529. /**
  530. * Generate a .MP3 box
  531. *
  532. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  533. * @return {!Uint8Array}
  534. * @private
  535. */
  536. mp3_(streamInfo) {
  537. const Mp4Generator = shaka.util.Mp4Generator;
  538. return Mp4Generator.box('.mp3', this.audioStsd_(streamInfo));
  539. }
  540. /**
  541. * Generate a AC-3 box
  542. *
  543. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  544. * @return {!Uint8Array}
  545. * @private
  546. */
  547. ac3_(streamInfo) {
  548. const Mp4Generator = shaka.util.Mp4Generator;
  549. const dac3Box = Mp4Generator.box('dac3', streamInfo.audioConfig);
  550. let boxName = 'ac-3';
  551. let sinfBox = new Uint8Array([]);
  552. if (streamInfo.encrypted) {
  553. sinfBox = this.sinf_(streamInfo);
  554. boxName = 'enca';
  555. }
  556. return Mp4Generator.box(boxName,
  557. this.audioStsd_(streamInfo), dac3Box, sinfBox);
  558. }
  559. /**
  560. * Generate a EC-3 box
  561. *
  562. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  563. * @return {!Uint8Array}
  564. * @private
  565. */
  566. ec3_(streamInfo) {
  567. const Mp4Generator = shaka.util.Mp4Generator;
  568. const dec3Box = Mp4Generator.box('dec3', streamInfo.audioConfig);
  569. let boxName = 'ec-3';
  570. let sinfBox = new Uint8Array([]);
  571. if (streamInfo.encrypted) {
  572. sinfBox = this.sinf_(streamInfo);
  573. boxName = 'enca';
  574. }
  575. return Mp4Generator.box(boxName,
  576. this.audioStsd_(streamInfo), dec3Box, sinfBox);
  577. }
  578. /**
  579. * Generate a Opus box
  580. *
  581. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  582. * @return {!Uint8Array}
  583. * @private
  584. */
  585. opus_(streamInfo) {
  586. const Mp4Generator = shaka.util.Mp4Generator;
  587. const dopsBox = Mp4Generator.box('dOps', streamInfo.audioConfig);
  588. let boxName = 'Opus';
  589. let sinfBox = new Uint8Array([]);
  590. if (streamInfo.encrypted) {
  591. sinfBox = this.sinf_(streamInfo);
  592. boxName = 'enca';
  593. }
  594. return Mp4Generator.box(boxName,
  595. this.audioStsd_(streamInfo), dopsBox, sinfBox);
  596. }
  597. /**
  598. * Generate a MP4A box
  599. *
  600. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  601. * @return {!Uint8Array}
  602. * @private
  603. */
  604. mp4a_(streamInfo) {
  605. const Mp4Generator = shaka.util.Mp4Generator;
  606. let esdsBox;
  607. if (streamInfo.audioConfig.byteLength > 0) {
  608. esdsBox = Mp4Generator.box('esds', streamInfo.audioConfig);
  609. } else {
  610. esdsBox = Mp4Generator.box('esds', this.esds_(streamInfo));
  611. }
  612. let boxName = 'mp4a';
  613. let sinfBox = new Uint8Array([]);
  614. if (streamInfo.encrypted) {
  615. sinfBox = this.sinf_(streamInfo);
  616. boxName = 'enca';
  617. }
  618. return Mp4Generator.box(boxName,
  619. this.audioStsd_(streamInfo), esdsBox, sinfBox);
  620. }
  621. /**
  622. * Generate a ESDS box
  623. *
  624. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  625. * @return {!Uint8Array}
  626. * @private
  627. */
  628. esds_(streamInfo) {
  629. const ContentType = shaka.util.ManifestParserUtils.ContentType;
  630. const id = streamInfo.id + 1;
  631. const channelsCount = streamInfo.stream.channelsCount || 2;
  632. const audioSamplingRate = streamInfo.stream.audioSamplingRate || 44100;
  633. const audioCodec = shaka.util.ManifestParserUtils.guessCodecs(
  634. ContentType.AUDIO, streamInfo.codecs.split(','));
  635. const samplingFrequencyIndex = {
  636. 96000: 0x0,
  637. 88200: 0x1,
  638. 64000: 0x2,
  639. 48000: 0x3,
  640. 44100: 0x4,
  641. 32000: 0x5,
  642. 24000: 0x6,
  643. 22050: 0x7,
  644. 16000: 0x8,
  645. 12000: 0x9,
  646. 11025: 0xA,
  647. 8000: 0xB,
  648. 7350: 0xC,
  649. };
  650. let indexFreq = samplingFrequencyIndex[audioSamplingRate];
  651. // In HE AAC Sampling frequence equals to SamplingRate * 2
  652. if (audioCodec === 'mp4a.40.5' || audioCodec === 'mp4a.40.29') {
  653. indexFreq = samplingFrequencyIndex[audioSamplingRate * 2];
  654. }
  655. const audioObjectType = parseInt(audioCodec.split('.').pop(), 10);
  656. return new Uint8Array([
  657. 0x00, // version
  658. 0x00, 0x00, 0x00, // flags
  659. // ES_Descriptor
  660. 0x03, // tag, ES_DescrTag
  661. 0x19, // length
  662. ...this.breakNumberIntoBytes_(id, 2), // ES_ID
  663. 0x00, // streamDependenceFlag, URL_flag, reserved, streamPriority
  664. // DecoderConfigDescriptor
  665. 0x04, // tag, DecoderConfigDescrTag
  666. 0x11, // length
  667. 0x40, // object type
  668. 0x15, // streamType
  669. 0x00, 0x00, 0x00, // bufferSizeDB
  670. 0x00, 0x00, 0x00, 0x00, // maxBitrate
  671. 0x00, 0x00, 0x00, 0x00, // avgBitrate
  672. // DecoderSpecificInfo
  673. 0x05, // tag, DecoderSpecificInfoTag
  674. 0x02, // length
  675. // ISO/IEC 14496-3, AudioSpecificConfig
  676. // for samplingFrequencyIndex see
  677. // ISO/IEC 13818-7:2006, 8.1.3.2.2, Table 35
  678. (audioObjectType << 3) | (indexFreq >>> 1),
  679. (indexFreq << 7) | (channelsCount << 3),
  680. 0x06, 0x01, 0x02, // GASpecificConfig
  681. ]);
  682. }
  683. /**
  684. * Generate a MVEX box
  685. *
  686. * @return {!Uint8Array}
  687. * @private
  688. */
  689. mvex_() {
  690. const Mp4Generator = shaka.util.Mp4Generator;
  691. const trexArrays = [];
  692. for (const streamInfo of this.streamInfos_) {
  693. trexArrays.push(this.trex_(streamInfo));
  694. }
  695. const trexs = shaka.util.Uint8ArrayUtils.concat(...trexArrays);
  696. return Mp4Generator.box('mvex', trexs);
  697. }
  698. /**
  699. * Generate a TREX box
  700. *
  701. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  702. * @return {!Uint8Array}
  703. * @private
  704. */
  705. trex_(streamInfo) {
  706. const Mp4Generator = shaka.util.Mp4Generator;
  707. const id = streamInfo.id + 1;
  708. const bytes = new Uint8Array([
  709. 0x00, // version 0
  710. 0x00, 0x00, 0x00, // flags
  711. ...this.breakNumberIntoBytes_(id, 4), // track_ID
  712. 0x00, 0x00, 0x00, 0x01, // default_sample_description_index
  713. 0x00, 0x00, 0x00, 0x00, // default_sample_duration
  714. 0x00, 0x00, 0x00, 0x00, // default_sample_size
  715. 0x00, 0x01, 0x00, 0x01, // default_sample_flags
  716. ]);
  717. return Mp4Generator.box('trex', bytes);
  718. }
  719. /**
  720. * Generate a PSSH box
  721. *
  722. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  723. * @return {!Uint8Array}
  724. * @private
  725. */
  726. pssh_(streamInfo) {
  727. const initDatas = [];
  728. if (!streamInfo.encrypted) {
  729. return new Uint8Array([]);
  730. }
  731. for (const drmInfo of streamInfo.stream.drmInfos) {
  732. if (!drmInfo.initData) {
  733. continue;
  734. }
  735. for (const initData of drmInfo.initData) {
  736. initDatas.push(initData.initData);
  737. }
  738. }
  739. const boxes = shaka.util.Uint8ArrayUtils.concat(...initDatas);
  740. return boxes;
  741. }
  742. /**
  743. * Generate a SINF box
  744. *
  745. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  746. * @return {!Uint8Array}
  747. * @private
  748. */
  749. sinf_(streamInfo) {
  750. const Mp4Generator = shaka.util.Mp4Generator;
  751. return Mp4Generator.box('sinf',
  752. this.frma_(streamInfo), this.schm_(), this.schi_(streamInfo));
  753. }
  754. /**
  755. * Generate a FRMA box
  756. *
  757. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  758. * @return {!Uint8Array}
  759. * @private
  760. */
  761. frma_(streamInfo) {
  762. const codec = streamInfo.codecs.substring(
  763. 0, streamInfo.codecs.indexOf('.'));
  764. const Mp4Generator = shaka.util.Mp4Generator;
  765. const codecNumber = this.stringToCharCode_(codec);
  766. const bytes = new Uint8Array([
  767. ...this.breakNumberIntoBytes_(codecNumber, 4),
  768. ]);
  769. return Mp4Generator.box('frma', bytes);
  770. }
  771. /**
  772. * Generate a SCHM box
  773. *
  774. * @return {!Uint8Array}
  775. * @private
  776. */
  777. schm_() {
  778. const Mp4Generator = shaka.util.Mp4Generator;
  779. const bytes = new Uint8Array([
  780. 0x00, // version 0
  781. 0x00, 0x00, 0x00, // flags
  782. 0x63, 0x65, 0x6e, 0x63, // Scheme: cenc
  783. 0x00, 0x01, 0x00, 0x00, // Scheme version: 1.0
  784. ]);
  785. return Mp4Generator.box('schm', bytes);
  786. }
  787. /**
  788. * Generate a SCHI box
  789. *
  790. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  791. * @return {!Uint8Array}
  792. * @private
  793. */
  794. schi_(streamInfo) {
  795. const Mp4Generator = shaka.util.Mp4Generator;
  796. return Mp4Generator.box('schi', this.tenc_(streamInfo));
  797. }
  798. /**
  799. * Generate a TENC box
  800. *
  801. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  802. * @return {!Uint8Array}
  803. * @private
  804. */
  805. tenc_(streamInfo) {
  806. // Default key ID: all zeros (dummy)
  807. let defaultKeyId = new Uint8Array([
  808. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  809. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  810. ]);
  811. for (const drmInfo of streamInfo.stream.drmInfos) {
  812. if (drmInfo && drmInfo.keyIds && drmInfo.keyIds.size) {
  813. for (const keyId of drmInfo.keyIds) {
  814. defaultKeyId = this.hexStringToBuffer_(keyId);
  815. }
  816. }
  817. }
  818. const Mp4Generator = shaka.util.Mp4Generator;
  819. const bytes = new Uint8Array([
  820. 0x00, // version 0
  821. 0x00, 0x00, 0x00, // flags
  822. 0x00, 0x00, // Reserved fields
  823. 0x01, // Default protected: true
  824. 0x08, // Default per-sample IV size: 8
  825. ]);
  826. goog.asserts.assert(defaultKeyId, 'Default KID should be non-null');
  827. return Mp4Generator.box('tenc', bytes, defaultKeyId);
  828. }
  829. /**
  830. * Generate a Segment Data (MP4).
  831. *
  832. * @return {!Uint8Array}
  833. */
  834. segmentData() {
  835. const segmentDataArray = [];
  836. for (const streamInfo of this.streamInfos_) {
  837. segmentDataArray.push(
  838. ...[this.moof_(streamInfo), this.mdat_(streamInfo)]);
  839. }
  840. const result = shaka.util.Uint8ArrayUtils.concat(...segmentDataArray);
  841. return result;
  842. }
  843. /**
  844. * Generate a MOOF box
  845. *
  846. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  847. * @return {!Uint8Array}
  848. * @private
  849. */
  850. moof_(streamInfo) {
  851. const Mp4Generator = shaka.util.Mp4Generator;
  852. return Mp4Generator.box('moof',
  853. this.mfhd_(streamInfo), this.traf_(streamInfo));
  854. }
  855. /**
  856. * Generate a MOOF box
  857. *
  858. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  859. * @return {!Uint8Array}
  860. * @private
  861. */
  862. mfhd_(streamInfo) {
  863. const Mp4Generator = shaka.util.Mp4Generator;
  864. const sequenceNumber =
  865. streamInfo.data ? streamInfo.data.sequenceNumber : 0;
  866. const bytes = new Uint8Array([
  867. 0x00, // version 0
  868. 0x00, 0x00, 0x00, // flags
  869. ...this.breakNumberIntoBytes_(sequenceNumber, 4),
  870. ]);
  871. return Mp4Generator.box('mfhd', bytes);
  872. }
  873. /**
  874. * Generate a TRAF box
  875. *
  876. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  877. * @return {!Uint8Array}
  878. * @private
  879. */
  880. traf_(streamInfo) {
  881. const Mp4Generator = shaka.util.Mp4Generator;
  882. const sampleDependencyTable = this.sdtp_(streamInfo);
  883. const offset = sampleDependencyTable.length +
  884. 32 + // tfhd
  885. 20 + // tfdt
  886. 8 + // traf header
  887. 16 + // mfhd
  888. 8 + // moof header
  889. 8; // mdat header;
  890. return Mp4Generator.box('traf',
  891. this.tfhd_(streamInfo),
  892. this.tfdt_(streamInfo),
  893. this.trun_(streamInfo, offset),
  894. sampleDependencyTable);
  895. }
  896. /**
  897. * Generate a SDTP box
  898. *
  899. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  900. * @return {!Uint8Array}
  901. * @private
  902. */
  903. sdtp_(streamInfo) {
  904. const Mp4Generator = shaka.util.Mp4Generator;
  905. const samples = streamInfo.data ? streamInfo.data.samples : [];
  906. const bytes = new Uint8Array(4 + samples.length);
  907. // leave the full box header (4 bytes) all zero
  908. // write the sample table
  909. for (let i = 0; i < samples.length; i++) {
  910. const flags = samples[i].flags;
  911. bytes[i + 4] = (flags.dependsOn << 4) |
  912. (flags.isDependedOn << 2) |
  913. flags.hasRedundancy;
  914. }
  915. return Mp4Generator.box('sdtp', bytes);
  916. }
  917. /**
  918. * Generate a TFHD box
  919. *
  920. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  921. * @return {!Uint8Array}
  922. * @private
  923. */
  924. tfhd_(streamInfo) {
  925. const Mp4Generator = shaka.util.Mp4Generator;
  926. const id = streamInfo.id + 1;
  927. const bytes = new Uint8Array([
  928. 0x00, // version 0
  929. 0x00, 0x00, 0x3a, // flags
  930. ...this.breakNumberIntoBytes_(id, 4), // track_ID
  931. 0x00, 0x00, 0x00, 0x01, // sample_description_index
  932. 0x00, 0x00, 0x00, 0x00, // default_sample_duration
  933. 0x00, 0x00, 0x00, 0x00, // default_sample_size
  934. 0x00, 0x00, 0x00, 0x00, // default_sample_flags
  935. ]);
  936. return Mp4Generator.box('tfhd', bytes);
  937. }
  938. /**
  939. * Generate a TFDT box
  940. *
  941. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  942. * @return {!Uint8Array}
  943. * @private
  944. */
  945. tfdt_(streamInfo) {
  946. const Mp4Generator = shaka.util.Mp4Generator;
  947. const baseMediaDecodeTime =
  948. streamInfo.data ? streamInfo.data.baseMediaDecodeTime : 0;
  949. const upperWordBaseMediaDecodeTime =
  950. Math.floor(baseMediaDecodeTime / (Mp4Generator.UINT32_MAX_ + 1));
  951. const lowerWordBaseMediaDecodeTime =
  952. Math.floor(baseMediaDecodeTime % (Mp4Generator.UINT32_MAX_ + 1));
  953. const bytes = new Uint8Array([
  954. 0x01, // version 1
  955. 0x00, 0x00, 0x00, // flags
  956. ...this.breakNumberIntoBytes_(upperWordBaseMediaDecodeTime, 4),
  957. ...this.breakNumberIntoBytes_(lowerWordBaseMediaDecodeTime, 4),
  958. ]);
  959. return Mp4Generator.box('tfdt', bytes);
  960. }
  961. /**
  962. * Generate a TRUN box
  963. *
  964. * @param {shaka.util.Mp4Generator.StreamInfo} streamInfo
  965. * @param {number} offset
  966. * @return {!Uint8Array}
  967. * @private
  968. */
  969. trun_(streamInfo, offset) {
  970. const ContentType = shaka.util.ManifestParserUtils.ContentType;
  971. const Mp4Generator = shaka.util.Mp4Generator;
  972. const samples = streamInfo.data ? streamInfo.data.samples : [];
  973. const samplesLength = samples.length;
  974. const byteslen = 12 + 16 * samplesLength;
  975. const bytes = new Uint8Array(byteslen);
  976. offset += 8 + byteslen;
  977. const isVideo = streamInfo.type === ContentType.VIDEO;
  978. bytes.set(
  979. [
  980. // version 1 for video with signed-int sample_composition_time_offset
  981. isVideo ? 0x01 : 0x00,
  982. 0x00, 0x0f, 0x01, // flags
  983. ...this.breakNumberIntoBytes_(samplesLength, 4), // sample_count
  984. ...this.breakNumberIntoBytes_(offset, 4), // data_offset
  985. ],
  986. 0,
  987. );
  988. for (let i = 0; i < samplesLength; i++) {
  989. const sample = samples[i];
  990. const duration = this.breakNumberIntoBytes_(sample.duration, 4);
  991. const size = this.breakNumberIntoBytes_(sample.size, 4);
  992. const flags = sample.flags;
  993. const cts = this.breakNumberIntoBytes_(sample.cts, 4);
  994. bytes.set(
  995. [
  996. ...duration, // sample_duration
  997. ...size, // sample_size
  998. (flags.isLeading << 2) | flags.dependsOn,
  999. (flags.isDependedOn << 6) | (flags.hasRedundancy << 4) |
  1000. flags.isNonSync,
  1001. flags.degradPrio & (0xf0 << 8),
  1002. flags.degradPrio & 0x0f, // sample_flags
  1003. ...cts, // sample_composition_time_offset
  1004. ],
  1005. 12 + 16 * i,
  1006. );
  1007. }
  1008. return Mp4Generator.box('trun', bytes);
  1009. }
  1010. /**
  1011. * Generate a MDAT box
  1012. *
  1013. * @return {!Uint8Array}
  1014. * @private
  1015. */
  1016. mdat_(streamInfo) {
  1017. const Mp4Generator = shaka.util.Mp4Generator;
  1018. const samples = streamInfo.data ? streamInfo.data.samples : [];
  1019. const allData = samples.map((sample) => sample.data);
  1020. const bytes = shaka.util.Uint8ArrayUtils.concat(...allData);
  1021. return Mp4Generator.box('mdat', bytes);
  1022. }
  1023. /**
  1024. * @param {number} number
  1025. * @param {number} numBytes
  1026. * @return {!Array.<number>}
  1027. * @private
  1028. */
  1029. breakNumberIntoBytes_(number, numBytes) {
  1030. const bytes = [];
  1031. for (let byte = numBytes - 1; byte >= 0; byte--) {
  1032. bytes.push((number >> (8 * byte)) & 0xff);
  1033. }
  1034. return bytes;
  1035. }
  1036. /**
  1037. * Convert a hex string to buffer.
  1038. *
  1039. * @param {string} str
  1040. * @return {Uint8Array}
  1041. * @private
  1042. */
  1043. hexStringToBuffer_(str) {
  1044. const buf = new Uint8Array(str.length / 2);
  1045. for (let i = 0; i < str.length / 2; i += 1) {
  1046. buf[i] = parseInt(String(str[i * 2] + str[i * 2 + 1]), 16);
  1047. }
  1048. return buf;
  1049. }
  1050. /**
  1051. * Convert a string to char code.
  1052. *
  1053. * @param {string} str
  1054. * @return {number}
  1055. * @private
  1056. */
  1057. stringToCharCode_(str) {
  1058. let code = 0;
  1059. for (let i = 0; i < str.length; i += 1) {
  1060. code |= str.charCodeAt(i) << ((str.length - i - 1) * 8);
  1061. }
  1062. return code;
  1063. }
  1064. /**
  1065. * @private
  1066. */
  1067. static initStaticProperties_() {
  1068. const Mp4Generator = shaka.util.Mp4Generator;
  1069. if (Mp4Generator.initializated_) {
  1070. return;
  1071. }
  1072. Mp4Generator.initializated_ = true;
  1073. const majorBrand = new Uint8Array([105, 115, 111, 109]); // isom
  1074. const avc1Brand = new Uint8Array([97, 118, 99, 49]); // avc1
  1075. const minorVersion = new Uint8Array([0, 0, 0, 1]);
  1076. Mp4Generator.FTYP_ = Mp4Generator.box(
  1077. 'ftyp', majorBrand, minorVersion, majorBrand, avc1Brand);
  1078. const drefBox = Mp4Generator.box('dref', Mp4Generator.DREF_);
  1079. Mp4Generator.DINF_ = Mp4Generator.box('dinf', drefBox);
  1080. }
  1081. /**
  1082. * Generate a box
  1083. *
  1084. * @param {string} boxName
  1085. * @param {...!Uint8Array} payload
  1086. * @return {!Uint8Array}
  1087. */
  1088. static box(boxName, ...payload) {
  1089. let type = shaka.util.Mp4Generator.BOX_TYPES_[boxName];
  1090. if (!type) {
  1091. type = [
  1092. boxName.charCodeAt(0),
  1093. boxName.charCodeAt(1),
  1094. boxName.charCodeAt(2),
  1095. boxName.charCodeAt(3),
  1096. ];
  1097. shaka.util.Mp4Generator.BOX_TYPES_[boxName] = type;
  1098. }
  1099. // make the header for the box
  1100. let size = 8;
  1101. // calculate the total size we need to allocate
  1102. for (let i = payload.length - 1; i >= 0; i--) {
  1103. size += payload[i].byteLength;
  1104. }
  1105. const result = new Uint8Array(size);
  1106. result[0] = (size >> 24) & 0xff;
  1107. result[1] = (size >> 16) & 0xff;
  1108. result[2] = (size >> 8) & 0xff;
  1109. result[3] = size & 0xff;
  1110. result.set(type, 4);
  1111. // copy the payload into the result
  1112. for (let i = 0, pointer = 8; i < payload.length; i++) {
  1113. // copy payload[i] array @ offset pointer
  1114. result.set(payload[i], pointer);
  1115. pointer += payload[i].byteLength;
  1116. }
  1117. return result;
  1118. }
  1119. };
  1120. /**
  1121. * @private {boolean}
  1122. */
  1123. shaka.util.Mp4Generator.initializated_ = false;
  1124. /**
  1125. * @private {number}
  1126. */
  1127. shaka.util.Mp4Generator.UINT32_MAX_ = Math.pow(2, 32) - 1;
  1128. /**
  1129. * @private {!Object.<string, !Array.<number>>}
  1130. */
  1131. shaka.util.Mp4Generator.BOX_TYPES_ = {};
  1132. /**
  1133. * @private {{video: !Uint8Array, audio: !Uint8Array}}
  1134. */
  1135. shaka.util.Mp4Generator.HDLR_TYPES_ = {
  1136. video: new Uint8Array([
  1137. 0x00, // version 0
  1138. 0x00, 0x00, 0x00, // flags
  1139. 0x00, 0x00, 0x00, 0x00, // pre_defined
  1140. 0x76, 0x69, 0x64, 0x65, // handler_type: 'vide'
  1141. 0x00, 0x00, 0x00, 0x00, // reserved
  1142. 0x00, 0x00, 0x00, 0x00, // reserved
  1143. 0x00, 0x00, 0x00, 0x00, // reserved
  1144. 0x56, 0x69, 0x64, 0x65,
  1145. 0x6f, 0x48, 0x61, 0x6e,
  1146. 0x64, 0x6c, 0x65, 0x72, 0x00, // name: 'VideoHandler'
  1147. ]),
  1148. audio: new Uint8Array([
  1149. 0x00, // version 0
  1150. 0x00, 0x00, 0x00, // flags
  1151. 0x00, 0x00, 0x00, 0x00, // pre_defined
  1152. 0x73, 0x6f, 0x75, 0x6e, // handler_type: 'soun'
  1153. 0x00, 0x00, 0x00, 0x00, // reserved
  1154. 0x00, 0x00, 0x00, 0x00, // reserved
  1155. 0x00, 0x00, 0x00, 0x00, // reserved
  1156. 0x53, 0x6f, 0x75, 0x6e,
  1157. 0x64, 0x48, 0x61, 0x6e,
  1158. 0x64, 0x6c, 0x65, 0x72, 0x00, // name: 'SoundHandler'
  1159. ]),
  1160. };
  1161. /**
  1162. * @private {!Uint8Array}
  1163. */
  1164. shaka.util.Mp4Generator.STTS_ = new Uint8Array([
  1165. 0x00, // version
  1166. 0x00, 0x00, 0x00, // flags
  1167. 0x00, 0x00, 0x00, 0x00, // entry_count
  1168. ]);
  1169. /**
  1170. * @private {!Uint8Array}
  1171. */
  1172. shaka.util.Mp4Generator.STSC_ = new Uint8Array([
  1173. 0x00, // version
  1174. 0x00, 0x00, 0x00, // flags
  1175. 0x00, 0x00, 0x00, 0x00, // entry_count
  1176. ]);
  1177. /**
  1178. * @private {!Uint8Array}
  1179. */
  1180. shaka.util.Mp4Generator.STCO_ = new Uint8Array([
  1181. 0x00, // version
  1182. 0x00, 0x00, 0x00, // flags
  1183. 0x00, 0x00, 0x00, 0x00, // entry_count
  1184. ]);
  1185. /**
  1186. * @private {!Uint8Array}
  1187. */
  1188. shaka.util.Mp4Generator.STSZ_ = new Uint8Array([
  1189. 0x00, // version
  1190. 0x00, 0x00, 0x00, // flags
  1191. 0x00, 0x00, 0x00, 0x00, // sample_size
  1192. 0x00, 0x00, 0x00, 0x00, // sample_count
  1193. ]);
  1194. /**
  1195. * @private {!Uint8Array}
  1196. */
  1197. shaka.util.Mp4Generator.VMHD_ = new Uint8Array([
  1198. 0x00, // version
  1199. 0x00, 0x00, 0x01, // flags
  1200. 0x00, 0x00, // graphicsmode
  1201. 0x00, 0x00,
  1202. 0x00, 0x00,
  1203. 0x00, 0x00, // opcolor
  1204. ]);
  1205. /**
  1206. * @private {!Uint8Array}
  1207. */
  1208. shaka.util.Mp4Generator.SMHD_ = new Uint8Array([
  1209. 0x00, // version
  1210. 0x00, 0x00, 0x00, // flags
  1211. 0x00, 0x00, // balance, 0 means centered
  1212. 0x00, 0x00, // reserved
  1213. ]);
  1214. /**
  1215. * @private {!Uint8Array}
  1216. */
  1217. shaka.util.Mp4Generator.STSD_ = new Uint8Array([
  1218. 0x00, // version 0
  1219. 0x00, 0x00, 0x00, // flags
  1220. 0x00, 0x00, 0x00, 0x01, // entry_count
  1221. ]);
  1222. /**
  1223. * @private {!Uint8Array}
  1224. */
  1225. shaka.util.Mp4Generator.FTYP_ = new Uint8Array([]);
  1226. /**
  1227. * @private {!Uint8Array}
  1228. */
  1229. shaka.util.Mp4Generator.DREF_ = new Uint8Array([
  1230. 0x00, // version 0
  1231. 0x00, 0x00, 0x00, // flags
  1232. 0x00, 0x00, 0x00, 0x01, // entry_count
  1233. 0x00, 0x00, 0x00, 0x0c, // entry_size
  1234. 0x75, 0x72, 0x6c, 0x20, // 'url' type
  1235. 0x00, // version 0
  1236. 0x00, 0x00, 0x01, // entry_flags
  1237. ]);
  1238. /**
  1239. * @private {!Uint8Array}
  1240. */
  1241. shaka.util.Mp4Generator.DINF_ = new Uint8Array([]);
  1242. /**
  1243. * @typedef {{
  1244. * id: number,
  1245. * type: string,
  1246. * codecs: string,
  1247. * encrypted: boolean,
  1248. * timescale: number,
  1249. * duration: number,
  1250. * videoNalus: !Array.<string>,
  1251. * audioConfig: !Uint8Array,
  1252. * videoConfig: !Uint8Array,
  1253. * hSpacing: number,
  1254. * vSpacing: number,
  1255. * data: ?shaka.util.Mp4Generator.Data,
  1256. * stream: !shaka.extern.Stream
  1257. * }}
  1258. *
  1259. * @property {number} id
  1260. * A unique ID
  1261. * @property {string} type
  1262. * Indicate the content type: 'video' or 'audio'.
  1263. * @property {string} codecs
  1264. * <i>Defaults to '' (i.e., unknown / not needed).</i> <br>
  1265. * The Stream's codecs, e.g., 'avc1.4d4015' or 'vp9'<br>
  1266. * See {@link https://tools.ietf.org/html/rfc6381}
  1267. * @property {boolean} encrypted
  1268. * Indicate if the stream is encrypted.
  1269. * @property {number} timescale
  1270. * The Stream's timescale.
  1271. * @property {number} duration
  1272. * The Stream's duration.
  1273. * @property {!Array.<string>} videoNalus
  1274. * The stream's video nalus.
  1275. * @property {!Uint8Array} audioConfig
  1276. * The stream's audio config.
  1277. * @property {!Uint8Array} videoConfig
  1278. * The stream's video config.
  1279. * @property {number} hSpacing
  1280. * The stream's video horizontal spacing of pixels.
  1281. * @property {number} vSpacing
  1282. * The stream's video vertial spacing of pixels.
  1283. * @property {?shaka.util.Mp4Generator.Data} data
  1284. * The stream's data.
  1285. * @property {!shaka.extern.Stream} stream
  1286. * The Stream.
  1287. */
  1288. shaka.util.Mp4Generator.StreamInfo;
  1289. /**
  1290. * @typedef {{
  1291. * sequenceNumber: number,
  1292. * baseMediaDecodeTime: number,
  1293. * samples: !Array.<shaka.util.Mp4Generator.Mp4Sample>
  1294. * }}
  1295. *
  1296. * @property {number} sequenceNumber
  1297. * The sequence number.
  1298. * @property {number} baseMediaDecodeTime
  1299. * The base media decode time.
  1300. * @property {!Array.<shaka.util.Mp4Generator.Mp4Sample>} samples
  1301. * The data samples.
  1302. */
  1303. shaka.util.Mp4Generator.Data;
  1304. /**
  1305. * @typedef {{
  1306. * data: !Uint8Array,
  1307. * size: number,
  1308. * duration: number,
  1309. * cts: number,
  1310. * flags: !shaka.util.Mp4Generator.Mp4SampleFlags
  1311. * }}
  1312. *
  1313. * @property {!Uint8Array} data
  1314. * The sample data.
  1315. * @property {number} size
  1316. * The sample size.
  1317. * @property {number} duration
  1318. * The sample duration.
  1319. * @property {number} cts
  1320. * The sample composition time.
  1321. * @property {!shaka.util.Mp4Generator.Mp4SampleFlags} flags
  1322. * The sample flags.
  1323. */
  1324. shaka.util.Mp4Generator.Mp4Sample;
  1325. /**
  1326. * @typedef {{
  1327. * isLeading: number,
  1328. * isDependedOn: number,
  1329. * hasRedundancy: number,
  1330. * degradPrio: number,
  1331. * dependsOn: number,
  1332. * isNonSync: number
  1333. * }}
  1334. *
  1335. * @property {number} isLeading
  1336. * @property {number} isDependedOn
  1337. * @property {number} hasRedundancy
  1338. * @property {number} degradPrio
  1339. * @property {number} dependsOn
  1340. * @property {number} isNonSync
  1341. */
  1342. shaka.util.Mp4Generator.Mp4SampleFlags;