/* -*- 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 "WMF.h" #include "WMFDecoderModule.h" #include "WMFVideoMFTManager.h" #include "WMFAudioMFTManager.h" #include "MFTDecoder.h" #include "mozilla/Preferences.h" #include "mozilla/DebugOnly.h" #include "mozilla/Services.h" #include "WMFMediaDataDecoder.h" #include "nsIWindowsRegKey.h" #include "nsComponentManagerUtils.h" #include "nsServiceManagerUtils.h" #include "nsIGfxInfo.h" #include "GfxDriverInfo.h" #include "gfxWindowsPlatform.h" #include "nsServiceManagerUtils.h" // for do_GetService #include "MediaInfo.h" #include "prsystem.h" namespace mozilla { static bool sDXVAEnabled = false; static int sNumDecoderThreads = -1; static bool sIsIntelDecoderEnabled = false; WMFDecoderModule::WMFDecoderModule() : mWMFInitialized(false) { } WMFDecoderModule::~WMFDecoderModule() { if (mWMFInitialized) { DebugOnly hr = wmf::MFShutdown(); NS_ASSERTION(SUCCEEDED(hr), "MFShutdown failed"); } } void WMFDecoderModule::DisableHardwareAcceleration() { sDXVAEnabled = false; sIsIntelDecoderEnabled = false; } static void SetNumOfDecoderThreads() { MOZ_ASSERT(NS_IsMainThread(), "Preferences can only be read on main thread"); int32_t numCores = PR_GetNumberOfProcessors(); // If we have more than 4 cores, let the decoder decide how many threads. // On an 8 core machine, WMF chooses 4 decoder threads const int WMF_DECODER_DEFAULT = -1; int32_t prefThreadCount = Preferences::GetInt("media.wmf.decoder.thread-count", -1); if (prefThreadCount != WMF_DECODER_DEFAULT) { sNumDecoderThreads = std::max(prefThreadCount, 1); } else if (numCores > 4) { sNumDecoderThreads = WMF_DECODER_DEFAULT; } else { sNumDecoderThreads = std::max(numCores - 1, 1); } } /* static */ void WMFDecoderModule::Init() { MOZ_ASSERT(NS_IsMainThread(), "Must be on main thread."); sDXVAEnabled = gfxPlatform::GetPlatform()->CanUseHardwareVideoDecoding(); sIsIntelDecoderEnabled = Preferences::GetBool("media.webm.intel_decoder.enabled", false); SetNumOfDecoderThreads(); } /* static */ int WMFDecoderModule::GetNumDecoderThreads() { return sNumDecoderThreads; } nsresult WMFDecoderModule::Startup() { mWMFInitialized = SUCCEEDED(wmf::MFStartup()); return mWMFInitialized ? NS_OK : NS_ERROR_FAILURE; } already_AddRefed WMFDecoderModule::CreateVideoDecoder(const VideoInfo& aConfig, layers::LayersBackend aLayersBackend, layers::ImageContainer* aImageContainer, FlushableTaskQueue* aVideoTaskQueue, MediaDataDecoderCallback* aCallback) { nsAutoPtr manager = new WMFVideoMFTManager(aConfig, aLayersBackend, aImageContainer, sDXVAEnabled && ShouldUseDXVA(aConfig)); nsRefPtr mft = manager->Init(); if (!mft) { return nullptr; } nsRefPtr decoder = new WMFMediaDataDecoder(manager.forget(), mft, aVideoTaskQueue, aCallback); return decoder.forget(); } already_AddRefed WMFDecoderModule::CreateAudioDecoder(const AudioInfo& aConfig, FlushableTaskQueue* aAudioTaskQueue, MediaDataDecoderCallback* aCallback) { nsAutoPtr manager = new WMFAudioMFTManager(aConfig); nsRefPtr mft = manager->Init(); if (!mft) { return nullptr; } nsRefPtr decoder = new WMFMediaDataDecoder(manager.forget(), mft, aAudioTaskQueue, aCallback); return decoder.forget(); } bool WMFDecoderModule::ShouldUseDXVA(const VideoInfo& aConfig) const { static bool isAMD = false; static bool initialized = false; if (!initialized) { nsCOMPtr gfxInfo = services::GetGfxInfo(); nsAutoString vendor; gfxInfo->GetAdapterVendorID(vendor); isAMD = vendor.Equals(widget::GfxDriverInfo::GetDeviceVendor(widget::VendorAMD), nsCaseInsensitiveStringComparator()) || vendor.Equals(widget::GfxDriverInfo::GetDeviceVendor(widget::VendorATI), nsCaseInsensitiveStringComparator()); initialized = true; } if (!isAMD) { return true; } // Don't use DXVA for 4k videos or above, since it seems to perform poorly. return aConfig.mDisplay.width <= 1920 && aConfig.mDisplay.height <= 1200; } bool WMFDecoderModule::SupportsSharedDecoders(const VideoInfo& aConfig) const { // If DXVA is enabled, but we're not going to use it for this specific config, then // we can't use the shared decoder. return !AgnosticMimeType(aConfig.mMimeType) && (!sDXVAEnabled || ShouldUseDXVA(aConfig)); } bool WMFDecoderModule::SupportsMimeType(const nsACString& aMimeType) { return aMimeType.EqualsLiteral("video/mp4") || aMimeType.EqualsLiteral("video/avc") || aMimeType.EqualsLiteral("audio/mp4a-latm") || aMimeType.EqualsLiteral("audio/mpeg") || (sIsIntelDecoderEnabled && (aMimeType.EqualsLiteral("video/webm; codecs=vp8") || aMimeType.EqualsLiteral("video/webm; codecs=vp9"))); } PlatformDecoderModule::ConversionRequired WMFDecoderModule::DecoderNeedsConversion(const TrackInfo& aConfig) const { if (aConfig.IsVideo() && (aConfig.mMimeType.EqualsLiteral("video/avc") || aConfig.mMimeType.EqualsLiteral("video/mp4"))) { return kNeedAnnexB; } else { return kNeedNone; } } static bool ClassesRootRegKeyExists(const nsAString& aRegKeyPath) { nsresult rv; nsCOMPtr regKey = do_CreateInstance("@mozilla.org/windows-registry-key;1", &rv); if (NS_WARN_IF(NS_FAILED(rv))) { return false; } rv = regKey->Open(nsIWindowsRegKey::ROOT_KEY_CLASSES_ROOT, aRegKeyPath, nsIWindowsRegKey::ACCESS_READ); if (NS_FAILED(rv)) { return false; } regKey->Close(); return true; } /* static */ bool WMFDecoderModule::HasH264() { // CLSID_CMSH264DecoderMFT return ClassesRootRegKeyExists( NS_LITERAL_STRING("CLSID\\{32D186A7-218F-4C75-8876-DD77273A8999}")); } /* static */ bool WMFDecoderModule::HasAAC() { // CLSID_CMSAACDecMFT return ClassesRootRegKeyExists( NS_LITERAL_STRING("CLSID\\{62CE7E72-4C71-4D20-B15D-452831A87D9D}")); } } // namespace mozilla