Source: lib/media/adaptation_set.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.media.AdaptationSet');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.log');
  9. goog.require('shaka.util.MimeUtils');
  10. /**
  11. * A set of variants that we want to adapt between.
  12. *
  13. * @final
  14. */
  15. shaka.media.AdaptationSet = class {
  16. /**
  17. * @param {shaka.extern.Variant} root
  18. * The variant that all other variants will be tested against when being
  19. * added to the adaptation set. If a variant is not compatible with the
  20. * root, it will not be added.
  21. * @param {!Iterable.<shaka.extern.Variant>=} candidates
  22. * Variants that may be compatible with the root and should be added if
  23. * compatible. If a candidate is not compatible, it will not end up in the
  24. * adaptation set.
  25. */
  26. constructor(root, candidates) {
  27. /** @private {shaka.extern.Variant} */
  28. this.root_ = root;
  29. /** @private {!Set.<shaka.extern.Variant>} */
  30. this.variants_ = new Set([root]);
  31. // Try to add all the candidates. If they cannot be added (because they
  32. // are not compatible with the root, they will be rejected by |add|.
  33. candidates = candidates || [];
  34. for (const candidate of candidates) {
  35. this.add(candidate);
  36. }
  37. }
  38. /**
  39. * @param {shaka.extern.Variant} variant
  40. * @return {boolean}
  41. */
  42. add(variant) {
  43. if (this.canInclude(variant)) {
  44. this.variants_.add(variant);
  45. return true;
  46. }
  47. // To be nice, issue a warning if someone is trying to add something that
  48. // they shouldn't.
  49. shaka.log.warning('Rejecting variant - not compatible with root.');
  50. return false;
  51. }
  52. /**
  53. * Check if |variant| can be included with the set. If |canInclude| returns
  54. * |false|, calling |add| will result in it being ignored.
  55. *
  56. * @param {shaka.extern.Variant} variant
  57. * @return {boolean}
  58. */
  59. canInclude(variant) {
  60. return shaka.media.AdaptationSet.areAdaptable(this.root_, variant);
  61. }
  62. /**
  63. * @param {shaka.extern.Variant} a
  64. * @param {shaka.extern.Variant} b
  65. * @return {boolean}
  66. */
  67. static areAdaptable(a, b) {
  68. const AdaptationSet = shaka.media.AdaptationSet;
  69. // All variants should have audio or should all not have audio.
  70. if (!!a.audio != !!b.audio) {
  71. return false;
  72. }
  73. // All variants should have video or should all not have video.
  74. if (!!a.video != !!b.video) {
  75. return false;
  76. }
  77. // If the languages don't match, we should not adapt between them.
  78. if (a.language != b.language) {
  79. return false;
  80. }
  81. goog.asserts.assert(
  82. !!a.audio == !!b.audio,
  83. 'Both should either have audio or not have audio.');
  84. if (a.audio && b.audio &&
  85. !AdaptationSet.areAudiosCompatible_(a.audio, b.audio)) {
  86. return false;
  87. }
  88. goog.asserts.assert(
  89. !!a.video == !!b.video,
  90. 'Both should either have video or not have video.');
  91. if (a.video && b.video &&
  92. !AdaptationSet.areVideosCompatible_(a.video, b.video)) {
  93. return false;
  94. }
  95. return true;
  96. }
  97. /**
  98. * @return {!Iterable.<shaka.extern.Variant>}
  99. */
  100. values() {
  101. return this.variants_.values();
  102. }
  103. /**
  104. * Check if we can switch between two audio streams.
  105. *
  106. * @param {shaka.extern.Stream} a
  107. * @param {shaka.extern.Stream} b
  108. * @return {boolean}
  109. * @private
  110. */
  111. static areAudiosCompatible_(a, b) {
  112. const AdaptationSet = shaka.media.AdaptationSet;
  113. // Don't adapt between channel counts, which could annoy the user
  114. // due to volume changes on downmixing. An exception is made for
  115. // stereo and mono, which should be fine to adapt between.
  116. if (!a.channelsCount || !b.channelsCount ||
  117. a.channelsCount > 2 || b.channelsCount > 2) {
  118. if (a.channelsCount != b.channelsCount) {
  119. return false;
  120. }
  121. }
  122. // We can only adapt between base-codecs.
  123. if (!AdaptationSet.canTransitionBetween_(a, b)) {
  124. return false;
  125. }
  126. // Audio roles must not change between adaptations.
  127. if (!AdaptationSet.areRolesEqual_(a.roles, b.roles)) {
  128. return false;
  129. }
  130. return true;
  131. }
  132. /**
  133. * Check if we can switch between two video streams.
  134. *
  135. * @param {shaka.extern.Stream} a
  136. * @param {shaka.extern.Stream} b
  137. * @return {boolean}
  138. * @private
  139. */
  140. static areVideosCompatible_(a, b) {
  141. const AdaptationSet = shaka.media.AdaptationSet;
  142. // We can only adapt between base-codecs.
  143. if (!AdaptationSet.canTransitionBetween_(a, b)) {
  144. return false;
  145. }
  146. // Video roles must not change between adaptations.
  147. if (!AdaptationSet.areRolesEqual_(a.roles, b.roles)) {
  148. return false;
  149. }
  150. return true;
  151. }
  152. /**
  153. * Check if we can switch between two streams based on their codec and mime
  154. * type.
  155. *
  156. * @param {shaka.extern.Stream} a
  157. * @param {shaka.extern.Stream} b
  158. * @return {boolean}
  159. * @private
  160. */
  161. static canTransitionBetween_(a, b) {
  162. if (a.mimeType != b.mimeType) {
  163. return false;
  164. }
  165. // Get the base codec of each codec in each stream.
  166. const codecsA = shaka.util.MimeUtils.splitCodecs(a.codecs).map((codec) => {
  167. return shaka.util.MimeUtils.getCodecBase(codec);
  168. });
  169. const codecsB = shaka.util.MimeUtils.splitCodecs(b.codecs).map((codec) => {
  170. return shaka.util.MimeUtils.getCodecBase(codec);
  171. });
  172. // We don't want to allow switching between transmuxed and non-transmuxed
  173. // content so the number of codecs should be the same.
  174. //
  175. // To avoid the case where an codec is used for audio and video we will
  176. // codecs using arrays (not sets). While at this time, there are no codecs
  177. // that work for audio and video, it is possible for "raw" codecs to be
  178. // which would share the same name.
  179. if (codecsA.length != codecsB.length) {
  180. return false;
  181. }
  182. // Sort them so that we can walk through them and compare them
  183. // element-by-element.
  184. codecsA.sort();
  185. codecsB.sort();
  186. for (let i = 0; i < codecsA.length; i++) {
  187. if (codecsA[i] != codecsB[i]) {
  188. return false;
  189. }
  190. }
  191. return true;
  192. }
  193. /**
  194. * Check if two role lists are the equal. This will take into account all
  195. * unique behaviours when comparing roles.
  196. *
  197. * @param {!Iterable.<string>} a
  198. * @param {!Iterable.<string>} b
  199. * @return {boolean}
  200. * @private
  201. */
  202. static areRolesEqual_(a, b) {
  203. const aSet = new Set(a);
  204. const bSet = new Set(b);
  205. // Remove the main role from the role lists (we expect to see them only
  206. // in dash manifests).
  207. const mainRole = 'main';
  208. aSet.delete(mainRole);
  209. bSet.delete(mainRole);
  210. // Make sure that we have the same number roles in each list. Make sure to
  211. // do it after correcting for 'main'.
  212. if (aSet.size != bSet.size) {
  213. return false;
  214. }
  215. // Because we know the two sets are the same size, if any item is missing
  216. // if means that they are not the same.
  217. for (const x of aSet) {
  218. if (!bSet.has(x)) {
  219. return false;
  220. }
  221. }
  222. return true;
  223. }
  224. };