/* -*- 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 "DecoderTraits.h" #include "MediaDecoder.h" #include "nsCharSeparatedTokenizer.h" #include "nsMimeTypes.h" #include "mozilla/Preferences.h" #ifdef MOZ_ANDROID_OMX #include "AndroidMediaPluginHost.h" #endif #include "OggDecoder.h" #include "OggReader.h" #ifdef MOZ_WAVE #include "WaveDecoder.h" #include "WaveReader.h" #endif #ifdef MOZ_WEBM #include "WebMDecoder.h" #include "WebMReader.h" #endif #ifdef MOZ_RAW #include "RawDecoder.h" #include "RawReader.h" #endif #ifdef MOZ_GSTREAMER #include "GStreamerDecoder.h" #include "GStreamerReader.h" #endif #ifdef MOZ_ANDROID_OMX #include "AndroidMediaPluginHost.h" #include "AndroidMediaDecoder.h" #include "AndroidMediaReader.h" #include "AndroidMediaPluginHost.h" #endif #ifdef MOZ_OMX_DECODER #include "MediaOmxDecoder.h" #include "MediaOmxReader.h" #include "nsIPrincipal.h" #include "mozilla/dom/HTMLMediaElement.h" #if ANDROID_VERSION >= 18 #include "MediaCodecDecoder.h" #include "MediaCodecReader.h" #endif #endif #ifdef NECKO_PROTOCOL_rtsp #if ANDROID_VERSION >= 18 #include "RtspMediaCodecDecoder.h" #include "RtspMediaCodecReader.h" #endif #include "RtspOmxDecoder.h" #include "RtspOmxReader.h" #endif #ifdef MOZ_WMF #include "WMFDecoder.h" #include "WMFReader.h" #endif #ifdef MOZ_DIRECTSHOW #include "DirectShowDecoder.h" #include "DirectShowReader.h" #endif #ifdef MOZ_APPLEMEDIA #include "AppleDecoder.h" #include "AppleMP3Reader.h" #endif #ifdef MOZ_FMP4 #include "MP4Reader.h" #include "MP4Decoder.h" #endif namespace mozilla { template static bool CodecListContains(char const *const * aCodecs, const String& aCodec) { for (int32_t i = 0; aCodecs[i]; ++i) { if (aCodec.EqualsASCII(aCodecs[i])) return true; } return false; } #ifdef MOZ_RAW static const char* gRawTypes[3] = { "video/x-raw", "video/x-raw-yuv", nullptr }; static const char* gRawCodecs[1] = { nullptr }; static bool IsRawType(const nsACString& aType) { if (!MediaDecoder::IsRawEnabled()) { return false; } return CodecListContains(gRawTypes, aType); } #endif // See http://www.rfc-editor.org/rfc/rfc5334.txt for the definitions // of Ogg media types and codec types static const char* const gOggTypes[4] = { "video/ogg", "audio/ogg", "application/ogg", nullptr }; static char const *const gOggCodecs[3] = { "vorbis", "theora", nullptr }; static char const *const gOggCodecsWithOpus[4] = { "vorbis", "opus", "theora", nullptr }; static bool IsOggType(const nsACString& aType) { if (!MediaDecoder::IsOggEnabled()) { return false; } return CodecListContains(gOggTypes, aType); } #ifdef MOZ_WAVE // See http://www.rfc-editor.org/rfc/rfc2361.txt for the definitions // of WAVE media types and codec types. However, the audio/vnd.wave // MIME type described there is not used. static const char* const gWaveTypes[5] = { "audio/x-wav", "audio/wav", "audio/wave", "audio/x-pn-wav", nullptr }; static char const *const gWaveCodecs[2] = { "1", // Microsoft PCM Format nullptr }; static bool IsWaveType(const nsACString& aType) { if (!MediaDecoder::IsWaveEnabled()) { return false; } return CodecListContains(gWaveTypes, aType); } #endif #ifdef MOZ_WEBM static const char* const gWebMTypes[3] = { "video/webm", "audio/webm", nullptr }; static char const *const gWebMCodecs[7] = { "vp8", "vp8.0", "vp9", "vp9.0", "vorbis", "opus", nullptr }; static bool IsWebMType(const nsACString& aType) { if (!MediaDecoder::IsWebMEnabled()) { return false; } return CodecListContains(gWebMTypes, aType); } #endif #ifdef MOZ_GSTREAMER static bool IsGStreamerSupportedType(const nsACString& aMimeType) { if (!MediaDecoder::IsGStreamerEnabled()) return false; #ifdef MOZ_WEBM if (IsWebMType(aMimeType) && !Preferences::GetBool("media.prefer-gstreamer", false)) return false; #endif if (IsOggType(aMimeType) && !Preferences::GetBool("media.prefer-gstreamer", false)) return false; return GStreamerDecoder::CanHandleMediaType(aMimeType, nullptr); } #endif #ifdef MOZ_OMX_DECODER static const char* const gOmxTypes[] = { "audio/mpeg", "audio/mp4", "audio/amr", "audio/3gpp", "audio/flac", "video/mp4", "video/3gpp", "video/3gpp2", "video/quicktime", #ifdef MOZ_OMX_WEBM_DECODER "video/webm", "audio/webm", #endif "audio/x-matroska", "video/mp2t", "video/avi", "video/x-matroska", nullptr }; static const char* const gB2GOnlyTypes[] = { "audio/3gpp", "audio/amr", "audio/x-matroska", "video/mp2t", "video/avi", "video/x-matroska", nullptr }; static bool IsOmxSupportedType(const nsACString& aType) { if (!MediaDecoder::IsOmxEnabled()) { return false; } return CodecListContains(gOmxTypes, aType); } static bool IsB2GSupportOnlyType(const nsACString& aType) { return CodecListContains(gB2GOnlyTypes, aType); } static char const *const gH264Codecs[9] = { "avc1.42E01E", // H.264 Constrained Baseline Profile Level 3.0 "avc1.42001E", // H.264 Baseline Profile Level 3.0 "avc1.58A01E", // H.264 Extended Profile Level 3.0 "avc1.4D401E", // H.264 Main Profile Level 3.0 "avc1.64001E", // H.264 High Profile Level 3.0 "avc1.64001F", // H.264 High Profile Level 3.1 "mp4v.20.3", // 3GPP "mp4a.40.2", // AAC-LC nullptr }; static char const *const gMpegAudioCodecs[2] = { "mp3", // MP3 nullptr }; #ifdef MOZ_OMX_WEBM_DECODER static char const *const gOMXWebMCodecs[4] = { "vorbis", "vp8", "vp8.0", nullptr }; #endif //MOZ_OMX_WEBM_DECODER #endif #ifdef NECKO_PROTOCOL_rtsp static const char* const gRtspTypes[2] = { "RTSP", nullptr }; static bool IsRtspSupportedType(const nsACString& aMimeType) { return MediaDecoder::IsRtspEnabled() && CodecListContains(gRtspTypes, aMimeType); } #endif /* static */ bool DecoderTraits::DecoderWaitsForOnConnected(const nsACString& aMimeType) { #ifdef NECKO_PROTOCOL_rtsp return CodecListContains(gRtspTypes, aMimeType); #else return false; #endif } #ifdef MOZ_ANDROID_OMX static bool IsAndroidMediaType(const nsACString& aType) { if (!MediaDecoder::IsAndroidMediaEnabled()) { return false; } static const char* supportedTypes[] = { "audio/mpeg", "audio/mp4", "video/mp4", nullptr }; return CodecListContains(supportedTypes, aType); } #endif #ifdef MOZ_WMF static bool IsWMFSupportedType(const nsACString& aType) { return WMFDecoder::CanPlayType(aType, NS_LITERAL_STRING("")); } #endif #ifdef MOZ_DIRECTSHOW static bool IsDirectShowSupportedType(const nsACString& aType) { return DirectShowDecoder::GetSupportedCodecs(aType, nullptr); } #endif #ifdef MOZ_FMP4 static bool IsMP4SupportedType(const nsACString& aType, const nsAString& aCodecs = EmptyString()) { // MP4Decoder/Reader is currently used for MSE and mp4 files local playback. bool haveAAC, haveMP3, haveH264; return Preferences::GetBool("media.fragmented-mp4.exposed", false) && MP4Decoder::CanHandleMediaType(aType, aCodecs, haveAAC, haveH264, haveMP3); } #endif #ifdef MOZ_APPLEMEDIA static const char * const gAppleMP3Types[] = { "audio/mp3", "audio/mpeg", nullptr, }; static const char * const gAppleMP3Codecs[] = { "mp3", nullptr }; static bool IsAppleMediaSupportedType(const nsACString& aType, const char * const ** aCodecs = nullptr) { if (MediaDecoder::IsAppleMP3Enabled() && CodecListContains(gAppleMP3Types, aType)) { if (aCodecs) { *aCodecs = gAppleMP3Codecs; } return true; } // TODO MP4 return false; } #endif /* static */ bool DecoderTraits::ShouldHandleMediaType(const char* aMIMEType) { #ifdef MOZ_WAVE if (IsWaveType(nsDependentCString(aMIMEType))) { // We should not return true for Wave types, since there are some // Wave codecs actually in use in the wild that we don't support, and // we should allow those to be handled by plugins or helper apps. // Furthermore people can play Wave files on most platforms by other // means. return false; } #endif return CanHandleMediaType(aMIMEType, false, EmptyString()) != CANPLAY_NO; } /* static */ CanPlayStatus DecoderTraits::CanHandleMediaType(const char* aMIMEType, bool aHaveRequestedCodecs, const nsAString& aRequestedCodecs) { MOZ_ASSERT(NS_IsMainThread()); char const* const* codecList = nullptr; CanPlayStatus result = CANPLAY_NO; #ifdef MOZ_RAW if (IsRawType(nsDependentCString(aMIMEType))) { codecList = gRawCodecs; result = CANPLAY_MAYBE; } #endif if (IsOggType(nsDependentCString(aMIMEType))) { codecList = MediaDecoder::IsOpusEnabled() ? gOggCodecsWithOpus : gOggCodecs; result = CANPLAY_MAYBE; } #ifdef MOZ_WAVE if (IsWaveType(nsDependentCString(aMIMEType))) { codecList = gWaveCodecs; result = CANPLAY_MAYBE; } #endif #if defined(MOZ_WEBM) && !defined(MOZ_OMX_WEBM_DECODER) if (IsWebMType(nsDependentCString(aMIMEType))) { codecList = gWebMCodecs; result = CANPLAY_MAYBE; } #endif #ifdef MOZ_FMP4 if (IsMP4SupportedType(nsDependentCString(aMIMEType), aRequestedCodecs)) { return aHaveRequestedCodecs ? CANPLAY_YES : CANPLAY_MAYBE; } #endif #ifdef MOZ_GSTREAMER if (GStreamerDecoder::CanHandleMediaType(nsDependentCString(aMIMEType), aHaveRequestedCodecs ? &aRequestedCodecs : nullptr)) { if (aHaveRequestedCodecs) return CANPLAY_YES; return CANPLAY_MAYBE; } #endif #ifdef MOZ_OMX_DECODER if (IsOmxSupportedType(nsDependentCString(aMIMEType))) { result = CANPLAY_MAYBE; if (nsDependentCString(aMIMEType).EqualsASCII("audio/mpeg")) { codecList = gMpegAudioCodecs; #ifdef MOZ_OMX_WEBM_DECODER } else if (nsDependentCString(aMIMEType).EqualsASCII("audio/webm") || nsDependentCString(aMIMEType).EqualsASCII("video/webm")) { codecList = gOMXWebMCodecs; #endif } else { codecList = gH264Codecs; } } #endif #ifdef MOZ_DIRECTSHOW // Note: DirectShow should come before WMF, so that we prefer DirectShow's // MP3 support over WMF's. if (DirectShowDecoder::GetSupportedCodecs(nsDependentCString(aMIMEType), &codecList)) { result = CANPLAY_MAYBE; } #endif #ifdef MOZ_WMF if (!Preferences::GetBool("media.fragmented-mp4.exposed", false) && IsWMFSupportedType(nsDependentCString(aMIMEType))) { if (!aHaveRequestedCodecs) { return CANPLAY_MAYBE; } return WMFDecoder::CanPlayType(nsDependentCString(aMIMEType), aRequestedCodecs) ? CANPLAY_YES : CANPLAY_NO; } #endif #ifdef MOZ_APPLEMEDIA if (IsAppleMediaSupportedType(nsDependentCString(aMIMEType), &codecList)) { result = CANPLAY_MAYBE; } #endif #ifdef MOZ_ANDROID_OMX if (MediaDecoder::IsAndroidMediaEnabled() && EnsureAndroidMediaPluginHost()->FindDecoder(nsDependentCString(aMIMEType), &codecList)) result = CANPLAY_MAYBE; #endif #ifdef NECKO_PROTOCOL_rtsp if (IsRtspSupportedType(nsDependentCString(aMIMEType))) { result = CANPLAY_MAYBE; } #endif if (result == CANPLAY_NO || !aHaveRequestedCodecs || !codecList) { return result; } // See http://www.rfc-editor.org/rfc/rfc4281.txt for the description // of the 'codecs' parameter nsCharSeparatedTokenizer tokenizer(aRequestedCodecs, ','); bool expectMoreTokens = false; while (tokenizer.hasMoreTokens()) { const nsSubstring& token = tokenizer.nextToken(); if (!CodecListContains(codecList, token)) { // Totally unsupported codec return CANPLAY_NO; } expectMoreTokens = tokenizer.separatorAfterCurrentToken(); } if (expectMoreTokens) { // Last codec name was empty return CANPLAY_NO; } return CANPLAY_YES; } // Instantiates but does not initialize decoder. static already_AddRefed InstantiateDecoder(const nsACString& aType, MediaDecoderOwner* aOwner) { MOZ_ASSERT(NS_IsMainThread()); nsRefPtr decoder; #ifdef MOZ_FMP4 if (IsMP4SupportedType(aType)) { decoder = new MP4Decoder(); return decoder.forget(); } #endif #ifdef MOZ_GSTREAMER if (IsGStreamerSupportedType(aType)) { decoder = new GStreamerDecoder(); return decoder.forget(); } #endif #ifdef MOZ_RAW if (IsRawType(aType)) { decoder = new RawDecoder(); return decoder.forget(); } #endif if (IsOggType(aType)) { decoder = new OggDecoder(); return decoder.forget(); } #ifdef MOZ_WAVE if (IsWaveType(aType)) { decoder = new WaveDecoder(); return decoder.forget(); } #endif #ifdef MOZ_OMX_DECODER if (IsOmxSupportedType(aType)) { // we are discouraging Web and App developers from using those formats in // gB2GOnlyTypes, thus we only allow them to be played on WebApps. if (IsB2GSupportOnlyType(aType)) { dom::HTMLMediaElement* element = aOwner->GetMediaElement(); if (!element) { return nullptr; } nsIPrincipal* principal = element->NodePrincipal(); if (!principal) { return nullptr; } if (principal->GetAppStatus() < nsIPrincipal::APP_STATUS_PRIVILEGED) { return nullptr; } } #if ANDROID_VERSION >= 18 decoder = MediaDecoder::IsOmxAsyncEnabled() ? static_cast(new MediaCodecDecoder()) : static_cast(new MediaOmxDecoder()); #else decoder = new MediaOmxDecoder(); #endif return decoder.forget(); } #endif #ifdef NECKO_PROTOCOL_rtsp if (IsRtspSupportedType(aType)) { #if ANDROID_VERSION >= 18 decoder = MediaDecoder::IsOmxAsyncEnabled() ? static_cast(new RtspMediaCodecDecoder()) : static_cast(new RtspOmxDecoder()); #else decoder = new RtspOmxDecoder(); #endif return decoder.forget(); } #endif #ifdef MOZ_ANDROID_OMX if (MediaDecoder::IsAndroidMediaEnabled() && EnsureAndroidMediaPluginHost()->FindDecoder(aType, nullptr)) { decoder = new AndroidMediaDecoder(aType); return decoder.forget(); } #endif #ifdef MOZ_WEBM if (IsWebMType(aType)) { decoder = new WebMDecoder(); return decoder.forget(); } #endif #ifdef MOZ_DIRECTSHOW // Note: DirectShow should come before WMF, so that we prefer DirectShow's // MP3 support over WMF's. if (IsDirectShowSupportedType(aType)) { decoder = new DirectShowDecoder(); return decoder.forget(); } #endif #ifdef MOZ_WMF if (IsWMFSupportedType(aType)) { decoder = new WMFDecoder(); return decoder.forget(); } #endif #ifdef MOZ_APPLEMEDIA if (IsAppleMediaSupportedType(aType)) { decoder = new AppleDecoder(); return decoder.forget(); } #endif NS_ENSURE_TRUE(decoder != nullptr, nullptr); NS_ENSURE_TRUE(decoder->Init(aOwner), nullptr); return nullptr; } /* static */ already_AddRefed DecoderTraits::CreateDecoder(const nsACString& aType, MediaDecoderOwner* aOwner) { MOZ_ASSERT(NS_IsMainThread()); nsRefPtr decoder(InstantiateDecoder(aType, aOwner)); NS_ENSURE_TRUE(decoder != nullptr, nullptr); NS_ENSURE_TRUE(decoder->Init(aOwner), nullptr); return decoder.forget(); } /* static */ MediaDecoderReader* DecoderTraits::CreateReader(const nsACString& aType, AbstractMediaDecoder* aDecoder) { MOZ_ASSERT(NS_IsMainThread()); MediaDecoderReader* decoderReader = nullptr; if (!aDecoder) { return decoderReader; } #ifdef MOZ_FMP4 if (IsMP4SupportedType(aType)) { decoderReader = new MP4Reader(aDecoder); } else #endif #ifdef MOZ_GSTREAMER if (IsGStreamerSupportedType(aType)) { decoderReader = new GStreamerReader(aDecoder); } else #endif #ifdef MOZ_RAW if (IsRawType(aType)) { decoderReader = new RawReader(aDecoder); } else #endif if (IsOggType(aType)) { decoderReader = new OggReader(aDecoder); } else #ifdef MOZ_WAVE if (IsWaveType(aType)) { decoderReader = new WaveReader(aDecoder); } else #endif #ifdef MOZ_OMX_DECODER if (IsOmxSupportedType(aType)) { #if ANDROID_VERSION >= 18 decoderReader = MediaDecoder::IsOmxAsyncEnabled() ? static_cast(new MediaCodecReader(aDecoder)) : static_cast(new MediaOmxReader(aDecoder)); #else decoderReader = new MediaOmxReader(aDecoder); #endif } else #endif #ifdef MOZ_ANDROID_OMX if (MediaDecoder::IsAndroidMediaEnabled() && EnsureAndroidMediaPluginHost()->FindDecoder(aType, nullptr)) { decoderReader = new AndroidMediaReader(aDecoder, aType); } else #endif #ifdef MOZ_WEBM if (IsWebMType(aType)) { decoderReader = new WebMReader(aDecoder); } else #endif #ifdef MOZ_DIRECTSHOW // Note: DirectShowReader is preferred for MP3, but if it's disabled we // fallback to the WMFReader. if (IsDirectShowSupportedType(aType)) { decoderReader = new DirectShowReader(aDecoder); } else #endif #ifdef MOZ_WMF if (IsWMFSupportedType(aType)) { decoderReader = new WMFReader(aDecoder); } else #endif #ifdef MOZ_APPLEMEDIA if (IsAppleMediaSupportedType(aType)) { decoderReader = new AppleMP3Reader(aDecoder); } else #endif if (false) {} // dummy if to take care of the dangling else return decoderReader; } /* static */ bool DecoderTraits::IsSupportedInVideoDocument(const nsACString& aType) { // Forbid playing media in video documents if the user has opted // not to, using either the legacy WMF specific pref, or the newer // catch-all pref. if (!Preferences::GetBool("media.windows-media-foundation.play-stand-alone", true) || !Preferences::GetBool("media.play-stand-alone", true)) { return false; } return IsOggType(aType) || #ifdef MOZ_OMX_DECODER // We support the formats in gB2GOnlyTypes only inside WebApps on firefoxOS // but not in general web content. Ensure we dont create a VideoDocument // when accessing those format URLs directly. (IsOmxSupportedType(aType) && !IsB2GSupportOnlyType(aType)) || #endif #ifdef MOZ_WEBM IsWebMType(aType) || #endif #ifdef MOZ_GSTREAMER IsGStreamerSupportedType(aType) || #endif #ifdef MOZ_ANDROID_OMX (MediaDecoder::IsAndroidMediaEnabled() && IsAndroidMediaType(aType)) || #endif #ifdef MOZ_FMP4 IsMP4SupportedType(aType) || #endif #ifdef MOZ_WMF IsWMFSupportedType(aType) || #endif #ifdef MOZ_DIRECTSHOW IsDirectShowSupportedType(aType) || #endif #ifdef MOZ_APPLEMEDIA IsAppleMediaSupportedType(aType) || #endif #ifdef NECKO_PROTOCOL_rtsp IsRtspSupportedType(aType) || #endif false; } }