mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:30:27 +00:00
44d3f3aca1
- Bug 1148102: P5. Add Opus/VPX/Vorbis MediaDataDecoder. r=jya (b07fea1ea) - Bug 1148102: P6. Add WebMDemuxer object. r=jya (225cee37f) - reformat comment (9d12af823) - reformat comment (80765706d) - Bug 1148102: P7. Hookup WebMDemuxer. r=jya (78f973b17) - Bug 1185782 - Remove media.windows-media-foundation.enabled pref. r=jya (2fb2c5e4a) - Bug 1163486 - Remove MP4Reader. r=jya (0376d6d60) and some custom changes for our tree: - change MediaReadAt to SilentReadAt in new file dom/media/webm/WebMDemuxer.cpp - add missing export of "vpx_codec_err_to_string" in layout/media/symbols.def.in
216 lines
6.4 KiB
C++
216 lines
6.4 KiB
C++
/* -*- 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 "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<HRESULT> 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<MediaDataDecoder>
|
|
WMFDecoderModule::CreateVideoDecoder(const VideoInfo& aConfig,
|
|
layers::LayersBackend aLayersBackend,
|
|
layers::ImageContainer* aImageContainer,
|
|
FlushableTaskQueue* aVideoTaskQueue,
|
|
MediaDataDecoderCallback* aCallback)
|
|
{
|
|
nsRefPtr<MediaDataDecoder> decoder =
|
|
new WMFMediaDataDecoder(new WMFVideoMFTManager(aConfig,
|
|
aLayersBackend,
|
|
aImageContainer,
|
|
sDXVAEnabled && ShouldUseDXVA(aConfig)),
|
|
aVideoTaskQueue,
|
|
aCallback);
|
|
return decoder.forget();
|
|
}
|
|
|
|
already_AddRefed<MediaDataDecoder>
|
|
WMFDecoderModule::CreateAudioDecoder(const AudioInfo& aConfig,
|
|
FlushableTaskQueue* aAudioTaskQueue,
|
|
MediaDataDecoderCallback* aCallback)
|
|
{
|
|
nsRefPtr<MediaDataDecoder> decoder =
|
|
new WMFMediaDataDecoder(new WMFAudioMFTManager(aConfig),
|
|
aAudioTaskQueue,
|
|
aCallback);
|
|
return decoder.forget();
|
|
}
|
|
|
|
bool
|
|
WMFDecoderModule::ShouldUseDXVA(const VideoInfo& aConfig) const
|
|
{
|
|
static bool isAMD = false;
|
|
static bool initialized = false;
|
|
if (!initialized) {
|
|
nsCOMPtr<nsIGfxInfo> 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<nsIWindowsRegKey> 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
|