diff options
Diffstat (limited to 'dom/media/platforms/PDMFactory.cpp')
-rw-r--r-- | dom/media/platforms/PDMFactory.cpp | 482 |
1 files changed, 482 insertions, 0 deletions
diff --git a/dom/media/platforms/PDMFactory.cpp b/dom/media/platforms/PDMFactory.cpp new file mode 100644 index 000000000..a72d910f5 --- /dev/null +++ b/dom/media/platforms/PDMFactory.cpp @@ -0,0 +1,482 @@ +/* -*- 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 "PDMFactory.h" + +#ifdef XP_WIN +#include "WMFDecoderModule.h" + +#endif +#ifdef MOZ_FFVPX +#include "FFVPXRuntimeLinker.h" +#endif +#ifdef MOZ_FFMPEG +#include "FFmpegRuntimeLinker.h" +#endif +#ifdef MOZ_APPLEMEDIA +#include "AppleDecoderModule.h" +#endif +#ifdef MOZ_GONK_MEDIACODEC +#include "GonkDecoderModule.h" +#endif +#ifdef MOZ_WIDGET_ANDROID +#include "AndroidDecoderModule.h" +#endif +#include "GMPDecoderModule.h" + +#include "mozilla/CDMProxy.h" +#include "mozilla/ClearOnShutdown.h" +#include "mozilla/SharedThreadPool.h" +#include "mozilla/StaticPtr.h" +#include "mozilla/SyncRunnable.h" +#include "mozilla/TaskQueue.h" + +#include "MediaInfo.h" +#include "MediaPrefs.h" +#include "FuzzingWrapper.h" +#include "H264Converter.h" + +#include "AgnosticDecoderModule.h" +#include "EMEDecoderModule.h" + +#include "DecoderDoctorDiagnostics.h" + +#include "MP4Decoder.h" +#include "mozilla/dom/RemoteVideoDecoder.h" + +#ifdef XP_WIN +#include "mozilla/WindowsVersion.h" +#endif + +#include "mp4_demuxer/H264.h" + +namespace mozilla { + +extern already_AddRefed<PlatformDecoderModule> CreateAgnosticDecoderModule(); +extern already_AddRefed<PlatformDecoderModule> CreateBlankDecoderModule(); + +class PDMFactoryImpl final { +public: + PDMFactoryImpl() + { +#ifdef XP_WIN + WMFDecoderModule::Init(); +#endif +#ifdef MOZ_APPLEMEDIA + AppleDecoderModule::Init(); +#endif +#ifdef MOZ_FFVPX + FFVPXRuntimeLinker::Init(); +#endif +#ifdef MOZ_FFMPEG + FFmpegRuntimeLinker::Init(); +#endif + } +}; + +StaticAutoPtr<PDMFactoryImpl> PDMFactory::sInstance; +StaticMutex PDMFactory::sMonitor; + +class SupportChecker +{ +public: + enum class Reason : uint8_t + { + kSupported, + kVideoFormatNotSupported, + kAudioFormatNotSupported, + kUnknown, + }; + + struct CheckResult + { + explicit CheckResult(Reason aReason, + MediaResult aResult = MediaResult(NS_OK)) + : mReason(aReason), + mMediaResult(mozilla::Move(aResult)) + {} + CheckResult(const CheckResult& aOther) = default; + CheckResult(CheckResult&& aOther) = default; + CheckResult& operator=(const CheckResult& aOther) = default; + CheckResult& operator=(CheckResult&& aOther) = default; + + Reason mReason; + MediaResult mMediaResult; + }; + + template<class Func> + void + AddToCheckList(Func&& aChecker) + { + mCheckerList.AppendElement(mozilla::Forward<Func>(aChecker)); + } + + void + AddMediaFormatChecker(const TrackInfo& aTrackConfig) + { + if (aTrackConfig.IsVideo()) { + auto mimeType = aTrackConfig.GetAsVideoInfo()->mMimeType; + RefPtr<MediaByteBuffer> extraData = aTrackConfig.GetAsVideoInfo()->mExtraData; + AddToCheckList( + [mimeType, extraData]() { + if (MP4Decoder::IsH264(mimeType)) { + mp4_demuxer::SPSData spsdata; + // WMF H.264 Video Decoder and Apple ATDecoder + // do not support YUV444 format. + // For consistency, all decoders should be checked. + if (mp4_demuxer::H264::DecodeSPSFromExtraData(extraData, spsdata) && + (spsdata.profile_idc == 244 /* Hi444PP */ || + spsdata.chroma_format_idc == PDMFactory::kYUV444)) { + return CheckResult( + SupportChecker::Reason::kVideoFormatNotSupported, + MediaResult( + NS_ERROR_DOM_MEDIA_FATAL_ERR, + RESULT_DETAIL("Decoder may not have the capability to handle" + " the requested video format" + " with YUV444 chroma subsampling."))); + } + } + return CheckResult(SupportChecker::Reason::kSupported); + }); + } + } + + SupportChecker::CheckResult + Check() + { + for (auto& checker : mCheckerList) { + auto result = checker(); + if (result.mReason != SupportChecker::Reason::kSupported) { + return result; + } + } + return CheckResult(SupportChecker::Reason::kSupported); + } + + void Clear() { mCheckerList.Clear(); } + +private: + nsTArray<mozilla::function<CheckResult()>> mCheckerList; +}; // SupportChecker + +PDMFactory::PDMFactory() +{ + EnsureInit(); + CreatePDMs(); + CreateBlankPDM(); +} + +PDMFactory::~PDMFactory() +{ +} + +void +PDMFactory::EnsureInit() const +{ + { + StaticMutexAutoLock mon(sMonitor); + if (sInstance) { + // Quick exit if we already have an instance. + return; + } + if (NS_IsMainThread()) { + // On the main thread and holding the lock -> Create instance. + sInstance = new PDMFactoryImpl(); + ClearOnShutdown(&sInstance); + return; + } + } + + // Not on the main thread -> Sync-dispatch creation to main thread. + nsCOMPtr<nsIThread> mainThread = do_GetMainThread(); + nsCOMPtr<nsIRunnable> runnable = + NS_NewRunnableFunction([]() { + StaticMutexAutoLock mon(sMonitor); + if (!sInstance) { + sInstance = new PDMFactoryImpl(); + ClearOnShutdown(&sInstance); + } + }); + SyncRunnable::DispatchToThread(mainThread, runnable); +} + +already_AddRefed<MediaDataDecoder> +PDMFactory::CreateDecoder(const CreateDecoderParams& aParams) +{ + if (aParams.mUseBlankDecoder) { + MOZ_ASSERT(mBlankPDM); + return CreateDecoderWithPDM(mBlankPDM, aParams); + } + + const TrackInfo& config = aParams.mConfig; + bool isEncrypted = mEMEPDM && config.mCrypto.mValid; + + if (isEncrypted) { + return CreateDecoderWithPDM(mEMEPDM, aParams); + } + + DecoderDoctorDiagnostics* diagnostics = aParams.mDiagnostics; + if (diagnostics) { + // If libraries failed to load, the following loop over mCurrentPDMs + // will not even try to use them. So we record failures now. + if (mWMFFailedToLoad) { + diagnostics->SetWMFFailedToLoad(); + } + if (mFFmpegFailedToLoad) { + diagnostics->SetFFmpegFailedToLoad(); + } + if (mGMPPDMFailedToStartup) { + diagnostics->SetGMPPDMFailedToStartup(); + } + } + + for (auto& current : mCurrentPDMs) { + if (!current->SupportsMimeType(config.mMimeType, diagnostics)) { + continue; + } + RefPtr<MediaDataDecoder> m = CreateDecoderWithPDM(current, aParams); + if (m) { + return m.forget(); + } + } + NS_WARNING("Unable to create a decoder, no platform found."); + return nullptr; +} + +already_AddRefed<MediaDataDecoder> +PDMFactory::CreateDecoderWithPDM(PlatformDecoderModule* aPDM, + const CreateDecoderParams& aParams) +{ + MOZ_ASSERT(aPDM); + RefPtr<MediaDataDecoder> m; + MediaResult* result = aParams.mError; + + SupportChecker supportChecker; + const TrackInfo& config = aParams.mConfig; + supportChecker.AddMediaFormatChecker(config); + + auto checkResult = supportChecker.Check(); + if (checkResult.mReason != SupportChecker::Reason::kSupported) { + DecoderDoctorDiagnostics* diagnostics = aParams.mDiagnostics; + if (checkResult.mReason == SupportChecker::Reason::kVideoFormatNotSupported) { + if (diagnostics) { + diagnostics->SetVideoNotSupported(); + } + if (result) { + *result = checkResult.mMediaResult; + } + } else if (checkResult.mReason == SupportChecker::Reason::kAudioFormatNotSupported) { + if (diagnostics) { + diagnostics->SetAudioNotSupported(); + } + if (result) { + *result = checkResult.mMediaResult; + } + } + return nullptr; + } + + if (config.IsAudio()) { + m = aPDM->CreateAudioDecoder(aParams); + return m.forget(); + } + + if (!config.IsVideo()) { + *result = MediaResult(NS_ERROR_DOM_MEDIA_FATAL_ERR, + RESULT_DETAIL("Decoder configuration error, expected audio or video.")); + return nullptr; + } + + MediaDataDecoderCallback* callback = aParams.mCallback; + RefPtr<DecoderCallbackFuzzingWrapper> callbackWrapper; + if (MediaPrefs::PDMFuzzingEnabled()) { + callbackWrapper = new DecoderCallbackFuzzingWrapper(callback); + callbackWrapper->SetVideoOutputMinimumInterval( + TimeDuration::FromMilliseconds(MediaPrefs::PDMFuzzingInterval())); + callbackWrapper->SetDontDelayInputExhausted(!MediaPrefs::PDMFuzzingDelayInputExhausted()); + callback = callbackWrapper.get(); + } + + CreateDecoderParams params = aParams; + params.mCallback = callback; + + if (MP4Decoder::IsH264(config.mMimeType) && !aParams.mUseBlankDecoder) { + RefPtr<H264Converter> h = new H264Converter(aPDM, params); + const nsresult rv = h->GetLastError(); + if (NS_SUCCEEDED(rv) || rv == NS_ERROR_NOT_INITIALIZED) { + // The H264Converter either successfully created the wrapped decoder, + // or there wasn't enough AVCC data to do so. Otherwise, there was some + // problem, for example WMF DLLs were missing. + m = h.forget(); + } + } else { + m = aPDM->CreateVideoDecoder(params); + } + + if (callbackWrapper && m) { + m = new DecoderFuzzingWrapper(m.forget(), callbackWrapper.forget()); + } + + return m.forget(); +} + +bool +PDMFactory::SupportsMimeType(const nsACString& aMimeType, + DecoderDoctorDiagnostics* aDiagnostics) const +{ + UniquePtr<TrackInfo> trackInfo = CreateTrackInfoWithMIMEType(aMimeType); + if (!trackInfo) { + return false; + } + return Supports(*trackInfo, aDiagnostics); +} + +bool +PDMFactory::Supports(const TrackInfo& aTrackInfo, + DecoderDoctorDiagnostics* aDiagnostics) const +{ + if (mEMEPDM) { + return mEMEPDM->Supports(aTrackInfo, aDiagnostics); + } + RefPtr<PlatformDecoderModule> current = GetDecoder(aTrackInfo, aDiagnostics); + return !!current; +} + +void +PDMFactory::CreatePDMs() +{ + RefPtr<PlatformDecoderModule> m; + + if (MediaPrefs::PDMUseBlankDecoder()) { + m = CreateBlankDecoderModule(); + StartupPDM(m); + // The Blank PDM SupportsMimeType reports true for all codecs; the creation + // of its decoder is infallible. As such it will be used for all media, we + // can stop creating more PDM from this point. + return; + } + +#ifdef MOZ_WIDGET_ANDROID + if(MediaPrefs::PDMAndroidMediaCodecPreferred() && + MediaPrefs::PDMAndroidMediaCodecEnabled()) { + m = new AndroidDecoderModule(); + StartupPDM(m); + } +#endif +#ifdef XP_WIN + if (MediaPrefs::PDMWMFEnabled() && IsVistaOrLater() && !IsWin7AndPre2000Compatible()) { + // *Only* use WMF on Vista and later, as if Firefox is run in Windows 95 + // compatibility mode on Windows 7 (it does happen!) we may crash trying + // to startup WMF. So we need to detect the OS version here, as in + // compatibility mode IsVistaOrLater() and friends behave as if we're on + // the emulated version of Windows. See bug 1279171. + // Additionally, we don't want to start the RemoteDecoderModule if we + // expect it's not going to work (i.e. on Windows older than Vista). + // IsWin7AndPre2000Compatible() uses GetVersionEx as the user specified OS version can + // be reflected when compatibility mode is in effect. + m = new WMFDecoderModule(); + RefPtr<PlatformDecoderModule> remote = new dom::RemoteDecoderModule(m); + StartupPDM(remote); + mWMFFailedToLoad = !StartupPDM(m); + } else { + mWMFFailedToLoad = MediaPrefs::DecoderDoctorWMFDisabledIsFailure(); + } +#endif +#ifdef MOZ_FFVPX + if (MediaPrefs::PDMFFVPXEnabled()) { + m = FFVPXRuntimeLinker::CreateDecoderModule(); + StartupPDM(m); + } +#endif +#ifdef MOZ_FFMPEG + if (MediaPrefs::PDMFFmpegEnabled()) { + m = FFmpegRuntimeLinker::CreateDecoderModule(); + mFFmpegFailedToLoad = !StartupPDM(m); + } else { + mFFmpegFailedToLoad = false; + } +#endif +#ifdef MOZ_APPLEMEDIA + m = new AppleDecoderModule(); + StartupPDM(m); +#endif +#ifdef MOZ_GONK_MEDIACODEC + if (MediaPrefs::PDMGonkDecoderEnabled()) { + m = new GonkDecoderModule(); + StartupPDM(m); + } +#endif +#ifdef MOZ_WIDGET_ANDROID + if(MediaPrefs::PDMAndroidMediaCodecEnabled()){ + m = new AndroidDecoderModule(); + StartupPDM(m); + } +#endif + + m = new AgnosticDecoderModule(); + StartupPDM(m); + + if (MediaPrefs::PDMGMPEnabled()) { + m = new GMPDecoderModule(); + mGMPPDMFailedToStartup = !StartupPDM(m); + } else { + mGMPPDMFailedToStartup = false; + } +} + +void +PDMFactory::CreateBlankPDM() +{ + mBlankPDM = CreateBlankDecoderModule(); + MOZ_ASSERT(mBlankPDM && NS_SUCCEEDED(mBlankPDM->Startup())); +} + +bool +PDMFactory::StartupPDM(PlatformDecoderModule* aPDM) +{ + if (aPDM && NS_SUCCEEDED(aPDM->Startup())) { + mCurrentPDMs.AppendElement(aPDM); + return true; + } + return false; +} + +already_AddRefed<PlatformDecoderModule> +PDMFactory::GetDecoder(const TrackInfo& aTrackInfo, + DecoderDoctorDiagnostics* aDiagnostics) const +{ + if (aDiagnostics) { + // If libraries failed to load, the following loop over mCurrentPDMs + // will not even try to use them. So we record failures now. + if (mWMFFailedToLoad) { + aDiagnostics->SetWMFFailedToLoad(); + } + if (mFFmpegFailedToLoad) { + aDiagnostics->SetFFmpegFailedToLoad(); + } + if (mGMPPDMFailedToStartup) { + aDiagnostics->SetGMPPDMFailedToStartup(); + } + } + + RefPtr<PlatformDecoderModule> pdm; + for (auto& current : mCurrentPDMs) { + if (current->Supports(aTrackInfo, aDiagnostics)) { + pdm = current; + break; + } + } + return pdm.forget(); +} + +void +PDMFactory::SetCDMProxy(CDMProxy* aProxy) +{ + RefPtr<PDMFactory> m = new PDMFactory(); + mEMEPDM = new EMEDecoderModule(aProxy, m); +} + +} // namespace mozilla |