Source: lib/util/data_view_reader.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.util.DataViewReader');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.util.BufferUtils');
  9. goog.require('shaka.util.Error');
  10. goog.require('shaka.util.StringUtils');
  11. /**
  12. * @summary DataViewReader abstracts a DataView object.
  13. * @export
  14. */
  15. shaka.util.DataViewReader = class {
  16. /**
  17. * @param {BufferSource} data
  18. * @param {shaka.util.DataViewReader.Endianness} endianness The endianness.
  19. */
  20. constructor(data, endianness) {
  21. /** @private {!DataView} */
  22. this.dataView_ = shaka.util.BufferUtils.toDataView(data);
  23. /** @private {boolean} */
  24. this.littleEndian_ =
  25. endianness == shaka.util.DataViewReader.Endianness.LITTLE_ENDIAN;
  26. /** @private {number} */
  27. this.position_ = 0;
  28. }
  29. /** @return {!DataView} The underlying DataView instance. */
  30. getDataView() {
  31. return this.dataView_;
  32. }
  33. /**
  34. * @return {boolean} True if the reader has more data, false otherwise.
  35. * @export
  36. */
  37. hasMoreData() {
  38. return this.position_ < this.dataView_.byteLength;
  39. }
  40. /**
  41. * Gets the current byte position.
  42. * @return {number}
  43. * @export
  44. */
  45. getPosition() {
  46. return this.position_;
  47. }
  48. /**
  49. * Gets the byte length of the DataView.
  50. * @return {number}
  51. * @export
  52. */
  53. getLength() {
  54. return this.dataView_.byteLength;
  55. }
  56. /**
  57. * Reads an unsigned 8 bit integer, and advances the reader.
  58. * @return {number} The integer.
  59. * @export
  60. */
  61. readUint8() {
  62. try {
  63. const value = this.dataView_.getUint8(this.position_);
  64. this.position_ += 1;
  65. return value;
  66. } catch (exception) {
  67. throw this.outOfBounds_();
  68. }
  69. }
  70. /**
  71. * Reads an unsigned 16 bit integer, and advances the reader.
  72. * @return {number} The integer.
  73. * @export
  74. */
  75. readUint16() {
  76. try {
  77. const value =
  78. this.dataView_.getUint16(this.position_, this.littleEndian_);
  79. this.position_ += 2;
  80. return value;
  81. } catch (exception) {
  82. throw this.outOfBounds_();
  83. }
  84. }
  85. /**
  86. * Reads an unsigned 32 bit integer, and advances the reader.
  87. * @return {number} The integer.
  88. * @export
  89. */
  90. readUint32() {
  91. try {
  92. const value =
  93. this.dataView_.getUint32(this.position_, this.littleEndian_);
  94. this.position_ += 4;
  95. return value;
  96. } catch (exception) {
  97. throw this.outOfBounds_();
  98. }
  99. }
  100. /**
  101. * Reads a signed 32 bit integer, and advances the reader.
  102. * @return {number} The integer.
  103. * @export
  104. */
  105. readInt32() {
  106. try {
  107. const value = this.dataView_.getInt32(this.position_, this.littleEndian_);
  108. this.position_ += 4;
  109. return value;
  110. } catch (exception) {
  111. throw this.outOfBounds_();
  112. }
  113. }
  114. /**
  115. * Reads an unsigned 64 bit integer, and advances the reader.
  116. * @return {number} The integer.
  117. * @export
  118. */
  119. readUint64() {
  120. /** @type {number} */
  121. let low;
  122. /** @type {number} */
  123. let high;
  124. try {
  125. if (this.littleEndian_) {
  126. low = this.dataView_.getUint32(this.position_, true);
  127. high = this.dataView_.getUint32(this.position_ + 4, true);
  128. } else {
  129. high = this.dataView_.getUint32(this.position_, false);
  130. low = this.dataView_.getUint32(this.position_ + 4, false);
  131. }
  132. } catch (exception) {
  133. throw this.outOfBounds_();
  134. }
  135. if (high > 0x1FFFFF) {
  136. throw new shaka.util.Error(
  137. shaka.util.Error.Severity.CRITICAL,
  138. shaka.util.Error.Category.MEDIA,
  139. shaka.util.Error.Code.JS_INTEGER_OVERFLOW);
  140. }
  141. this.position_ += 8;
  142. // NOTE: This is subtle, but in JavaScript you can't shift left by 32
  143. // and get the full range of 53-bit values possible.
  144. // You must multiply by 2^32.
  145. return (high * Math.pow(2, 32)) + low;
  146. }
  147. /**
  148. * Reads the specified number of raw bytes.
  149. * @param {number} bytes The number of bytes to read.
  150. * @return {!Uint8Array}
  151. * @export
  152. */
  153. readBytes(bytes) {
  154. goog.asserts.assert(bytes >= 0, 'Bad call to DataViewReader.readBytes');
  155. if (this.position_ + bytes > this.dataView_.byteLength) {
  156. throw this.outOfBounds_();
  157. }
  158. const value =
  159. shaka.util.BufferUtils.toUint8(this.dataView_, this.position_, bytes);
  160. this.position_ += bytes;
  161. return value;
  162. }
  163. /**
  164. * Skips the specified number of bytes.
  165. * @param {number} bytes The number of bytes to skip.
  166. * @export
  167. */
  168. skip(bytes) {
  169. goog.asserts.assert(bytes >= 0, 'Bad call to DataViewReader.skip');
  170. if (this.position_ + bytes > this.dataView_.byteLength) {
  171. throw this.outOfBounds_();
  172. }
  173. this.position_ += bytes;
  174. }
  175. /**
  176. * Rewinds the specified number of bytes.
  177. * @param {number} bytes The number of bytes to rewind.
  178. * @export
  179. */
  180. rewind(bytes) {
  181. goog.asserts.assert(bytes >= 0, 'Bad call to DataViewReader.rewind');
  182. if (this.position_ < bytes) {
  183. throw this.outOfBounds_();
  184. }
  185. this.position_ -= bytes;
  186. }
  187. /**
  188. * Seeks to a specified position.
  189. * @param {number} position The desired byte position within the DataView.
  190. * @export
  191. */
  192. seek(position) {
  193. goog.asserts.assert(position >= 0, 'Bad call to DataViewReader.seek');
  194. if (position < 0 || position > this.dataView_.byteLength) {
  195. throw this.outOfBounds_();
  196. }
  197. this.position_ = position;
  198. }
  199. /**
  200. * Keeps reading until it reaches a byte that equals to zero. The text is
  201. * assumed to be UTF-8.
  202. * @return {string}
  203. * @export
  204. */
  205. readTerminatedString() {
  206. const start = this.position_;
  207. while (this.hasMoreData()) {
  208. const value = this.dataView_.getUint8(this.position_);
  209. if (value == 0) {
  210. break;
  211. }
  212. this.position_ += 1;
  213. }
  214. const ret = shaka.util.BufferUtils.toUint8(
  215. this.dataView_, start, this.position_ - start);
  216. // Skip string termination.
  217. this.position_ += 1;
  218. return shaka.util.StringUtils.fromUTF8(ret);
  219. }
  220. /**
  221. * @return {!shaka.util.Error}
  222. * @private
  223. */
  224. outOfBounds_() {
  225. return new shaka.util.Error(
  226. shaka.util.Error.Severity.CRITICAL,
  227. shaka.util.Error.Category.MEDIA,
  228. shaka.util.Error.Code.BUFFER_READ_OUT_OF_BOUNDS);
  229. }
  230. };
  231. /**
  232. * Endianness.
  233. * @enum {number}
  234. * @export
  235. */
  236. shaka.util.DataViewReader.Endianness = {
  237. 'BIG_ENDIAN': 0,
  238. 'LITTLE_ENDIAN': 1,
  239. };