/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim:set ts=2 sw=2 sts=2 et cindent: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "WebMDecoder.h" #include #include "mozilla/Preferences.h" #include "mozilla/StaticPrefs_media.h" #ifdef MOZ_AV1 # include "AOMDecoder.h" #endif #include "MediaContainerType.h" #include "PDMFactory.h" #include "PlatformDecoderModule.h" #include "VideoUtils.h" namespace mozilla { /* static */ nsTArray> WebMDecoder::GetTracksInfo( const MediaContainerType& aType, MediaResult& aError) { nsTArray> tracks; const bool isVideo = aType.Type() == MEDIAMIMETYPE("video/webm"); if (aType.Type() != MEDIAMIMETYPE("audio/webm") && !isVideo) { aError = MediaResult( NS_ERROR_DOM_MEDIA_FATAL_ERR, RESULT_DETAIL("Invalid type:%s", aType.Type().AsString().get())); return tracks; } aError = NS_OK; const MediaCodecs& codecs = aType.ExtendedType().Codecs(); if (codecs.IsEmpty()) { return tracks; } for (const auto& codec : codecs.Range()) { if (codec.EqualsLiteral("opus") || codec.EqualsLiteral("vorbis")) { tracks.AppendElement( CreateTrackInfoWithMIMETypeAndContainerTypeExtraParameters( "audio/"_ns + NS_ConvertUTF16toUTF8(codec), aType)); continue; } if (isVideo) { UniquePtr trackInfo; if (IsVP9CodecString(codec)) { trackInfo = CreateTrackInfoWithMIMETypeAndContainerTypeExtraParameters( "video/vp9"_ns, aType); } else if (IsVP8CodecString(codec)) { trackInfo = CreateTrackInfoWithMIMETypeAndContainerTypeExtraParameters( "video/vp8"_ns, aType); } if (trackInfo) { uint8_t profile = 0; uint8_t level = 0; uint8_t bitDepth = 0; if (ExtractVPXCodecDetails(codec, profile, level, bitDepth)) { trackInfo->GetAsVideoInfo()->mColorDepth = gfx::ColorDepthForBitDepth(bitDepth); } tracks.AppendElement(std::move(trackInfo)); continue; } } #ifdef MOZ_AV1 if (StaticPrefs::media_av1_enabled() && IsAV1CodecString(codec)) { tracks.AppendElement( CreateTrackInfoWithMIMETypeAndContainerTypeExtraParameters( "video/av1"_ns, aType)); continue; } #endif // Unknown codec aError = MediaResult( NS_ERROR_DOM_MEDIA_FATAL_ERR, RESULT_DETAIL("Unknown codec:%s", NS_ConvertUTF16toUTF8(codec).get())); } return tracks; } /* static */ bool WebMDecoder::IsSupportedType(const MediaContainerType& aContainerType) { if (!StaticPrefs::media_webm_enabled()) { return false; } MediaResult rv = NS_OK; auto tracks = GetTracksInfo(aContainerType, rv); if (NS_FAILED(rv)) { return false; } if (tracks.IsEmpty()) { // WebM guarantees that the only codecs it contained are vp8, vp9, opus or // vorbis. return true; } // Verify that we have a PDM that supports the whitelisted types, include // color depth RefPtr platform = new PDMFactory(); for (const auto& track : tracks) { if (!track || !platform->Supports(SupportDecoderParams(*track), nullptr /* diagnostic */)) { return false; } } return true; } /* static */ nsTArray> WebMDecoder::GetTracksInfo( const MediaContainerType& aType) { MediaResult rv = NS_OK; return GetTracksInfo(aType, rv); } } // namespace mozilla