Introduce "--with-system-libdrm" option, allowing to build Firefox with system drm library and to find system drm header files. Also unify all drm includes to match both system and bundled paths. Differential Revision: https://phabricator.services.mozilla.com/D242099
471 lines
16 KiB
C++
471 lines
16 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 "FFmpegVideoFramePool.h"
|
|
#include "PlatformDecoderModule.h"
|
|
#include "FFmpegLog.h"
|
|
#include "mozilla/widget/DMABufLibWrapper.h"
|
|
#include "libavutil/pixfmt.h"
|
|
#include "mozilla/StaticPrefs_media.h"
|
|
#include "mozilla/gfx/gfxVars.h"
|
|
#include "mozilla/widget/va_drmcommon.h"
|
|
|
|
// DMABufLibWrapper defines its own version of this which collides with the
|
|
// official version in drm_fourcc.h
|
|
#ifdef DRM_FORMAT_MOD_INVALID
|
|
# undef DRM_FORMAT_MOD_INVALID
|
|
#endif
|
|
#include <libdrm/drm_fourcc.h>
|
|
|
|
#ifdef MOZ_LOGGING
|
|
# undef DMABUF_LOG
|
|
extern mozilla::LazyLogModule gDmabufLog;
|
|
# define DMABUF_LOG(str, ...) \
|
|
MOZ_LOG(gDmabufLog, mozilla::LogLevel::Debug, (str, ##__VA_ARGS__))
|
|
#else
|
|
# define DMABUF_LOG(args)
|
|
#endif /* MOZ_LOGGING */
|
|
|
|
// Start copying surfaces when free ffmpeg surface count is below 1/4 of all
|
|
// available surfaces.
|
|
#define SURFACE_COPY_THRESHOLD (1.0f / 4.0f)
|
|
|
|
namespace mozilla {
|
|
|
|
RefPtr<layers::Image> VideoFrameSurface<LIBAV_VER>::GetAsImage() {
|
|
return new layers::DMABUFSurfaceImage(mSurface);
|
|
}
|
|
|
|
VideoFrameSurface<LIBAV_VER>::VideoFrameSurface(DMABufSurface* aSurface)
|
|
: mSurface(aSurface),
|
|
mLib(nullptr),
|
|
mAVHWFrameContext(nullptr),
|
|
mHWAVBuffer(nullptr) {
|
|
// Create global refcount object to track mSurface usage over
|
|
// gects rendering engine. We can't release it until it's used
|
|
// by GL compositor / WebRender.
|
|
MOZ_ASSERT(mSurface);
|
|
MOZ_RELEASE_ASSERT(mSurface->GetAsDMABufSurfaceYUV());
|
|
mSurface->GlobalRefCountCreate();
|
|
DMABUF_LOG("VideoFrameSurface: creating surface UID %d", mSurface->GetUID());
|
|
}
|
|
|
|
VideoFrameSurface<LIBAV_VER>::~VideoFrameSurface() {
|
|
DMABUF_LOG("~VideoFrameSurface: deleting dmabuf surface UID %d",
|
|
mSurface->GetUID());
|
|
mSurface->GlobalRefCountDelete();
|
|
// We're about to quit, no need to recycle the frames.
|
|
if (mFFMPEGSurfaceID) {
|
|
ReleaseVAAPIData(/* aForFrameRecycle */ false);
|
|
}
|
|
}
|
|
|
|
void VideoFrameSurface<LIBAV_VER>::LockVAAPIData(
|
|
AVCodecContext* aAVCodecContext, AVFrame* aAVFrame,
|
|
FFmpegLibWrapper* aLib) {
|
|
mLib = aLib;
|
|
|
|
// V4L2 frames don't have hw_frames_ctx because the v4l2-wrapper codecs
|
|
// don't actually use hwaccel. In this case we don't need to add a
|
|
// HW frame context reference
|
|
if (aAVCodecContext->hw_frames_ctx) {
|
|
mAVHWFrameContext = aLib->av_buffer_ref(aAVCodecContext->hw_frames_ctx);
|
|
mHWAVBuffer = aLib->av_buffer_ref(aAVFrame->buf[0]);
|
|
DMABUF_LOG(
|
|
"VideoFrameSurface: VAAPI locking dmabuf surface UID %d FFMPEG ID 0x%x "
|
|
"mAVHWFrameContext %p mHWAVBuffer %p",
|
|
mSurface->GetUID(), mFFMPEGSurfaceID.value(), mAVHWFrameContext,
|
|
mHWAVBuffer);
|
|
} else {
|
|
mAVHWFrameContext = nullptr;
|
|
mHWAVBuffer = aLib->av_buffer_ref(aAVFrame->buf[0]);
|
|
DMABUF_LOG(
|
|
"VideoFrameSurface: V4L2 locking dmabuf surface UID %d FFMPEG ID 0x%x "
|
|
"mHWAVBuffer %p",
|
|
mSurface->GetUID(), mFFMPEGSurfaceID.value(), mHWAVBuffer);
|
|
}
|
|
}
|
|
|
|
void VideoFrameSurface<LIBAV_VER>::ReleaseVAAPIData(bool aForFrameRecycle) {
|
|
DMABUF_LOG(
|
|
"VideoFrameSurface: Releasing dmabuf surface UID %d FFMPEG ID 0x%x "
|
|
"aForFrameRecycle %d mLib %p mAVHWFrameContext %p mHWAVBuffer %p",
|
|
mSurface->GetUID(), mFFMPEGSurfaceID.value(), aForFrameRecycle, mLib,
|
|
mAVHWFrameContext, mHWAVBuffer);
|
|
// It's possible to unref GPU data while IsUsed() is still set.
|
|
// It can happen when VideoFramePool is deleted while decoder shutdown
|
|
// but related dmabuf surfaces are still used in another process.
|
|
// In such case we don't care as the dmabuf surface will not be
|
|
// recycled for another frame and stays here untill last fd of it
|
|
// is closed.
|
|
if (mLib) {
|
|
mLib->av_buffer_unref(&mHWAVBuffer);
|
|
if (mAVHWFrameContext) {
|
|
mLib->av_buffer_unref(&mAVHWFrameContext);
|
|
}
|
|
mLib = nullptr;
|
|
}
|
|
|
|
mFFMPEGSurfaceID = Nothing();
|
|
mSurface->ReleaseSurface();
|
|
|
|
if (aForFrameRecycle && IsUsed()) {
|
|
NS_WARNING("Reusing live dmabuf surface, visual glitches ahead");
|
|
}
|
|
}
|
|
|
|
VideoFramePool<LIBAV_VER>::VideoFramePool(int aFFMPEGPoolSize)
|
|
: mSurfaceLock("VideoFramePoolSurfaceLock"),
|
|
mFFMPEGPoolSize(aFFMPEGPoolSize) {
|
|
DMABUF_LOG("VideoFramePool::VideoFramePool() pool size %d", mFFMPEGPoolSize);
|
|
}
|
|
|
|
VideoFramePool<LIBAV_VER>::~VideoFramePool() {
|
|
DMABUF_LOG("VideoFramePool::~VideoFramePool()");
|
|
MutexAutoLock lock(mSurfaceLock);
|
|
mDMABufSurfaces.Clear();
|
|
|
|
DMABufSurface::DeleteSnapshotGLContext();
|
|
}
|
|
|
|
void VideoFramePool<LIBAV_VER>::ReleaseUnusedVAAPIFrames() {
|
|
MutexAutoLock lock(mSurfaceLock);
|
|
for (const auto& surface : mDMABufSurfaces) {
|
|
#ifdef DEBUG
|
|
if (!surface->mFFMPEGSurfaceID && surface->IsUsed()) {
|
|
NS_WARNING("Untracked but still used dmabug surface!");
|
|
}
|
|
#endif
|
|
if (surface->mFFMPEGSurfaceID && !surface->IsUsed()) {
|
|
surface->ReleaseVAAPIData();
|
|
}
|
|
}
|
|
}
|
|
|
|
RefPtr<VideoFrameSurface<LIBAV_VER>>
|
|
VideoFramePool<LIBAV_VER>::GetFreeVideoFrameSurface() {
|
|
for (auto& surface : mDMABufSurfaces) {
|
|
if (!surface->mFFMPEGSurfaceID) {
|
|
return surface;
|
|
}
|
|
if (surface->IsUsed()) {
|
|
continue;
|
|
}
|
|
surface->ReleaseVAAPIData();
|
|
return surface;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void VideoFramePool<LIBAV_VER>::CheckNewFFMPEGSurface(
|
|
VASurfaceID aNewSurfaceID) {
|
|
for (const auto& surface : mDMABufSurfaces) {
|
|
if (surface->IsUsed() && surface->IsFFMPEGSurface()) {
|
|
MOZ_DIAGNOSTIC_ASSERT(surface->mFFMPEGSurfaceID.value() != aNewSurfaceID);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool VideoFramePool<LIBAV_VER>::ShouldCopySurface() {
|
|
// Number of used HW surfaces.
|
|
int surfacesUsed = 0;
|
|
int surfacesUsedFFmpeg = 0;
|
|
for (const auto& surface : mDMABufSurfaces) {
|
|
if (surface->IsUsed()) {
|
|
surfacesUsed++;
|
|
if (surface->IsFFMPEGSurface()) {
|
|
DMABUF_LOG(
|
|
"Used HW surface UID %d FFMPEG ID 0x%x\n",
|
|
surface->mSurface->GetUID(),
|
|
surface->mFFMPEGSurfaceID ? surface->mFFMPEGSurfaceID.value() : -1);
|
|
surfacesUsedFFmpeg++;
|
|
}
|
|
}
|
|
}
|
|
float freeRatio = 1.0f - (surfacesUsedFFmpeg / (float)mFFMPEGPoolSize);
|
|
DMABUF_LOG(
|
|
"Surface pool size %d used copied %d used ffmpeg %d (max %d) free ratio "
|
|
"%f",
|
|
(int)mDMABufSurfaces.Length(), surfacesUsed - surfacesUsedFFmpeg,
|
|
surfacesUsedFFmpeg, mFFMPEGPoolSize, freeRatio);
|
|
if (!gfx::gfxVars::HwDecodedVideoZeroCopy()) {
|
|
return true;
|
|
}
|
|
return freeRatio < SURFACE_COPY_THRESHOLD;
|
|
}
|
|
|
|
RefPtr<VideoFrameSurface<LIBAV_VER>>
|
|
VideoFramePool<LIBAV_VER>::GetVideoFrameSurface(
|
|
VADRMPRIMESurfaceDescriptor& aVaDesc, int aWidth, int aHeight,
|
|
AVCodecContext* aAVCodecContext, AVFrame* aAVFrame,
|
|
FFmpegLibWrapper* aLib) {
|
|
if (aVaDesc.fourcc != VA_FOURCC_NV12 && aVaDesc.fourcc != VA_FOURCC_YV12 &&
|
|
aVaDesc.fourcc != VA_FOURCC_P010 && aVaDesc.fourcc != VA_FOURCC_P016) {
|
|
DMABUF_LOG("Unsupported VA-API surface format %d", aVaDesc.fourcc);
|
|
return nullptr;
|
|
}
|
|
|
|
MutexAutoLock lock(mSurfaceLock);
|
|
|
|
RefPtr<DMABufSurfaceYUV> surface;
|
|
RefPtr<VideoFrameSurface<LIBAV_VER>> videoSurface =
|
|
GetFreeVideoFrameSurface();
|
|
if (!videoSurface) {
|
|
surface = new DMABufSurfaceYUV();
|
|
videoSurface = new VideoFrameSurface<LIBAV_VER>(surface);
|
|
mDMABufSurfaces.AppendElement(videoSurface);
|
|
} else {
|
|
surface = videoSurface->GetDMABufSurface();
|
|
}
|
|
VASurfaceID ffmpegSurfaceID = (uintptr_t)aAVFrame->data[3];
|
|
DMABUF_LOG("Using VA-API DMABufSurface UID %d FFMPEG ID 0x%x",
|
|
surface->GetUID(), ffmpegSurfaceID);
|
|
|
|
bool copySurface = mTextureCopyWorks && ShouldCopySurface();
|
|
if (!surface->UpdateYUVData(aVaDesc, aWidth, aHeight, copySurface)) {
|
|
if (!copySurface) {
|
|
// Failed without texture copy. We can't do more here.
|
|
return nullptr;
|
|
}
|
|
// Try again without texture copy
|
|
DMABUF_LOG(" DMABuf texture copy is broken");
|
|
copySurface = mTextureCopyWorks = false;
|
|
if (!surface->UpdateYUVData(aVaDesc, aWidth, aHeight, copySurface)) {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
if (MOZ_UNLIKELY(!mTextureCreationWorks)) {
|
|
mTextureCreationWorks = Some(surface->VerifyTextureCreation());
|
|
if (!*mTextureCreationWorks) {
|
|
DMABUF_LOG(" failed to create texture over DMABuf memory!");
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
videoSurface->MarkAsUsed(ffmpegSurfaceID);
|
|
|
|
if (!copySurface) {
|
|
// Check that newly added ffmpeg surface isn't already used by different
|
|
// VideoFrameSurface.
|
|
CheckNewFFMPEGSurface(ffmpegSurfaceID);
|
|
videoSurface->LockVAAPIData(aAVCodecContext, aAVFrame, aLib);
|
|
}
|
|
return videoSurface;
|
|
}
|
|
|
|
static gfx::SurfaceFormat GetSurfaceFormat(enum AVPixelFormat aPixFmt) {
|
|
switch (aPixFmt) {
|
|
case AV_PIX_FMT_YUV420P10LE:
|
|
return gfx::SurfaceFormat::YUV420P10;
|
|
case AV_PIX_FMT_YUV420P:
|
|
return gfx::SurfaceFormat::YUV420;
|
|
default:
|
|
return gfx::SurfaceFormat::UNKNOWN;
|
|
}
|
|
}
|
|
|
|
// TODO: Add support for AV_PIX_FMT_YUV444P / AV_PIX_FMT_GBRP
|
|
RefPtr<VideoFrameSurface<LIBAV_VER>>
|
|
VideoFramePool<LIBAV_VER>::GetVideoFrameSurface(
|
|
const layers::PlanarYCbCrData& aData, AVCodecContext* aAVCodecContext) {
|
|
static gfx::SurfaceFormat format = GetSurfaceFormat(aAVCodecContext->pix_fmt);
|
|
if (format == gfx::SurfaceFormat::UNKNOWN) {
|
|
DMABUF_LOG("Unsupported FFmpeg DMABuf format %x", aAVCodecContext->pix_fmt);
|
|
return nullptr;
|
|
}
|
|
|
|
MutexAutoLock lock(mSurfaceLock);
|
|
|
|
RefPtr<DMABufSurfaceYUV> surface;
|
|
RefPtr<VideoFrameSurface<LIBAV_VER>> videoSurface =
|
|
GetFreeVideoFrameSurface();
|
|
if (!videoSurface) {
|
|
surface = new DMABufSurfaceYUV();
|
|
videoSurface = new VideoFrameSurface<LIBAV_VER>(surface);
|
|
mDMABufSurfaces.AppendElement(videoSurface);
|
|
} else {
|
|
surface = videoSurface->GetDMABufSurface();
|
|
}
|
|
|
|
DMABUF_LOG("Using SW DMABufSurface UID %d", surface->GetUID());
|
|
|
|
if (!surface->UpdateYUVData(aData, format)) {
|
|
DMABUF_LOG(" failed to convert YUV data to DMABuf memory!");
|
|
return nullptr;
|
|
}
|
|
|
|
if (MOZ_UNLIKELY(!mTextureCreationWorks)) {
|
|
mTextureCreationWorks = Some(surface->VerifyTextureCreation());
|
|
if (!*mTextureCreationWorks) {
|
|
DMABUF_LOG(" failed to create texture over DMABuf memory!");
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
videoSurface->MarkAsUsed(/* Any random value as FFmpegID */ 1);
|
|
return videoSurface;
|
|
}
|
|
|
|
// Convert an FFmpeg-specific DRM descriptor into a
|
|
// VADRMPRIMESurfaceDescriptor. There is no fundamental difference between
|
|
// the descriptor structs and using the latter means this can use all the
|
|
// existing machinery in DMABufSurfaceYUV.
|
|
static Maybe<VADRMPRIMESurfaceDescriptor> FFmpegDescToVA(
|
|
AVDRMFrameDescriptor& aDesc, AVFrame* aAVFrame) {
|
|
VADRMPRIMESurfaceDescriptor vaDesc{};
|
|
|
|
if (aAVFrame->format != AV_PIX_FMT_DRM_PRIME) {
|
|
DMABUF_LOG("Got non-DRM-PRIME frame from FFmpeg V4L2");
|
|
return Nothing();
|
|
}
|
|
|
|
if (aAVFrame->crop_top != 0 || aAVFrame->crop_left != 0) {
|
|
DMABUF_LOG("Top and left-side cropping are not supported");
|
|
return Nothing();
|
|
}
|
|
|
|
// Width and height after crop
|
|
vaDesc.width = aAVFrame->width;
|
|
vaDesc.height = aAVFrame->height - aAVFrame->crop_bottom;
|
|
|
|
// Native width and height before crop is applied
|
|
unsigned int uncrop_width = aDesc.layers[0].planes[0].pitch;
|
|
unsigned int uncrop_height = aAVFrame->height;
|
|
|
|
unsigned int offset = aDesc.layers[0].planes[0].offset;
|
|
|
|
if (aDesc.layers[0].format == DRM_FORMAT_YUV420) {
|
|
vaDesc.fourcc = VA_FOURCC_YV12;
|
|
|
|
// V4L2 expresses YUV420 as a single contiguous buffer containing
|
|
// all three planes. DMABufSurfaceYUV expects the three planes
|
|
// separately, so we have to split them out
|
|
MOZ_ASSERT(aDesc.nb_objects == 1);
|
|
MOZ_ASSERT(aDesc.nb_layers == 1);
|
|
|
|
vaDesc.num_objects = 1;
|
|
vaDesc.objects[0].drm_format_modifier = aDesc.objects[0].format_modifier;
|
|
vaDesc.objects[0].size = aDesc.objects[0].size;
|
|
vaDesc.objects[0].fd = aDesc.objects[0].fd;
|
|
|
|
vaDesc.num_layers = 3;
|
|
for (int i = 0; i < 3; i++) {
|
|
vaDesc.layers[i].drm_format = DRM_FORMAT_R8;
|
|
vaDesc.layers[i].num_planes = 1;
|
|
vaDesc.layers[i].object_index[0] = 0;
|
|
}
|
|
vaDesc.layers[0].offset[0] = offset;
|
|
vaDesc.layers[0].pitch[0] = uncrop_width;
|
|
vaDesc.layers[1].offset[0] = offset + uncrop_width * uncrop_height;
|
|
vaDesc.layers[1].pitch[0] = uncrop_width / 2;
|
|
vaDesc.layers[2].offset[0] = offset + uncrop_width * uncrop_height * 5 / 4;
|
|
vaDesc.layers[2].pitch[0] = uncrop_width / 2;
|
|
} else if (aDesc.layers[0].format == DRM_FORMAT_NV12) {
|
|
vaDesc.fourcc = VA_FOURCC_NV12;
|
|
|
|
// V4L2 expresses NV12 as a single contiguous buffer containing both
|
|
// planes. DMABufSurfaceYUV expects the two planes separately, so we have
|
|
// to split them out
|
|
MOZ_ASSERT(aDesc.nb_objects == 1);
|
|
MOZ_ASSERT(aDesc.nb_layers == 1);
|
|
|
|
vaDesc.num_objects = 1;
|
|
vaDesc.objects[0].drm_format_modifier = aDesc.objects[0].format_modifier;
|
|
vaDesc.objects[0].size = aDesc.objects[0].size;
|
|
vaDesc.objects[0].fd = aDesc.objects[0].fd;
|
|
|
|
vaDesc.num_layers = 2;
|
|
for (int i = 0; i < 2; i++) {
|
|
vaDesc.layers[i].num_planes = 1;
|
|
vaDesc.layers[i].object_index[0] = 0;
|
|
vaDesc.layers[i].pitch[0] = uncrop_width;
|
|
}
|
|
vaDesc.layers[0].drm_format = DRM_FORMAT_R8; // Y plane
|
|
vaDesc.layers[0].offset[0] = offset;
|
|
vaDesc.layers[1].drm_format = DRM_FORMAT_GR88; // UV plane
|
|
vaDesc.layers[1].offset[0] = offset + uncrop_width * uncrop_height;
|
|
} else {
|
|
DMABUF_LOG("Don't know how to deal with FOURCC 0x%x",
|
|
aDesc.layers[0].format);
|
|
return Nothing();
|
|
}
|
|
|
|
return Some(vaDesc);
|
|
}
|
|
|
|
RefPtr<VideoFrameSurface<LIBAV_VER>>
|
|
VideoFramePool<LIBAV_VER>::GetVideoFrameSurface(AVDRMFrameDescriptor& aDesc,
|
|
int aWidth, int aHeight,
|
|
AVCodecContext* aAVCodecContext,
|
|
AVFrame* aAVFrame,
|
|
FFmpegLibWrapper* aLib) {
|
|
MOZ_ASSERT(aDesc.nb_layers > 0);
|
|
|
|
auto layerDesc = FFmpegDescToVA(aDesc, aAVFrame);
|
|
if (layerDesc.isNothing()) {
|
|
return nullptr;
|
|
}
|
|
|
|
// Width and height, after cropping
|
|
int crop_width = (int)layerDesc->width;
|
|
int crop_height = (int)layerDesc->height;
|
|
|
|
// Use the descriptor's address as an id to track ffmpeg surfaces
|
|
unsigned int ffmpegSurfaceID = (uintptr_t)&aDesc;
|
|
|
|
MutexAutoLock lock(mSurfaceLock);
|
|
|
|
RefPtr<DMABufSurfaceYUV> surface;
|
|
RefPtr<VideoFrameSurface<LIBAV_VER>> videoSurface =
|
|
GetFreeVideoFrameSurface();
|
|
if (!videoSurface) {
|
|
surface = new DMABufSurfaceYUV();
|
|
videoSurface = new VideoFrameSurface<LIBAV_VER>(surface);
|
|
mDMABufSurfaces.AppendElement(videoSurface);
|
|
} else {
|
|
surface = videoSurface->GetDMABufSurface();
|
|
}
|
|
DMABUF_LOG("Using V4L2 DMABufSurface UID %d FFMPEG ID 0x%x",
|
|
surface->GetUID(), ffmpegSurfaceID);
|
|
|
|
bool copySurface = mTextureCopyWorks && ShouldCopySurface();
|
|
if (!surface->UpdateYUVData(layerDesc.value(), crop_width, crop_height,
|
|
copySurface)) {
|
|
if (!copySurface) {
|
|
// Failed without texture copy. We can't do more here.
|
|
return nullptr;
|
|
}
|
|
// Try again without texture copy
|
|
DMABUF_LOG(" DMABuf texture copy is broken");
|
|
copySurface = mTextureCopyWorks = false;
|
|
if (!surface->UpdateYUVData(layerDesc.value(), crop_width, crop_height,
|
|
copySurface)) {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
if (MOZ_UNLIKELY(!mTextureCreationWorks)) {
|
|
mTextureCreationWorks = Some(surface->VerifyTextureCreation());
|
|
if (!*mTextureCreationWorks) {
|
|
DMABUF_LOG(" failed to create texture over DMABuf memory!");
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
videoSurface->MarkAsUsed(ffmpegSurfaceID);
|
|
|
|
if (!copySurface) {
|
|
// Check that newly added ffmpeg surface isn't already used by different
|
|
// VideoFrameSurface.
|
|
CheckNewFFMPEGSurface(ffmpegSurfaceID);
|
|
videoSurface->LockVAAPIData(aAVCodecContext, aAVFrame, aLib);
|
|
}
|
|
return videoSurface;
|
|
}
|
|
|
|
} // namespace mozilla
|