Source: lib/util/mp4_parser.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.util.Mp4Parser');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.log');
  9. goog.require('shaka.util.DataViewReader');
  10. /**
  11. * @export
  12. */
  13. shaka.util.Mp4Parser = class {
  14. /** */
  15. constructor() {
  16. /** @private {!Object.<number, shaka.util.Mp4Parser.BoxType_>} */
  17. this.headers_ = [];
  18. /** @private {!Object.<number, !shaka.util.Mp4Parser.CallbackType>} */
  19. this.boxDefinitions_ = [];
  20. /** @private {boolean} */
  21. this.done_ = false;
  22. }
  23. /**
  24. * Declare a box type as a Box.
  25. *
  26. * @param {string} type
  27. * @param {!shaka.util.Mp4Parser.CallbackType} definition
  28. * @return {!shaka.util.Mp4Parser}
  29. * @export
  30. */
  31. box(type, definition) {
  32. const typeCode = shaka.util.Mp4Parser.typeFromString_(type);
  33. this.headers_[typeCode] = shaka.util.Mp4Parser.BoxType_.BASIC_BOX;
  34. this.boxDefinitions_[typeCode] = definition;
  35. return this;
  36. }
  37. /**
  38. * Declare a box type as a Full Box.
  39. *
  40. * @param {string} type
  41. * @param {!shaka.util.Mp4Parser.CallbackType} definition
  42. * @return {!shaka.util.Mp4Parser}
  43. * @export
  44. */
  45. fullBox(type, definition) {
  46. const typeCode = shaka.util.Mp4Parser.typeFromString_(type);
  47. this.headers_[typeCode] = shaka.util.Mp4Parser.BoxType_.FULL_BOX;
  48. this.boxDefinitions_[typeCode] = definition;
  49. return this;
  50. }
  51. /**
  52. * Stop parsing. Useful for extracting information from partial segments and
  53. * avoiding an out-of-bounds error once you find what you are looking for.
  54. *
  55. * @export
  56. */
  57. stop() {
  58. this.done_ = true;
  59. }
  60. /**
  61. * Parse the given data using the added callbacks.
  62. *
  63. * @param {!BufferSource} data
  64. * @param {boolean=} partialOkay If true, allow reading partial payloads
  65. * from some boxes. If the goal is a child box, we can sometimes find it
  66. * without enough data to find all child boxes.
  67. * @param {boolean=} stopOnPartial If true, stop reading if an incomplete
  68. * box is detected.
  69. * @export
  70. */
  71. parse(data, partialOkay, stopOnPartial) {
  72. const reader = new shaka.util.DataViewReader(
  73. data, shaka.util.DataViewReader.Endianness.BIG_ENDIAN);
  74. this.done_ = false;
  75. while (reader.hasMoreData() && !this.done_) {
  76. this.parseNext(0, reader, partialOkay, stopOnPartial);
  77. }
  78. }
  79. /**
  80. * Parse the next box on the current level.
  81. *
  82. * @param {number} absStart The absolute start position in the original
  83. * byte array.
  84. * @param {!shaka.util.DataViewReader} reader
  85. * @param {boolean=} partialOkay If true, allow reading partial payloads
  86. * from some boxes. If the goal is a child box, we can sometimes find it
  87. * without enough data to find all child boxes.
  88. * @param {boolean=} stopOnPartial If true, stop reading if an incomplete
  89. * box is detected.
  90. * @export
  91. */
  92. parseNext(absStart, reader, partialOkay, stopOnPartial) {
  93. const start = reader.getPosition();
  94. // size(4 bytes) + type(4 bytes) = 8 bytes
  95. if (stopOnPartial && start + 8 > reader.getLength()) {
  96. this.done_ = true;
  97. return;
  98. }
  99. let size = reader.readUint32();
  100. const type = reader.readUint32();
  101. const name = shaka.util.Mp4Parser.typeToString(type);
  102. let has64BitSize = false;
  103. shaka.log.v2('Parsing MP4 box', name);
  104. switch (size) {
  105. case 0:
  106. size = reader.getLength() - start;
  107. break;
  108. case 1:
  109. if (stopOnPartial && reader.getPosition() + 8 > reader.getLength()) {
  110. this.done_ = true;
  111. return;
  112. }
  113. size = reader.readUint64();
  114. has64BitSize = true;
  115. break;
  116. }
  117. const boxDefinition = this.boxDefinitions_[type];
  118. if (boxDefinition) {
  119. let version = null;
  120. let flags = null;
  121. if (this.headers_[type] == shaka.util.Mp4Parser.BoxType_.FULL_BOX) {
  122. if (stopOnPartial && reader.getPosition() + 4 > reader.getLength()) {
  123. this.done_ = true;
  124. return;
  125. }
  126. const versionAndFlags = reader.readUint32();
  127. version = versionAndFlags >>> 24;
  128. flags = versionAndFlags & 0xFFFFFF;
  129. }
  130. // Read the whole payload so that the current level can be safely read
  131. // regardless of how the payload is parsed.
  132. let end = start + size;
  133. if (partialOkay && end > reader.getLength()) {
  134. // For partial reads, truncate the payload if we must.
  135. end = reader.getLength();
  136. }
  137. if (stopOnPartial && end > reader.getLength()) {
  138. this.done_ = true;
  139. return;
  140. }
  141. const payloadSize = end - reader.getPosition();
  142. const payload =
  143. (payloadSize > 0) ? reader.readBytes(payloadSize) : new Uint8Array(0);
  144. const payloadReader = new shaka.util.DataViewReader(
  145. payload, shaka.util.DataViewReader.Endianness.BIG_ENDIAN);
  146. /** @type {shaka.extern.ParsedBox} */
  147. const box = {
  148. parser: this,
  149. partialOkay: partialOkay || false,
  150. version,
  151. flags,
  152. reader: payloadReader,
  153. size,
  154. start: start + absStart,
  155. has64BitSize,
  156. };
  157. boxDefinition(box);
  158. } else {
  159. // Move the read head to be at the end of the box.
  160. // If the box is longer than the remaining parts of the file, e.g. the
  161. // mp4 is improperly formatted, or this was a partial range request that
  162. // ended in the middle of a box, just skip to the end.
  163. const skipLength = Math.min(
  164. start + size - reader.getPosition(),
  165. reader.getLength() - reader.getPosition());
  166. reader.skip(skipLength);
  167. }
  168. }
  169. /**
  170. * A callback that tells the Mp4 parser to treat the body of a box as a series
  171. * of boxes. The number of boxes is limited by the size of the parent box.
  172. *
  173. * @param {!shaka.extern.ParsedBox} box
  174. * @export
  175. */
  176. static children(box) {
  177. // The "reader" starts at the payload, so we need to add the header to the
  178. // start position. The header size varies.
  179. const headerSize = shaka.util.Mp4Parser.headerSize(box);
  180. while (box.reader.hasMoreData() && !box.parser.done_) {
  181. box.parser.parseNext(box.start + headerSize, box.reader, box.partialOkay);
  182. }
  183. }
  184. /**
  185. * A callback that tells the Mp4 parser to treat the body of a box as a sample
  186. * description. A sample description box has a fixed number of children. The
  187. * number of children is represented by a 4 byte unsigned integer. Each child
  188. * is a box.
  189. *
  190. * @param {!shaka.extern.ParsedBox} box
  191. * @export
  192. */
  193. static sampleDescription(box) {
  194. // The "reader" starts at the payload, so we need to add the header to the
  195. // start position. The header size varies.
  196. const headerSize = shaka.util.Mp4Parser.headerSize(box);
  197. const count = box.reader.readUint32();
  198. for (let i = 0; i < count; i++) {
  199. box.parser.parseNext(box.start + headerSize, box.reader, box.partialOkay);
  200. if (box.parser.done_) {
  201. break;
  202. }
  203. }
  204. }
  205. /**
  206. * Create a callback that tells the Mp4 parser to treat the body of a box as a
  207. * binary blob and to parse the body's contents using the provided callback.
  208. *
  209. * @param {function(!Uint8Array)} callback
  210. * @return {!shaka.util.Mp4Parser.CallbackType}
  211. * @export
  212. */
  213. static allData(callback) {
  214. return (box) => {
  215. const all = box.reader.getLength() - box.reader.getPosition();
  216. callback(box.reader.readBytes(all));
  217. };
  218. }
  219. /**
  220. * Convert an ascii string name to the integer type for a box.
  221. *
  222. * @param {string} name The name of the box. The name must be four
  223. * characters long.
  224. * @return {number}
  225. * @private
  226. */
  227. static typeFromString_(name) {
  228. goog.asserts.assert(
  229. name.length == 4,
  230. 'Mp4 box names must be 4 characters long');
  231. let code = 0;
  232. for (const chr of name) {
  233. code = (code << 8) | chr.charCodeAt(0);
  234. }
  235. return code;
  236. }
  237. /**
  238. * Convert an integer type from a box into an ascii string name.
  239. * Useful for debugging.
  240. *
  241. * @param {number} type The type of the box, a uint32.
  242. * @return {string}
  243. * @export
  244. */
  245. static typeToString(type) {
  246. const name = String.fromCharCode(
  247. (type >> 24) & 0xff,
  248. (type >> 16) & 0xff,
  249. (type >> 8) & 0xff,
  250. type & 0xff);
  251. return name;
  252. }
  253. /**
  254. * Find the header size of the box.
  255. * Useful for modifying boxes in place or finding the exact offset of a field.
  256. *
  257. * @param {shaka.extern.ParsedBox} box
  258. * @return {number}
  259. * @export
  260. */
  261. static headerSize(box) {
  262. const basicHeaderSize = 8;
  263. const _64BitFieldSize = box.has64BitSize ? 8 : 0;
  264. const versionAndFlagsSize = box.flags != null ? 4 : 0;
  265. return basicHeaderSize + _64BitFieldSize + versionAndFlagsSize;
  266. }
  267. };
  268. /**
  269. * @typedef {function(!shaka.extern.ParsedBox)}
  270. * @exportInterface
  271. */
  272. shaka.util.Mp4Parser.CallbackType;
  273. /**
  274. * An enum used to track the type of box so that the correct values can be
  275. * read from the header.
  276. *
  277. * @enum {number}
  278. * @private
  279. */
  280. shaka.util.Mp4Parser.BoxType_ = {
  281. BASIC_BOX: 0,
  282. FULL_BOX: 1,
  283. };