fune/dom/media/platforms/ffmpeg/ffvpx/FFVPXRuntimeLinker.cpp
Cristian Tuns 48a036458a Backed out 5 changesets (bug 1879873) for causing reftest failures in asan_malloc_win.cpp CLOSED TREE
Backed out changeset e519654e5d8b (bug 1879873)
Backed out changeset 9b17ac4800c6 (bug 1879873)
Backed out changeset 2e5121a988a8 (bug 1879873)
Backed out changeset 9973c75413d6 (bug 1879873)
Backed out changeset 7c947114da80 (bug 1879873)
2024-02-27 10:39:59 -05:00

163 lines
4.6 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 "FFVPXRuntimeLinker.h"
#include "FFmpegLibWrapper.h"
#include "FFmpegLog.h"
#include "BinaryPath.h"
#include "mozilla/FileUtils.h"
#include "nsLocalFile.h"
#include "prmem.h"
#include "prlink.h"
#ifdef XP_WIN
# include <windows.h>
#endif
namespace mozilla {
template <int V>
class FFmpegDecoderModule {
public:
static already_AddRefed<PlatformDecoderModule> Create(FFmpegLibWrapper*);
};
template <int V>
class FFmpegEncoderModule {
public:
static already_AddRefed<PlatformEncoderModule> Create(FFmpegLibWrapper*);
};
static FFmpegLibWrapper sFFVPXLib;
StaticMutex FFVPXRuntimeLinker::sMutex;
FFVPXRuntimeLinker::LinkStatus FFVPXRuntimeLinker::sLinkStatus =
LinkStatus_INIT;
static PRLibrary* MozAVLink(nsIFile* aFile) {
PRLibSpec lspec;
PathString path = aFile->NativePath();
#ifdef XP_WIN
lspec.type = PR_LibSpec_PathnameU;
lspec.value.pathname_u = path.get();
#else
# if defined(XP_OPENBSD)
/* on OpenBSD, libmozavcodec.so and libmozavutil.so are preloaded before
* sandboxing so make sure only the filename is passed to
* PR_LoadLibraryWithFlags(), dlopen() will return the preloaded library
* handle instead of failing to find it due to sandboxing */
nsAutoCString leaf;
nsresult rv = aFile->GetNativeLeafName(leaf);
if (NS_SUCCEEDED(rv)) {
path = PathString(leaf);
}
# endif // defined(XP_OPENBSD)
lspec.type = PR_LibSpec_Pathname;
lspec.value.pathname = path.get();
#endif
#ifdef MOZ_WIDGET_ANDROID
PRLibrary* lib = PR_LoadLibraryWithFlags(lspec, PR_LD_NOW | PR_LD_GLOBAL);
#else
PRLibrary* lib = PR_LoadLibraryWithFlags(lspec, PR_LD_NOW | PR_LD_LOCAL);
#endif
if (!lib) {
FFMPEGV_LOG("unable to load library %s", aFile->HumanReadablePath().get());
}
return lib;
}
/* static */
bool FFVPXRuntimeLinker::Init() {
// Enter critical section to set up ffvpx.
StaticMutexAutoLock lock(sMutex);
if (sLinkStatus) {
return sLinkStatus == LinkStatus_SUCCEEDED;
}
sLinkStatus = LinkStatus_FAILED;
#ifdef MOZ_WIDGET_GTK
sFFVPXLib.LinkVAAPILibs();
#endif
nsCOMPtr<nsIFile> libFile;
if (NS_FAILED(mozilla::BinaryPath::GetFile(getter_AddRefs(libFile)))) {
return false;
}
#ifdef XP_DARWIN
if (!XRE_IsParentProcess() &&
(XRE_GetChildProcBinPathType(XRE_GetProcessType()) ==
BinPathType::PluginContainer)) {
// On macOS, PluginContainer processes have their binary in a
// plugin-container.app/Content/MacOS/ directory.
nsCOMPtr<nsIFile> parentDir1, parentDir2;
if (NS_FAILED(libFile->GetParent(getter_AddRefs(parentDir1)))) {
return false;
}
if (NS_FAILED(parentDir1->GetParent(getter_AddRefs(parentDir2)))) {
return false;
}
if (NS_FAILED(parentDir2->GetParent(getter_AddRefs(libFile)))) {
return false;
}
}
#endif
if (NS_FAILED(libFile->SetNativeLeafName(MOZ_DLL_PREFIX
"mozavutil" MOZ_DLL_SUFFIX ""_ns))) {
return false;
}
sFFVPXLib.mAVUtilLib = MozAVLink(libFile);
if (NS_FAILED(libFile->SetNativeLeafName(
MOZ_DLL_PREFIX "mozavcodec" MOZ_DLL_SUFFIX ""_ns))) {
return false;
}
sFFVPXLib.mAVCodecLib = MozAVLink(libFile);
FFmpegLibWrapper::LinkResult res = sFFVPXLib.Link();
FFMPEGP_LOG("Link result: %s", FFmpegLibWrapper::LinkResultToString(res));
if (res == FFmpegLibWrapper::LinkResult::Success) {
sLinkStatus = LinkStatus_SUCCEEDED;
return true;
}
return false;
}
/* static */
already_AddRefed<PlatformDecoderModule> FFVPXRuntimeLinker::CreateDecoder() {
if (!Init()) {
return nullptr;
}
return FFmpegDecoderModule<FFVPX_VERSION>::Create(&sFFVPXLib);
}
/* static */
already_AddRefed<PlatformEncoderModule> FFVPXRuntimeLinker::CreateEncoder() {
if (!Init()) {
return nullptr;
}
return FFmpegEncoderModule<FFVPX_VERSION>::Create(&sFFVPXLib);
}
/* static */
void FFVPXRuntimeLinker::GetRDFTFuncs(FFmpegRDFTFuncs* aOutFuncs) {
[]() MOZ_NO_THREAD_SAFETY_ANALYSIS {
MOZ_ASSERT(sLinkStatus != LinkStatus_INIT);
}();
if (sFFVPXLib.av_rdft_init && sFFVPXLib.av_rdft_calc &&
sFFVPXLib.av_rdft_end) {
aOutFuncs->init = sFFVPXLib.av_rdft_init;
aOutFuncs->calc = sFFVPXLib.av_rdft_calc;
aOutFuncs->end = sFFVPXLib.av_rdft_end;
} else {
NS_WARNING("RDFT functions expected but not found");
*aOutFuncs = FFmpegRDFTFuncs(); // zero
}
}
} // namespace mozilla