Source: lib/transmuxer/ac3.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.transmuxer.Ac3');
  7. /**
  8. * AC3 utils
  9. */
  10. shaka.transmuxer.Ac3 = class {
  11. /**
  12. * @param {!Uint8Array} data
  13. * @param {!number} offset
  14. * @return {?{sampleRate: number, channelCount: number,
  15. * audioConfig: !Uint8Array, frameLength: number}}
  16. */
  17. static parseFrame(data, offset) {
  18. if (offset + 8 > data.length) {
  19. // not enough bytes left
  20. return null;
  21. }
  22. if (data[offset] !== 0x0b || data[offset + 1] !== 0x77) {
  23. // invalid magic
  24. return null;
  25. }
  26. // get sample rate
  27. const samplingRateCode = data[offset + 4] >> 6;
  28. if (samplingRateCode >= 3) {
  29. // invalid sampling rate
  30. return null;
  31. }
  32. const samplingRateMap = [48000, 44100, 32000];
  33. // get frame size
  34. const frameSizeCode = data[offset + 4] & 0x3f;
  35. const frameSizeMap = [
  36. 64, 69, 96, 64, 70, 96, 80, 87, 120, 80, 88, 120, 96, 104, 144, 96, 105,
  37. 144, 112, 121, 168, 112, 122, 168, 128, 139, 192, 128, 140, 192, 160,
  38. 174, 240, 160, 175, 240, 192, 208, 288, 192, 209, 288, 224, 243, 336,
  39. 224, 244, 336, 256, 278, 384, 256, 279, 384, 320, 348, 480, 320, 349,
  40. 480, 384, 417, 576, 384, 418, 576, 448, 487, 672, 448, 488, 672, 512,
  41. 557, 768, 512, 558, 768, 640, 696, 960, 640, 697, 960, 768, 835, 1152,
  42. 768, 836, 1152, 896, 975, 1344, 896, 976, 1344, 1024, 1114, 1536, 1024,
  43. 1115, 1536, 1152, 1253, 1728, 1152, 1254, 1728, 1280, 1393, 1920, 1280,
  44. 1394, 1920,
  45. ];
  46. const frameLength = frameSizeMap[frameSizeCode * 3 + samplingRateCode] * 2;
  47. if (offset + frameLength > data.length) {
  48. return null;
  49. }
  50. // get channel count
  51. const channelMode = data[offset + 6] >> 5;
  52. let skipCount = 0;
  53. if (channelMode === 2) {
  54. skipCount += 2;
  55. } else {
  56. if ((channelMode & 1) && channelMode !== 1) {
  57. skipCount += 2;
  58. }
  59. if (channelMode & 4) {
  60. skipCount += 2;
  61. }
  62. }
  63. const lowFrequencyEffectsChannelOn =
  64. (((data[offset + 6] << 8) | data[offset + 7]) >> (12 - skipCount)) & 1;
  65. const channelsMap = [2, 1, 2, 3, 3, 4, 4, 5];
  66. // Audio config for DAC3 box
  67. const bitStreamIdentification = data[offset + 5] >> 3;
  68. const bitStreamMode = data[offset + 5] & 7;
  69. const config = new Uint8Array([
  70. (samplingRateCode << 6) |
  71. (bitStreamIdentification << 1) |
  72. (bitStreamMode >> 2),
  73. ((bitStreamMode & 3) << 6) |
  74. (channelMode << 3) |
  75. (lowFrequencyEffectsChannelOn << 2) |
  76. (frameSizeCode >> 4),
  77. (frameSizeCode << 4) & 0xe0,
  78. ]);
  79. return {
  80. sampleRate: samplingRateMap[samplingRateCode],
  81. channelCount: channelsMap[channelMode] + lowFrequencyEffectsChannelOn,
  82. audioConfig: config,
  83. frameLength: frameLength,
  84. };
  85. }
  86. /**
  87. * @param {!Uint8Array} data
  88. * @param {!number} offset
  89. * @return {boolean}
  90. */
  91. static probe(data, offset) {
  92. // look for the ac-3 sync bytes
  93. if (data[offset] === 0x0b &&
  94. data[offset + 1] === 0x77) {
  95. // check the bsid (bitStreamIdentification) to confirm ac-3
  96. let bsid = 0;
  97. let numBits = 5;
  98. offset += numBits;
  99. /** @type {?number} */
  100. let temp = null;
  101. /** @type {?number} */
  102. let mask = null;
  103. /** @type {?number} */
  104. let byte = null;
  105. while (numBits > 0) {
  106. byte = data[offset];
  107. // read remaining bits, upto 8 bits at a time
  108. const bits = Math.min(numBits, 8);
  109. const shift = 8 - bits;
  110. mask = (0xff000000 >>> (24 + shift)) << shift;
  111. temp = (byte & mask) >> shift;
  112. bsid = !bsid ? temp : (bsid << bits) | temp;
  113. offset += 1;
  114. numBits -= bits;
  115. }
  116. if (bsid < 16) {
  117. return true;
  118. }
  119. }
  120. return false;
  121. }
  122. };
  123. /**
  124. * @const {number}
  125. */
  126. shaka.transmuxer.Ac3.AC3_SAMPLES_PER_FRAME = 1536;