Source: lib/polyfill/media_capabilities.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.polyfill.MediaCapabilities');
  7. goog.require('shaka.log');
  8. goog.require('shaka.media.Capabilities');
  9. goog.require('shaka.polyfill');
  10. goog.require('shaka.util.DrmUtils');
  11. goog.require('shaka.util.Platform');
  12. /**
  13. * @summary A polyfill to provide navigator.mediaCapabilities on all browsers.
  14. * This is necessary for Tizen 3, Xbox One and possibly others we have yet to
  15. * discover.
  16. * @export
  17. */
  18. shaka.polyfill.MediaCapabilities = class {
  19. /**
  20. * Install the polyfill if needed.
  21. * @suppress {const}
  22. * @export
  23. */
  24. static install() {
  25. // We can enable MediaCapabilities in Android and Fuchsia devices, but not
  26. // in Linux devices because the implementation is buggy.
  27. // Since MediaCapabilities implementation is buggy in Apple browsers, we
  28. // should always install polyfill for Apple browsers.
  29. // See: https://github.com/shaka-project/shaka-player/issues/3530
  30. // TODO: re-evaluate MediaCapabilities in the future versions of Apple
  31. // Browsers.
  32. // Since MediaCapabilities implementation is buggy in PS5 browsers, we
  33. // should always install polyfill for PS5 browsers.
  34. // See: https://github.com/shaka-project/shaka-player/issues/3582
  35. // TODO: re-evaluate MediaCapabilities in the future versions of PS5
  36. // Browsers.
  37. // Since MediaCapabilities implementation does not exist in PS4 browsers, we
  38. // should always install polyfill.
  39. // Since MediaCapabilities implementation is buggy in Tizen browsers, we
  40. // should always install polyfill for Tizen browsers.
  41. // Since MediaCapabilities implementation is buggy in WebOS browsers, we
  42. // should always install polyfill for WebOS browsers.
  43. // Since MediaCapabilities implementation is buggy in EOS browsers, we
  44. // should always install polyfill for EOS browsers.
  45. // Since MediaCapabilities implementation is buggy in Hisense browsers, we
  46. // should always install polyfill for Hisense browsers.
  47. let canUseNativeMCap = true;
  48. if (shaka.util.Platform.isChromecast() &&
  49. !shaka.util.Platform.isAndroidCastDevice() &&
  50. !shaka.util.Platform.isFuchsiaCastDevice()) {
  51. canUseNativeMCap = false;
  52. }
  53. if (shaka.util.Platform.isApple() ||
  54. shaka.util.Platform.isPS5() ||
  55. shaka.util.Platform.isPS4() ||
  56. shaka.util.Platform.isWebOS() ||
  57. shaka.util.Platform.isTizen() ||
  58. shaka.util.Platform.isEOS() ||
  59. shaka.util.Platform.isHisense()) {
  60. canUseNativeMCap = false;
  61. }
  62. if (canUseNativeMCap && navigator.mediaCapabilities) {
  63. shaka.log.info(
  64. 'MediaCapabilities: Native mediaCapabilities support found.');
  65. return;
  66. }
  67. shaka.log.info('MediaCapabilities: install');
  68. if (!navigator.mediaCapabilities) {
  69. navigator.mediaCapabilities = /** @type {!MediaCapabilities} */ ({});
  70. }
  71. // Keep the patched MediaCapabilities object from being garbage-collected in
  72. // Safari.
  73. // See https://github.com/shaka-project/shaka-player/issues/3696#issuecomment-1009472718
  74. shaka.polyfill.MediaCapabilities.originalMcap =
  75. navigator.mediaCapabilities;
  76. navigator.mediaCapabilities.decodingInfo =
  77. shaka.polyfill.MediaCapabilities.decodingInfo_;
  78. }
  79. /**
  80. * @param {!MediaDecodingConfiguration} mediaDecodingConfig
  81. * @return {!Promise.<!MediaCapabilitiesDecodingInfo>}
  82. * @private
  83. */
  84. static async decodingInfo_(mediaDecodingConfig) {
  85. /** @type {!MediaCapabilitiesDecodingInfo} */
  86. const res = {
  87. supported: false,
  88. powerEfficient: true,
  89. smooth: true,
  90. keySystemAccess: null,
  91. configuration: mediaDecodingConfig,
  92. };
  93. const videoConfig = mediaDecodingConfig['video'];
  94. const audioConfig = mediaDecodingConfig['audio'];
  95. if (mediaDecodingConfig.type == 'media-source') {
  96. if (!shaka.util.Platform.supportsMediaSource()) {
  97. return res;
  98. }
  99. if (videoConfig) {
  100. const isSupported =
  101. await shaka.polyfill.MediaCapabilities.checkVideoSupport_(
  102. videoConfig);
  103. if (!isSupported) {
  104. return res;
  105. }
  106. }
  107. if (audioConfig) {
  108. const isSupported =
  109. shaka.polyfill.MediaCapabilities.checkAudioSupport_(audioConfig);
  110. if (!isSupported) {
  111. return res;
  112. }
  113. }
  114. } else if (mediaDecodingConfig.type == 'file') {
  115. if (videoConfig) {
  116. const contentType = videoConfig.contentType;
  117. const isSupported = shaka.util.Platform.supportsMediaType(contentType);
  118. if (!isSupported) {
  119. return res;
  120. }
  121. }
  122. if (audioConfig) {
  123. const contentType = audioConfig.contentType;
  124. const isSupported = shaka.util.Platform.supportsMediaType(contentType);
  125. if (!isSupported) {
  126. return res;
  127. }
  128. }
  129. } else {
  130. // Otherwise not supported.
  131. return res;
  132. }
  133. if (!mediaDecodingConfig.keySystemConfiguration) {
  134. // The variant is supported if it's unencrypted.
  135. res.supported = true;
  136. return res;
  137. } else {
  138. const mcapKeySystemConfig = mediaDecodingConfig.keySystemConfiguration;
  139. const keySystemAccess =
  140. await shaka.polyfill.MediaCapabilities.checkDrmSupport_(
  141. videoConfig, audioConfig, mcapKeySystemConfig);
  142. if (keySystemAccess) {
  143. res.supported = true;
  144. res.keySystemAccess = keySystemAccess;
  145. }
  146. }
  147. return res;
  148. }
  149. /**
  150. * @param {!VideoConfiguration} videoConfig The 'video' field of the
  151. * MediaDecodingConfiguration.
  152. * @return {!Promise<boolean>}
  153. * @private
  154. */
  155. static async checkVideoSupport_(videoConfig) {
  156. // Use 'shaka.media.Capabilities.isTypeSupported' to check if
  157. // the stream is supported.
  158. // Cast platforms will additionally check canDisplayType(), which
  159. // accepts extended MIME type parameters.
  160. // See: https://github.com/shaka-project/shaka-player/issues/4726
  161. if (shaka.util.Platform.isChromecast()) {
  162. const isSupported =
  163. await shaka.polyfill.MediaCapabilities.canCastDisplayType_(
  164. videoConfig);
  165. return isSupported;
  166. } else if (shaka.util.Platform.isTizen()) {
  167. let extendedType = videoConfig.contentType;
  168. if (videoConfig.width && videoConfig.height) {
  169. extendedType += `; width=${videoConfig.width}`;
  170. extendedType += `; height=${videoConfig.height}`;
  171. }
  172. if (videoConfig.framerate) {
  173. extendedType += `; framerate=${videoConfig.framerate}`;
  174. }
  175. if (videoConfig.bitrate) {
  176. extendedType += `; bitrate=${videoConfig.bitrate}`;
  177. }
  178. return shaka.media.Capabilities.isTypeSupported(extendedType);
  179. }
  180. return shaka.media.Capabilities.isTypeSupported(videoConfig.contentType);
  181. }
  182. /**
  183. * @param {!AudioConfiguration} audioConfig The 'audio' field of the
  184. * MediaDecodingConfiguration.
  185. * @return {boolean}
  186. * @private
  187. */
  188. static checkAudioSupport_(audioConfig) {
  189. let extendedType = audioConfig.contentType;
  190. if (shaka.util.Platform.isChromecast() && audioConfig.spatialRendering) {
  191. extendedType += '; spatialRendering=true';
  192. }
  193. return shaka.media.Capabilities.isTypeSupported(extendedType);
  194. }
  195. /**
  196. * @param {VideoConfiguration} videoConfig The 'video' field of the
  197. * MediaDecodingConfiguration.
  198. * @param {AudioConfiguration} audioConfig The 'audio' field of the
  199. * MediaDecodingConfiguration.
  200. * @param {!MediaCapabilitiesKeySystemConfiguration} mcapKeySystemConfig The
  201. * 'keySystemConfiguration' field of the MediaDecodingConfiguration.
  202. * @return {Promise<MediaKeySystemAccess>}
  203. * @private
  204. */
  205. static async checkDrmSupport_(videoConfig, audioConfig, mcapKeySystemConfig) {
  206. const audioCapabilities = [];
  207. const videoCapabilities = [];
  208. if (mcapKeySystemConfig.audio) {
  209. const capability = {
  210. robustness: mcapKeySystemConfig.audio.robustness || '',
  211. contentType: audioConfig.contentType,
  212. };
  213. // Some Tizen devices seem to misreport AC-3 support, but correctly
  214. // report EC-3 support. So query EC-3 as a fallback for AC-3.
  215. // See https://github.com/shaka-project/shaka-player/issues/2989 for
  216. // details.
  217. if (shaka.util.Platform.isTizen() &&
  218. audioConfig.contentType.includes('codecs="ac-3"')) {
  219. capability.contentType = 'audio/mp4; codecs="ec-3"';
  220. }
  221. if (mcapKeySystemConfig.audio.encryptionScheme) {
  222. capability.encryptionScheme =
  223. mcapKeySystemConfig.audio.encryptionScheme;
  224. }
  225. audioCapabilities.push(capability);
  226. }
  227. if (mcapKeySystemConfig.video) {
  228. const capability = {
  229. robustness: mcapKeySystemConfig.video.robustness || '',
  230. contentType: videoConfig.contentType,
  231. };
  232. if (mcapKeySystemConfig.video.encryptionScheme) {
  233. capability.encryptionScheme =
  234. mcapKeySystemConfig.video.encryptionScheme;
  235. }
  236. videoCapabilities.push(capability);
  237. }
  238. /** @type {MediaKeySystemConfiguration} */
  239. const mediaKeySystemConfig = {
  240. initDataTypes: [mcapKeySystemConfig.initDataType],
  241. distinctiveIdentifier: mcapKeySystemConfig.distinctiveIdentifier,
  242. persistentState: mcapKeySystemConfig.persistentState,
  243. sessionTypes: mcapKeySystemConfig.sessionTypes,
  244. };
  245. // Only add audio / video capabilities if they have valid data.
  246. // Otherwise the query will fail.
  247. if (audioCapabilities.length) {
  248. mediaKeySystemConfig.audioCapabilities = audioCapabilities;
  249. }
  250. if (videoCapabilities.length) {
  251. mediaKeySystemConfig.videoCapabilities = videoCapabilities;
  252. }
  253. const videoCodec = videoConfig ? videoConfig.contentType : '';
  254. const audioCodec = audioConfig ? audioConfig.contentType : '';
  255. const keySystem = mcapKeySystemConfig.keySystem;
  256. /** @type {MediaKeySystemAccess} */
  257. let keySystemAccess = null;
  258. try {
  259. if (shaka.util.DrmUtils.hasMediaKeySystemAccess(
  260. videoCodec, audioCodec, keySystem)) {
  261. keySystemAccess = shaka.util.DrmUtils.getMediaKeySystemAccess(
  262. videoCodec, audioCodec, keySystem);
  263. } else {
  264. keySystemAccess = await navigator.requestMediaKeySystemAccess(
  265. mcapKeySystemConfig.keySystem, [mediaKeySystemConfig]);
  266. shaka.util.DrmUtils.setMediaKeySystemAccess(
  267. videoCodec, audioCodec, keySystem, keySystemAccess);
  268. }
  269. } catch (e) {
  270. shaka.log.info('navigator.requestMediaKeySystemAccess failed.');
  271. }
  272. return keySystemAccess;
  273. }
  274. /**
  275. * Checks if the given media parameters of the video or audio streams are
  276. * supported by the Cast platform.
  277. * @param {!VideoConfiguration} videoConfig The 'video' field of the
  278. * MediaDecodingConfiguration.
  279. * @return {!Promise<boolean>} `true` when the stream can be displayed on a
  280. * Cast device.
  281. * @private
  282. */
  283. static async canCastDisplayType_(videoConfig) {
  284. if (!(window.cast &&
  285. cast.__platform__ && cast.__platform__.canDisplayType)) {
  286. shaka.log.warning('Expected cast APIs to be available! Falling back to ' +
  287. 'shaka.media.Capabilities.isTypeSupported() for type support.');
  288. return shaka.media.Capabilities.isTypeSupported(videoConfig.contentType);
  289. }
  290. let displayType = videoConfig.contentType;
  291. if (videoConfig.width && videoConfig.height) {
  292. displayType +=
  293. `; width=${videoConfig.width}; height=${videoConfig.height}`;
  294. }
  295. if (videoConfig.framerate) {
  296. displayType += `; framerate=${videoConfig.framerate}`;
  297. }
  298. if (videoConfig.transferFunction === 'pq') {
  299. // A "PQ" transfer function indicates this is an HDR-capable stream;
  300. // "smpte2084" is the published standard. We need to inform the platform
  301. // this query is specifically for HDR.
  302. displayType += '; eotf=smpte2084';
  303. }
  304. let result = false;
  305. if (displayType in shaka.polyfill.MediaCapabilities
  306. .memoizedCanDisplayTypeRequests_) {
  307. result = shaka.polyfill.MediaCapabilities
  308. .memoizedCanDisplayTypeRequests_[displayType];
  309. } else {
  310. result = await cast.__platform__.canDisplayType(displayType);
  311. shaka.polyfill.MediaCapabilities
  312. .memoizedCanDisplayTypeRequests_[displayType] = result;
  313. }
  314. return result;
  315. }
  316. };
  317. /**
  318. * A copy of the MediaCapabilities instance, to prevent Safari from
  319. * garbage-collecting the polyfilled method on it. We make it public and export
  320. * it to ensure that it is not stripped out by the compiler.
  321. *
  322. * @type {MediaCapabilities}
  323. * @export
  324. */
  325. shaka.polyfill.MediaCapabilities.originalMcap = null;
  326. /**
  327. * A cache that stores the canDisplayType result of calling
  328. * `cast.__platform__.canDisplayType`.
  329. *
  330. * @type {(Object<(!string), (!boolean)>)}
  331. * @export
  332. */
  333. shaka.polyfill.MediaCapabilities.memoizedCanDisplayTypeRequests_ = {};
  334. // Install at a lower priority than MediaSource polyfill, so that we have
  335. // MediaSource available first.
  336. shaka.polyfill.register(shaka.polyfill.MediaCapabilities.install, -1);