mirror of
https://github.com/Ryujinx/Ryujinx.git
synced 2025-01-01 17:25:59 +00:00
16c649934f
* ffmpeg: Add extra checks and error messages This PR adds some checks and logging error messages related to the ffmpeg context creation, that will prevent users to open issues because they don't have the correct packages installed. Close #2762 * Update FFmpegContext.cs
187 lines
5.8 KiB
C#
187 lines
5.8 KiB
C#
using FFmpeg.AutoGen;
|
|
using Ryujinx.Common.Logging;
|
|
using System;
|
|
using System.Diagnostics;
|
|
using System.IO;
|
|
using System.Runtime.InteropServices;
|
|
|
|
namespace Ryujinx.Graphics.Nvdec.FFmpeg
|
|
{
|
|
unsafe class FFmpegContext : IDisposable
|
|
{
|
|
private readonly AVCodec_decode _decodeFrame;
|
|
private static readonly av_log_set_callback_callback _logFunc;
|
|
private readonly AVCodec* _codec;
|
|
private AVPacket* _packet;
|
|
private AVCodecContext* _context;
|
|
|
|
public FFmpegContext(AVCodecID codecId)
|
|
{
|
|
_codec = ffmpeg.avcodec_find_decoder(codecId);
|
|
if (_codec == null)
|
|
{
|
|
Logger.Error?.PrintMsg(LogClass.FFmpeg, $"Codec wasn't found. Make sure you have the {codecId} codec present in your FFmpeg installation.");
|
|
|
|
return;
|
|
}
|
|
|
|
_context = ffmpeg.avcodec_alloc_context3(_codec);
|
|
if (_context == null)
|
|
{
|
|
Logger.Error?.PrintMsg(LogClass.FFmpeg, "Codec context couldn't be allocated.");
|
|
|
|
return;
|
|
}
|
|
|
|
if (ffmpeg.avcodec_open2(_context, _codec, null) != 0)
|
|
{
|
|
Logger.Error?.PrintMsg(LogClass.FFmpeg, "Codec couldn't be opened.");
|
|
|
|
return;
|
|
}
|
|
|
|
_packet = ffmpeg.av_packet_alloc();
|
|
if (_packet == null)
|
|
{
|
|
Logger.Error?.PrintMsg(LogClass.FFmpeg, "Packet couldn't be allocated.");
|
|
|
|
return;
|
|
}
|
|
|
|
_decodeFrame = Marshal.GetDelegateForFunctionPointer<AVCodec_decode>(_codec->decode.Pointer);
|
|
}
|
|
|
|
static FFmpegContext()
|
|
{
|
|
SetRootPath();
|
|
|
|
_logFunc = Log;
|
|
|
|
// Redirect log output.
|
|
ffmpeg.av_log_set_level(ffmpeg.AV_LOG_MAX_OFFSET);
|
|
ffmpeg.av_log_set_callback(_logFunc);
|
|
}
|
|
|
|
private static void SetRootPath()
|
|
{
|
|
if (OperatingSystem.IsLinux())
|
|
{
|
|
// Configure FFmpeg search path
|
|
Process lddProcess = Process.Start(new ProcessStartInfo
|
|
{
|
|
FileName = "/bin/sh",
|
|
Arguments = "-c \"ldd $(which ffmpeg 2>/dev/null) | grep libavfilter\" 2>/dev/null",
|
|
UseShellExecute = false,
|
|
RedirectStandardOutput = true
|
|
});
|
|
|
|
string lddOutput = lddProcess.StandardOutput.ReadToEnd();
|
|
|
|
lddProcess.WaitForExit();
|
|
lddProcess.Close();
|
|
|
|
if (lddOutput.Contains(" => "))
|
|
{
|
|
ffmpeg.RootPath = Path.GetDirectoryName(lddOutput.Split(" => ")[1]);
|
|
}
|
|
else
|
|
{
|
|
Logger.Error?.PrintMsg(LogClass.FFmpeg, "FFmpeg wasn't found. Make sure that you have it installed and up to date.");
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void Log(void* p0, int level, string format, byte* vl)
|
|
{
|
|
if (level > ffmpeg.av_log_get_level())
|
|
{
|
|
return;
|
|
}
|
|
|
|
int lineSize = 1024;
|
|
byte* lineBuffer = stackalloc byte[lineSize];
|
|
int printPrefix = 1;
|
|
|
|
ffmpeg.av_log_format_line(p0, level, format, vl, lineBuffer, lineSize, &printPrefix);
|
|
|
|
string line = Marshal.PtrToStringAnsi((IntPtr)lineBuffer).Trim();
|
|
|
|
switch (level)
|
|
{
|
|
case ffmpeg.AV_LOG_PANIC:
|
|
case ffmpeg.AV_LOG_FATAL:
|
|
case ffmpeg.AV_LOG_ERROR:
|
|
Logger.Error?.Print(LogClass.FFmpeg, line);
|
|
break;
|
|
case ffmpeg.AV_LOG_WARNING:
|
|
Logger.Warning?.Print(LogClass.FFmpeg, line);
|
|
break;
|
|
case ffmpeg.AV_LOG_INFO:
|
|
Logger.Info?.Print(LogClass.FFmpeg, line);
|
|
break;
|
|
case ffmpeg.AV_LOG_VERBOSE:
|
|
case ffmpeg.AV_LOG_DEBUG:
|
|
case ffmpeg.AV_LOG_TRACE:
|
|
Logger.Debug?.Print(LogClass.FFmpeg, line);
|
|
break;
|
|
}
|
|
}
|
|
|
|
public int DecodeFrame(Surface output, ReadOnlySpan<byte> bitstream)
|
|
{
|
|
ffmpeg.av_frame_unref(output.Frame);
|
|
|
|
int result;
|
|
int gotFrame;
|
|
|
|
fixed (byte* ptr = bitstream)
|
|
{
|
|
_packet->data = ptr;
|
|
_packet->size = bitstream.Length;
|
|
result = _decodeFrame(_context, output.Frame, &gotFrame, _packet);
|
|
}
|
|
|
|
if (gotFrame == 0)
|
|
{
|
|
ffmpeg.av_frame_unref(output.Frame);
|
|
|
|
// If the frame was not delivered, it was probably delayed.
|
|
// Get the next delayed frame by passing a 0 length packet.
|
|
_packet->data = null;
|
|
_packet->size = 0;
|
|
result = _decodeFrame(_context, output.Frame, &gotFrame, _packet);
|
|
|
|
// We need to set B frames to 0 as we already consumed all delayed frames.
|
|
// This prevents the decoder from trying to return a delayed frame next time.
|
|
_context->has_b_frames = 0;
|
|
}
|
|
|
|
ffmpeg.av_packet_unref(_packet);
|
|
|
|
if (gotFrame == 0)
|
|
{
|
|
ffmpeg.av_frame_unref(output.Frame);
|
|
|
|
return -1;
|
|
}
|
|
|
|
return result < 0 ? result : 0;
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
fixed (AVPacket** ppPacket = &_packet)
|
|
{
|
|
ffmpeg.av_packet_free(ppPacket);
|
|
}
|
|
|
|
ffmpeg.avcodec_close(_context);
|
|
|
|
fixed (AVCodecContext** ppContext = &_context)
|
|
{
|
|
ffmpeg.avcodec_free_context(ppContext);
|
|
}
|
|
}
|
|
}
|
|
}
|